Android过渡动画,发现掘金小秘密

  1. 在两个Activity定义两个相同类型的View;
  2. 给两个View设置相同的transitionName属性;
  3. 通过ActivityOptions.makeSceneTransitionAnimation()函数生成Bundle对象;
  4. startActivity()函数传递bundle对象。

「栗子讲解,清晰易懂:」

  1. 分别在activity_first.xmlactivity_second.xml布局文件定义ImageView组件,并将transitionName属性设为activityTransform

`

<?xml version="1.0" encoding="utf-8"?>




<?xml version="1.0" encoding="utf-8"?>




`

「预览图」 外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 activityTransform属性也可以通过代码设置。

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { ivImage.transitionName="activityTransform" }

  1. FirstActivity中给ImageView设置点击事件,跳转到第二个Activity。

ivImage.setOnClickListener { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {//判断Android版本 val bundle = ActivityOptions.makeSceneTransitionAnimation(this, ivImage, "activityTransform") .toBundle() startActivity(Intent(this, SecondActivity::class.java), bundle) } else { startActivity(Intent(this, SecondActivity::class.java)) } }

代码中,先判断当前Android版本是否大于等于5.0,大于或等于Android 5.0的话就设置共享元素动画,小于5.0 就正常启动第二个Activity

通过ActivityOptions.makeSceneTransitionAnimation()创建启动Activity过渡的一些参数,makeSceneTransitionAnimation()函数第一个参数为Activity对象;第二个参数为共享元素组件,这里设置为idivImageImageView视图;第三个参数为transitionName属性的值,这里是activityTransform。在调用AcivityOptions对象toBundle函数,包装成Bundle对象。

「效果图:」

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 「多个共享元素过渡」

多个共享元素过渡也很简单,只需要调用makeSceneTransitionAnimation()函数的另外一个重载函数即可。

  1. 在前面XML布局的基础上,给TextView增加transitionName属性:textTransform

`

<?xml version="1.0" encoding="utf-8"?>




<?xml version="1.0" encoding="utf-8"?>




`

  1. 构建多个Pair对象,并传递给makeSceneTransitionAnimation()函数,启动Activity

ivImage.setOnClickListener { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { val imagePair=Pair<View,String>(ivImage,"activityTransform") val textPair=Pair<View,String>(ivImage,"textTransform") val bundle = ActivityOptions.makeSceneTransitionAnimation(this, imagePair,textPair).toBundle() startActivity(Intent(this, SecondActivity::class.java), bundle) } else { startActivity(Intent(this, SecondActivity::class.java)) } }

这里主要是通过将共享视图和transitionName属性的值包装到Pair对象,其他操作和一个共享元素的操作步骤并无区别。

「效果图:」

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

「深坑提醒」

有时从RecyclerView界面进入到详情页,由于详情页加载延迟,可能出现没有效果。例如ImageView从网络加载图片,可能A界面到B界面没效果,B回到A界面有效果。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 解决步骤:

  1. setContentView后添加下面代码,延迟加载过渡动画。

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { postponeEnterTransition() }

  1. 在共享元素视图加载完毕,或者图片加载完毕后调用下面代码,开始加载过渡动画。

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { startPostponedEnterTransition() }

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 例如我是在Glide加载完再调用:

Glide.with(mContext) .asBitmap() .load(value?.avatar ?: "") .listener(object : RequestListener<Bitmap> { override fun onResourceReady(resource: Bitmap?, model: Any?, target: Target<Bitmap>?, dataSource: DataSource?, isFirstResource: Boolean): Boolean { animatorCallback?.invoke()//回调开始加载过渡动画 return false } override fun onLoadFailed(e: GlideException?, model: Any?, target: Target<Bitmap>?, isFirstResource: Boolean): Boolean { animatorCallback?.invoke()//回调开始加载过渡动画 return false } }) .apply(RequestOptions.circleCropTransform()) .placeholder(R.mipmap.ic_default) .error(R.mipmap.ic_default) .into(authorBinding!!.ivAvatar)

大家也可以考虑下面代码:

shareElement.viewTreeObserver.addOnPreDrawListener(object : ViewTreeObserver.OnPreDrawListener { override fun onPreDraw(): Boolean { shareElement!!.viewTreeObserver.removeOnPreDrawListener(this) if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { animatorCallback?.invoke() } return true } })

二)、进入过渡与退出过渡动画

与共享元素相反的,就是Activity进入与退出过渡动画,两个Activity之间在没有共享的视图情况下进行动画切换。下面先看三种动画效果图:「爆炸式效果」「淡入淡出式效果」「滑动式效果」

  • 「爆炸式」:将视图移入场景中心或从中移出;
  • 「滑动式」:将视图从场景的其中一个边缘移入或移出;
  • 「爆炸式」:通过更改视图的不透明度,在场景中添加视图或从中移除视图;

第一个界面采用Fade淡入淡出效果,第二个界面采用了Explode爆炸效果。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 前后两个界面都采用了Slide滑入滑出效果。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 利用Android现有的过渡框架,实现起来是很简单的,步骤如下:

  1. ActivityonCreate()方法中调用 setContentView()前设置启用窗口过渡属性;

window.requestFeature(Window.FEATURE_CONTENT_TRANSITIONS)

  1. 创建过渡效果对象SlideExplodeFade;

val slide=Slide() slide.slideEdge=Gravity.START slide.duration=300//效果时长,一般Activity切换时间很短,不建议设置过长

如果是Slide效果,可以设置slideEdge属性来指定滑动方向,默认是Gravity.BOTTOM

  1. 将过渡效果设置给window相关属性,设置;

//退出当前界面的过渡动画 window.exitTransition = slide //进入当前界面的过渡动画 window.enterTransition = slide //重新进入界面的过渡动画 window.reenterTransition = slide

  1. 调用第二个Activity界面,使用过渡效果。

startActivity( Intent(this, SecondActivity::class.java), ActivityOptions.makeSceneTransitionAnimation(this).toBundle())

那么ActivityOnCreate()方法看起来是这样子的。

override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { window.requestFeature(Window.FEATURE_CONTENT_TRANSITIONS) window.allowEnterTransitionOverlap=false Slide().apply { duration = 300 excludeTarget(android.R.id.statusBarBackground, true) excludeTarget(android.R.id.navigationBarBackground, true) }.also { window.exitTransition = it window.enterTransition = it window.reenterTransition = it } } setContentView(R.layout.activity_first) ivContent.setOnClickListener { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { startActivity( Intent(this, SecondActivity::class.java), ActivityOptions.makeSceneTransitionAnimation(this).toBundle() ) } } }

上面代码中调用 了excludeTarget()方法将状态栏和导航栏排除在过渡动画效果之外。否则会跟着一起起动画效果,不是很美观。

正常情况,退出与进入过渡动画会有一小段交叉的过程,而window.allowEnterTransitionOverlap=false就是禁止交叉,只有退出过渡动画结束后才会再显示进入过渡动画。

如果第二个Activityfinish掉后,回到第一个Activity界面也想有过渡效果,就不要手动调用finish(),可以调用finishAfterTransition ()方法。

三)、兼容Android 5.0前

如果Android 5.0前也想要有切换动画怎么办?

  1. res/anim文件夹下创建想要的效果:

<alpha xmlns:android="http://schemas.android.com/apk/res/android" android:interpolator="@interpolator/decelerate_quad" android:fromAlpha="0.0" android:toAlpha="1.0" android:duration="@android:integer/config_longAnimTime" />

  1. 在启动Activity后调用overridePendingTransition()方法。

val intent = Intent(this, TestActivity2::class.java) startActivity(intent) overridePendingTransition(R.anim.fade_in, R.anim.fade_out)

overridePendingTransition()方法第一个参数为下一个界面进入动画,第二个参数为当前界面退出动画。

到这里,Activity的切换使用过渡动画基本就结束了。有朋友可能会问,只有Activity切换才能应用过渡效果么?

二、布局变化过渡动画

在上一节要理解一个概念:场景。布局的显示与隐藏可以理解分别为一个场景,过渡动画就是解决场景切换带来的生硬视觉感受。Activity切换过渡动画指在两个Activity之间,而布局变化过渡动画,是指同个Activity之间View的变化过渡动画。

一)、手动创建Scene

手动创建场景的话,需要我们自己创建起始和结束场景,利用现有的过渡效果来达到两个场景的切换。默认情况下,当前界面就是起始场景。

  1. 创建起始场景和结束场景的xml布局。起始场景和结束场景需要有相同根元素,例如下面代码idflConatentFrameLayout布局。

`<?xml version="1.0" encoding="utf-8"?>




最后

其实要轻松掌握很简单,要点就两个:

  1. 找到一套好的视频资料,紧跟大牛梳理好的知识框架进行学习。
  2. 多练。 (视频优势是互动感强,容易集中注意力)

你不需要是天才,也不需要具备强悍的天赋,只要做到这两点,短期内成功的概率是非常高的。

对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

阿里P7Android高级教程

下面资料部分截图,诚意满满:特别适合有3-5年开发经验的Android程序员们学习。

附送高清脑图,高清知识点讲解教程,以及一些面试真题及答案解析。送给需要的提升技术、近期面试跳槽、自身职业规划迷茫的朋友们。

Android核心高级技术PDF资料,BAT大厂面试真题解析;

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

593520)]

附送高清脑图,高清知识点讲解教程,以及一些面试真题及答案解析。送给需要的提升技术、近期面试跳槽、自身职业规划迷茫的朋友们。

Android核心高级技术PDF资料,BAT大厂面试真题解析;
[外链图片转存中…(img-737k6NrC-1714459593520)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值