小程序模板網(wǎng)

任務(wù)列表中顯示"小程序"的原理分析

發(fā)布時(shí)間:2018-01-03 09:53 所屬欄目:小程序開發(fā)教程

今天被小程序刷屏了^^ 我也來湊湊熱鬧. 談?wù)勎⑿攀窃趺磳?shí)現(xiàn)在任務(wù)列表中顯示"小程序"的.

效果

微信中打開了"滴滴(小程序)"后, 可以看到,任務(wù)列表不僅顯示了"微信", 還顯示了"滴滴(小程序)"的人口. 通過這個(gè)入口,就可以很方面的切換小程序了, 體驗(yàn)和原生程序也一樣一樣的.

 
 

分析

下面簡(jiǎn)單分析一下他的實(shí)現(xiàn).

1.Android系統(tǒng)中,顯示最近程序列表的View是 RecentsPanelView. 他通過refreshRecentTasksList()加載程序列表,我們來看看代碼: 可以看到RecentTasksLoader mRecentTasksLoader負(fù)責(zé)真正處理數(shù)據(jù)加載.

private RecentTasksLoader mRecentTasksLoader;  

***  

private void refreshRecentTasksList(
    ArrayList<TaskDescription> recentTasksList, boolean firstScreenful) {
    if (mRecentTaskDescriptions == null && recentTasksList != null){
      onTasksLoaded(recentTasksList, firstScreenful);
    } else {
      //加載最近的列表
      mRecentTasksLoader.loadTasksInBackground();
    }
  }

2.順著看看RecentTasksLoader的實(shí)現(xiàn):
這里可以清楚看到,加載的數(shù)據(jù)是ActivityManager.getRecentTasks().
也就是說顯示的是Task列表.

 ArrayList<TaskDescription>  getRecentTasks() {
       cancelLoadingThumbnails();

       ArrayList<TaskDescription> tasks = new ArrayList<TaskDescription>();
       final PackageManager pm = mContext.getPackageManager();
       final ActivityManager am = (ActivityManager)
               mContext.getSystemService(Context.ACTIVITY_SERVICE);

       //獲取最近的Task
       final List<ActivityManager.RecentTaskInfo> recentTasks =
               am.getRecentTasks(MAX_TASKS, ActivityManager.RECENT_IGNORE_UNAVAILABLE);

        ***
       return tasks;
   }

到這里,已經(jīng)很清楚了. 要顯示"小程序"入口, 只需要新建一個(gè)Task啟動(dòng)就好了~
是不是很簡(jiǎn)單啊.

來來來驗(yàn)證一下我們的想法.
這里使用 adb shell dumpsys activity activities查看一下Task狀態(tài)

1. 只開啟微信
 
 
2. 開啟小程序

如下圖所示, 微信新開啟了一個(gè).AppBrandUI的task棧


 
 

實(shí)現(xiàn)

知道原理,實(shí)現(xiàn)就很簡(jiǎn)單了.

假設(shè)小程序的Activity是 TaskTestActivity,整個(gè)實(shí)現(xiàn)分兩步:

1. AndroidManifest.xml中為Activity設(shè)置taskAffinity
<activity
        android:icon="@drawable/didi"
        android:name=".TaskTestActivity"
        android:label="小程序測(cè)試"
        android:taskAffinity=".NewTask" />    

2. 以NEW_TASK方式啟動(dòng)Activity
    //以Intent.FLAG_ACTIVITY_NEW_TASK方式啟動(dòng)Activity
    Intent intent = new Intent(this, TaskTestActivity.class);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);

為什么要設(shè)置taskAffinity呢,

這和FLAG_ACTIVITY_NEW_TASK的特性有關(guān):

FLAG_ACTIVITY_NEW_TASK: 設(shè)置此狀態(tài),首先會(huì)查找是否存在和被啟動(dòng)的Activity具有相同的taskAffinity的task(注意同一個(gè)應(yīng)用程序中默認(rèn)所有activity 的taskAffinity是一樣的),如果有,剛直接把這個(gè)棧整體移動(dòng)到前臺(tái),并保持棧中的狀態(tài)不變,即棧中的activity順序不變,如果沒有,則新建一個(gè)棧來存放被啟動(dòng)的activity.

也就是說,如果App已經(jīng)啟動(dòng),即使用FLAG_ACTIVITY_NEW_TASK新起Activity, 因?yàn)閠askAffinity相同,也會(huì)被壓到一個(gè)task中, 自然recent panel 就看不到兩個(gè)入口了.

所以我們需要為小程序設(shè)置一個(gè)新的taskAffinity

下面是demo效果

 
 

拓展

讓自己的程序不顯示在任務(wù)列表中

有時(shí)候我們做一個(gè)工具, 或者后臺(tái)界面, 不希望顯示在程序列表中.
也很簡(jiǎn)單,只要設(shè)置task的屬性就好了

方法1:

在AndroidManifest.xml設(shè)置Activity的 android:excludeFromRecents為true

<activity
      android:excludeFromRecents="true"
      android:icon="@drawable/didi"
      android:name=".TaskTestActivity"
      android:label="小程序測(cè)試" />
方法2:

在啟動(dòng)Activity的時(shí)候加上Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS FLAG

//Activity不顯示在recent列表中.
Intent intent = new Intent(this, TaskTestActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
startActivity(intent);


作者:菜刀文
鏈接:https://www.jianshu.com/p/c089e13e57b5
來源:簡(jiǎn)書
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請(qǐng)注明出處。


易優(yōu)小程序(企業(yè)版)+靈活api+前后代碼開源 碼云倉庫:starfork
本文地址:http://22321a.com/wxmini/doc/course/18365.html 復(fù)制鏈接 如需定制請(qǐng)聯(lián)系易優(yōu)客服咨詢:800182392 點(diǎn)擊咨詢
QQ在線咨詢