WMShell finish transition
当 Launcher 执行完远程动画之时,会通知 WMShell,最终调用 Transitions#onFinish(),如下
java
// Transitions.java
// wct 为 null
// wctCB 为 null
private void onFinish(ActiveTransition active,
@Nullable WindowContainerTransaction wct,
@Nullable WindowContainerTransactionCallback wctCB) {
final Track track = mTracks.get(active.getTrack());
track.mActiveTransition = null;
// ...
// 1. apply finish transaction
SurfaceControl.Transaction fullFinish = active.mFinishT;
if (active.mMerged != null) {
// ...
}
if (fullFinish != null) {
fullFinish.apply();
}
// ...
// 2. 通知 WMCore finish transition
mOrganizer.finishTransition(active.mToken, wct, wctCB);
// ...
}
WMShell 在 apply finish transaction 后,通知 WMCore finish transition。
WMCore finish transition
java
// WindowOrganizerController.java
// t 为 null
// callback 为 null
public int finishTransition(@NonNull IBinder transitionToken,
@Nullable WindowContainerTransaction t,
@Nullable IWindowContainerTransactionCallback callback) {
enforceTaskPermission("finishTransition()");
final CallerInfo caller = new CallerInfo();
final long ident = Binder.clearCallingIdentity();
try {
synchronized (mGlobalLock) {
// ...
// 根据 token 获取 Transition
final Transition transition = Transition.fromBinder(transitionToken);
// ...
// 由 TransitionController 执行 finish transition
mTransitionController.finishTransition(transition);
mTransitionController.mFinishingTransition = null;
// ...
}
} finally {
Binder.restoreCallingIdentity(ident);
}
}
java
// TransitionController.java
void finishTransition(Transition record) {
// ...
ProtoLog.v(ProtoLogGroup.WM_DEBUG_WINDOW_TRANSITIONS, "Finish Transition: %s", record);
// 注意,先从 mPlayingTransitions 移除,在执行 Transition#finishTransition()
mPlayingTransitions.remove(record);
// ...
record.finishTransition();
// ...
}
java
// Transition.java
void finishTransition() {
// ...
// apply cleanup transaction
if (mCleanupTransaction != null) {
mCleanupTransaction.apply();
mCleanupTransaction = null;
}
// ...
// 标记 finishing transition
mController.mFinishingTransition = this;
// ...
// 1. 提交即将不可见的 ActivityRecord
// Commit all going-invisible containers
for (int i = 0; i < mParticipants.size(); ++i) {
final WindowContainer<?> participant = mParticipants.valueAt(i);
final ActivityRecord ar = participant.asActivityRecord();
if (ar != null) {
final Task task = ar.getTask();
if (task == null) continue;
// 检测哪些 WindowToken 在 transitin finish 是还可见
boolean visibleAtTransitionEnd = mVisibleAtTransitionEndTokens.contains(ar);
if (isTransientLaunch(ar) && !ar.isVisibleRequested()
&& mController.inCollectingTransition(ar)) {
// ...
}
final boolean isScreenOff = ar.mDisplayContent == null
|| ar.mDisplayContent.getDisplayInfo().state == Display.STATE_OFF;
if ((!visibleAtTransitionEnd || isScreenOff) && !ar.isVisibleRequested()) {
// 检测 Activity 是否进入PIP
final boolean commitVisibility = !checkEnterPipOnFinish(ar);
if (commitVisibility) { // Activity不进入 PIP
ProtoLog.v(ProtoLogGroup.WM_DEBUG_WINDOW_TRANSITIONS,
" Commit activity becoming invisible: %s", ar);
final SnapshotController snapController = mController.mSnapshotController;
if (mTransientLaunches != null && !task.isVisibleRequested()
&& !task.isActivityTypeHome()) {
// ...
}
// 提交 ActivityRecord 不可见
ar.commitVisibility(false /* visible */, false /* performLayout */,
true /* fromTransition */);
committedSomeInvisible = true;
} else {
}
}
// ...
continue;
}
// ...
}
// 2. wallpaper target 不可见,那么提交 WallpaperWindowToken 不可见
// Commit wallpaper visibility after activity, because usually the wallpaper target token is
// an activity, and wallpaper's visibility depends on activity's visibility.
for (int i = mParticipants.size() - 1; i >= 0; --i) {
final WallpaperWindowToken wt = mParticipants.valueAt(i).asWallpaperToken();
if (wt == null) continue;
final WindowState target = wt.mDisplayContent.mWallpaperController.getWallpaperTarget();
final boolean isTargetInvisible = target == null || !target.mToken.isVisible();
final boolean isWallpaperVisibleAtEnd =
wt.isVisibleRequested() || mVisibleAtTransitionEndTokens.contains(wt);
// wallpaper target 不可见,提交 WallpaperWindowToken 不可见
if (isTargetInvisible || !isWallpaperVisibleAtEnd) {
ProtoLog.v(ProtoLogGroup.WM_DEBUG_WINDOW_TRANSITIONS,
" Commit wallpaper becoming invisible: %s", wt);
wt.commitVisibility(false /* visible */);
}
// 在 layout change 标记 FINISH_LAYOUT_REDO_WALLPAPER
// 在窗口刷新时,会重新计算 wallpaper target
if (isTargetInvisible) {
// Our original target went invisible, so we should look for a new target.
wt.mDisplayContent.pendingLayoutChanges |= FINISH_LAYOUT_REDO_WALLPAPER;
}
}
// ...
// 3. 通知监听者 transition finish
// dispatch legacy callback in a different loop. This is because multiple legacy handlers
// (fixed-rotation/displaycontent) make global changes, so we want to ensure that we've
// processed all the participants first (in particular, we want to trigger pip-enter first)
for (int i = 0; i < mParticipants.size(); ++i) {
// 必须有 ActivityRecord 参与到 Transition
final ActivityRecord ar = mParticipants.valueAt(i).asActivityRecord();
if (ar == null) continue;
// 此时,待启动的 ActivityRecord 是可见的
if ((ar.isVisibleRequested() || !ar.isState(ActivityRecord.State.INITIALIZING))
&& !ar.isAnimating(PARENTS, ANIMATION_TYPE_PREDICT_BACK)) {
mController.dispatchLegacyAppTransitionFinished(ar);
}
// ...
}
// ...
// 4.调度 stop activity
// Always schedule stop processing when transition finishes because activities don't
// stop while they are in a transition thus their stop could still be pending.
mController.mAtm.mTaskSupervisor
.scheduleProcessStoppingAndFinishingActivitiesIfNeeded();
// ...
// Transition 状态切换到 STATE_FINISHED
mState = STATE_FINISHED;
// ...
// 此时没有 finishing transition
mController.mFinishingTransition = null;
// ...
}
finish transition
- Transition 的参与者中,如果有不可见的 ActivityRecord,提交其不可见。对于本案例来说,Launcher ActivityRecord 不可见,因此对它提交了不可见,即更新 ActivityRecord#mVisible 和 ActivityRecord#mClientVisible 为 false。
- Launcher ActivityRecord 现已被提交不可见,导致其下的 wallpaper target 窗口即将不可见。因此,得提交 WallpaperWindowToken 的不可见。
- 通知监听者 transition finish。在 play transition 前,DisplayContent update orientation 时,执行了 Fixed Rotation,导致推迟了 update orientation。现在 transition finish,DisplayContent 需继续执行 update orientation,更新系统 rotation。
- 调度 stop activity。根据 Android V app 冷启动 (1) Activity生命周期 分析,Launcher ActivityRecord 处于 transition 中,无法执行 stop activity。还需正在启动的 Activity 上报 idle ,才可以执行 Launcher Activity 的 stop activity 流程。
提交 WallpaperWindowToken 不可见
java
// WallpaperWindowToken.java
// 参数 visible 为 false
/** Commits the visibility of this token. This will directly update the visibility. */
void commitVisibility(boolean visible) {
if (visible == isVisible()) return;
ProtoLog.v(WM_DEBUG_APP_TRANSITIONS,
"commitVisibility: %s: visible=%b mVisibleRequested=%b", this,
isVisible(), mVisibleRequested);
// 更新 mVisibleRequested 为 false
setVisibleRequested(visible);
// 只更新 mClientVisible 为 false
// 注意,WallpaperWindowToken 的 mClientVisible 就是 mVisible
setVisible(visible);
}
// 参数 visible 为 false
private void setVisible(boolean visible) {
final boolean wasClientVisible = isClientVisible();
// 更新 mClientVisible 为 false
setClientVisible(visible);
if (visible && !wasClientVisible) {
}
}
WallpaperWindowToken 的提交不可见,只需更新 mVisibleRequested 和 mClientVisible 为 false。WallpaperWindowToken 并没有 mVisible 属性,因为它是 ActivityRecord 专有的属性。
注意,mClientVisible 被更新为 false,会通知 app 端窗口不可见,app 端不会再绘制壁纸窗口。
DisplayContent 处理 transition finish
java
// DisplayContent.java
class FixedRotationTransitionListener extends WindowManagerInternal.AppTransitionListener {
public void onAppTransitionFinishedLocked(IBinder token) {
// 这个是正在启动的 ActivityRecord
final ActivityRecord r = ActivityRecord.forTokenLocked(token);
// ...
continueUpdateOrientationForDiffOrienLaunchingApp();
}
}
/**
* Continue updating the orientation change of display if it was deferred by a top activity
* launched in a different orientation.
*/
void continueUpdateOrientationForDiffOrienLaunchingApp() {
// ...
// 1.继续 update orientation
// Update directly because the app which will change the orientation of display is ready.
if (mDisplayRotation.updateOrientation(getOrientation(), false /* forceUpdate */)) {
// If a transition is collecting, let the transition apply the rotation change on
// display thread. See Transition#shouldApplyOnDisplayThread().
if (!mTransitionController.isCollecting(this)) {
// 2. 发送新配置
sendNewConfiguration();
}
return;
}
// ...
}
play transition 前的 DisplayContent 的 update orientation,由于执行了 fixed rotation 而被推迟。
现在,transition finish,得继续执行 DisplayContent update orientation。此时,会导致系统 rotation 被更新,从而触发配置更新。
play transition 前的 apply fixed rotation,对正在启动的 ActivityRecord surface 设置了 transform,使横屏绘制的 Activity 窗口,可以"正常地"显示在竖屏的屏幕上。
现在,系统 rotation 的更新,会导致屏幕方向的更新,此时需要移除正在启动的 ActivityRecord surface 的 transform,使横屏绘制的窗口,真正地可以正常的显示在横屏的屏幕上。这就是配置更新中的 finish fixed rotation 所做的。
update orientation
java
// DisplayRotation.java
// newOrientation 是计算出的 orientation
// forceUpdate 为 false
boolean updateOrientation(@ScreenOrientation int newOrientation, boolean forceUpdate) {
if (newOrientation == mLastOrientation && !forceUpdate) {
return false;
}
// 更新 orientation
mLastOrientation = newOrientation;
if (newOrientation != mCurrentAppOrientation) {
mCurrentAppOrientation = newOrientation;
if (isDefaultDisplay) {
updateOrientationListenerLw();
}
}
// 更新 rotation
return updateRotationUnchecked(forceUpdate);
}
boolean updateRotationUnchecked(boolean forceUpdate) {
// ...
final int oldRotation = mRotation;
final int lastOrientation = mLastOrientation;
// 根据 orientation 计算 rotation
int rotation = rotationForOrientation(lastOrientation, oldRotation);
// ...
// 保存 rotation
mRotation = rotation;
// 标记 DisplayContent 需要 layout
mDisplayContent.setLayoutNeeded();
// 标记 DisplayContent 正在等待配置更新
mDisplayContent.mWaitingForConfig = true;
if (mDisplayContent.mTransitionController.isShellTransitionsEnabled()) {
// 此时已经没有处于 collecting 的 Transition
final boolean wasCollecting = mDisplayContent.mTransitionController.isCollecting();
if (!wasCollecting) {
if (mDisplayContent.getLastHasContent()) {
// 请求类型为 CHANGE 的 transition
final TransitionRequestInfo.DisplayChange change =
new TransitionRequestInfo.DisplayChange(mDisplayContent.getDisplayId(),
oldRotation, mRotation);
mDisplayContent.requestChangeTransition(
ActivityInfo.CONFIG_WINDOW_CONFIGURATION, change);
}
} else {
// ...
}
return true;
}
// ...
}
update orientation 触发了 rotation 更新,导致请求了一个类型为 CHANGE transition,这是一个无缝的屏幕旋转动画。屏幕旋转动画,我在前面的文章中分析过,它会导致屏幕方向的更新。
我在前面的文章中,详细分析过屏幕旋转动画,而无缝的屏幕旋转动画,只是看不到动画而已。
更新配置
由于 update orientation 触发了 rotation 更新,因此需要更新系统配置。我在屏幕旋转动画的系列文章中,已经详细分析过配置更新,这里只展示与 Fixed Rotation 相关的配置更新
java
// DisplayContent.java
public void onRequestedOverrideConfigurationChanged(Configuration overrideConfiguration) {
final Configuration currOverrideConfig = getRequestedOverrideConfiguration();
final int currRotation = currOverrideConfig.windowConfiguration.getRotation();
final int overrideRotation = overrideConfiguration.windowConfiguration.getRotation();
if (currRotation != ROTATION_UNDEFINED && overrideRotation != ROTATION_UNDEFINED
&& currRotation != overrideRotation) {
// rotation 改变,执行 finish fixed rotation
applyRotationAndFinishFixedRotation(currRotation, overrideRotation);
}
// ...
// 通过基类方法,更新配置
super.onRequestedOverrideConfigurationChanged(overrideConfiguration);
// ...
}
/**
* If the launching rotated activity ({@link #mFixedRotationLaunchingApp}) is null, it simply
* applies the rotation to display. Otherwise because the activity has shown as rotated, the
* fixed rotation transform also needs to be cleared to make sure the rotated activity fits
* the display naturally.
*/
private void applyRotationAndFinishFixedRotation(int oldRotation, int newRotation) {
final WindowToken rotatedLaunchingApp = mFixedRotationLaunchingApp;
if (rotatedLaunchingApp == null) {
// ...
}
// finish fixed rotation,并且 apply rotation
rotatedLaunchingApp.finishFixedRotationTransform(
() -> applyRotation(oldRotation, newRotation));
// 重置 mFixedRotationLaunchingApp 为 null
setFixedRotationLaunchingAppUnchecked(null);
}
java
// WindowToken.java
/**
* Finishes the transform and apply display rotation if the action is given. If the display will
* not rotate, the transformed containers are restored to their original states.
*/
void finishFixedRotationTransform(Runnable applyDisplayRotation) {
final FixedRotationTransformState state = mFixedRotationTransformState;
if (state == null) {
return;
}
// 重置 FixedRotationTransformState
state.resetTransform();
// Clear the flag so if the display will be updated to the same orientation, the transform
// won't take effect.
state.mIsTransforming = false;
// 执行回调 applyRotation()
if (applyDisplayRotation != null) {
applyDisplayRotation.run();
}
// 清理相关 WindowToken#mFixedRotationTransformState
// The state is cleared at the end, because it is used to indicate that other windows can
// use seamless rotation when applying rotation to display.
for (int i = state.mAssociatedTokens.size() - 1; i >= 0; i--) {
final WindowToken token = state.mAssociatedTokens.get(i);
token.mFixedRotationTransformState = null;
if (applyDisplayRotation == null) {
// ...
}
}
}
注意,finish fixed rotation 是在 DisplayContent 正式更新配置前执行的。而它只是清理了正在启动 ActivityRecord 的 FixedRotationTransformState。当 ActivityRecord 正式更新配置时,会 update surface position,正式清理 ActivityRecord 的 Fixed Rotation transform。