美文网首页Android开发经验
从wap页面跳转app(Android)

从wap页面跳转app(Android)

作者: 李科吐温 | 来源:发表于2015-05-18 18:10 被阅读6440次

    注:代码块中出现了代码块中的代码块样式,请大家忽略这些,因为第一次用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)

    相关文章

      网友评论

        本文标题:从wap页面跳转app(Android)

        本文链接:https://www.haomeiwen.com/subject/cpflqttx.html