属性动画是API 11加进来的一个新特性,其实在现在来说也没什么新的了。属性动画可以对任意view的属性做动画,实现动画的原理就是在给定的时间内把属性从一个值变为另一个值。因此可以说属性动画什么都可以干,只要view有这个属性。

所以我们这里对Button来做一个简单的属性动画:改变这个Button的宽度。也可以用Tween Animation,但是明显有一点不能满足要求的地方是Tween Animation只能做Scale动画,也就是缩放。你可以对这个button做缩放来达到增加宽度的效果,但是这个时候按钮的文字也会跟着出现缩放和变形。同时很重要的一点,Tween Animation不改变view的本来位置和大小。看起来这个按钮变大了,但是点击动画执行前的按钮没有覆盖的位置是没有效果的。

我们简略的看一下这个Tween动画是怎么样的。

首先,用xml的文件定义一个Scale动画,对宽度扩大为原来的两倍,高度扩大为原来的六倍。在动画之后填充。动画执行完成后就可以清楚的看到按钮文字跟着按钮动画执行完成后之后的效果。

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="500"
android:fillAfter="true"
android:interpolator="@android:anim/accelerate_decelerate_interpolator">
<scale
android:fromXScale="100%"
android:fromYScale="100%"
android:pivotX="50%"
android:pivotY="50%"
android:toXScale="200%"
android:toYScale="600%" />
</set>

执行这个Tween动画:

var button = findViewById(R.id.tween_button) as Button
button.setOnClickListener { v -> var anim = AnimationUtils.loadAnimation(this@TweenAnimActvity, R.anim.scale_anim)
v.startAnimation(anim)
}

这里必须说明,上面这段代码是Kotlin语言写的。自从用了之后就再不想用java了。只要你有一定的java基础,阅读这段代码并没有什么难度。

点击原来按钮区域以外的地方,按钮是不会有任何的反应的。

看看效果:

试试Property动画吧

直接看看按照同样的缩放大小生成的效果吧:

前后两者相差还是很明显的。越明显越突出了属性动画存在的必要。这种必要不止是效果上看到的,
还有交互和开发的时候代码相关的。

所以无论如何都要使用属性动画了。这里使用最简答的方法: ObjectAnimator来做这个动画:

ObjectAnimator.ofInt(mAnimateButton, "width", mAnimateButton.getWidth(), 1000)
.setDuration(1000)
.start();

看起来很简单就实现了按钮的动画。但是运行的时候就会出现问题。因为,属性动画在执行的时候需要改变指定的属性,这里是width,的值。使用的就是属性对应的getWidthsetWidth方法。getWidth在没有给定动画的初值时,使用这个方法获得初始值。setWidth则在给定的时间内不断地被用来修改属性值来达到动画的效果。注意,这个方法不是只是用一次

但是来看看ButtongetWidthsetWidth两个方法的代码:

    /**
* Return the width of the your view.
*
* @return The width of your view, in pixels.
*/
@ViewDebug.ExportedProperty(category = "layout")
public final int getWidth() {
return mRight - mLeft;
}
    /**
* Makes the TextView exactly this many pixels wide.
* You could do the same thing by specifying this number in the
* LayoutParams.
*
* @see #setMaxWidth(int)
* @see #setMinWidth(int)
* @see #getMinWidth()
* @see #getMaxWidth()
*
* @attr ref android.R.styleable#TextView_width
*/
@android.view.RemotableViewMethod
public void setWidth(int pixels) {
mMaxWidth = mMinWidth = pixels;
mMaxWidthMode = mMinWidthMode = PIXELS; requestLayout();
invalidate();
}

显然在setWidth的时候,并没有用给定的值去修改按钮layout param的宽度。

在这种情况下Google给了三种解决方法:

  1. 给你的view加上get和set方法。但是这需要你有这个权限。
  2. 用一个类来包装目标view,间接的给这个view来添加get和set方法。
  3. ValueAnimatorAnimatorUpdateListener监听动画,自己修改每个时间片的属性修改。

Button添加get和set方法不是很现实,所以只能选择后两者。

下面一一介绍后面两个方法。

间接给出get、set方法

这个方法看起来很简单,定义一个类间接给出get、set方法就是这样的:

class ViewWrapper {
View mTargetView; public ViewWrapper(View v) {
mTargetView = v;
} public void setWidth(int width) {
mTargetView.getLayoutParams().width = width;
mTargetView.requestLayout();
} // for view's width
public int getWidth() {
int width = mTargetView.getLayoutParams().width;
return width;
} // for view's height
public void setHeight(int height) {
mTargetView.getLayoutParams().height = height;
mTargetView.requestLayout();
} public int getHeight() {
int height = mTargetView.getLayoutParams().height;
return height;
}
}
  1. 既然动画是需要修改layout params的宽度,那么我们在这个set方法里就修改layout params的宽度。
  2. 返回layout params的宽度。这个值是view在动画之前的宽度。

然后在按钮点击之后开始这个修改宽度的动画:

 @Override
public void onClick(View v) {
Log.d("##ViewWrapperActivity", "width is " + v.getWidth()); // 1
ViewWrapper viewWrapper = new ViewWrapper(v);
// 2
ObjectAnimator animator = ObjectAnimator.ofInt(viewWrapper, "width", /*viewWrapper.getWidth(),*/ 1500);
// 3
animator.addListener(new Animator.AnimatorListener() {
@Override
public void onAnimationStart(Animator animation) {
Log.d("##ANIM", "started");
} @Override
public void onAnimationEnd(Animator animation) {
Log.d("##ANIM", "stopped");
} @Override
public void onAnimationCancel(Animator animation) { } @Override
public void onAnimationRepeat(Animator animation) { }
});
// 4
animator.setDuration(3000).start();
}
  1. 用包装类包装view,这里是按钮。
  2. 开始动画,动画的对象现在为包装类对象。这里可以修改属性动画的定义了,属性动画可以对任何对象修改属性。这里的包装类对象明显不是一个view
  3. 这里增加了一个监听器,监听动画是刚开始还是已经结束。
  4. 开始动画。在三秒钟的时间内修改按钮的宽度,从初始值修改为1500像素宽。

看起来已经很完美了,运行这个段代码。点击按钮后。好吧,这个动画很奇怪,并没有运行“完全”。点一下动一点,但是没有达到宽度为1500像素。虽然动画监听器AnimatorListener的方法onAnimationEnd已经执行,而且也打出了执行完成的log,但是宽度始终达不到。所以说动画执行并不“完全”。

那么这是为什么呢?先给出正确的代码各位可以参考着考虑一下:

public class ViewWrapperActivity extends Activity implements View.OnClickListener {

    private Button mAnimateButton;
// 1
private ViewWrapper mWrapper; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_view_wrapper); mAnimateButton = (Button) findViewById(R.id.animate_button);
mAnimateButton.setOnClickListener(this);
// 2
mWrapper = new ViewWrapper(mAnimateButton);
} @Override
public void onClick(View v) {
Log.d("##ViewWrapperActivity", "width is " + v.getWidth());
// 3
int width = v.getLayoutParams().width;
int height = v.getHeight(); // current height
// 4
PropertyValuesHolder widthHolder = PropertyValuesHolder.ofInt("width", width * 2);
PropertyValuesHolder heightHolder = PropertyValuesHolder.ofInt("height", height * 6);
// 5
ObjectAnimator animator = ObjectAnimator.ofPropertyValuesHolder(mWrapper, widthHolder, heightHolder);
animator.setInterpolator(new LinearInterpolator());
animator.addListener(new Animator.AnimatorListener() {
// ...
});
animator.setDuration(3000).start();
}
}
  1. 声明包装类对象类成员。
  2. onCreate方法里初始化包装类对象。
  3. widthheight获取Button当前的宽度和高度。
  4. 在这定义对宽度做2倍的扩大,对高度做6倍的扩大。两个动画的定义都存放在PropertyValuesHolder中,并在后面的实现中使用。使用这个类存放对不同属性的动画定义,方便使用。这两个动画会同时并行执行。
  5. mWrapper执行前面定义的两个动画。这两个动画同时执行。要使两个动画顺序执行可以AnimatorSet来实现:
    int width = v.getWidth();
int height = v.getHeight(); AnimatorSet animSet = new AnimatorSet(); ObjectAnimator widthAnim = ObjectAnimator.ofInt(mSequenceWrapper, "width", width * 2);
ObjectAnimator heightAnim = ObjectAnimator.ofInt(mSequenceWrapper, "height", height * 6); animSet.play(widthAnim).before(heightAnim);
animSet.setDuration(1000);
animSet.setInterpolator(new AccelerateDecelerateInterpolator());
animSet.start();

这样就可以一次动画达到指定宽度和高度了。具体是为什么呢?欢迎再后面的评论中一起讨论。

从点到面,给Button的属性动画的更多相关文章

  1. 属性动画 LayoutTransition AnimatorInflater Keyframe 新特性

    LayoutTransition设置动画 使用LayoutTransition可为布局的容器设置动画,当容器中的视图层次发生变化时产生相应的过渡的动画效果 过渡的类型一共有四种: LayoutTran ...

  2. Android属性动画

    这几天看郭神的博客 Android属性动画完全解析(上),初识属性动画的基本用法之后,我自己突然想实现一种动画功能,就是我们在携程网.阿里旅行等等手机APP端买火车票的时候,看到有选择城市,那么就有出 ...

  3. android 帧动画,补间动画,属性动画的简单总结

      帧动画——FrameAnimation 将一系列图片有序播放,形成动画的效果.其本质是一个Drawable,是一系列图片的集合,本身可以当做一个图片一样使用 在Drawable文件夹下,创建ani ...

  4. View动画和属性动画

    在应用中, 动画效果提升用户体验, 主要分为View动画和属性动画. View动画变换场景图片效果, 效果包括平移(translate), 缩放(scale), 旋转(rotate), 透明(alph ...

  5. 使用属性动画 — Property Animation

    属性动画,就是通过控制对象中的属性值产生的动画.属性动画是目前最高级的2D动画系统. 在API Level 11中添加.Property Animation号称能控制一切对象的动画,包括可见的和不可见 ...

  6. 【转】android 属性动画之 ObjectAnimator

    原文网址:http://blog.csdn.net/feiduclear_up/article/details/39255083 前面一篇博客讲解了 android 简单动画之 animtion,这里 ...

  7. Android属性动画之ValueAnimation

    ValueAnimation是ObjectAnimation类的父类,经过前几天的介绍,相信大家对ObjectAnimation有了 一定的认识,今天就为大家最后介绍一下ValueAnimation, ...

  8. Android属性动画之ObjectAnimator

    相信对于Android初学者,对于Android中的动画效果一定很感兴趣,今天为大家总结一下刚刚学到的属性动画案例. 首先和一般的Android应用一样,我们先建一个工程,为了方便,我们的布局文件中就 ...

  9. Android Animation学习(四) ApiDemos解析:多属性动画

    Android Animation学习(四) ApiDemos解析:多属性动画 如果想同时改变多个属性,根据前面所学的,比较显而易见的一种思路是构造多个对象Animator , ( Animator可 ...

随机推荐

  1. ROS launch总结

    1 运行Launch文件2 新建Launch文件3  在namespace中启动nodes 4   remapping names 5 其他的launch元素 1 运行Launch文件 Launch文 ...

  2. SQL truncate 、delete与drop区别[z]

    [z]https://www.cnblogs.com/8765h/archive/2011/11/25/2374167.html 相同点: 1.truncate和不带where子句的delete.以及 ...

  3. Windows下adobe Reader中pdf字体helvetica被替换为ArialMT

    笔者最近java项目中用itext-2.1.7导出pdf,使用了Helvetica,这个字体是内置的字体,本地Adobe Reader版本9.0 导出的文字是img.nur.cn实际效果 查看了pdf ...

  4. join 子句(C# 参考)

    参考:https://msdn.microsoft.com/zh-cn/library/vstudio/bb311040%28v=vs.110%29.aspx 使用 join 子句可以将来自不同源序列 ...

  5. 完全卸载jdk

    完全卸载jdk  如(卸载jdk1.7.0_80),  当卸载jdk时出现删除信息不全,导致无法安装,可通过此方法实现完全卸载. 1.打开系统运行输入regedit(注册表)分别查找以下路径  (1) ...

  6. 9.12 h5日记

    9.12 知识点补充: 属性继承例子,color.font(font-size/style/family/weight) 1.浏览器的默认字体大小是16px,谷歌浏览器的最小字体是10px,其他浏览器 ...

  7. js 逻辑运算符

    两个逻辑运算符的操作顺序在自己的脑海里一直理不清,用js做了个实验 <script type="text/javascript">    if(false && ...

  8. redis 哨兵(sentinel)

    redis哨兵 哨兵自动故障转移 自动通知应用最新master信息 无需担心,master挂了,程序不需要修改IP啥的,由哨兵自动完成 修改sentinel.conf protected-mode n ...

  9. hdu 1254(两个BFS) 推箱子

    http://acm.hdu.edu.cn/showproblem.php?pid=1254 首先,要判断人是不是可以从4到达箱子的位置2,而且不止判断一次,因为推动箱子一步后,人的位置也会改变,所以 ...

  10. javascript 高级程序设计 八

    function 类型: 1.ECMAscript中函数和类C语言的函数有这很多不同.其中之一就是js的函数没有重载.并且多次定义一个同名的函数,当调用这个函数的时候, 会调用最后一次定义的函数. 2 ...