Material Animations 4:Circular Reveal Animations

   2016-10-13 0
核心提示:Circular Reveal Animations,官方称之为循环揭露动画效果,是一种用来显示/隐藏一组UI界面元素的动画效果,它是在API 21引入的,对应的类是 ViewAnimationUtils 。 循环揭露动画效果可以和共享元素变换动画组合,用来创造一些有意义的动画效果,自然地告诉用

Circular Reveal Animations,官方称之为循环揭露动画效果,是一种用来显示/隐藏一组UI界面元素的动画效果,它是在API 21引入的,对应的类是 ViewAnimationUtils

循环揭露动画效果可以和共享元素变换动画组合,用来创造一些有意义的动画效果,自然地告诉用户这个app有些什么东西,将会产生怎样的效果。

二、效果图

Material Animations 4:Circular Reveal Animations

三、实现

在上面的例子中,依次发生了:

  • 橘色的圆是一个共享元素,从 MainActivity 变换到 CircularRevealActivity
  • CircularRevealActivity 中有一个监听器(listener),用来监听共享元素转换动画的结束,当动画结束时,做了这么两件事:
    • 为Toolbar执行了一个循环揭露动画
    • CircularRevealActivity 中的视图(Views)执行了一个放大动画,使用的是以前的 ViewPropertyAnimator

监听共享元素进入动画的结束

Transition transition = TransitionInflater.from(this).inflateTransition(R.transition.changebounds_with_arcmotion);
getWindow().setSharedElementEnterTransition(transition);
transition.addListener(new Transition.TransitionListener() {
@Override
public void onTransitionEnd(Transition transition) {
animateRevealShow(mToolbar);
animateButtonsIn();
}
...
});

animateRevealShow(mToolbar)

private void animateRevealShow(View viewRoot) {
int centerX = (viewRoot.getLeft() + viewRoot.getRight()) / 2;
int centerY = (viewRoot.getTop() + viewRoot.getBottom()) / 2;
int endRadius = Math.max(viewRoot.getWidth(), viewRoot.getHeight());

Animator animator = ViewAnimationUtils.createCircularReveal(viewRoot, centerX, centerY, 0, endRadius);
viewRoot.setVisibility(View.VISIBLE);
animator.setDuration(1000);
animator.setInterpolator(new AccelerateInterpolator());
animator.start();
}

上述方法的重点是 createCircularReveal (View view, int centerX, int centerY, float startRadius, float endRadius)

view :要执行循环揭露动画的View

centerX :循环揭露动画中心位置的X坐标

centerY :循环揭露动画中心位置的Y坐标

startRadius :循环揭露动画的起始半径

endRadius :循环揭露动画的结束半径

animateButtonsIn()

private void animateButtonsIn() {
for (int i = 0; i < bgViewGroup.getChildCount(); i++) {
View child = bgViewGroup.getChildAt(i);
child.animate()
.setStartDelay(100 + i*DELAY)
.setInterpolator(interpolator)
.alpha(1)
.scaleX(1)
.scaleY(1);
}
}

上述方法为底部的4个圆执行了一个放大动画,使用 ViewPropertyAnimator 类。

四、更多

还有一些不同的方式来创建循环揭露动画,关键是使用动画效果让用户更好地理解这个app有些什么东西,将会产生怎样的效果。

1. 从目标视图的中心创建循环揭露动画

Material Animations 4:Circular Reveal Animations

public void revealGreenAtMiddle(View view) {
int centerX = (bgViewGroup.getLeft() + bgViewGroup.getRight()) / 2;
int centerY = (bgViewGroup.getTop() + bgViewGroup.getBottom()) / 2;
int endRadius = (int) Math.hypot(bgViewGroup.getWidth()/2, bgViewGroup.getHeight()/2);

Animator animator = ViewAnimationUtils.createCircularReveal(bgViewGroup, centerX, centerY, 0, endRadius);
bgViewGroup.setBackgroundResource(R.color.green);
animator.setDuration(getResources().getInteger(android.R.integer.config_longAnimTime));
animator.setInterpolator(new AccelerateDecelerateInterpolator());
animator.start();
}

2. 从目标视图的顶部创建循环揭露动画+底部按钮动画

Material Animations 4:Circular Reveal Animations

public void revealBlueAtTop(View view) {
animateButtonsOut();

int centerX = (bgViewGroup.getLeft() + bgViewGroup.getRight()) / 2;
int centerY = 0;
int endRadius = (int) Math.hypot(bgViewGroup.getWidth()/2, bgViewGroup.getHeight());

Animator animator = ViewAnimationUtils.createCircularReveal(bgViewGroup, centerX, centerY, 0, endRadius);
bgViewGroup.setBackgroundResource(R.color.blue);
animator.addListener(new Animator.AnimatorListener() {
@Override
public void onAnimationEnd(Animator animation) {
animateButtonsIn();
}
...
});
animator.setDuration(getResources().getInteger(android.R.integer.config_longAnimTime));
animator.setInterpolator(new AccelerateDecelerateInterpolator());
animator.start();
}

此处动画效果经历了以下3个步骤:

  • 隐藏底部按钮(通过控制按钮的透明度、缩放比例)
  • 从顶部执行循环揭露动画
  • 监听器监听到揭露动画执行完后,显示底部按钮(还是通过控制按钮的透明度、缩放比例)

3. 在点击位置创建循环揭露动画

Material Animations 4:Circular Reveal Animations

首先,给橘色圆添加触摸监听事件,获取点击到的橘色圆的位置坐标:

findViewById(R.id.iv_square_orange).setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
if (v.getId() == R.id.iv_square_orange) {
revealOrangeAtPoint(event.getRawX(), event.getRawY());
}
return false;
}
});

接着,就跟前面一样了,根据获取到的坐标位置创建循环揭露动画:

private void revealOrangeAtPoint(float rawX, float rawY) {
int centerX = (int) rawX;
int centerY = (int) rawY;
int endRadius = (int) Math.hypot(bgViewGroup.getWidth(), bgViewGroup.getHeight());

Animator animator = ViewAnimationUtils.createCircularReveal(bgViewGroup, centerX, centerY, 0, endRadius);
bgViewGroup.setBackgroundResource(R.color.orange);
animator.setDuration(getResources().getInteger(android.R.integer.config_longAnimTime));
animator.setInterpolator(new AccelerateDecelerateInterpolator());
animator.start();
}

4. 属性变化动画+循环揭露动画

Material Animations 4:Circular Reveal Animations

这个会难那么一丢丢,毕竟是两个动画效果的组合技,但是只要抓住上一篇讲的 属性变化 动画和上面讲的 循环揭露 动画这两个点,就不难理解了。

private void revealRedAtCenter() {
final ViewGroup.LayoutParams originalParams = ivSquareRed.getLayoutParams();

Transition transition = TransitionInflater.from(this).inflateTransition(R.transition.changebounds_with_arcmotion);
transition.addListener(new Transition.TransitionListener() {
@Override
public void onTransitionEnd(Transition transition) {
int centerX = (bgViewGroup.getLeft() + bgViewGroup.getRight()) / 2;
int centerY = (bgViewGroup.getTop() + bgViewGroup.getBottom()) / 2;
int endRadius = (int) Math.hypot(bgViewGroup.getWidth(), bgViewGroup.getHeight());

Animator animator = ViewAnimationUtils.createCircularReveal(bgViewGroup, centerX, centerY, 0, endRadius);
bgViewGroup.setBackgroundResource(R.color.red);
animator.setDuration(getResources().getInteger(android.R.integer.config_longAnimTime));
animator.setInterpolator(new AccelerateDecelerateInterpolator());
animator.start();

ivSquareRed.setLayoutParams(originalParams);
}
...
});

TransitionManager.beginDelayedTransition(bgViewGroup, transition);
RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
params.addRule(RelativeLayout.CENTER_IN_PARENT);
ivSquareRed.setLayoutParams(params);
}

五、总结

本篇的重点就1个内容:

createCircularReveal (View view, int centerX, int centerY, float startRadius, float endRadius)

只要抓住这两条主线,其它的内容都可以按主线来抽丝拨茧,一切难题都可以迎刃而解。

项目代码已分享到Github: https://github.com/SherlockShi/AndroidMaterialAnimationPractise

 
标签: 安卓开发
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
  • 安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件 作者: 飞龙 这个教程中我们要分析一个锁机软件。像这种软件都比较简单,完全可以顺着入口看下去,但我这里还是用关键点来定位。首先这个软件的截图是这样,进入这个界面之后,除非退出模拟器,否则没办法回到桌面。上面那
    02-05 安卓开发
  • Android插件化(二):OpenAtlas插件安装过程分析
    Android插件化(二):OpenAtlas插件安装过程分析
    在前一篇博客 Android插件化(一):OpenAtlas架构以及实现原理概要 中,我们对应Android插件化存在的问题,实现原理,以及目前的实现方案进行了简单的叙述。从这篇开始,我们要深入到OpenAtlas的源码中进行插件安装过程的分析。 插件的安装分为3种:宿主启动时立
    02-05 安卓开发
  • [译] Android API 指南
    [译] Android API 指南
    众所周知,Android开发者有中文网站了,API 指南一眼看去最左侧的菜单都是中文,然而点进去内容还是很多是英文,并没有全部翻译,我这里整理了API 指南的目录,便于查看,如果之前还没有通读,现在可以好好看一遍。注意,如果标题带有英文,说明官方还没有翻
  • 使用FileProvider解决file:// URI引起的FileUriExposedException
    使用FileProvider解决file:// URI引起的FileUri
    问题以下是一段简单的代码,它调用系统的相机app来拍摄照片:void takePhoto(String cameraPhotoPath) {File cameraPhoto = new File(cameraPhotoPath);Intent takePhotoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);takePhotoIntent.putExtra(Medi
    02-05 安卓开发
点击排行