本篇文章給大家分享的是有關(guān)使用ObjectAnimator屬性怎么實(shí)現(xiàn)動(dòng)畫(huà),小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說(shuō),跟著小編一起來(lái)看看吧。
成都創(chuàng)新互聯(lián)主營(yíng)新區(qū)網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營(yíng)網(wǎng)站建設(shè)方案,成都APP應(yīng)用開(kāi)發(fā),新區(qū)h5小程序開(kāi)發(fā)搭建,新區(qū)網(wǎng)站營(yíng)銷推廣歡迎新區(qū)等地區(qū)企業(yè)咨詢
創(chuàng)建
首先看一下今天所有用到的背景:
寫(xiě)了一個(gè)自定義的View——PointView,用來(lái)實(shí)現(xiàn)一個(gè)小球的移動(dòng)效果,PointView代碼如下(可以不用看-。+):
public class PointView extends View { private Point mCurrentPoint; private Paint paint; /** * 兩個(gè)構(gòu)造方法 **/ public PointView(Context context) { super(context); paint = new Paint(Paint.ANTI_ALIAS_FLAG); paint.setColor(Color.CYAN); paint.setStrokeWidth(5); paint.setStyle(Paint.Style.STROKE); } public PointView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); paint = new Paint(Paint.ANTI_ALIAS_FLAG); paint.setColor(Color.CYAN); paint.setStrokeWidth(5); paint.setStyle(Paint.Style.STROKE); } /** * onDraw開(kāi)始使用畫(huà)筆,如果mCurrentPoint為空,就創(chuàng)建Point對(duì)象, * 否則就直接調(diào)用drawPoint方法 **/ @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); if (mCurrentPoint == null) { mCurrentPoint = new Point(500, 500); drawPoint(canvas); } else { drawPoint(canvas); } } //設(shè)置一個(gè)屬性添加的方法 public void setMove(Point point) { mCurrentPoint = point; invalidate(); } //啟動(dòng)動(dòng)畫(huà) private void startAnimation() { ObjectAnimator animator = ObjectAnimator.ofObject(PointView.this, "Move", new BallEvaluator(), new Point(500, 500), new Point(500, 800), new Point(700, 800)); animator.setInterpolator(new LinearInterpolator()); animator.setDuration(5000); animator.start(); } //在外部調(diào)用的方法,通過(guò)此方法,開(kāi)啟小球動(dòng)作的動(dòng)畫(huà)。 public void moveBall() { startAnimation(); } private void drawPoint(Canvas canvas) { canvas.drawCircle(mCurrentPoint.getX(), mCurrentPoint.getY(), 30, paint); } //小球?qū)ο蟮墓乐灯? public class BallEvaluator implements TypeEvaluator{ float x; float y; @Override public Point evaluate(float fraction, Point startValue, Point endValue) { float startX = startValue.getX(); float startY = startValue.getY(); float endX = endValue.getX(); float endY = endValue.getY(); x = fraction * (endX - startX); y = fraction * (endY - startY); Log.e("ASDSAD", "x = " + x + "y = " + y); return new Point(startX + x, startY + y); } } }
代碼可能有點(diǎn)多,不過(guò)這不是主要的,我們今天關(guān)注的只是屬性動(dòng)畫(huà),所以我們只需要看里面的startAnimation方法和setMove方法就好:
setMove:由于我們知道屬性動(dòng)畫(huà)ObjectAnimator類是通過(guò)將propertyName拼接成對(duì)應(yīng)的set方法,然后通過(guò)反射機(jī)制去調(diào)用該方法,所以我們需要有一個(gè)對(duì)應(yīng)的set方法。
startAnimation:這個(gè)方法我們用來(lái)設(shè)置我們的動(dòng)畫(huà)以及啟動(dòng)動(dòng)畫(huà)。setMove方法,很簡(jiǎn)單,我們只是將傳入的新的小球?qū)ο筚x值給了mCurrentBall,然后調(diào)用invalidate方法重新繪制。下面看一下startAnimation方法:
ObjectAnimator animator = ObjectAnimator.ofObject(PointView.this, "Move", new BallEvaluator(), new Point(500, 500), new Point(500, 800), new Point(700, 800)); animator.setInterpolator(new LinearInterpolator()); animator.setDuration(5000); animator.start();
我們看一下先進(jìn)入ofObject方法(相信ofObject里面的參數(shù)大家都看得懂):
public static ObjectAnimator ofObject(Object target, String propertyName, TypeEvaluator evaluator, Object... values) { ObjectAnimator anim = new ObjectAnimator(target, propertyName); anim.setObjectValues(values); anim.setEvaluator(evaluator); return anim; }
我們發(fā)現(xiàn)ofObject是一個(gè)靜態(tài)方法:他在里面創(chuàng)建了一個(gè)ObjectAnimator對(duì)象。然后調(diào)用了setObjeectValues和setEvaluator方法,分別添加了數(shù)據(jù)和估值器。
也就是這個(gè)ofObject里面有三個(gè)入口等著我們進(jìn)去:
ObjectAnimator的構(gòu)造方法
setObjectValues方法
setEvaluator方法
那我們先從ObjectAnimator的構(gòu)造方法開(kāi)始看吧。
ObjectAnimator構(gòu)造方法:
private ObjectAnimator(Object target, String propertyName) { setTarget(target); setPropertyName(propertyName); }
又是兩個(gè)方法,一個(gè)一個(gè)去看,先進(jìn)入setTarget:
@Override public void setTarget(@Nullable Object target) { final Object oldTarget = getTarget(); if (oldTarget != target) { if (isStarted()) { cancel(); } mTarget = target == null ? null : new WeakReference
這些代碼都能夠知道什么意思,這個(gè)方法我們只需要注意兩點(diǎn):
我們將傳入的target傳給了mTarget這個(gè)弱引用。
mInitialized = false;這個(gè)屬性我們可以這么理解,在他的ObjectAnimator沒(méi)有準(zhǔn)備就緒(初始化過(guò)程尚未完成時(shí)),他一直都是false。
這個(gè)方法跳過(guò)。
然后我們看setPropertyName方法:
public void setPropertyName(@NonNull String propertyName) { // mValues could be null if this is being constructed piecemeal. Just record the // propertyName to be used later when setValues() is called if so. if (mValues != null) { PropertyValuesHolder valuesHolder = mValues[0]; String oldName = valuesHolder.getPropertyName(); valuesHolder.setPropertyName(propertyName); mValuesMap.remove(oldName); mValuesMap.put(propertyName, valuesHolder); } mPropertyName = propertyName; // New property/values/target should cause re-initialization prior to starting mInitialized = false; }
首先介紹一下mValues和mValuesMap這兩個(gè)屬性,他們都是存儲(chǔ)PropertyValueHolder的屬性,而且儲(chǔ)存的都一樣,只是mValuesMap可以讓我們通過(guò)propertyName來(lái)查找對(duì)應(yīng)的PropertyValueHolder。
PropertyValuesHolder[] mValues; HashMapmValuesMap;
這個(gè)方法只是將propertyName放入PropertyValueHolder中(具體邏輯如上,先判斷mValues是否為空,如果不為空就將propertyName放入mValues和mValuesMap中,最后將propertyName賦值給mPropertyName),可以過(guò)了。
現(xiàn)在我們的ObjectAnimator構(gòu)造方法看完了,我們接著看setObjectValues方法:
anim.setObjectValues:
@Override public void setObjectValues(Object... values) { if (mValues == null || mValues.length == 0) { // No values yet - this animator is being constructed piecemeal. Init the values with // whatever the current propertyName is if (mProperty != null) { setValues(PropertyValuesHolder.ofObject(mProperty, (TypeEvaluator) null, values)); } else { setValues(PropertyValuesHolder.ofObject(mPropertyName, (TypeEvaluator) null, values)); } } else { super.setObjectValues(values); } }
這段代碼的總體邏輯只有一個(gè):如果mValues沒(méi)有值,那么就調(diào)用setValues方法,否則就調(diào)用父類的setObjectValues方法。
感覺(jué)很亂啊,穩(wěn)?。? 我們這是第一次創(chuàng)建的對(duì)象,所以肯定是為空的,所以我們只需要看setValues方法就好了,但是注意,這里還有PropertyValueHolder,所以我們決定先看一下PropertyValueHolder的ofObject方法:
PropertyValueHolder.ofObject:
public staticPropertyValuesHolder ofObject(Property property, TypeEvaluator evaluator, V... values) { PropertyValuesHolder pvh = new PropertyValuesHolder(property); pvh.setObjectValues(values); pvh.setEvaluator(evaluator); return pvh; }
跟上面ObjectAnimator的ofObject差異不多,我們就不多說(shuō)了,有兩條路可以選:
setObjectValues
setEvaluator
先看setObjectValues:
PropertyValueHolder.setObjectValues:
public void setObjectValues(Object... values) { mValueType = values[0].getClass(); mKeyframes = KeyframeSet.ofObject(values); if (mEvaluator != null) { mKeyframes.setEvaluator(mEvaluator); } }
我知道大家都快吐了,現(xiàn)在KeyFrames又出來(lái)了,頭皮發(fā)麻對(duì)吧,穩(wěn)住,我們堅(jiān)持??!
這個(gè)KeyFrames是KeyFrameSet的接口,我們看一下KeyframeSet的ofObject方法:
KeyframeSet.ofObject方法:
public static KeyframeSet ofObject(Object... values) { int numKeyframes = values.length; ObjectKeyframe keyframes[] = new ObjectKeyframe[Math.max(numKeyframes,2)];//創(chuàng)建了一個(gè)至少為兩位的ObjectKeyFrame對(duì)象 if (numKeyframes == 1) { keyframes[0] = (ObjectKeyframe) Keyframe.ofObject(0f); keyframes[1] = (ObjectKeyframe) Keyframe.ofObject(1f, values[0]); } else { keyframes[0] = (ObjectKeyframe) Keyframe.ofObject(0f, values[0]); for (int i = 1; i < numKeyframes; ++i) { keyframes[i] = (ObjectKeyframe) Keyframe.ofObject((float) i / (numKeyframes - 1), values[i]); } } return new KeyframeSet(keyframes); }
我們終于看到了一個(gè)比較熟悉的類,KeyFrame,這個(gè)叫做關(guān)鍵幀的類終于出現(xiàn)了,我們簡(jiǎn)單分析一下這個(gè)方法:
首先創(chuàng)建了一個(gè)至少為兩位的ObjectKeyFrame對(duì)象,然后對(duì)values的長(zhǎng)度進(jìn)行判斷,如果只有一個(gè)值,那么就將唯一的一個(gè)值添加到最后一位(此時(shí)也就是第二位),否則就依次添加。最后將ObjectKeyFrame的數(shù)組轉(zhuǎn)換成KeyFrameSet類型返回。
現(xiàn)在我們回到PropertyValueHolder的setObjectValues方法中,接下來(lái)我們要看一下setEvaluator方法(需要在KeyFrameSet中查看)
KeyFrameSet.setEvaluator方法:
public void setEvaluator(TypeEvaluator evaluator) { mEvaluator = evaluator; }
這個(gè)不用多說(shuō),直接過(guò)。
現(xiàn)在我們的PropertyValueHolder的ofObject方法已經(jīng)看完了,我們跳回anim.setObjectValues方法,看一下setValues方法:
setValues:
public void setValues(PropertyValuesHolder... values) { int numValues = values.length; mValues = values; mValuesMap = new HashMap(numValues); for (int i = 0; i < numValues; ++i) { PropertyValuesHolder valuesHolder = values[i]; mValuesMap.put(valuesHolder.getPropertyName(), valuesHolder); } // New property/values/target should cause re-initialization prior to starting mInitialized = false; }
這個(gè)完全就是我們剛才說(shuō)的儲(chǔ)存PropertyValueHolder的兩個(gè)屬性mValues和mValuesMap添加數(shù)據(jù)的過(guò)程,過(guò)。
現(xiàn)在我們只差最后一個(gè)大方法了,anim.setEvaluator了,激動(dòng)?。?!
anim.setEvaluator方法:
public void setEvaluator(TypeEvaluator value) { if (value != null && mValues != null && mValues.length > 0) { mValues[0].setEvaluator(value); } }
這個(gè)方法是給每個(gè)PropertyValueHolder對(duì)象都執(zhí)行setEvaluator方法,我們點(diǎn)進(jìn)去這個(gè)方法看一下:
PropertyValueHolder.setEvaluator方法:
public void setEvaluator(TypeEvaluator evaluator) { mEvaluator = evaluator; mKeyframes.setEvaluator(evaluator); }
又進(jìn)入了Keyframes的setEvaluator,我們接著看一下KeyFrameSet的setEvaluator方法:
KeyFrameSet.setEvaluator方法:
public void setEvaluator(TypeEvaluator evaluator) { mEvaluator = evaluator; }
這個(gè)方法不用說(shuō)了。
最后我們返回了創(chuàng)建的anim的對(duì)象,到現(xiàn)在為止,我們得到我們想要的ObjectAnimator對(duì)象了。
這個(gè)過(guò)程是有點(diǎn)繁瑣,我們現(xiàn)在屢一下思路:
調(diào)用了ObjectAnimator.ofObject之后
首先new一個(gè)ObjectAnimator對(duì)象,進(jìn)入ObjectAnimator的構(gòu)造方法中:在構(gòu)造方法中,我們執(zhí)行了兩個(gè)方法:setTarget和setPropertyName。
然后調(diào)用了ObjectAnimator的setObjectValues方法:在這個(gè)方法中我們首先實(shí)例化了PropertyValueHolder對(duì)象,然后調(diào)用setValues方法將PropertyValueHolder傳入。
之后調(diào)用了ObjectAnimator的setEvaluator方法:添加了估值器。
最后返回了ObjectAnimator對(duì)象。
到這里我們就完成了ObjectAnimator對(duì)象實(shí)例的創(chuàng)建。
到這里創(chuàng)建部分就全部完成了,接下來(lái)我們看一下添加屬性,這個(gè)就很簡(jiǎn)單了。
添加屬性
我們就舉一個(gè)方法為例吧,拿setInterpolator方法為例:
public void setInterpolator(TimeInterpolator value) { if (value != null) { mInterpolator = value; } else { mInterpolator = new LinearInterpolator(); } }
這個(gè)方法是我們添加插值器的方法,我們注意到他只是給mInterpolator賦值而已,如果傳入為空,則添加線性插值器。
其他的添加屬性的方法,像setDuration、setRepeatCount等都是如此,大家下去就自己看一下吧。
由于還不會(huì)用 starUML,所以現(xiàn)在還沒(méi)發(fā)畫(huà)一張時(shí)序圖(只是手畫(huà)的,估計(jì)大家也不想看哈哈),等學(xué)完UML之后會(huì)給大家補(bǔ)上的,希望這篇文章大家喜歡。
以上就是使用ObjectAnimator屬性怎么實(shí)現(xiàn)動(dòng)畫(huà),小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。