注:代碼塊中出現(xiàn)了代碼塊中的代碼塊樣式俩滥,請(qǐng)大家忽略這些嘉蕾,因?yàn)榈谝淮斡胢arkdown不知道如何導(dǎo)致已經(jīng)如何去掉,知道的朋友還請(qǐng)回復(fù)我霜旧。感謝错忱!
概述
最近產(chǎn)品提出了新的需求:通過(guò)wap打開(kāi)手機(jī)本地APP。
功能的實(shí)現(xiàn)主要涉及了以下幾個(gè)知識(shí)點(diǎn):
- scheme打開(kāi)應(yīng)用
- Theme.NoDisplay的使用
- activity父級(jí)activity的重建
scheme
scheme類似自定義url協(xié)議挂据,我們可以通過(guò)自定義的協(xié)議來(lái)打開(kāi)自己的應(yīng)用以清,形如:
<pre><code>txvideo://xxxx</code></pre>
Theme.NoDisplay
在Android中想進(jìn)行一些無(wú)界面的處理又不適合使用service時(shí),此時(shí)可以在項(xiàng)目的AndroidManifest.xml文件中相應(yīng)的Activity標(biāo)簽中添加這樣一行:
<pre><code>android:theme=”@android:style/Theme.NoDisplay</code></pre>
activity父級(jí)activity的重建
當(dāng)我們從wap頁(yè)跳轉(zhuǎn)到應(yīng)用內(nèi)部時(shí)崎逃,可能我們跳轉(zhuǎn)到的不是應(yīng)用的第一層級(jí)頁(yè)面而是深層次的頁(yè)面掷倔,這時(shí)候我們需要在關(guān)閉應(yīng)用內(nèi)部頁(yè)面時(shí),對(duì)高層的頁(yè)面進(jìn)行重建个绍。
- 為activity指定父級(jí)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> - 二級(jí)頁(yè)面返回時(shí)勒葱,重建任務(wù)棧
<pre><code>
@Override
public void onBackPressed() {
// 獲得指向父級(jí)activity的intent,NavUtils在support v4 包中
Intent upIntent = NavUtils.getParentActivityIntent(this);
// 判斷是否需要重建任務(wù)棧
if (NavUtils.shouldUpRecreateTask(this, upIntent)) {
// 這個(gè)activity不是這個(gè)app任務(wù)的一部分, 所以當(dāng)向上導(dǎo)航時(shí)創(chuàng)建
// 用合成后退棧(synthesized back stack)創(chuàng)建一個(gè)新任務(wù)巴柿。
TaskStackBuilder.create(this)
// 添加這個(gè)activity的所有父activity到后退棧中
.addNextIntentWithParentStack(upIntent)
// 向上導(dǎo)航到最近的一個(gè)父activity
.startActivities();
} else {
// 這個(gè)activity是這個(gè)app任務(wù)的一部分, 所以
// 向上導(dǎo)航至邏輯父activity.
NavUtils.navigateUpTo(this, upIntent);
}
super.onBackPressed();
}
</code></pre> </br>
效果與分析

第一次錄屏凛虽,效果不大好(有好的錄屏方法請(qǐng)告訴我),建議下載代碼測(cè)試
我們點(diǎn)擊網(wǎng)頁(yè)中的立即打開(kāi)->彈出secondActivity->點(diǎn)擊closeactivity按鈕广恢,程序?qū)econdActivity的父級(jí)activity進(jìn)行重建
編碼與實(shí)現(xiàn)
html頁(yè)面
<pre><code>
<a id="openJD" href="appscheme://contentId">立即打開(kāi)//</span></a></br>
<a id="openJD" href="appscheme:///contentId">立即打開(kāi)///</span></a>
</code></pre>
簡(jiǎn)單的兩個(gè)超鏈接</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中定義了凯旋,二級(jí)activity 'SecondeActivity',一級(jí)activity ‘MainActivity’ 它是SecondeActivity的父級(jí)activity。
定義了不可顯示的activity‘SchemeActivity’來(lái)處理scheme的調(diào)起钉迷,可對(duì)scheme的path部分進(jìn)行處理在跳轉(zhuǎn)到不同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" 他們的唯一差別就是 '/'的數(shù)量。
在使用三個(gè)'/'時(shí)糠聪,是Android可以自動(dòng)解析的格式(可產(chǎn)考參考鏈接中的intent-filter之data)睡蟋,此時(shí)url.getPath()為“/contentId”。
在使用兩個(gè)'/'時(shí)枷颊,Android不能自動(dòng)解析這個(gè)path,此時(shí)url.getPath()獲取失敗。因?yàn)?a target="_blank" rel="nofollow">appscheme://contentId在去掉協(xié)議頭‘a(chǎn)ppscheme://’之后剩下的‘contentId’中沒(méi)有path起始的標(biāo)示‘/’夭苗,所以解析失敗信卡。由于IOS的scheme的調(diào)用格式為href="appscheme://?contentId=xx",所以在IOS與Android同時(shí)開(kāi)發(fā)時(shí),需要自行處理字符串“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 響應(yīng)函數(shù)
*/
public void closeActivity(View view) {
finishActivity();
}
private void finishActivity() {
// 獲得指向父級(jí)activity的intent题造,NavUtils在support v4 包中
Intent upIntent = NavUtils.getParentActivityIntent(this);
// 判斷是否需要重建任務(wù)棧,有時(shí)“NavUtils.shouldUpRecreateTask(this, upIntent)”
// 判斷返回為false傍菇,個(gè)人感覺(jué)自己根據(jù)情景來(lái)判斷是否需要重建棧更準(zhǔn)確()
Toast.makeText(this, "shouldUpRecreateTask:"+NavUtils.shouldUpRecreateTask(this, upIntent), Toast.LENGTH_SHORT).show();
if (NavUtils.shouldUpRecreateTask(this, upIntent) || fromWap == true) {// 這個(gè)activity不是這個(gè)app任務(wù)的一部分, 所以當(dāng)向上導(dǎo)航時(shí)創(chuàng)建 // 用合成后退棧(synthesized back stack)創(chuàng)建一個(gè)新任務(wù)。 TaskStackBuilder.create(this) // 添加這個(gè)activity的所有父activity到后退棧中 .addNextIntentWithParentStack(upIntent) // 向上導(dǎo)航到最近的一個(gè)父activity .startActivities(); } else { // 這個(gè)activity是這個(gè)app任務(wù)的一部分, 所以 // 向上導(dǎo)航至邏輯父activity. NavUtils.navigateUpTo(this, upIntent); }
}
}
</code></pre></br>
activity的返回最終都是經(jīng)過(guò)了方法‘finishActivity’
方法‘finishActivity’中對(duì)是否需要重建棧及父級(jí)activity進(jìn)行了判斷界赔,但是個(gè)人感覺(jué)還是自己通過(guò)情景分析是否需要重建棧更有效(stackflow中發(fā)現(xiàn)其他開(kāi)發(fā)者也遇到了shouldUpRecreateTask方法判斷不準(zhǔn)確的情況)
源碼下載
參考鏈接:
intent-filter之data (scheme使用)
提供向上導(dǎo)航(向上導(dǎo)航至父級(jí)activity)