1.5.39 Android Animation学习(二) ApiDemos解析:基本Animatiors使用

本文主要是介绍1.5.39 Android Animation学习(二) ApiDemos解析:基本Animatiors使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Animator类提供了创建动画的基本结构,但是一般使用的是它的子类:

 ValueAnimator、ObjectAnimator、AnimatorSet

  ApiDemos中Animation部分是单独的一个包。

下面代码来自ApiDemos中的AnimationCloning类,加了一个使用ValueAnimator的动画,还有一些注释。

完整的项目见:URL:https://github.com/mengdd/AnimationApiDemos.git

 

package com.example.helloanimation.demo1;import java.util.ArrayList;import com.example.helloanimation.R;
import com.example.helloanimation.demo.ShapeHolder;import android.animation.AnimatorSet;
import android.animation.ObjectAnimator;
import android.animation.ValueAnimator;
import android.animation.ValueAnimator.AnimatorUpdateListener;
import android.app.Activity;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.RadialGradient;
import android.graphics.Shader;
import android.graphics.drawable.ShapeDrawable;
import android.graphics.drawable.shapes.OvalShape;
import android.os.Bundle;
import android.view.View;
import android.view.animation.AccelerateInterpolator;
import android.view.animation.DecelerateInterpolator;
import android.widget.Button;
import android.widget.LinearLayout;public class BasicAnimationActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);// 设置布局,布局xml中只包含了一个线性布局和一个ButtonsetContentView(R.layout.animation_basic);LinearLayout container = (LinearLayout) findViewById(R.id.container);// 将自定义的View加入到线性布局中final MyAnimationView animView = new MyAnimationView(this);container.addView(animView);// Button的点击事件即动画开始Button starter = (Button) findViewById(R.id.startButton);starter.setOnClickListener(new View.OnClickListener() {public void onClick(View v) {animView.startAnimation();}});}/*** 自定义的View类* 其中包含了一系列的球形对象* */public class MyAnimationView extends View implementsValueAnimator.AnimatorUpdateListener {// 圆形球public final ArrayList<ShapeHolder> balls = new ArrayList<ShapeHolder>();// 总的动画集合AnimatorSet animation = null;// 屏幕密度private float mDensity;public MyAnimationView(Context context) {super(context);// 得到密度值mDensity = getContext().getResources().getDisplayMetrics().density;addBall(50f, 25f);addBall(150f, 25f);addBall(250f, 25f);addBall(350f, 25f);addBall(450f, 25f);}private void createAnimation() {if (animation == null) {// ===============================================// 第1个球球的动画效果:用ObjectAnimator// 用工厂方法构造对象:用ofFloat()因为属性值是float类型// 第1个参数为目标对象:balls.get(0)// 第2个参数为属性名:y 这里要求目标对象要有“set属性名()”的方法。// 后面是可变参数,表明属性目标值,一个参数表明是终止值(对象要有get属性方法)// 可变参数的个数为2时,表明第一个是起始值,第二个是终止值;更多个参数时,动画属性值会逐个经过这些值ObjectAnimator anim1 = ObjectAnimator.ofFloat(balls.get(0),"y", 0f, getHeight() - balls.get(0).getHeight()).setDuration(500);// ===============================================// 第二个球球的动画效果:clone动画效果1,但是重新设置目标物体ObjectAnimator anim2 = anim1.clone();anim2.setTarget(balls.get(1));anim1.addUpdateListener(this);// 因为前两个动画完全相同,所以设置刷新监听的时候就只设置了一个(它们刷新的是同一个View)// ===============================================// 第三个球球的动画效果:先加速下落,再减速上升ShapeHolder ball2 = balls.get(2);// 动画效果:落下效果ObjectAnimator animDown = ObjectAnimator.ofFloat(ball2, "y",0f, getHeight() - ball2.getHeight()).setDuration(500);// 落下效果改变了Interpolator,设置为加速animDown.setInterpolator(new AccelerateInterpolator());// 动画效果:上升效果ObjectAnimator animUp = ObjectAnimator.ofFloat(ball2, "y",getHeight() - ball2.getHeight(), 0f).setDuration(500);// 上升效果设置为减速上升animUp.setInterpolator(new DecelerateInterpolator());// 用一个AnimatorSet对象将下落效果和上升效果顺序播放AnimatorSet s1 = new AnimatorSet();s1.playSequentially(animDown, animUp);// 顺序播放效果,参数个数可变// 下落动画刷新ViewanimDown.addUpdateListener(this);// 上升动画刷新ViewanimUp.addUpdateListener(this);// ===============================================// 第四个球球的动画效果// 另一个AnimatorSet克隆了上一个set,更换了对象AnimatorSet s2 = (AnimatorSet) s1.clone();s2.setTarget(balls.get(3));// ===============================================// 第五个球球的动画效果:使用ValueAnimatorfinal ShapeHolder ball5 = balls.get(4);ValueAnimator valueAnimator5 = ValueAnimator.ofFloat(0f,getHeight() - ball5.getHeight());valueAnimator5.setDuration(500);valueAnimator5.addUpdateListener(new AnimatorUpdateListener() {// ValueAnimator需要自己在监听处理中设置对象参数@Overridepublic void onAnimationUpdate(ValueAnimator animation) {// 用animation.getAnimatedValue()得到当前的属性值,设置进动画对象中ball5.setY((Float) animation.getAnimatedValue());// 记得要刷新View否则不会调用重新绘制invalidate();}});// =============================================================// 用一个总的AnimatorSet对象管理以上所有动画animation = new AnimatorSet();animation.playTogether(anim1, anim2, s1);// 并行animation.playSequentially(s1, s2, valueAnimator5);// 串行}}// 在指定位置加上球形private ShapeHolder addBall(float x, float y) {OvalShape circle = new OvalShape();circle.resize(50f * mDensity, 50f * mDensity);ShapeDrawable drawable = new ShapeDrawable(circle);ShapeHolder shapeHolder = new ShapeHolder(drawable);shapeHolder.setX(x - 25f);shapeHolder.setY(y - 25f);int red = (int) (100 + Math.random() * 155);int green = (int) (100 + Math.random() * 155);int blue = (int) (100 + Math.random() * 155);int color = 0xff000000 | red << 16 | green << 8 | blue;Paint paint = drawable.getPaint(); // new// Paint(Paint.ANTI_ALIAS_FLAG);int darkColor = 0xff000000 | red / 4 << 16 | green / 4 << 8 | blue/ 4;RadialGradient gradient = new RadialGradient(37.5f, 12.5f, 50f,color, darkColor, Shader.TileMode.CLAMP);paint.setShader(gradient);shapeHolder.setPaint(paint);balls.add(shapeHolder);return shapeHolder;}@Overrideprotected void onDraw(Canvas canvas) {// 遍历并绘制每一个球形对象for (int i = 0; i < balls.size(); ++i) {ShapeHolder shapeHolder = balls.get(i);canvas.save();canvas.translate(shapeHolder.getX(), shapeHolder.getY());shapeHolder.getShape().draw(canvas);canvas.restore();}}public void startAnimation() {createAnimation();animation.start();// 这里开始播放动画}@Overridepublic void onAnimationUpdate(ValueAnimator animation) {// 在参数更新的时候invalidate,刷新整个View的绘制// 否则onDraw不会被调用,即看不到View外观的改变invalidate();}}
}

 

其中ShapeHolder:

 

package com.example.helloanimation.demo;import android.graphics.Paint;
import android.graphics.RadialGradient;
import android.graphics.drawable.ShapeDrawable;
import android.graphics.drawable.shapes.Shape;
import android.view.View;/*** * A data structure that holds a Shape and various properties that can be used to define* how the shape is drawn.* (从API Demos中直接搬过来的类)* */
public class ShapeHolder {private float x = 0, y = 0;private ShapeDrawable shape;private int color;private RadialGradient gradient;private float alpha = 1f;private Paint paint;public void setPaint(Paint value) {paint = value;}public Paint getPaint() {return paint;}public void setX(float value) {x = value;}public float getX() {return x;}public void setY(float value) {y = value;}public float getY() {return y;}public void setShape(ShapeDrawable value) {shape = value;}public ShapeDrawable getShape() {return shape;}public int getColor() {return color;}public void setColor(int value) {shape.getPaint().setColor(value);color = value;}public void setGradient(RadialGradient value) {gradient = value;}public RadialGradient getGradient() {return gradient;}public void setAlpha(float alpha) {this.alpha = alpha;shape.setAlpha((int)((alpha * 255f) + .5f));}public float getWidth() {return shape.getShape().getWidth();}public void setWidth(float width) {Shape s = shape.getShape();s.resize(width, s.getHeight());}public float getHeight() {return shape.getShape().getHeight();}public void setHeight(float height) {Shape s = shape.getShape();s.resize(s.getWidth(), height);}public ShapeHolder(ShapeDrawable s) {shape = s;}
}

 

效果如图:

          

一、ValueAnimator

  ValueAnimator使用时可以需要自己设置监听,将变动的值设置给目标对象:

  ValueAnimator构造使用工厂方法。

  上面例子中相应的代码片段:

 

                // ===============================================// 第五个球球的动画效果:使用ValueAnimatorfinal ShapeHolder ball5 = balls.get(4);ValueAnimator valueAnimator5 = ValueAnimator.ofFloat(0f,getHeight() - ball5.getHeight());valueAnimator5.setDuration(500);valueAnimator5.addUpdateListener(new AnimatorUpdateListener() {// ValueAnimator需要自己在监听处理中设置对象参数@Overridepublic void onAnimationUpdate(ValueAnimator animation) {// 用animation.getAnimatedValue()得到当前的属性值,设置进动画对象中ball5.setY((Float) animation.getAnimatedValue());// 记得要刷新View否则不会调用重新绘制invalidate();}});

 

二、ObjectAnimator

  ObjectAnimatorValueAnimator的子类,构造时也用工厂方法。

  ObjectAnimator不用自己设置监听来设置对象的值,要动画的对象和要改变的属性都在构造的时候设置好。

   比如前两个球球的动画:

 

                // ===============================================// 第1个球球的动画效果:用ObjectAnimator// 用工厂方法构造对象:用ofFloat()因为属性值是float类型// 第1个参数为目标对象:balls.get(0)// 第2个参数为属性名:y 这里要求目标对象要有“set属性名()”的方法。// 后面是可变参数,表明属性目标值,一个参数表明是终止值(对象要有get属性方法)// 可变参数的个数为2时,表明第一个是起始值,第二个是终止值;更多个参数时,动画属性值会逐个经过这些值ObjectAnimator anim1 = ObjectAnimator.ofFloat(balls.get(0),"y", 0f, getHeight() - balls.get(0).getHeight()).setDuration(500);// ===============================================// 第二个球球的动画效果:clone动画效果1,但是重新设置目标物体ObjectAnimator anim2 = anim1.clone();anim2.setTarget(balls.get(1));anim1.addUpdateListener(this);// 因为前两个动画完全相同,所以设置刷新监听的时候就只设置了一个(它们刷新的是同一个View)

 

三、AnimatorSet

  AnimatorSet用来组织动画,动画可以同时播放,顺序播放,也可以设定一定的延迟之后播放。

  playTogether()表示动画同时播放。

  playSequentially() 表示动画顺序播放。

  比如第三个球球先加速下降再减速上升的动画:

 

                // ===============================================// 第三个球球的动画效果:先加速下落,再减速上升ShapeHolder ball2 = balls.get(2);// 动画效果:落下效果ObjectAnimator animDown = ObjectAnimator.ofFloat(ball2, "y",0f, getHeight() - ball2.getHeight()).setDuration(500);// 落下效果改变了Interpolator,设置为加速animDown.setInterpolator(new AccelerateInterpolator());// 动画效果:上升效果ObjectAnimator animUp = ObjectAnimator.ofFloat(ball2, "y",getHeight() - ball2.getHeight(), 0f).setDuration(500);// 上升效果设置为减速上升animUp.setInterpolator(new DecelerateInterpolator());// 用一个AnimatorSet对象将下落效果和上升效果顺序播放AnimatorSet s1 = new AnimatorSet();s1.playSequentially(animDown, animUp);// 顺序播放效果,参数个数可变// 下落动画刷新ViewanimDown.addUpdateListener(this);// 上升动画刷新ViewanimUp.addUpdateListener(this);

 

  因为参数是Animator类型的对象集合或者可变参数,所以表示AnimationSet是可嵌套使用的,因为AnimationSet是Animator的子类。

 

<span style="white-space:pre">		</span>// =============================================================// 用一个总的AnimatorSet对象管理以上所有动画animation = new AnimatorSet();animation.playTogether(anim1, anim2, s1);// 并行animation.playSequentially(s1, s2, valueAnimator5);// 串行

 

  Demo中就是将所有的动画都放在一个AnimationSet对象中,最后调用start()方法播放。

 

以上转载:http://www.cnblogs.com/mengdd/archive/2013/09/05/3303191.html

这篇关于1.5.39 Android Animation学习(二) ApiDemos解析:基本Animatiors使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1094207

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用