Android 13 WMS-动画流程

 

动画的类型如下

    @IntDef(flag = true, prefix = { "ANIMATION_TYPE_" }, value = {ANIMATION_TYPE_NONE,ANIMATION_TYPE_APP_TRANSITION,ANIMATION_TYPE_SCREEN_ROTATION,ANIMATION_TYPE_DIMMER,ANIMATION_TYPE_RECENTS,ANIMATION_TYPE_WINDOW_ANIMATION,ANIMATION_TYPE_INSETS_CONTROL,ANIMATION_TYPE_TOKEN_TRANSFORM,ANIMATION_TYPE_STARTING_REVEAL})

ANIMATION_TYPE_APP_TRANSITION动画

当点击一个二级页面时, 会有如下调用栈

动画的type = ANIMATION_TYPE_APP_TRANSITION

03-05 14:55:37.752  3059  3163 D jinyanmeianimation: SurfaceAnimator startAnimation mAnimation:com.android.server.wm.LocalAnimationAdapter@71444ae
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: SurfaceAnimator startAnimation mLeash:Surface(name=Surface(name=ActivityRecord{8b5f0f2 u0 com.android.settings/.MainSettings} t76})/@0xaa929f3 - animation-leash of app_transition)/@0x9c116b0
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: SurfaceAnimator startAnimation t:android.view.SurfaceControl$Transaction@8d14824
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: SurfaceAnimator startAnimation type:1
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: SurfaceAnimator startAnimation mInnerAnimationFinishedCallback:com.android.server.wm.SurfaceAnimator$$ExternalSyntheticLambda0@d22724f
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: java.lang.RuntimeException: jinyanmeianimation
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.SurfaceAnimator.startAnimation(SurfaceAnimator.java:200)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowContainer.startAnimation(WindowContainer.java:2912)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowContainer$AnimationRunnerBuilder.lambda$build$4$com-android-server-wm-WindowContainer$AnimationRunnerBuilder(WindowContainer.java:4257)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowContainer$AnimationRunnerBuilder$$ExternalSyntheticLambda4.startAnimation(Unknown Source:7)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowContainer.applyAnimationUnchecked(WindowContainer.java:3384)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowContainer.applyAnimation(WindowContainer.java:3072)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.ActivityRecord.applyAnimation(ActivityRecord.java:6074)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.AppTransitionController.applyAnimations(AppTransitionController.java:876)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.AppTransitionController.applyAnimations(AppTransitionController.java:1083)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.AppTransitionController.handleAppTransitionReady(AppTransitionController.java:293)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.RootWindowContainer.checkAppTransitionReady(RootWindowContainer.java:1070)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.RootWindowContainer.performSurfacePlacementNoTrace(RootWindowContainer.java:937)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.RootWindowContainer.performSurfacePlacement(RootWindowContainer.java:877)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowSurfacePlacer.performSurfacePlacementLoop(WindowSurfacePlacer.java:199)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowSurfacePlacer.performSurfacePlacement(WindowSurfacePlacer.java:148)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowSurfacePlacer.performSurfacePlacement(WindowSurfacePlacer.java:137)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at com.android.server.wm.WindowSurfacePlacer$Traverser.run(WindowSurfacePlacer.java:79)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at android.os.Handler.handleCallback(Handler.java:942)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at android.os.Handler.dispatchMessage(Handler.java:99)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at android.os.Looper.loopOnce(Looper.java:211)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at android.os.Looper.loop(Looper.java:300)
03-05 14:55:37.752  3059  3163 D jinyanmeianimation: 	at android.os.HandlerThread.run(HandlerThread.java:67)

每次循环都要检查是否开始一个动画

void handleAppTransitionReady() {
183          mTempTransitionReasons.clear();
//        //检查app transition是否已经准备好184          if (!transitionGoodToGo(mDisplayContent.mOpeningApps, mTempTransitionReasons)
185                  || !transitionGoodToGo(mDisplayContent.mChangingContainers, mTempTransitionReasons)
186                  || !transitionGoodToGoForTaskFragments()) {
187              return;
188          }
189          Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "AppTransitionReady");
190  
191          ProtoLog.v(WM_DEBUG_APP_TRANSITIONS, "**** GOOD TO GO");
192          // TODO(b/205335975): Remove window which stuck in animatingExit status. Find actual cause.
193          mDisplayContent.forAllWindows(WindowState::cleanupAnimatingExitWindow,
194                  true /* traverseTopToBottom */);
195          // TODO(new-app-transition): Remove code using appTransition.getAppTransition()
196          final AppTransition appTransition = mDisplayContent.mAppTransition;
197  
198          mDisplayContent.mNoAnimationNotifyOnTransitionFinished.clear();
199  
200          appTransition.removeAppTransitionTimeoutCallbacks();
201  
202          mDisplayContent.mWallpaperMayChange = false;
203  
204          int appCount = mDisplayContent.mOpeningApps.size();
205          for (int i = 0; i < appCount; ++i) {
206              // Clearing the mAnimatingExit flag before entering animation. It's set to true if app
207              // window is removed, or window relayout to invisible. This also affects window
208              // visibility. We need to clear it *before* maybeUpdateTransitToWallpaper() as the
209              // transition selection depends on wallpaper target visibility.
210              mDisplayContent.mOpeningApps.valueAtUnchecked(i).clearAnimatingFlags();
211          }
212          appCount = mDisplayContent.mChangingContainers.size();
213          for (int i = 0; i < appCount; ++i) {
214              // Clearing for same reason as above.
215              final ActivityRecord activity = getAppFromContainer(
216                      mDisplayContent.mChangingContainers.valueAtUnchecked(i));
217              if (activity != null) {
218                  activity.clearAnimatingFlags();
219              }
220          }
221  
222          // Adjust wallpaper before we pull the lower/upper target, since pending changes
223          // (like the clearAnimatingFlags() above) might affect wallpaper target result.
224          // Or, the opening app window should be a wallpaper target.
225          mWallpaperControllerLocked.adjustWallpaperWindowsForAppTransitionIfNeeded(
226                  mDisplayContent.mOpeningApps);
227  
228          // Remove launcher from app transition animation while recents is running. Recents animation
229          // is managed outside of app transition framework, so we just need to commit visibility.
230          final boolean excludeLauncherFromAnimation =
231                  mDisplayContent.mOpeningApps.stream().anyMatch(
232                          (app) -> app.isAnimating(PARENTS, ANIMATION_TYPE_RECENTS))
233                  || mDisplayContent.mClosingApps.stream().anyMatch(
234                          (app) -> app.isAnimating(PARENTS, ANIMATION_TYPE_RECENTS));
235          final ArraySet<ActivityRecord> openingAppsForAnimation = getAppsForAnimation(
236                  mDisplayContent.mOpeningApps, excludeLauncherFromAnimation);
237          final ArraySet<ActivityRecord> closingAppsForAnimation = getAppsForAnimation(
238                  mDisplayContent.mClosingApps, excludeLauncherFromAnimation);
239  
240          @TransitionOldType final int transit = getTransitCompatType(
241                  mDisplayContent.mAppTransition, openingAppsForAnimation, closingAppsForAnimation,
242                  mDisplayContent.mChangingContainers,
243                  mWallpaperControllerLocked.getWallpaperTarget(), getOldWallpaper(),
244                  mDisplayContent.mSkipAppTransitionAnimation);
245          mDisplayContent.mSkipAppTransitionAnimation = false;
246  
247          ProtoLog.v(WM_DEBUG_APP_TRANSITIONS,
248                  "handleAppTransitionReady: displayId=%d appTransition={%s}"
249                  + " excludeLauncherFromAnimation=%b openingApps=[%s] closingApps=[%s] transit=%s",
250                  mDisplayContent.mDisplayId, appTransition.toString(), excludeLauncherFromAnimation,
251                  mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps,
252                  AppTransition.appTransitionOldToString(transit));
253  
254          // Find the layout params of the top-most application window in the tokens, which is
255          // what will control the animation theme. If all closing windows are obscured, then there is
256          // no need to do an animation. This is the case, for example, when this transition is being
257          // done behind a dream window.
258          final ArraySet<Integer> activityTypes = collectActivityTypes(openingAppsForAnimation,
259                  closingAppsForAnimation, mDisplayContent.mChangingContainers);
//首先会获取当前需要opening和closing的app window列表(ActivityRecord类型)
260          final ActivityRecord animLpActivity = findAnimLayoutParamsToken(transit, activityTypes,
261                  openingAppsForAnimation, closingAppsForAnimation,
262                  mDisplayContent.mChangingContainers);
263          final ActivityRecord topOpeningApp =
264                  getTopApp(openingAppsForAnimation, false /* ignoreHidden */);
265          final ActivityRecord topClosingApp =
266                  getTopApp(closingAppsForAnimation, false /* ignoreHidden */);
267          final ActivityRecord topChangingApp =
268                  getTopApp(mDisplayContent.mChangingContainers, false /* ignoreHidden */);
269          final WindowManager.LayoutParams animLp = getAnimLp(animLpActivity);
270  
271          // Check if there is any override
272          if (!overrideWithTaskFragmentRemoteAnimation(transit, activityTypes)) {
273              // Unfreeze the windows that were previously frozen for TaskFragment animation.
274              unfreezeEmbeddedChangingWindows();
275              overrideWithRemoteAnimationIfSet(animLpActivity, transit, activityTypes);
276          }
277  
278          final boolean voiceInteraction = containsVoiceInteraction(closingAppsForAnimation)
279                  || containsVoiceInteraction(openingAppsForAnimation);
280  
281          final int layoutRedo;
282          mService.mSurfaceAnimationRunner.deferStartingAnimations();
283          try {
//然后在applyAnimations方法里面对window列表进行遍历WindowContainer的动画applyAnimation方法的调用
284              applyAnimations(openingAppsForAnimation, closingAppsForAnimation, transit, animLp,
285                      voiceInteraction);
286              handleClosingApps();
287              handleOpeningApps();
288              handleChangingApps(transit);
289  
290              appTransition.setLastAppTransition(transit, topOpeningApp,
291                      topClosingApp, topChangingApp);
292  
293              final int flags = appTransition.getTransitFlags();
294              layoutRedo = appTransition.goodToGo(transit, topOpeningApp);
295              handleNonAppWindowsInTransition(transit, flags);
296              appTransition.postAnimationCallback();
297              appTransition.clear();
298          } finally {
299              mService.mSurfaceAnimationRunner.continueStartingAnimations();
300          }
301  
302          mService.mTaskSnapshotController.onTransitionStarting(mDisplayContent);
303  
304          mDisplayContent.mOpeningApps.clear();
305          mDisplayContent.mClosingApps.clear();
306          mDisplayContent.mChangingContainers.clear();
307          mDisplayContent.mUnknownAppVisibilityController.clear();
308  
309          // This has changed the visibility of windows, so perform
310          // a new layout to get them all up-to-date.
311          mDisplayContent.setLayoutNeeded();
312  
313          mDisplayContent.computeImeTarget(true /* updateImeTarget */);
314  
315          mService.mAtmService.mTaskSupervisor.getActivityMetricsLogger().notifyTransitionStarting(
316                  mTempTransitionReasons);
317  
318          Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
319  
320          mDisplayContent.pendingLayoutChanges |=
321                  layoutRedo | FINISH_LAYOUT_REDO_LAYOUT | FINISH_LAYOUT_REDO_CONFIG;
322      }

applyAnimationUnchecked包含两部分, 

第一先通过getAnimationAdapter加载合适的动画

如果是普通的窗口动画,比如app内部activity的切换,当前的场景是设置主菜单跳转子菜单,根据当前场景获取到具体的transit,transit=TRANSIT_OLD_ACTIVITY_OPEN,然后再结合enter为true或者false,可以最终可以找到设置主菜单的动画xml资源是activity_open_exit.xml,设置子菜单的动画xml资源是activity_open_enter.xml,在获取到具体xml资源名字后,通过AnimationUtils.loadAnimation方法把xml资源转成Animation对象。
之后就会创建一个WindowAnimationSpec对象,并把Animation对象作为构造方法的第一个参数传给了WindowAnimationSpec

第二开始动画 

protected void applyAnimationUnchecked(WindowManager.LayoutParams lp, boolean enter,
2994              @TransitionOldType int transit, boolean isVoiceInteraction,
2995              @Nullable ArrayList<WindowContainer> sources) {
2996          final Task task = asTask();
2997          if (task != null && !enter && !task.isActivityTypeHomeOrRecents()) {
2998              final InsetsControlTarget imeTarget = mDisplayContent.getImeTarget(IME_TARGET_LAYERING);
2999              final boolean isImeLayeringTarget = imeTarget != null && imeTarget.getWindow() != null
3000                      && imeTarget.getWindow().getTask() == task;
3001              // Attach and show the IME screenshot when the task is the IME target and performing
3002              // task closing transition to the next task.
3003              if (isImeLayeringTarget && AppTransition.isTaskCloseTransitOld(transit)) {
3004                  mDisplayContent.showImeScreenshot();
3005              }
3006          }
3007          final Pair<AnimationAdapter, AnimationAdapter> adapters = getAnimationAdapter(lp,
3008                  transit, enter, isVoiceInteraction);
3009          AnimationAdapter adapter = adapters.first;
3010          AnimationAdapter thumbnailAdapter = adapters.second;
3011          if (adapter != null) {
3012              if (sources != null) {
3013                  mSurfaceAnimationSources.addAll(sources);
3014              }
3015  
3016              AnimationRunnerBuilder animationRunnerBuilder = new AnimationRunnerBuilder();
3017  
3018              if (isTaskTransitOld(transit)) {
3019                  animationRunnerBuilder.setTaskBackgroundColor(getTaskAnimationBackgroundColor());
3020                  // TODO: Remove when we migrate to shell (b/202383002)
3021                  if (mWmService.mTaskTransitionSpec != null) {
3022                      animationRunnerBuilder.hideInsetSourceViewOverflows(
3023                              mWmService.mTaskTransitionSpec.animationBoundInsets);
3024                  }
3025              }
3026  
3027              final ActivityRecord activityRecord = asActivityRecord();
3028              if (activityRecord != null && isActivityTransitOld(transit)
3029                      && adapter.getShowBackground()) {
3030                  final @ColorInt int backgroundColorForTransition;
3031                  if (adapter.getBackgroundColor() != 0) {
3032                      // If available use the background color provided through getBackgroundColor
3033                      // which if set originates from a call to overridePendingAppTransition.
3034                      backgroundColorForTransition = adapter.getBackgroundColor();
3035                  } else {
3036                      // Otherwise default to the window's background color if provided through
3037                      // the theme as the background color for the animation - the top most window
3038                      // with a valid background color and showBackground set takes precedence.
3039                      final Task arTask = activityRecord.getTask();
3040                      backgroundColorForTransition = ColorUtils.setAlphaComponent(
3041                              arTask.getTaskDescription().getBackgroundColor(), 255);
3042                  }
3043                  animationRunnerBuilder.setTaskBackgroundColor(backgroundColorForTransition);
3044              }
3045  
3046              animationRunnerBuilder.build()
3047                      .startAnimation(getPendingTransaction(), adapter, !isVisible(),
3048                              ANIMATION_TYPE_APP_TRANSITION, thumbnailAdapter);
3049  
3050              if (adapter.getShowWallpaper()) {
3051                  getDisplayContent().pendingLayoutChanges |= FINISH_LAYOUT_REDO_WALLPAPER;
3052              }
3053          }
3054      }
3055  

 计算transition的类型

在getTransitCompatType中会对动画做转换,将动画转化为old类型的动画

 @TransitionOldType static int getTransitCompatType(AppTransition appTransition,ArraySet<ActivityRecord> openingApps, ArraySet<ActivityRecord> closingApps,ArraySet<WindowContainer> changingContainers, @Nullable WindowState wallpaperTarget,@Nullable WindowState oldWallpaper, boolean skipAppTransitionAnimation) {Slog.d("jinyanmeianimation","getTransitCompatType appTransition :" + appTransition );Slog.d("jinyanmeianimation","getTransitCompatType appTransition :" + appTransition , new RuntimeException("jinyanmeianimation"));// Determine if closing and opening app token sets are wallpaper targets, in which case// special animations are needed.final boolean openingAppHasWallpaper = canBeWallpaperTarget(openingApps)&& wallpaperTarget != null;final boolean closingAppHasWallpaper = canBeWallpaperTarget(closingApps)&& wallpaperTarget != null;// Keyguard transit has highest priority.switch (appTransition.getKeyguardTransition()) {case TRANSIT_KEYGUARD_GOING_AWAY:return openingAppHasWallpaper ? TRANSIT_OLD_KEYGUARD_GOING_AWAY_ON_WALLPAPER: TRANSIT_OLD_KEYGUARD_GOING_AWAY;case TRANSIT_KEYGUARD_OCCLUDE:// When there is a closing app, the keyguard has already been occluded by an// activity, and another activity has started on top of that activity, so normal// app transition animation should be used.return closingApps.isEmpty() ? TRANSIT_OLD_KEYGUARD_OCCLUDE: TRANSIT_OLD_ACTIVITY_OPEN;case TRANSIT_KEYGUARD_UNOCCLUDE:return TRANSIT_OLD_KEYGUARD_UNOCCLUDE;}// This is not keyguard transition and one of the app has request to skip app transition.// MIUI MOD: START// For Stage Split Screen: Drag to enter split-screen feature.// if (skipAppTransitionAnimation) {if (skipAppTransitionAnimation ||ActivityTaskManagerServiceStub.get().removeSplitTaskShotIfNeed()) {// ENDreturn WindowManager.TRANSIT_OLD_UNSET;}@TransitionFlags final int flags = appTransition.getTransitFlags();@TransitionType final int firstTransit = appTransition.getFirstAppTransition();// Special transitions// TODO(new-app-transitions): Revisit if those can be rewritten by using flags.if (appTransition.containsTransitRequest(TRANSIT_CHANGE) && !changingContainers.isEmpty()) {// MIUI ADD: START Activity Embedding Resizingif (MiuiEmbeddingWindowServiceStub.get().skipAppTransitionAnimationForEmbeddingDivider()) {return WindowManager.TRANSIT_OLD_UNSET;}// END@TransitContainerType int changingType =getTransitContainerType(changingContainers.valueAt(0));switch (changingType) {case TYPE_TASK:return TRANSIT_OLD_TASK_CHANGE_WINDOWING_MODE;case TYPE_TASK_FRAGMENT:return TRANSIT_OLD_TASK_FRAGMENT_CHANGE;default:throw new IllegalStateException("TRANSIT_CHANGE with unrecognized changing type=" + changingType);}}if ((flags & TRANSIT_FLAG_APP_CRASHED) != 0) {return TRANSIT_OLD_CRASHING_ACTIVITY_CLOSE;}if (firstTransit == TRANSIT_NONE) {return TRANSIT_OLD_NONE;}/** There are cases where we open/close a new task/activity, but in reality only a* translucent activity on top of existing activities is opening/closing. For that one, we* have a different animation because non of the task/activity animations actually work well* with translucent apps.*/if (isNormalTransit(firstTransit)) {boolean allOpeningVisible = true;boolean allTranslucentOpeningApps = !openingApps.isEmpty();for (int i = openingApps.size() - 1; i >= 0; i--) {final ActivityRecord activity = openingApps.valueAt(i);if (!activity.isVisible()) {allOpeningVisible = false;if (activity.fillsParent()) {allTranslucentOpeningApps = false;}}}boolean allTranslucentClosingApps = !closingApps.isEmpty();for (int i = closingApps.size() - 1; i >= 0; i--) {if (closingApps.valueAt(i).fillsParent()) {allTranslucentClosingApps = false;break;}}if (allTranslucentClosingApps && allOpeningVisible) {return TRANSIT_OLD_TRANSLUCENT_ACTIVITY_CLOSE;}if (allTranslucentOpeningApps && closingApps.isEmpty()) {return TRANSIT_OLD_TRANSLUCENT_ACTIVITY_OPEN;}}final ActivityRecord topOpeningApp = getTopApp(openingApps,false /* ignoreHidden */);final ActivityRecord topClosingApp = getTopApp(closingApps,true /* ignoreHidden */);if (closingAppHasWallpaper && openingAppHasWallpaper) {ProtoLog.v(WM_DEBUG_APP_TRANSITIONS, "Wallpaper animation!");switch (firstTransit) {case TRANSIT_OPEN:case TRANSIT_TO_FRONT:return TRANSIT_OLD_WALLPAPER_INTRA_OPEN;case TRANSIT_CLOSE:case TRANSIT_TO_BACK:return TRANSIT_OLD_WALLPAPER_INTRA_CLOSE;}} else if (oldWallpaper != null && !openingApps.isEmpty()&& !openingApps.contains(oldWallpaper.mActivityRecord)&& closingApps.contains(oldWallpaper.mActivityRecord)&& topClosingApp == oldWallpaper.mActivityRecord) {// We are transitioning from an activity with a wallpaper to one without.return TRANSIT_OLD_WALLPAPER_CLOSE;} else if (wallpaperTarget != null && wallpaperTarget.isVisible()&& openingApps.contains(wallpaperTarget.mActivityRecord)&& topOpeningApp == wallpaperTarget.mActivityRecord/* && transit != TRANSIT_TRANSLUCENT_ACTIVITY_CLOSE */) {// We are transitioning from an activity without// a wallpaper to now showing the wallpaperreturn TRANSIT_OLD_WALLPAPER_OPEN;}final ArraySet<WindowContainer> openingWcs = getAnimationTargets(openingApps, closingApps, true /* visible */);final ArraySet<WindowContainer> closingWcs = getAnimationTargets(openingApps, closingApps, false /* visible */);final WindowContainer<?> openingContainer = !openingWcs.isEmpty()? openingWcs.valueAt(0) : null;final WindowContainer<?> closingContainer = !closingWcs.isEmpty()? closingWcs.valueAt(0) : null;@TransitContainerType int openingType = getTransitContainerType(openingContainer);@TransitContainerType int closingType = getTransitContainerType(closingContainer);if (appTransition.containsTransitRequest(TRANSIT_TO_FRONT) && openingType == TYPE_TASK) {return TRANSIT_OLD_TASK_TO_FRONT;}if (appTransition.containsTransitRequest(TRANSIT_TO_BACK) && closingType == TYPE_TASK) {return TRANSIT_OLD_TASK_TO_BACK;}if (appTransition.containsTransitRequest(TRANSIT_OPEN)) {if (openingType == TYPE_TASK) {return (appTransition.getTransitFlags() & TRANSIT_FLAG_OPEN_BEHIND) != 0? TRANSIT_OLD_TASK_OPEN_BEHIND : TRANSIT_OLD_TASK_OPEN;}if (openingType == TYPE_ACTIVITY) {return TRANSIT_OLD_ACTIVITY_OPEN;}if (openingType == TYPE_TASK_FRAGMENT) {return TRANSIT_OLD_TASK_FRAGMENT_OPEN;}}if (appTransition.containsTransitRequest(TRANSIT_CLOSE)) {if (closingType == TYPE_TASK) {return TRANSIT_OLD_TASK_CLOSE;}if (closingType == TYPE_TASK_FRAGMENT) {return TRANSIT_OLD_TASK_FRAGMENT_CLOSE;}if (closingType == TYPE_ACTIVITY) {for (int i = closingApps.size() - 1; i >= 0; i--) {if (closingApps.valueAt(i).visibleIgnoringKeyguard) {return TRANSIT_OLD_ACTIVITY_CLOSE;}}// Skip close activity transition since no closing app can be visiblereturn WindowManager.TRANSIT_OLD_UNSET;}}if (appTransition.containsTransitRequest(TRANSIT_RELAUNCH)&& !openingWcs.isEmpty() && !openingApps.isEmpty()) {return TRANSIT_OLD_ACTIVITY_RELAUNCH;}return TRANSIT_OLD_NONE;}

加载和开始动画

getAnimationAdapter加载动画

 

首先遍历所有的windowContainer 调用applyAnimation

    private void applyAnimations(ArraySet<WindowContainer> wcs, ArraySet<ActivityRecord> apps,@TransitionOldType int transit, boolean visible, LayoutParams animLp,boolean voiceInteraction) {final int wcsCount = wcs.size();for (int i = 0; i < wcsCount; i++) {final WindowContainer wc = wcs.valueAt(i);// If app transition animation target is promoted to higher level, SurfaceAnimator// triggers WC#onAnimationFinished only on the promoted target. So we need to take care// of triggering AR#onAnimationFinished on each ActivityRecord which is a part of the// app transition.final ArrayList<ActivityRecord> transitioningDescendants = new ArrayList<>();for (int j = 0; j < apps.size(); ++j) {final ActivityRecord app = apps.valueAt(j);if (app.isDescendantOf(wc)) {transitioningDescendants.add(app);}}wc.applyAnimation(animLp, transit, visible, voiceInteraction, transitioningDescendants);}}

然后调用getAnimationAdapter 根据窗口层次结构中给定的窗口布局属性获取动画适配器

startAnimation 创建leash开始动画

 

SurfaceAnimation.java
166      void startAnimation(Transaction t, AnimationAdapter anim, boolean hidden,
167              @AnimationType int type,
168              @Nullable OnAnimationFinishedCallback animationFinishedCallback,
169              @Nullable Runnable animationCancelledCallback,
170              @Nullable AnimationAdapter snapshotAnim, @Nullable SurfaceFreezer freezer) {
171          cancelAnimation(t, true /* restarting */, true /* forwardCancel */);
172          mAnimation = anim;
173          mAnimationType = type;
174          mSurfaceAnimationFinishedCallback = animationFinishedCallback;
175          mAnimationCancelledCallback = animationCancelledCallback;
176          final SurfaceControl surface = mAnimatable.getSurfaceControl();
177          if (surface == null) {
178              Slog.w(TAG, "Unable to start animation, surface is null or no children.");
179              cancelAnimation();
180              return;
181          }
182          mLeash = freezer != null ? freezer.takeLeashForAnimation() : null;
183          if (mLeash == null) {
184              mLeash = createAnimationLeash(mAnimatable, surface, t, type,
185                      mAnimatable.getSurfaceWidth(), mAnimatable.getSurfaceHeight(), 0 /* x */,
186                      0 /* y */, hidden, mService.mTransactionFactory);
187              mAnimatable.onAnimationLeashCreated(t, mLeash);
188          }
189          mAnimatable.onLeashAnimationStarting(t, mLeash);
190          if (mAnimationStartDelayed) {
191              ProtoLog.i(WM_DEBUG_ANIM, "Animation start delayed for %s", mAnimatable);
192              return;
193          }
194          mAnimation.startAnimation(mLeash, t, type, mInnerAnimationFinishedCallback);
195          if (ProtoLogImpl.isEnabled(WM_DEBUG_ANIM)) {
196              StringWriter sw = new StringWriter();
197              PrintWriter pw = new PrintWriter(sw);
198              mAnimation.dump(pw, "");
199              ProtoLog.d(WM_DEBUG_ANIM, "Animation start for %s, anim=%s", mAnimatable, sw);
200          }
201          if (snapshotAnim != null) {
202              mSnapshot = freezer.takeSnapshotForAnimation();
203              if (mSnapshot == null) {
204                  Slog.e(TAG, "No snapshot target to start animation on for " + mAnimatable);
205                  return;
206              }
207              mSnapshot.startAnimation(t, snapshotAnim, type);
208          }
209      }

可见创建一个SuefaceControl, 然后把 WindowContainer的surface挂在了leash上面, 然后对leash做动画

455      static SurfaceControl createAnimationLeash(Animatable animatable, SurfaceControl surface,
456              Transaction t, @AnimationType int type, int width, int height, int x, int y,
457              boolean hidden, Supplier<Transaction> transactionFactory) {
458          ProtoLog.i(WM_DEBUG_ANIM, "Reparenting to leash for %s", animatable);
459          final SurfaceControl.Builder builder = animatable.makeAnimationLeash()
460                  .setParent(animatable.getAnimationLeashParent())
461                  .setName(surface + " - animation-leash of " + animationTypeToString(type))
462                  // TODO(b/151665759) Defer reparent calls
463                  // We want the leash to be visible immediately because the transaction which shows
464                  // the leash may be deferred but the reparent will not. This will cause the leashed
465                  // surface to be invisible until the deferred transaction is applied. If this
466                  // doesn't work, you will can see the 2/3 button nav bar flicker during seamless
467                  // rotation.
468                  .setHidden(hidden)
469                  .setEffectLayer()
470                  .setCallsite("SurfaceAnimator.createAnimationLeash");
471          final SurfaceControl leash = builder.build();
472          t.setWindowCrop(leash, width, height);
473          t.setPosition(leash, x, y);
474          t.show(leash);
475          t.setAlpha(leash, hidden ? 0 : 1);
476  
477          t.reparent(surface, leash);
478          return leash;
479      }

 

SurfaceAnimationRunner.java171      void startAnimation(AnimationSpec a, SurfaceControl animationLeash, Transaction t,
172              Runnable finishCallback) {
173          synchronized (mLock) {
174              final RunningAnimation runningAnim = new RunningAnimation(a, animationLeash,
175                      finishCallback);
176              boolean requiresEdgeExtension = requiresEdgeExtension(a);
177  
178              if (requiresEdgeExtension) {
179                  final ArrayList<SurfaceControl> extensionSurfaces = new ArrayList<>();
180                  synchronized (mEdgeExtensionLock) {
181                      mEdgeExtensions.put(animationLeash, extensionSurfaces);
182                  }
183  
184                  mPreProcessingAnimations.put(animationLeash, runningAnim);
185  
186                  // We must wait for t to be committed since otherwise the leash doesn't have the
187                  // windows we want to screenshot and extend as children.
188                  t.addTransactionCommittedListener(mEdgeExtensionExecutor, () -> {
189                      final WindowAnimationSpec animationSpec = a.asWindowAnimationSpec();
190  
191                      final Transaction edgeExtensionCreationTransaction = new Transaction();
192                      edgeExtendWindow(animationLeash,
193                              animationSpec.getRootTaskBounds(), animationSpec.getAnimation(),
194                              edgeExtensionCreationTransaction);
195  
196                      synchronized (mLock) {
197                          // only run if animation is not yet canceled by this point
198                          if (mPreProcessingAnimations.get(animationLeash) == runningAnim) {
199                              // In the case the animation is cancelled, edge extensions are removed
200                              // onAnimationLeashLost which is called before onAnimationCancelled.
201                              // So we need to check if the edge extensions have already been removed
202                              // or not, and if so we don't want to apply the transaction.
203                              synchronized (mEdgeExtensionLock) {
204                                  if (!mEdgeExtensions.isEmpty()) {
205                                      edgeExtensionCreationTransaction.apply();
206                                  }
207                              }
208  
209                              mPreProcessingAnimations.remove(animationLeash);
210                              mPendingAnimations.put(animationLeash, runningAnim);
211                              if (!mAnimationStartDeferred && mPreProcessingAnimations.isEmpty()) {
212                                  mChoreographer.postFrameCallback(this::startAnimations);
213                              }
214                          }
215                      }
216                  });
217              }
218  
219              if (!requiresEdgeExtension) {
220                  mPendingAnimations.put(animationLeash, runningAnim);
221                  if (!mAnimationStartDeferred && mPreProcessingAnimations.isEmpty()) {
222                      mChoreographer.postFrameCallback(this::startAnimations);
223                  }
224              }
225  
226              // Some animations (e.g. move animations) require the initial transform to be
227              // applied immediately.
228              applyTransformation(runningAnim, t, 0 /* currentPlayTime */);
229          }
230      }

发送消息

scheduleFrameLocked:862, Choreographer (android.view)
postCallbackDelayedInternal:605, Choreographer (android.view)
postFrameCallbackDelayed:702, Choreographer (android.view)
postFrameCallback:682, Choreographer (android.view)
continueStartingAnimations:173, SurfaceAnimationRunner (com.android.server.wm)
handleAppTransitionReady:312, AppTransitionController (com.android.server.wm)
checkAppTransitionReady:1070, RootWindowContainer (com.android.server.wm)
performSurfacePlacementNoTrace:937, RootWindowContainer (com.android.server.wm)
performSurfacePlacement:877, RootWindowContainer (com.android.server.wm)
performSurfacePlacementLoop:199, WindowSurfacePlacer (com.android.server.wm)
performSurfacePlacement:148, WindowSurfacePlacer (com.android.server.wm)
performSurfacePlacement:137, WindowSurfacePlacer (com.android.server.wm)
handleAppTransitionTimeout:1814, AppTransition (com.android.server.wm)
lambda$new$0$com-android-server-wm-AppTransition:243, AppTransition (com.android.server.wm)
run:-1, AppTransition$$ExternalSyntheticLambda3 (com.android.server.wm)
handleCallback:942, Handler (android.os)
dispatchMessage:99, Handler (android.os)
loopOnce:211, Looper (android.os)
loop:300, Looper (android.os)
run:67, HandlerThread (android.os)
run:46, ServiceThread (com.android.server)

为动画添加listener

SurfaceAnimationRunner.javaprivate void startAnimationLocked(RunningAnimation a) {
270          final ValueAnimator anim = mAnimatorFactory.makeAnimator();
271  
272          // Animation length is already expected to be scaled.
273          anim.overrideDurationScale(1.0f);
274          anim.setDuration(a.mAnimSpec.getDuration());
275          anim.addUpdateListener(animation -> {
276              synchronized (mCancelLock) {
277                  if (!a.mCancelled) {
278                      final long duration = anim.getDuration();
279                      long currentPlayTime = anim.getCurrentPlayTime();
280                      if (currentPlayTime > duration) {
281                          currentPlayTime = duration;
282                      }
283                      applyTransformation(a, mFrameTransaction, currentPlayTime);
284                  }
285              }
286  
287              // Transaction will be applied in the commit phase.
288              scheduleApplyTransaction();
289          });
290  
291          anim.addListener(new AnimatorListenerAdapter() {
292              @Override
293              public void onAnimationStart(Animator animation) {
294                  synchronized (mCancelLock) {
295                      if (!a.mCancelled) {
296                          // TODO: change this back to use show instead of alpha when b/138459974 is
297                          // fixed.
298                          mFrameTransaction.setAlpha(a.mLeash, 1);
299                      }
300                  }
301              }
302  
303              @Override
304              public void onAnimationEnd(Animator animation) {
305                  synchronized (mLock) {
306                      mRunningAnimations.remove(a.mLeash);
307                      synchronized (mCancelLock) {
308                          if (!a.mCancelled) {
309  
310                              // Post on other thread that we can push final state without jank.
311                              mAnimationThreadHandler.post(a.mFinishCallback);
312                          }
313                      }
314                  }
315              }
316          });
317          a.mAnim = anim;
318          mRunningAnimations.put(a.mLeash, a);
319  
320          anim.start();
321          if (a.mAnimSpec.canSkipFirstFrame()) {
322              // If we can skip the first frame, we start one frame later.
323              anim.setCurrentPlayTime(mChoreographer.getFrameIntervalNanos() / NANOS_PER_MS);
324          }
325  
326          // Immediately start the animation by manually applying an animation frame. Otherwise, the
327          // start time would only be set in the next frame, leading to a delay.
328          anim.doAnimationFrame(mChoreographer.getFrameTime());
329      }
330  

 

doFrame时更新动画位置

放vsync到来后, 执行onAnimationUpdate

    private void applyTransformation(RunningAnimation a, Transaction t, long currentPlayTime) {a.mAnimSpec.apply(t, a.mLeash, currentPlayTime);}

再来看下apply方法的具体实现,通过之前以具体xml资源创建的mAnimation对象,根据当前时间片currentPlayTime获取到当前的tmp.transformation,对leash对象实现了Matrix(大小,位置),Alpha,Crop等transformation变化,再通过Transaction 交给surfaceflinger显示,从而实现了动画当前时间片的显示效果。对比旧动画机制,这个transformation变化是在WindowStateAnimator类里面实现的。为什么要重点关注这个方法呢?因为如果窗口动画出bug了(位置大小不对?透明度异常?),就可以在这个方法里面打印window的相关参数来初步定位原因。 

119      @Override
120      public void apply(Transaction t, SurfaceControl leash, long currentPlayTime) {
121          final TmpValues tmp = mThreadLocalTmps.get();
122          tmp.transformation.clear();
123          mAnimation.getTransformation(currentPlayTime, tmp.transformation);
124          tmp.transformation.getMatrix().postTranslate(mPosition.x, mPosition.y);
125          t.setMatrix(leash, tmp.transformation.getMatrix(), tmp.floats);
126          t.setAlpha(leash, tmp.transformation.getAlpha());
127  
128          boolean cropSet = false;
129          if (mRootTaskClipMode == ROOT_TASK_CLIP_NONE) {
130              if (tmp.transformation.hasClipRect()) {
131                  final Rect clipRect = tmp.transformation.getClipRect();
132                  accountForExtension(tmp.transformation, clipRect);
133                  t.setWindowCrop(leash, clipRect);
134                  cropSet = true;
135              }
136          } else {
137              mTmpRect.set(mRootTaskBounds);
138              if (tmp.transformation.hasClipRect()) {
139                  mTmpRect.intersect(tmp.transformation.getClipRect());
140              }
141              accountForExtension(tmp.transformation, mTmpRect);
142              t.setWindowCrop(leash, mTmpRect);
143              cropSet = true;
144          }
145  
146          // We can only apply rounded corner if a crop is set, as otherwise the value is meaningless,
147          // since it doesn't have anything it's relative to.
148          if (cropSet && mAnimation.hasRoundedCorners() && mWindowCornerRadius > 0) {
149              t.setCornerRadius(leash, mWindowCornerRadius);
150          }
151      }

6、ValueAnimator类,从上面的介绍可以得知,窗口动画的最终本质就是一个ValueAnimator属性动画,理解了这一点,就相当于把窗口动画简单化了,最终的实现就类比于我们普通app的属性动画的实现(app属性动画的对象是view,窗口属性动画的对象是window),只不过整个流程比较复杂而已,但是最终的实现原理是一样的,殊途同归,这个才是android窗口动画机制的精髓所在

Rotation动画 ANIMATION_TYPE_SCREEN_ROTATION

1. 根据sensor导致的旋转动画

最终也会到SurfaceAnimation

 拿到动画适配器:

FadeAnimationController.javapublic void fadeWindowToken(boolean show, WindowToken windowToken, int animationType) {if (windowToken == null || windowToken.getParent() == null) {return;}final Animation animation = show ? getFadeInAnimation() : getFadeOutAnimation();final FadeAnimationAdapter animationAdapter = animation != null? createAdapter(createAnimationSpec(animation), show, windowToken) : null;if (animationAdapter == null) {return;}windowToken.startAnimation(windowToken.getPendingTransaction(), animationAdapter,show /* hidden */, animationType, null /* finishedCallback */);}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/519780.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

片上网络(NoC)技术的发展及其给高端FPGA带来的优势

片上网络(NoC)技术的发展及其给高端FPGA带来的优势 1. 概述 在摩尔定律的推动下,集成电路工艺取得了高速发展,单位面积上的晶体管数量不断增加。 片上系统(System-on-Chip,SoC)具有集成度高、功耗低、成本低等优势,已经成为大规模集成电路系统设计的主流方向,解决了…

【Linux】Linux原生异步IO(一):libaio-介绍

1、IO模型 1.1 简述 相信大家在搜索的时候,都会看到下面这张图,IO的使用场景:同步、异步、阻塞、非阻塞,可以组合成四种情况: 同步阻塞I/O: 用户进程进行I/O操作,一直阻塞到I/O操作完成为止。同步非阻塞I/O: 用户程序可以通过设置文件描述符的属性O_NONBLOCK,I/O操作可…

量化交易日记 基础概念篇

联系方式 17710158550 NBEATS (Neural Basis Expansion Analysis for Time Series)、NHiTS (Neural Hierarchical Interpolation for Time Series Forecasting)、LSTNet (Long Short-Term Memory Network)、TCN (Temporal Convolutional Network)、Transformer、DeepAR (DeepAR…

2024.3.7每日一题

LeetCode 找出字符串的可整除数组 题目链接&#xff1a;2575. 找出字符串的可整除数组 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给你一个下标从 0 开始的字符串 word &#xff0c;长度为 n &#xff0c;由从 0 到 9 的数字组成。另给你一个正整数 m 。 word 的 …

今天分享一个好看的输入法皮肤相信每个人心里住着一个少女心我们美化一下她吧

标题&#xff1a; 白日梦皮肤上线&#xff0c;百度输入法助你开启梦幻之旅&#xff01; 正文&#xff1a; 大家好呀&#xff01;今天我来给大家安利一款超级梦幻的百度输入法皮肤——“白日梦”系列&#xff01; 这款皮肤的设计灵感来源于我们内心深处的白日梦&#xff0c;充…

Matlab 多项式插值(曲线拟合)

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 由于对曲线拟合有些兴趣,这里就找了一些资料从最基本的方法来看一下曲线拟合的效果: 二、实现代码 % **********

就业班 2401--3.7 Linux Day13--日志轮转+jumpserver堡垒机

一、日志轮转 日志重要性 Linux系统日志对管理员来说&#xff0c;是了解系统运行的主要途径&#xff0c;因此需要对 Linux 日志系统有个详细的了解。 Linux 系统内核和许多程序会产生各种错误信息、告警信息和其他的提示信息&#xff0c;这些各种信息都应该记录到日志文件中&a…

SQOOP安装与使用

SQOOP安装及使用 文章目录 SQOOP安装及使用SQOOP安装1、上传并解压2、修改配置文件3、修改环境变量4、添加MySQL连接驱动5、测试 准备MySQL数据登录MySQL数据库创建student数据库切换数据库并导入数据另外一种导入数据的方式使用Navicat运行SQL文件导出MySQL数据库 importMySQL…

Leetcode : 147. 对链表进行插入排序

给定单个链表的头 head &#xff0c;使用 插入排序 对链表进行排序&#xff0c;并返回 排序后链表的头 。 插入排序 算法的步骤: 插入排序是迭代的&#xff0c;每次只移动一个元素&#xff0c;直到所有元素可以形成一个有序的输出列表。 每次迭代中&#xff0c;插入排序只从输…

SpringBoot项目如何添加全局接口上下文

1. 定义Spring Boot应用的路由 首先&#xff0c;确保您的Spring Boot应用有一个统一的路由前缀。例如&#xff0c;可以在application.properties或application.yml配置文件中使用server.servlet.context-path属性来定义所有请求的基础路径。 # application.properties server…

[OpenCv]频域处理

目录 前言 一、频域变换 1.傅里叶变换 2.代码实现 二、频域中图像处理 1.理解数字图片的频谱 2.频域图像处理步骤 3.使用低通滤波器实现图像平滑 4.使用高通滤波器实现图像锐化 三、总结 前言 数字图像处理的方法有两大类&#xff1a;一种是空间域处理法&#xff0c;…

0x04_数组_指针_字符串

数组 数组的定义与使用 数组是具有一定顺序关系的若干相同类型变量的集合体&#xff0c;组成数组的变量称为该数组的元素。 给出下面程序的输出&#xff1a; #include <iostream> using namespace std; int main() {int a[10], b[10];for(int i 0; i < 10; i) {a[…