从wap页面跳转app(Android)

注:代码块中出现了代码块中的代码块样式,请大家忽略这些,因为第一次用markdown不知道如何导致已经如何去掉,知道的朋友还请回复我。感谢!

概述

最近产品提出了新的需求:通过wap打开手机本地APP。
功能的实现主要涉及了以下几个知识点:

  1. scheme打开应用
  2. Theme.NoDisplay的使用
  3. activity父级activity的重建
scheme

scheme类似自定义url协议,我们可以通过自定义的协议来打开自己的应用,形如:
<pre><code>txvideo://xxxx</code></pre>

Theme.NoDisplay

在Android中想进行一些无界面的处理又不适合使用service时,此时可以在项目的AndroidManifest.xml文件中相应的Activity标签中添加这样一行:
<pre><code>android:theme=”@android:style/Theme.NoDisplay</code></pre>

activity父级activity的重建

当我们从wap页跳转到应用内部时,可能我们跳转到的不是应用的第一层级页面而是深层次的页面,这时候我们需要在关闭应用内部页面时,对高层的页面进行重建。

  1. 为activity指定父级activity
    <pre><code>
    <application ...>
    <activity ...
    android:name="com.example.Mainactivity">
    </activity>
    <activity ...
    android:name="com.example.SecondeActivity">
    android:parentActivityName="com.example.Mainactivity" >

    <meta-data
    android:name="android.support.PARENT_ACTIVITY"
    android:value="com.example.Mainactivity" />
    </activity>
    </application>
    </code></pre> </br>
  2. 二级页面返回时,重建任务栈
    <pre><code>
    @Override
    public void onBackPressed() {
    // 获得指向父级activity的intent,NavUtils在support v4 包中
    Intent upIntent = NavUtils.getParentActivityIntent(this);
    // 判断是否需要重建任务栈
    if (NavUtils.shouldUpRecreateTask(this, upIntent)) {
    // 这个activity不是这个app任务的一部分, 所以当向上导航时创建
    // 用合成后退栈(synthesized back stack)创建一个新任务。
    TaskStackBuilder.create(this)
    // 添加这个activity的所有父activity到后退栈中
    .addNextIntentWithParentStack(upIntent)
    // 向上导航到最近的一个父activity
    .startActivities();
    } else {
    // 这个activity是这个app任务的一部分, 所以
    // 向上导航至逻辑父activity.
    NavUtils.navigateUpTo(this, upIntent);
    }
    super.onBackPressed();
    }
    </code></pre> </br>

效果与分析

效果图
效果图

第一次录屏,效果不大好(有好的录屏方法请告诉我),建议下载代码测试

我们点击网页中的立即打开->弹出secondActivity->点击closeactivity按钮,程序对secondActivity的父级activity进行重建

编码与实现

  • html页面
    <pre><code>
    <a id="openJD" href="appscheme://contentId">立即打开//</span></a></br>
    <a id="openJD" href="appscheme:///contentId">立即打开///</span></a>
    </code></pre>
    简单的两个超链接</br>

  • AndroidMainifast.xml
    <pre><code>
    ...
    <application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
    android:name=".MainActivity"
    android:label="@string/main_activity" >
    <intent-filter>
    <action android:name="android.intent.action.MAIN" />
    <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
    </activity>
    <activity
    android:name=".SecondActivity"
    android:label="@string/second_activity"
    android:parentActivityName=".MainActivity" >
    <meta-data
    android:name="android.support.PARENT_ACTIVITY"
    android:value=".MainActivity" />
    </activity>
    <activity
    android:name=".SchemeCenterActivity"
    android:label="@string/title_activity_scheme_center"
    android:theme="@android:style/Theme.NoDisplay">
    <intent-filter>
    <action android:name="android.intent.action.VIEW" />
    <category android:name="android.intent.category.DEFAULT" />
    <category android:name="android.intent.category.BROWSABLE" />
    <data android:scheme="appscheme" />
    </intent-filter>
    </activity>
    </application>
    ...
    </code></pre>
    在AndroidManifest中定义了,二级activity 'SecondeActivity',一级activity ‘MainActivity’ 它是SecondeActivity的父级activity。
    定义了不可显示的activity‘SchemeActivity’来处理scheme的调起,可对scheme的path部分进行处理在跳转到不同activity。

  • SchemeCenterActivity.java
    <pre><code>
    @Override
    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_scheme_center);
    Uri uri = getIntent().getData();
    Toast.makeText(this, "Uri:"+uri.toString(), Toast.LENGTH_SHORT).show();
    Toast.makeText(this, "Uri path:"+uri.getPath(), Toast.LENGTH_SHORT).show();
    Intent intent = new Intent(this, SecondActivity.class);
    intent.putExtra("wap", true);
    startActivity(intent);
    finish();
    }
    </code>
    </pre>
    </br>
    在html的代码中我使用了href="appscheme://contentId" 和href="appscheme:///contentId" 他们的唯一差别就是 '/'的数量。
    在使用三个'/'时,是Android可以自动解析的格式(可产考参考链接中的intent-filter之data),此时url.getPath()为“/contentId”。
    在使用两个'/'时,Android不能自动解析这个path,此时url.getPath()获取失败。因为appscheme://contentId在去掉协议头‘appscheme://’之后剩下的‘contentId’中没有path起始的标示‘/’,所以解析失败。由于IOS的scheme的调用格式为href="appscheme://?contentId=xx",所以在IOS与Android同时开发时,需要自行处理字符串“appscheme://contentId

  • SecondActivity.java
    <pre><code>
    public class SecondActivity extends ActionBarActivity {
    private boolean fromWap = false;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_second);
    Intent intent = getIntent();
    if (intent != null && intent.hasExtra("wap")) {
    fromWap = intent.getBooleanExtra("wap", false);
    }
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
    // Handle action bar item clicks here. The action bar will
    // automatically handle clicks on the Home/Up button, so long
    // as you specify a parent activity in AndroidManifest.xml.
    int id = item.getItemId();

      //noinspection SimplifiableIfStatement
      if (id == R.id.action_settings) {
          return true;
      }
      if (id == R.id.home) {
          finishActivity();
          return true;
      }
    
      return super.onOptionsItemSelected(item);
    

    }
    @Override
    public void onBackPressed() {
    Toast.makeText(this, "onBackPressed", Toast.LENGTH_SHORT).show();
    finishActivity();
    super.onBackPressed();
    }
    /**button 响应函数
    */
    public void closeActivity(View view) {
    finishActivity();
    }
    private void finishActivity() {
    // 获得指向父级activity的intent,NavUtils在support v4 包中
    Intent upIntent = NavUtils.getParentActivityIntent(this);
    // 判断是否需要重建任务栈,有时“NavUtils.shouldUpRecreateTask(this, upIntent)”
    // 判断返回为false,个人感觉自己根据情景来判断是否需要重建栈更准确()
    Toast.makeText(this, "shouldUpRecreateTask:"+NavUtils.shouldUpRecreateTask(this, upIntent), Toast.LENGTH_SHORT).show();
    if (NavUtils.shouldUpRecreateTask(this, upIntent) || fromWap == true) {

          // 这个activity不是这个app任务的一部分, 所以当向上导航时创建
          // 用合成后退栈(synthesized back stack)创建一个新任务。
          TaskStackBuilder.create(this)
                  // 添加这个activity的所有父activity到后退栈中
                  .addNextIntentWithParentStack(upIntent)
                          // 向上导航到最近的一个父activity
                  .startActivities();
      } else {
          // 这个activity是这个app任务的一部分, 所以
          // 向上导航至逻辑父activity.
          NavUtils.navigateUpTo(this, upIntent);
      }
    

    }
    }
    </code></pre></br>
    activity的返回最终都是经过了方法‘finishActivity’
    方法‘finishActivity’中对是否需要重建栈及父级activity进行了判断,但是个人感觉还是自己通过情景分析是否需要重建栈更有效(stackflow中发现其他开发者也遇到了shouldUpRecreateTask方法判断不准确的情况)

源码下载

下载:源代码
github:点我

参考链接:
intent-filter之data (scheme使用)
提供向上导航(向上导航至父级activity)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容