這個博客我們將討論畫中畫模式,如何實現(xiàn)它以及如果有多個活動時如何管理后臺堆棧。
現(xiàn)在闽撤,如果應(yīng)用程序僅包含一個活動并且您必須實現(xiàn) pip,這非常簡單脯颜,您只需要將活動設(shè)為單一任務(wù)即可 這是代碼片段-
<activity
android:name="HomeActivity"
android:launchMode="singleTask"
android:supportsPictureInPicture="true"
android:theme="@style/AppTheme">
<intent-filter>
<action android:name="android.intent. action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
覆蓋 Home 活動中的 onUserLeaveHint 方法哟旗。
override fun onUserLeaveHint() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O)
if (!isInPictureInPictureMode) {
var pictureInPictureParamsBuilder = PictureInPictureParams.Builder()
Logger.err(TAG, "onUserLeaveHintCalled")
var aspectRatio = Rational(ScreenUtils.getScreenWidth(this), ScreenUtils.getSceenHeight(this))
pictureInPictureParamsBuilder.setAspectRatio(aspectRatio).build()
enterPictureInPictureMode(pictureInPictureParamsBuilder.build())
}
}
如您所見,我們必須將我們的單個 Activity 作為單個任務(wù)栋操,并為圖中畫提供支持闸餐。如果用戶處于 pip 模式并且如果用戶單擊應(yīng)用程序圖標(biāo),則我們必須將我們的活動設(shè)置為 singleTask 背后的原因矾芙,將不會創(chuàng)建新任務(wù)舍沙,將重用相同的任務(wù)并將意圖傳遞給onNewIntent(Intent intent) 方法。通過這種方式剔宪,我們可以在單個活動應(yīng)用程序中實現(xiàn) pip拂铡,因為我們不必在單個活動中維護后臺堆棧壹无。
現(xiàn)在,如果我們在應(yīng)用程序中有多個 Activity感帅,并且您想在進(jìn)入 pip 后維護 backstack斗锭。
假設(shè),我們有一個類似 Login -> VideoList -> VideoDetail -> VideoPlayer Activity 的場景失球。在達(dá)到 VideoPlayer 活動后岖是,實際播放視頻的活動并且當(dāng)用戶進(jìn)入 pip 模式時,現(xiàn)在用戶最大化它并離開 pip 然后單擊后退按鈕实苞,用戶直接進(jìn)入另一個應(yīng)用程序的主屏幕/上次打開的活動豺撑,但我們希望我們的用戶轉(zhuǎn)到我們播放 VideoPlay 的 Vod 詳細(xì)信息頁面。對于這種情況硬梁,我們必須遵循以下步驟-
1 - 在 VideoPlayer 活動的清單中添加 pip 支持。
2 - 將此活動啟動模式設(shè)為 singleTask
3 - 為這個活動添加 Taskaffinity
4 - 從最近的 true 設(shè)置排除
<activity
android:name=".ui.videodetail.VodDetailActivity"
android:taskAffinity=".ui.videodetail.VodDetailActivity"
android:supportsPictureInPicture="true"
android:parentActivityName=".ui.dashboard.DashboardActivity"
android:launchMode="singleTask"
android:configChanges="screenSize|smallestScreenSize|screenLayout|orientation"
android:resizeableActivity="true"
android:autoRemoveFromRecents="true"
android:excludeFromRecents="true"
android:allowTaskReparenting="true"
android:theme="@style/AppTheme.NoActionBar"
android:screenOrientation="portrait" />
如您所見胞得,我已設(shè)置 taskAffinity 并設(shè)置 exclude from recent true 因為我們不想在 android 應(yīng)用程序堆棧中顯示兩個任務(wù)荧止。單任務(wù)啟動模式和不同的親和力將為 pip 啟動一個新任務(wù),我們必須從 android 應(yīng)用程序堆棧中刪除阶剑,因為我們不想為同一應(yīng)用程序顯示兩個堆棧
現(xiàn)在如果用戶點擊這個視頻播放器頁面上的后退圖標(biāo)跃巡,首先我們要檢查用戶是否直接點擊后退按鈕而不進(jìn)入 pip 模式,然后調(diào)用 super.finish()牧愁,但是如果 Activity 已經(jīng)在pip 模式素邪,然后我們必須檢查 android 應(yīng)用程序堆棧是否包含應(yīng)用程序堆棧中您的應(yīng)用程序的任務(wù),然后我們必須在單擊后退按鈕時將該任務(wù)移到前面猪半。代碼為此 -
/**
* Bring up launcher task to front
*/
public static void navToLauncherTask(Context appContext) {
ActivityManager activityManager = (ActivityManager) appContext.getSystemService(Context.ACTIVITY_SERVICE);
// iterate app tasks available and navigate to launcher task (browse task)
assert activityManager != null;
final List<ActivityManager.AppTask> appTasks = activityManager.getAppTasks();
for (ActivityManager.AppTask task : appTasks) {
final Intent baseIntent = task.getTaskInfo().baseIntent;
final Set<String> categories = baseIntent.getCategories();
if (categories != null && categories.contains(Intent.CATEGORY_LAUNCHER)) {
task.moveToFront();
return;
}
}
}
通過這種方式兔朦,我們也可以使用管道模式維護后臺堆棧,但這不是一個好習(xí)慣握童,如果您想為您的應(yīng)用程序?qū)崿F(xiàn) pip股淡,請嘗試制作包含多個片段的單個活動應(yīng)用程序杠娱。