从wap页面跳转app(Android)

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

概述

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

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

scheme类似自定义url协议,我们可以通过自定义的协议来打开自己的应用,形如:

txvideo://xxxx
Theme.NoDisplay

在Android中想进行一些无界面的处理又不适合使用service时,此时可以在项目的AndroidManifest.xml文件中相应的Activity标签中添加这样一行:

android:theme=”@android:style/Theme.NoDisplay
activity父级activity的重建

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

  1. 为activity指定父级activity
    <application ...> <activity ...
     android:name="com.example.Mainactivity"> <!-- 首层activity -->
     </activity> <activity ... android:name="com.example.SecondeActivity"> <!-- 第二层级activity -->
     android:parentActivityName="com.example.Mainactivity" > <!-- 指明父级activity类名 -->
     <!-- 父activity的meta-data,用来支持4.0以下版本 -->
     <meta-data
         android:name="android.support.PARENT_ACTIVITY"
         android:value="com.example.Mainactivity" />
     </activity></application>
  2. 二级页面返回时,重建任务栈
    @Overridepublic 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();}

效果与分析


效果图

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

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

编码与实现

  • html页面
    <a id="openJD" href="appscheme://contentId">立即打开//</span></a><a id="openJD" href="appscheme:///contentId">立即打开///</span></a>

    简单的两个超链接
  • AndroidMainifast.xml
    ...<application
      android:allowBackup="true"
      android:icon="@mipmap/ic_launcher"
      android:label="@string/app_name"
      android:theme="@style/AppTheme" >
      <activity <!-- 主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   <!-- 此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 <!-- 做sechame接收的activity-->
          android:name=".SchemeCenterActivity"
          android:label="@string/title_activity_scheme_center"
          android:theme="@android:style/Theme.NoDisplay">
          <intent-filter> <!-- 通过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" />  <!-- wap中声明的scheme -->
          </intent-filter>
      </activity>
      </application>...

    在AndroidManifest中定义了,二级activity 'SecondeActivity',一级activity ‘MainActivity’ 它是SecondeActivity的父级activity。
    定义了不可显示的activity‘SchemeActivity’来处理scheme的调起,可对scheme的path部分进行处理在跳转到不同activity。
  • SchemeCenterActivity.java

    @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();
      }
    


    在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

    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);
      }
      }}

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

源码下载

下载:源代码
github:点我

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值