注:代码块中出现了代码块中的代码块样式,请大家忽略这些,因为第一次用markdown不知道如何导致已经如何去掉,知道的朋友还请回复我。感谢!
概述
最近产品提出了新的需求:通过wap打开手机本地APP。
功能的实现主要涉及了以下几个知识点:
- scheme打开应用
- Theme.NoDisplay的使用
- 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页跳转到应用内部时,可能我们跳转到的不是应用的第一层级页面而是深层次的页面,这时候我们需要在关闭应用内部页面时,对高层的页面进行重建。
- 为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> - 二级页面返回时,重建任务栈
<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方法判断不准确的情况)
源码下载
参考链接:
intent-filter之data (scheme使用)
提供向上导航(向上导航至父级activity)