對Android近期任務列表(Recent Applications)的簡單分析

對Android近期任務列表(Recent Applications)的簡單分析

轉載自:http://www.cnblogs.com/coding-way/archive/2013/06/05/3118732.html

這裏的近期任務列表就是長按Home鍵出來的那個Dialog,裏面放着近期打開過的應用,當然3.0以上系統的多任務切換鍵也是。

這個Dialog的實現在Android源碼的/frameworks/base/policy/src/com/android/internal/policy/impl/RecentApplicationsDialog.java中。

接下來就對這個源碼分析一下。

先把整個源碼貼出來:

 RecentApplicationsDialog.java完整源碼

從源碼可以看出,關鍵部分有三處。

 

一個很關鍵的內部類:

// 每個任務都包含一個Tag,這個Tag保存着這個App的一些非常有用的信息
    class RecentTag {
        ActivityManager.RecentTaskInfo info;
        Intent intent;
    }

這個RecentTag保存在每個近期任務的圖標裏,並且保存着這個任務的原始信息。

 

剛啓動Dialog時對每個任務的初始化:

複製代碼
 1     private void reloadButtons() {
 2 
 3         final Context context = getContext();
 4         final PackageManager pm = context.getPackageManager();
 5         final ActivityManager am = (ActivityManager)
 6                 context.getSystemService(Context.ACTIVITY_SERVICE);
 7                 
 8         //拿到最近使用的應用的信息列表
 9         final List<ActivityManager.RecentTaskInfo> recentTasks =
10                 am.getRecentTasks(MAX_RECENT_TASKS, ActivityManager.RECENT_IGNORE_UNAVAILABLE);
11 
12         //自制一個home activity info,用來區分
13         ActivityInfo homeInfo = 
14             new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME)
15                     .resolveActivityInfo(pm, 0);
16 
17         IconUtilities iconUtilities = new IconUtilities(getContext());
18 
19         int index = 0;
20         int numTasks = recentTasks.size();
21         //開始初始化每個任務的信息
22         for (int i = 0; i < numTasks && (index < NUM_BUTTONS); ++i) {
23             final ActivityManager.RecentTaskInfo info = recentTasks.get(i);
24 
25             //複製一個任務的原始Intent
26             Intent intent = new Intent(info.baseIntent);
27             if (info.origActivity != null) {
28                 intent.setComponent(info.origActivity);
29             }
30 
31             //跳過home activity
32             if (homeInfo != null) {
33                 if (homeInfo.packageName.equals(
34                         intent.getComponent().getPackageName())
35                         && homeInfo.name.equals(
36                                 intent.getComponent().getClassName())) {
37                     continue;
38                 }
39             }
40 
41             intent.setFlags((intent.getFlags()&~Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED)
42                     | Intent.FLAG_ACTIVITY_NEW_TASK);
43             final ResolveInfo resolveInfo = pm.resolveActivity(intent, 0);
44             if (resolveInfo != null) {
45                 final ActivityInfo activityInfo = resolveInfo.activityInfo;
46                 final String title = activityInfo.loadLabel(pm).toString();
47                 Drawable icon = activityInfo.loadIcon(pm);
48 
49                 if (title != null && title.length() > 0 && icon != null) {
50                     final TextView tv = mIcons[index];
51                     tv.setText(title);
52                     icon = iconUtilities.createIconDrawable(icon);
53                     tv.setCompoundDrawables(null, icon, null, null);
54                     //new一個Tag,保存這個任務的RecentTaskInfo和Intent
55                     RecentTag tag = new RecentTag();
56                     tag.info = info;
57                     tag.intent = intent;
58                     tv.setTag(tag);
59                     tv.setVisibility(View.VISIBLE);
60                     tv.setPressed(false);
61                     tv.clearFocus();
62                     ++index;
63                 }
64             }
65         }
66 
67        ...//無關緊要的代碼
68     }
複製代碼

這裏的過程是:先用ActivityManager獲取RecentTasks並生成一個List,然後利用這個List爲Dialog中的每個任務初始化,並生成對應的信息RecentTag。

需要注意的是,RecentTag中的Intent是從對應任務的原始Intent複製過來的,這意味着那個原始Intent的一些Extra參數將會一併複製過來,

我來舉個例子:比如我的App支持從第三方啓動,並且第三方要提供一個token,當然這個token就以Extra參數的形式放進Intent裏,然後通過startActivity()啓動我的App;然後我的App根據這個token來處理,注意這裏,當我的App退出後,再從近期任務裏啓動這個App,之前的那個token還會傳遞給我的App,這裏就會出現錯誤了,原因就是上面分析的。這就是爲什麼從第三方跳轉的應用不會出現在近期任務的列表裏(比如點擊短信裏的url啓動一個瀏覽器,之後近期任務裏只有短信app,不會出現瀏覽器app)。要想不出現在近期任務裏,可以給Intent設置FLAG_ACTIVITY_NO_HISTORY標誌。

 

響應每個任務的點擊事件:

複製代碼
 1     private void switchTo(RecentTag tag) {
 2         if (tag.info.id >= 0) {
 3             // 這個Task沒有退出,直接移動到前臺
 4             final ActivityManager am = (ActivityManager)
 5                     getContext().getSystemService(Context.ACTIVITY_SERVICE);
 6             am.moveTaskToFront(tag.info.id, ActivityManager.MOVE_TASK_WITH_HOME);
 7         } else if (tag.intent != null) {
 8             //task退出了的話,id爲-1,則使用RecentTag中的Intent重新啓動
 9             tag.intent.addFlags(Intent.FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY
10                     | Intent.FLAG_ACTIVITY_TASK_ON_HOME);
11             try {
12                 getContext().startActivity(tag.intent);
13             } catch (ActivityNotFoundException e) {
14                 Log.w("Recent", "Unable to launch recent task", e);
15             }
16         }
17     }
複製代碼

如果該Task沒有退出,只是切到後臺,則切換到前臺;如果已經退出,就要重新啓動了。

這裏的Intent就是之前說的,重複使用的舊Intent了,這裏注意,系統添加了FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY和FLAG_ACTIVITY_TASK_ON_HOME標誌,所以我們可以在App中通過判斷Intent的flag是否包含這兩個來判斷是否是從近期任務裏啓動的。注意FLAG_ACTIVITY_TASK_ON_HOME標誌是Api 11添加的,所以11一下的之判斷FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY就行了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章