前言

早在Android 4.4,Transition 就已经引入,但在5.0才得以真正的实现。而究竟Transition是用来干嘛的呢。接下来我将通过实例和原理解析来分析下Google这个强大的动画框架。

先来张效果图镇住场面

这个效果下文会介绍如何实现,不过要先理解透这个框架的一些基础概念。

Transition Framework 核心就是根据Scene(场景,下文解释)的不同帮助开发者们自动生成动画。通常主要是通过以下几个方法开启动画。

  • TransitionManager.go()
  • beginDelayedTransition()
  • setEnterTransition()/setSharedElementEnterTransition()

我们来逐一解释以上各种情况

TransitionManager.go()

首先,先介绍下Scene这个类,看看官方的解释

A scene represents the collection of values that various properties in the View hierarchy will have when the scene is applied. A Scene can be configured to automatically run a Transition when it is applied, which will animate the various property changes that take place during the scene change.

通俗的解释就是这个类存储着一个根view下的各种view的属性。通常由getSceneForLayout (ViewGroup sceneRoot,int layoutId,Context context)获取实例。

  • sceneRoot

scene发生改变和动画执行的位置

  • layoutId

即上文所说的根view

可能这样解释有点无力,下面我举个例子。

  1. private Scene scene1;
  2. private Scene scene2;
  3. private boolean isScene2;
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_scene);
  8. initToolbar();
  9. initScene();
  10. }
  11. private void initScene() {
  12. ViewGroup sceneRoot= (ViewGroup) findViewById(R.id.scene_root);
  13. scene1=Scene.getSceneForLayout(sceneRoot,R.layout.scene_1,this);
  14. scene2=Scene.getSceneForLayout(sceneRoot,R.layout.scene_2,this);
  15. TransitionManager.go(scene1);
  16. }
  17. /**
  18. * scene1和scene2相互切换,播放动画 * @param view
  19. */
  20. public void change(View view){
  21. TransitionManager.go(isScene2?scene1:scene2,new ChangeBounds());
  22. isScene2=!isScene2;
  23. }

scene1:

scene2:

注意,两个scene布局中1和4,2和3除了图片位置大小不一样,其id是一样的。可以当成一个view.因为分析比较起始scene 的不同创建动画是针对于同一个view的。

上述简单的例子是通过第一种方式TransitionManager.go()触发动画。即在进入Activity的时候,手动将start scene通过TransitionManager.go(scene1)设置为scene1。点击button通过TransitionManager.go(scene2,new ChangeBounds())切换到end scene状态:scene2.Transition 框架通过ChangeBounds类分析start scene和end scene的不同创建并播放动画。由于ChangeBounds类是分析比较两个scene中view的位置边界创建移动和缩放动画。发现从scene1->scene2其实是1->4,2->3。于是就执行相应的动画,即是如下效果:

类似于ChangeBounds类的还有以下几种,他们都是继承Transiton类

  • ChangeBounds

检测view的位置边界创建移动和缩放动画

  • ChangeTransform

检测view的scale和rotation创建缩放和旋转动画

  • ChangeClipBounds

检测view的剪切区域的位置边界,和ChangeBounds类似。不过ChangeBounds针对的是view而ChangeClipBounds针对的是view的剪切区域(setClipBound(Rect rect)中的rect)。如果没有设置则没有动画效果

  • ChangeImageTransform

检测ImageView(这里是专指ImageView)的尺寸,位置以及ScaleType,并创建相应动画。

  • Fade,Slide,Explode

这三个都是根据view的visibility的不同分别创建渐入,滑动,爆炸动画。

以上各个动画类的实现效果如下:

  • AutoTransition

如果TransitionManager.go(scene1)不指定动画,则默认动画是AutoTransition类。它其实是一个动画集合,查看源码可知其实是动画集合中添加了Fade和ChangeBounds类。

  1. private void init() {
  2. setOrdering(ORDERING_SEQUENTIAL);
  3. addTransition(new Fade(Fade.OUT)).
  4. addTransition(new ChangeBounds()).
  5. addTransition(new Fade(Fade.IN));
  6. }

说到动画集合,其实动画类不仅可以通过类似new ChangeBounds()方法创建,也可以通过xml文件创建。且如果对于动画集合,xml方式可能会更加方便。

只需要两步,第一步在res/transition创建一个xml文件,如下:

res/transition/changebounds_and_fade.xml:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <transitionSet xmlns:android="http://schemas.android.com/apk/res/android">
  3. <changeBounds />
  4. <fade />
  5. </transitionSet>

然后再代码中调用:

  1. Transition sets=TransitionInflater.from(this).inflateTransition(R.transition.changebounds_and_fade);

最后补充一点,关于和TransitionManager.go(scene2)其实是调用当前的scene(scene1)的scene1.exit()以及下一个scene(scene2)的scene2.enter()

而它们又分别会触发scene1.setExitAction()和scene1.setEnterAction().可以在这两个方法中定制一些特别的效果.

beginDelayedTransition()

接下来介绍下一个触发方式,如果上面的理解透了话下面的就很简单了。之前的那种TransitionManager.go()一直都是根据xml文件创造start scene和end scene,这样未免有些麻烦。

而beginDelayedTransition()原理则是通过代码改变view的属性,然后通过之前介绍的ChangeBounds等类分析start scene和end Scene不同来创建动画。

依然举个例子:

  1. @Override
  2. protected void onCreate(Bundle savedInstanceState) {
  3. super.onCreate(savedInstanceState);
  4. setContentView(R.layout.activity_begin_delayed);
  5. initToolBar();
  6. initView();
  7. }
  8. @Override
  9. public void onClick(View v) {
  10. //start scene 是当前的scene
  11. TransitionManager.beginDelayedTransition(sceneRoot, TransitionInflater.from(this).inflateTransition(R.transition.explode_and_changebounds));
  12. //next scene 此时通过代码已改变了scene statue
  13. changeScene(v);
  14. }
  15. private void changeScene(View view) {
  16. changeSize(view);
  17. changeVisibility(cuteboy,cutegirl,hxy,lly);
  18. view.setVisibility(View.VISIBLE);
  19. }
  20. /**
  21. * view的宽高1.5倍和原尺寸大小切换 * 配合ChangeBounds实现缩放效果 * @param view
  22. */
  23. private void changeSize(View view) {
  24. isImageBigger=!isImageBigger;
  25. ViewGroup.LayoutParams layoutParams = view.getLayoutParams();
  26. if(isImageBigger){
  27. layoutParams.width=(int)(1.5*primarySize);
  28. layoutParams.height=(int)(1.5*primarySize);
  29. }else {
  30. layoutParams.width=primarySize;
  31. layoutParams.height=primarySize;
  32. }
  33. view.setLayoutParams(layoutParams);
  34. }
  35. /**
  36. * VISIBLE和INVISIBLE状态切换 * @param views
  37. */
  38. private void changeVisibility(View ...views){
  39. for (View view:views){
  40. view.setVisibility(view.getVisibility()==View.VISIBLE?View.INVISIBLE:View.VISIBLE);
  41. }
  42. }

当触发点击事件时候,此时记录下当前scene status,然后改变被点击view的尺寸,并改变其他view的visibility,再记录下改变后的scene status。而本例中beginDelayedTransition()第二个参数传的是一个ChangeBounds和Explode动画集合,所以这个集合的中改变尺寸的执行缩放动画,改变visibility的执行爆炸效果。整体效果如下:

界面切换动画

前面说了那么多终于到了重头戏了:Activity/Fragment之前的切换效果。界面切换有两种,一种是不带共享元素的Content Transition一种是带有共享元素的Shared Element Transition。

Content Transition

先解释下几个重要概念:

  • A.exitTransition(transition)

Transition框架会先遍历A界面确定要执行动画的view(非共享元素view),执行A.exitTransition()前A界面会获取界面的start scene(view 处于VISIBLE状态),然后将所有的要执行动画的view设置为INVISIBLE,并获取此时的end scene(view 处于INVISIBLE状态).根据transition分析差异的不同创建执行动画。

  • B.enterTransition()

Transition框架会先遍历B界面,确定要执行动画的view,设置为INVISIBLE。执行B.enterTransition()前获取此时的start scene(view 处于INVISIBLE状态),然后将所有的要执行动画的view设置为VISIBLE,并获取此时的end scene(view 处于VISIBLE状态).根据transition分析差异的不同创建执行动画。

根据上文解释,界面切换动画是建立在visibility的改变的基础上的,所以getWindow().setEnterTransition(transition);中的参数一般传的是Fade,Slide,Explode类的实例(因为这三个类是通过分析visibility不同创建动画的)。通常写一个完整的Activity Content Transiton有以下几个步骤:

  • 在style中添加
  1. <item name="android:windowActivityTransitions">true</item>

Material主题的应用自动设置为true.

  • 设置相应的A离开/B进入/B离开/A重新进入动画。
  1. //A 不设置默认为null
  2. getWindow().setExitTransition(transition);
  3. //B 不设置默认为Fade
  4. getWindow().setEnterTransition(transition);
  5. //B 不设置默认为EnterTransition
  6. getWindow().setReturnTransition(transition);
  7. //A 不设置默认为ExitTransition
  8. getWindow().setReenterTransition(transition);

当然也可以在主题中设置

  1. <item name="android:windowEnterTransition">@transition/slide_and_fade</item>
  2. <item name="android:windowReturnTransition">@transition/return_slide</item>
  • 跳转界面

这里的跳转界面不能仅仅startActivity(intent),需要

  1. Bundle bundle=ActivityOptionsCompat.makeSceneTransitionAnimation(activity).toBundle;
  2. startActivity(intent,bundle)

ok到这里为止既可以运行activity之间的切换动画了。

但是你会发现,在界面切换的时候,A退出时,过了一小会,B就进入了,(真是过分,不给A完全展示ExitTransition)如果你是想等A完全退出后B再进入可以通过设置setAllowEnterTransitionOverlap(false)(默认是true),同样可以在xml中设置:

  1. <item name="android:windowAllowEnterTransitionOverlap">false</item>
  2. <item name="android:windowAllowReturnTransitionOverlap">false</item>

说了这么多我觉得又得举个简单例子。

A.Activity:

  1. @Override
  2. protected void onCreate(Bundle savedInstanceState) {
  3. super.onCreate(savedInstanceState);
  4. setContentView(R.layout.activity_main);
  5. initToolBar();
  6. getWindow().setExitTransition(TransitionInflater.from(this).inflateTransition(R.transition.slide));
  7. //未设置setReenterTransition()默认和setExitTransition一样
  8. }
  9. public void goContentTransitions(View view){
  10. Intent intent = new Intent(this, ContentTransitionsActivity.class);
  11. ActivityOptionsCompat activityOptionsCompat = ActivityOptionsCompat.makeSceneTransitionAnimation(this);
  12. startActivity(intent,activityOptionsCompat.toBundle());
  13. }

res/translation/slide.xml:

  1. <transitionSet xmlns:android="http://schemas.android.com/apk/res/android">
  2. <slide android:duration="1000"></slide>
  3. </transitionSet>

B.Activity:

  1. @Override
  2. protected void onCreate(Bundle savedInstanceState) {
  3. super.onCreate(savedInstanceState);
  4. setContentView(R.layout.activity_content_transitions);
  5. initToolbar();
  6. Slide slide=new Slide();
  7. slide.setDuration(500);
  8. slide.setSlideEdge(Gravity.LEFT);
  9. getWindow().setEnterTransition(slide);
  10. getWindow().setReenterTransition(new Explode().setDuration(600));
  11. }

实现的效果如下:

仔细看着动画你其实可以发现A的状态栏也跟着下拉上拉了,而且和下面的视图有一定的间距。处女座表示不能忍。

其实从原理上来解释,Activity的切换动画针对的是整个界面的view的visibility,而有没有什么方法能让Transition框架只关注某一个view或者不关注某个view呢。当然,transition.addTarget()和transition.excludeTarget()可以分别实现上述功能。

方便的是也可以在xml设置该属性,那么我们现在要做的是将statusBar排除掉,可以在slide.xml这样写:

  1. <transitionSet xmlns:android="http://schemas.android.com/apk/res/android">
  2. <slide android:duration="1000">
  3. <targets >
  4. <!--表示除了状态栏-->
  5. <target android:excludeId="@android:id/statusBarBackground"/>
  6. <!--表示只针对状态栏-->
  7. <!--<target android:targetId="@android:id/statusBarBackground"/>-->  </targets>
  8. </slide>
  9. </transitionSet>

大功告成,效果我就不贴了,各位可以脑补一下...

Shared Element Transition

界面切换中往往Content Transition和Shared Element Transition是同时存在的,区别于Content Transition,主要有以下几个不同点:

  • startActivity()
  1. Bundle bundle=ActivityOptionsCompat.makeSceneTransitionAnimation(activity,pairs).toBundle;
  2. startActivity(intent,bundle)

这里的pairs是Pair<View, String>类的实例集合,存储着两个activity之间共享view和name。这里的name要和B界面的共享view的transitionName一致。就像这样:

  1. Intent intent = new Intent(this, WithSharedElementTransitionsActivity.class);
  2. ActivityOptionsCompat activityOptionsCompat = ActivityOptionsCompat.makeSceneTransitionAnimation(this
  3. ,new Pair<View, String>(shared_image,"shared_image_")
  4. ,new Pair<View, String>(shared_text,"shared_text_"));
  5. startActivity(intent,activityOptionsCompat.toBundle());//xml<TextView
  6. android:text="withShared"
  7. android:transitionName="shared_text_"
  8. style="@style/MaterialAnimations.TextAppearance.Title.Inverse"
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content" />
  11. ><de.hdodenhof.circleimageview.CircleImageView
  12. android:id="@+id/icon_gg"
  13. android:layout_centerInParent="true"
  14. android:src="@mipmap/xkl"
  15. android:transitionName="shared_image_"
  16. android:layout_width="150dp"
  17. android:layout_height="150dp" />
  • setSharedElementEnterTransition()/setSharedElementReturnTransition()

不设置的话默认是@android:transition/move动画。而setExitTransition()和setEnterTransition()默认为null和Fade.

其实Shared Element Transition原理和Content Transition类似都是根据始末scene status的不同创建动画。

不同的是Content Transition是通过改变view的visibility来改变scene状态从而进一步创建动画,而Shared Element Transition是分析A B界面共享view的尺寸,位置,样式的不同创建动画化的。所以前者通常设置Fade等Transition后者通常设置ChangeBounds等Transition.

最后的最后让我们来分析如何实现文章一开始的那个gif图效果。

  1. 整个动画包括Content Transition和Shared Element Transition。而A界面的setExitTransition()并没有设置为null。
  2. 当进入B界面,这里的共享view只是单纯的移动所以setSharedElementEnterTransition(transition)可以不用设置,默认为move。同时会执行一个水纹展开动画,这个可以通过ViewAnimationUtils.createCircularReveal()方法实现。在Shared Element Transition结束之后执行Content Transition,可以看出是Slide动画。所以可以通过设置setExitTransition(new Slide())完成。注意这里Slide只作用于底部的item(要设置target),否则就作用于一整个视图了。
  3. 最关键的来了,在B退出时候,可以看到屏幕上半部分向上滑过,下半部分向下滑过。一种从中间撕开的视觉效果。所以可以将布局一分为二并指定为用两个不同方向的Slide的Target,差不多像这样:
  1. <transitionSet
  2. android:duration="800" xmlns:android="http://schemas.android.com/apk/res/android">
  3. <slide android:slideEdge="top">
  4. <targets >
  5. <target android:targetId="@id/viewGroup_top"></target>
  6. </targets>
  7. </slide>
  8. <slide android:slideEdge="bottom">
  9. <targets >
  10. <target android:targetId="@id/viewGroup_bottom"></target>
  11. </targets>
  12. </slide>
  13. </transitionSet>

这里其实有个坑,我们先来看看isTransitionGroup()这个方法:

  1. public boolean isTransitionGroup() {
  2. if ((mGroupFlags & FLAG_IS_TRANSITION_GROUP_SET) != 0) {
  3. return ((mGroupFlags & FLAG_IS_TRANSITION_GROUP) != 0);
  4. } else {
  5. final ViewOutlineProvider outlineProvider = getOutlineProvider();
  6. return getBackground() != null || getTransitionName() != null ||
  7. (outlineProvider != null && outlineProvider != ViewOutlineProvider.BACKGROUND);
  8. }
  9. }

返回值为true表示这个ViewGroup作为一个整体执行Activity Transition,false表示这个ViewGroup中子view各自执行各自的。如果这个ViewGroup设置了background或者TransitionName,或者setTransitionGroup(true)则返回值为true表示作为一个整体执行动画.

所以这里的viewGroup_bottom和viewGroup_top最好设置下setTransitionGroup(true).

实现效果如下,自己加了点其他特效

具体代码我就不贴了,本文的所有的代码已上传Github(我是链接),包括Fragment的切换本文未作介绍代码中有写。希望大家能点个star。

如果你看了一遍还是不知所云那我强烈建议你结合代码运行下在看一遍,其实搞懂了还是蛮简单的。

最后我想说的是关于这个Transition Framework还有一些内容没说完,可能要等过段时间更新了,接下来还会写关于Dagger 2的相关文章以及NavigationBar的加强版,敬请期待吧。

作者:陈留年
来源:51CTO

Android Transition Framework详解---超炫的动画框架相关推荐

  1. Android系统目录结构详解

    Android系统基于linux内核.JAVA应用,算是一个小巧精致的系统.虽是开源,但不像Linux一般庞大,娇小可亲,于是国内厂商纷纷开发出自己基于Android的操作系统.在此呼吁各大厂商眼光放 ...

  2. [Android]多进程知识点详解

    作者:Android开发_Hua 链接:https://www.jianshu.com/p/e0f833151f66 多进程知识点汇总: 一:了解多进程 二:项目中多进程的实现 三:多进程的优缺点与使 ...

  3. Android学习笔记——Android 签名机制详解

    Android 签名机制详解 近期由于工作需要在学习 Android 的签名机制,因为没有现成资料,只能通过开发者文档和阅读博客的方式对 Android 签名机制进行大致了解.过程中查阅到的资料相对零 ...

  4. Android 全局大喇叭——详解广播机制

    Android 全局大喇叭--详解广播机制 一.广播机制简介 1. 标准广播(Normal broadcasts) 2. 有序广播(Ordered broadcasts) 二.接收系统广播 1. 动态 ...

  5. 《Android游戏开发详解》——第1章,第1.6节函数(在Java中称为“方法”更好)...

    本节书摘来自异步社区<Android游戏开发详解>一书中的第1章,第1.6节函数(在Java中称为"方法"更好),作者 [美]Jonathan S. Harbour,更 ...

  6. JMessage Android 端开发详解

    JMessage Android 端开发详解 目前越来越多的应用会需要集成即时通讯功能,这里就为大家详细讲一下如何通过集成 JMessage 来为你的 App 增加即时通讯功能. 首先,一个最基础的 ...

  7. 《Java和Android开发实战详解》——2.5节良好的Java程序代码编写风格

    本节书摘来自异步社区<Java和Android开发实战详解>一书中的第2章,第2.5节良好的Java程序代码编写风格,作者 陈会安,更多章节内容可以访问云栖社区"异步社区&quo ...

  8. Android事件流程详解

    Android事件流程详解 网络上有不少博客讲述了android的事件分发机制和处理流程机制,但是看过千遍,总还是觉得有些迷迷糊糊,因此特地抽出一天事件来亲测下,向像我一样的广大入门程序员详细讲述an ...

  9. Android Studio 插件开发详解二:工具类

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/78112856 本文出自[赵彦军的博客] 在插件开发过程中,我们按照开发一个正式的项 ...

  10. 《Android游戏开发详解》一2.16 区分类和对象

    本节书摘来异步社区<Android游戏开发详解>一书中的第2章,第2.16节,作者: [美]Jonathan S. Harbour 译者: 李强 责编: 陈冀康,更多章节内容可以访问云栖社 ...

最新文章

  1. 正式环境docker部署hyperf_应用部署 - Docker Swarm 集群搭建 - 《Hyperf v1.1.1 开发文档》 - 书栈网 · BookStack...
  2. JSP Tomcat8.0运行连接池时发生异常【AbstractMethodError oracle.jdbc.driver.T4CConnection.isValid(I)Z】...
  3. 【Python-ML】SKlearn库RANSAC拟合高鲁棒性回归模型
  4. 画火柴人动画的手机软件_王者荣耀 玩家画的火柴人造型的英雄 图五的英雄你能猜出是谁吗...
  5. JavaScript 常用函数
  6. php设计模式的六大原则(二):开闭原则
  7. python selenium po模式_Python+Selenium+Unittest实现PO模式web自动化框架
  8. (二十六)【2021 WWW】Knowledge-Preserving Incremental Social Event Detection via Heterogeneous GNNs
  9. django-反转路径时带上参数-适用于路径中通过命名分组的形式来捕捉参数的情景
  10. python function terminated un_绕过 RestrictedUnpickler
  11. Linux基础自学记录六-引导流程解析2
  12. android之uniapp弹出activity
  13. 阿里云移动推送的接入和踩坑
  14. Kali Linux 安装 COMFAST CF-WU925A Realtek RTL8811CU/RTL8821CU USB 无线网卡驱动
  15. c语言---25 扫雷游戏
  16. on device trainning
  17. android ibinder类接口编辑
  18. 计算机体系结构——1.1 计算机体系结构的概念与发展
  19. 【如何开发小程序】自己如何开发小程序?
  20. 投资学作业-投资咨询

热门文章

  1. word树状分支图_带你看懂市政管道工程图
  2. R语言在图上标出点坐标_利用R语言的leaflet包根据GPS信息在地图上标点
  3. cad注释比例和打印比例不一样_CAD中输出不同比例图纸效率不高,原因在这里
  4. 计算机管理固件在哪里,itunes固件位置在哪里
  5. android 登录 service_如何优雅的实现自己的Android组件化改造?
  6. gin框架的学习--golang
  7. matlab按图像边缘抠图_12. 泊松图像编辑
  8. html解决ajax调用跨域,JQuery Ajax执行跨域请求数据的解决方案
  9. 正则表达式收集(转)
  10. Debian 9 安装配置 Caddy Server