視圖動畫
實現(xiàn)原理:每次繪制視圖時View所在的ViewGroup中的drawChild函數(shù)獲取View的Animation的Transformation值控嗜,然后調(diào)用canvas.concat(transformToApply.getMatrix()),通過矩陣運算完成動畫幀。如果動畫沒有完成,就繼續(xù)調(diào)用invalidate()函數(shù),啟動下次繪制來驅(qū)動動畫,從而完成整個動畫的繪制评雌。
透明動畫
AlphaAnimation aa = new AlphaAnimation(0, 1); //透明變化范圍
aa.setDuration(1000); //持續(xù)時間
view.startAnimation(aa);
旋轉(zhuǎn)動畫
//構(gòu)造方法參數(shù)為旋轉(zhuǎn)起始角度和旋轉(zhuǎn)中心點坐標
//RotateAnimation ra = new RotateAnimation(0, 360, 100, 100);
//旋轉(zhuǎn)中心可直接設(shè)為定值,也可通過設(shè)置參考系選定直焙,這里設(shè)置為自身中心
RotateAnimation ra = new RotateAnimation(0, 360,
RotateAnimation.RELATIVE_TO_SELF, 0.5F,
RotateAnimation.RELATIVE_TO_SELF, 0.5F);
ra.setDuration(1000); //持續(xù)時間
view.startAnimation(ra);
位移動畫
TranslateAnimation ta = new TranslateAnimation(0, 200, 0, 300); //起始坐標和結(jié)束坐標
ta.setDuration(1000); //持續(xù)時間
view.startAnimation(ta);
縮放動畫
//構(gòu)造方法參數(shù)為x軸在動畫開始和結(jié)束時柳骄、Y軸在動畫開始和結(jié)束時的伸縮比例,1為正常大小
ScaleAnimation sa = new ScaleAnimation(0, 2, 0, 2);
/**
*另有8參數(shù)構(gòu)造方法箕般,可設(shè)置縮放的參考位置
*ScaleAnimation sa = new ScaleAnimation(0, 2, 0, 2,
* Animation.RELATIVE_TO_SELF, 0.5F,
* Animation.RELATIVE_TO_SELF, 0.5F);;
*/
sa.setDuration(1000); //持續(xù)時間
view.startAnimation(sa);
動畫集合
AnimationSet as = new AnimationSet(true);
as.setDuration(1000);
//
AlphaAnimation aa = new AlphaAnimation(0, 1);
aa.setDuration(1000);
as.addAnimation(aa);
TranslateAnimation ta = new TranslateAnimation(0, 200, 0, 300);
ta.setDuration(1000);
as.addAnimation(ta);
//
view.startAnimation(as);
屬性動畫
ObjectAnimator
/**
*通過靜態(tài)工廠類直接返回一個ObjectAnimator對象耐薯,參數(shù)包括:
*view對象、對象的屬性名丝里、屬性變化值
*該屬性必須有g(shù)et和set方法
*/
ObjectAnimator animator = ObjectAnimator.ofFloat(
view,
"translationX",
300);
animator.setDuration(300);
animator.start();
一些常用的可以直接使用屬性動畫的屬性值:
- translationX和translationY:作為增量曲初,控制View對象相對它所在布局容器的左上角坐標偏移的位置。
- rotation杯聚、rotationX和rotationY:控制View對象圍繞支點進行2D和3D旋轉(zhuǎn)臼婆。
- scaleX和scaleY:這兩個屬性控制著View對象圍繞它的支點進行2D縮放。
- pivotX和pivotY:這兩個屬性控制著View的支點位置幌绍,默認為View的中心點颁褂。
- x和y:描述View對象在它所在容器的最終位置。
- alpha:表示View對象的透明度傀广,默認為1(不透明)颁独,0表示完全透明。
對于沒有g(shù)et伪冰、set方法的屬性誓酒,可以自定義屬性類,來間接給屬性增加get贮聂、set方法靠柑;或者通過ValueAnimator來實現(xiàn)。
封裝屬性類方法示例:
private static class WrapperView {
private View mTarget;
public WrapperView(View target) {
mTarget = target;
}
public int getWidth() {
return mTarget.getLayoutParams().width;
}
public void setWidth(int width) {
mTarget.getLayoutParams().width = width;
mTarget.requestLayout();
}
}
ViewWrapper wrapper = new ViewWrapper(mButton);
ObjectAnimator.ofInt(wrapper, "width", 500).setDuration(5000).start();
PropertyValuesHolder
類似視圖動畫中的AnimationSet吓懈,在屬性動畫中歼冰,如果針對一個對象的多個屬性,要同時作用多種動畫耻警,可以使用PropertyValuesHolder來實現(xiàn)隔嫡。
PropertyValuesHolder pvh1 = PropertyValuesHolder.ofFloat("translationX", 300f);
PropertyValuesHolder pvh2 = PropertyValuesHolder.ofFloat("scaleX", 1f, 0, 1f);
PropertyValuesHolder pvh3 = PropertyValuesHolder.ofFloat("scaleY", 1f, 0, 1f);
ObjectAnimator.ofPropertyValuesHolder(view, pvh1, pvh2, pvh3).setDuration(1000).start();
AnimatorSet
對同一對象作用多個屬性動畫效果甸怕,除了PropertyValuesHolder,還可以通過AnimatorSet實現(xiàn)畔勤,同時也能實現(xiàn)更為精確的順序控制。顯示同上PropertyValuesHolder同樣的效果扒磁,用AnimatorSet實現(xiàn)代碼如下:
ObjectAnimator animator1 = ObjectAnimator.ofFloat(view, "translationX", 300f);
ObjectAnimator animator2 = ObjectAnimator.ofFloat(view, "scaleX", 1f, 0, 1f);
ObjectAnimator animator3 = ObjectAnimator.ofFloat(view, "scaleY", 1f, 0, 1f);
AnimatorSet set = new AnimatorSet();
set.setDuration(1000);
set.playTogther(animator1, animator2, animator3);
set.start();
屬性動畫中庆揪,AnimatorSet正是通過playTogether()、playSequentially()妨托、play()缸榛、with()、before()兰伤、after()這些方法控制多個動畫協(xié)同工作方式内颗。
ValueAnimator
ValueAnimator本身不提供任何動畫效果,而是產(chǎn)生有一定規(guī)律的數(shù)字敦腔,讓調(diào)用者來控制動畫的實現(xiàn)過程均澳,使用方法如下:
ValueAnimator animator = ValueAnimator.ofFloat(0, 100);
animator.setTarget(view);
animator.setDuration(1000).start();
animator.addUpdateListener(new AnimatorUpdateListener() {
@Override
public void onAnimationUpdate(ValueAnimator animation) {
Float value = (Float)animation.getAnimatedValue();
//TODO use the value
}
});
動畫事件監(jiān)聽
一個完整動畫具有Start、Repeat符衔、End找前、Cancel四個過程,通過Android提供了接口判族,可以很方便的監(jiān)聽四個事件:
ObjectAnimator anim = ObjectAnimator.ofFloat(view, "alpha", 0.5f);
anim.addListener(new AnimatorListener() {
@Override
public void onAnimationStart(Animator animation){
}
@Override
public void onAnimationRepeat(Animator animation){
}
@Override
public void onAnimationEnd(Animator animation){
}
@Override
public void onAnimationCancel(Animator animation){
}
});
//也可以只監(jiān)聽某一項事件
anim.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
}
});
xml中使用屬性動畫
屬性動畫同視圖動畫一樣躺盛,也可以直接寫在XML中,如下:
<?xml version="1.0" encoding="utf-8"?>
<objectAnimator xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="1000"
android:propertyName="scaleX"
android:valueFrom="1.0"
android:valueTo="2.0"
android:valueType="floatType">
</objectAnimator>
在程序中使用:
public void scaleX(View view) {
Animator anim = AnimatorInflater.loadAnimator(this, R.animator.scalex);
anim.setTarget(mMv);
anim.start();
}
Android布局動畫
所謂布局動畫是指用在ViewGroup上形帮,給ViewGroup增加View時添加一個動畫過渡效果槽惫。
最簡單的布局動畫是在ViewGroup的XML中,使用以下代碼打開布局動畫:
android:animateLayoutChanges="true"
此方法只是使用android默認過渡動畫辩撑,且無法替換這個效果界斜。
一般使用LayoutAnimationController類來自定義一個子view的過渡效果,以下是給子view添加縮放的動畫效果:
LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
// 設(shè)置過渡動畫
ScaleAnimation sa = new ScaleAnimation(0, 1, 0, 1);
sa.setDuration(2000);
//設(shè)置布局動畫的顯示屬性合冀,第二個參數(shù)為delay時間
LayoutAnimationController lac = new LayoutAnimationController(sa, 0.5f);
/**
*delay不為0時可以設(shè)置子view的顯示順序锄蹂,
*LayoutAnimationController.ORDER_NORMAL——順序
*LayoutAnimationController.ORDER_RANDOM——隨機
*LayoutAnimationController.ORDER_REVERSE——反序
*/
lac.setOrder(LayoutAnimationController.ORDER_NORMAL);
//為ViewGroup設(shè)置布局動畫
ll.setLayoutAnimation(lac);