注:代码块中出现了代码块中的代码块样式,请大家忽略这些,因为第一次用markdown不知道如何导致已经如何去掉,知道的朋友还请回复我。感谢!
概述
最近产品提出了新的需求:通过wap打开手机本地APP。
功能的实现主要涉及了以下几个知识点:
- scheme打开应用
- Theme.NoDisplay的使用
- activity父级activity的重建
scheme
scheme类似自定义url协议,我们可以通过自定义的协议来打开自己的应用,形如:
txvideo://xxxx
Theme.NoDisplay
在Android中想进行一些无界面的处理又不适合使用service时,此时可以在项目的AndroidManifest.xml文件中相应的Activity标签中添加这样一行:
android:theme=”@android:style/Theme.NoDisplay
activity父级activity的重建
当我们从wap页跳转到应用内部时,可能我们跳转到的不是应用的第一层级页面而是深层次的页面,这时候我们需要在关闭应用内部页面时,对高层的页面进行重建。
- 为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>
- 二级页面返回时,重建任务栈
@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方法判断不准确的情况)
源码下载
参考链接:
intent-filter之data (scheme使用)
提供向上导航(向上导航至父级activity)