Android 13 PIP画中画模式系统流程

学习笔记:

一、画中画创建

Activity调用系统api:ActivityTaskManagerService#enterPictureInPictureMode() 开始:

// ActivityTaskManagerService.java
    boolean enterPictureInPictureMode(@NonNull ActivityRecord r, PictureInPictureParams params) {
        // 如果 activity 已经处于画中画模式,则返回
        if (r.inPinnedWindowingMode()) {
            return true;
        }

        // Activity支持画中画,现在检查我们此时是否可以进入画中画,
        if (!r.checkEnterPictureInPictureState("enterPictureInPictureMode",
                false /* beforeStopping */)) {
            return false;
        }

        // 创建了一个 Runnable,在下方将会被调用
        final Runnable enterPipRunnable = () -> {
            synchronized (mGlobalLock) {
                if (r.getParent() == null) {
                    Slog.e(TAG, "Skip enterPictureInPictureMode, destroyed " + r);
                    return;
                }
                r.setPictureInPictureParams(params);
                // **--------重点关注-------------**
                mRootWindowContainer.moveActivityToPinnedRootTask(r,
                        null /* launchIntoPipHostActivity */, "enterPictureInPictureMode");
                final Task task = r.getTask();
                // Continue the pausing process after entering pip.
                if (task.getPausingActivity() == r) {
                    task.schedulePauseActivity(r, false /* userLeaving */,
                            false /* pauseImmediately */, "auto-pip");
                }
            }
        };

        if (r.isKeyguardLocked()) {
            // 如果键盘锁正在显示或被遮挡,则在进入画中画之前尝试关闭它
            // 如果设备当前已锁定(有锁的情况),这将提示用户进行身份验证(即弹出bouncer)。
            mActivityClientController.dismissKeyguard(r.token, new KeyguardDismissCallback() {
                @Override
                public void onDismissSucceeded() {
                    mH.post(enterPipRunnable);
                }
            }, null /* message */);
        } else {
            // 否则立即进入画中画
            enterPipRunnable.run();
        }
        return true;
    }

如果活动现在处于画中画模式,则为 return true;如果无法进入画中画模式,则为 return false。
App 端 binder 回调到 ATMS 的方法,如果有 keyguard,先 dismiss keyguard 再进入pip。

接着根据上述代码看 RootWindowContainer#moveActivityToPinnedRootTask():

// RootWindowContainer.java
    void moveActivityToPinnedRootTask(@NonNull ActivityRecord r,
            @Nullable ActivityRecord launchIntoPipHostActivity, String reason) {
        mService.deferWindowLayout();

        final TaskDisplayArea taskDisplayArea = r.getDisplayArea();

        try {
            // 获取任务栈
            final Task task = r.getTask();

            // 现在创建一个转换控制器以收集当前关闭的任务栈。
            // 由于进入 PIP 的任务(触发器)尚未准备好,因此在此处进行创建。
            final TransitionController transitionController = task.mTransitionController;
            Transition newTransition = null;
            if (transitionController.isCollecting()) {
                transitionController.setReady(task, false /* ready */);
            } else if (transitionController.getTransitionPlayer() != null) {
                newTransition = transitionController.createTransition(TRANSIT_PIP);
            }

            final Task rootPinnedTask = taskDisplayArea.getRootPinnedTask();
            if (rootPinnedTask != null) {
                transitionController.collect(rootPinnedTask);
                // 新的ActivityRecord应该取代现有的PiP,因此旧的PiP消失而不是同时转到全屏,
                // 正如Task#diseasePip试图做的那样。
                removeRootTasksInWindowingModes(WINDOWING_MODE_PINNED);
            }

            // 设置过渡以确保我们不会立即尝试更新可见性,进入 PIP 的活动
            r.getDisplayContent().prepareAppTransition(TRANSIT_NONE);

            final TaskFragment organizedTf = r.getOrganizedTaskFragment();
            final boolean singleActivity = task.getNonFinishingActivityCount() == 1;
            final Task rootTask;
            if (singleActivity) {
                rootTask = task;

                // 对进入PIP的任务应用最近一次的动画牵引变换
                rootTask.maybeApplyLastRecentsAnimationTransaction();
            } else {
                // 在多个活动的情况下,我们将为其创建一个新任务,
                // 然后将PIP活动移动到任务中。
                // 注意,我们明确地延迟了正在发送的任务,并将这个新创建的任务标记为可见
                rootTask = new Task.Builder(mService)
                        .setActivityType(r.getActivityType())
                        .setOnTop(true)
                        .setActivityInfo(r.info)
                        .setParent(taskDisplayArea)
                        .setIntent(r.intent)
                        .setDeferTaskAppear(true)
                        .setHasBeenVisible(true)
                        .build();
                // 在原始任务和固定任务之间建立双向链接。
                r.setLastParentBeforePip(launchIntoPipHostActivity);
                // 进入 PIP 的任务可能是自由格式的,因此请保存最后一个非全屏边界。
                // 然后当这个新的 PIP 任务退出 PIP 时,它可以恢复到它以前的自由形式边界
                rootTask.setLastNonFullscreenBounds(task.mLastNonFullscreenBounds);
                rootTask.setBounds(task.getBounds());

                // 将最后一个最近的动画事务从原始任务移动到新任务
                if (task.mLastRecentsAnimationTransaction != null) {
                    rootTask.setLastRecentsAnimationTransaction(
                            task.mLastRecentsAnimationTransaction,
                            task.mLastRecentsAnimationOverlay);
                    task.clearLastRecentsAnimationTransaction(false /* forceRemoveOverlay */);
                }

                if (organizedTf != null && organizedTf.getNonFinishingActivityCount() == 1
                        && organizedTf.getTopNonFinishingActivity() == r) {
                    // PIP 的状态,是否已清除 PIP 的TaskFragment。
                    // 一旦有改变将在 ActivityTaskManagerService#onPictureInPictureStateChanged() 收到回调
                    organizedTf.mClearedTaskFragmentForPip = true;
                }

                // Activity reparent 到新建的 task
                r.reparent(rootTask, MAX_VALUE, reason);

                // 确保修复后新任务的约束与其当前边界同步。
                rootTask.maybeApplyLastRecentsAnimationTransaction();

                final ActivityRecord oldTopActivity = task.getTopMostActivity();
                if (oldTopActivity != null && oldTopActivity.isState(STOPPED)
                        && task.getDisplayContent().mAppTransition.containsTransitRequest(
                        TRANSIT_TO_BACK)) {
                    task.getDisplayContent().mClosingApps.add(oldTopActivity);
                    oldTopActivity.mRequestForceTransition = true;
                }
            }
            final int intermediateWindowingMode = rootTask.getWindowingMode();
            if (rootTask.getParent() != taskDisplayArea) {
                rootTask.reparent(taskDisplayArea, true /* onTop */);
            }

            if (newTransition != null) {
                transitionController.requestStartTransition(newTransition, rootTask,
                        null /* remoteTransition */, null /* displayChange */);
            }
            transitionController.collect(rootTask);

            r.setWindowingMode(intermediateWindowingMode);
            r.mWaitForEnteringPinnedMode = true;
            rootTask.forAllTaskFragments(tf -> {
                if (!tf.isOrganizedTaskFragment()) {
                    return;
                }
                tf.resetAdjacentTaskFragment();
                if (tf.getTopNonFinishingActivity() != null) {

                    tf.updateRequestedOverrideConfiguration(EMPTY);
                }
            });
            rootTask.setWindowingMode(WINDOWING_MODE_PINNED);

            if (r.getOptions() != null && r.getOptions().isLaunchIntoPip()) {
                mWindowManager.mTaskSnapshotController.recordTaskSnapshot(
                        task, false /* allowSnapshotHome */);
                rootTask.setBounds(r.getOptions().getLaunchBounds());
            }
            rootTask.setDeferTaskAppear(false);
            r.supportsEnterPipOnTaskSwitch = false;

            if (organizedTf != null && organizedTf.mClearedTaskFragmentForPip
                    && organizedTf.isTaskVisibleRequested()) {
                mService.mTaskFragmentOrganizerController.dispatchPendingInfoChangedEvent(
                        organizedTf);
            }
        } finally {
            mService.continueWindowLayout();
        }
         // 显示
        ensureActivitiesVisible(null, 0, false /* preserveWindows */);
        resumeFocusedTasksTopActivities();

        notifyActivityPipModeChanged(r.getTask(), r);
    }
二、退出PIP模式

当 PIP 状态改变,将会在 ActivityTaskManagerService#onPictureInPictureStateChanged() 收到回调:

// ActivityTaskManagerService.java
    @Override
    public void onPictureInPictureStateChanged(PictureInPictureUiState pipState) {
        enforceTaskPermission("onPictureInPictureStateChanged");
        final Task rootPinnedTask = mRootWindowContainer.getDefaultTaskDisplayArea()
                .getRootPinnedTask();
        if (rootPinnedTask != null && rootPinnedTask.getTopMostActivity() != null) {
            // 这里将会去提醒客户端状态发生改变;
            mWindowManager.mAtmService.mActivityClientController.onPictureInPictureStateChanged(
                    rootPinnedTask.getTopMostActivity(), pipState);
        }
    }

ActivityClientController#onPictureInPictureStateChanged():

    void onPictureInPictureStateChanged(@NonNull ActivityRecord r,
            PictureInPictureUiState pipState) {
        if (!r.inPinnedWindowingMode()) {
            throw new IllegalStateException("Activity is not in PIP mode");
        }

        try {
            final ClientTransaction transaction = ClientTransaction.obtain(
                    r.app.getThread(), r.token);
            transaction.addCallback(PipStateTransactionItem.obtain(pipState));
            mService.getLifecycleManager().scheduleTransaction(transaction);
        } catch (Exception e) {
            Slog.w(TAG, "Failed to send pip state transaction item: "
                    + r.intent.getComponent(), e);
        }
    }
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容