Android自定義加載圈動(dòng)畫效果
本文實(shí)例為大家分享了Android自定義加載圈動(dòng)畫展示的具體代碼,供大家參考,具體內(nèi)容如下
實(shí)現(xiàn)如下效果:
該效果圖主要有3個(gè)動(dòng)畫:
1.旋轉(zhuǎn)動(dòng)畫
2.聚合動(dòng)畫
3.擴(kuò)散動(dòng)畫
以上3個(gè)動(dòng)畫都是通過(guò)ValueAnimator來(lái)實(shí)現(xiàn),配合自定義View的onDraw()方法實(shí)現(xiàn)不斷的刷新和繪制界面.
具體代碼如下:
package blog.csdn.net.mchenys.myanimationloading; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.ValueAnimator; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.PointF; import android.util.AttributeSet; import android.view.View; import android.view.animation.LinearInterpolator; import android.view.animation.OvershootInterpolator; /** * Created by mChenys on 2016/5/21. */ public class AnimationLoading extends View { private float mBigCircleRaduis = 90;//大圓的半徑 private float mSubCircleRadius = 20;//小圓的半徑 private PointF mBigCenterPoint;//大圓的圓心坐標(biāo) private Paint mBgPaint;//繪制背景的畫筆 private Paint mFgPaint;//繪制前景色的畫筆 private AnimatorTemplet mTemplet;//動(dòng)畫模板 float mBigCircleRotateAngle;//大圓旋轉(zhuǎn)的角度 float mDiagonalDist;//屏幕對(duì)角線一半的距離 float mBgStrokeCircleRadius;//用于作為繪制背景空心圓的半徑 //6個(gè)小圓的顏色 private int[] colors = new int[]{Color.RED, Color.DKGRAY, Color.YELLOW, Color.BLUE, Color.LTGRAY, Color.GREEN}; public AnimationLoading(Context context) { this(context, null); } public AnimationLoading(Context context, AttributeSet attrs) { super(context, attrs); init(); } @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); //確定大圓的圓心坐標(biāo) mBigCenterPoint.x = w / 2f; mBigCenterPoint.y = h / 2f; //屏幕對(duì)角線的一半 mDiagonalDist = (float) (Math.sqrt(w * w + h * h) / 2); } private void init() { mBigCenterPoint = new PointF(); mFgPaint = new Paint(); mFgPaint.setAntiAlias(true); mBgPaint = new Paint(mFgPaint); mBgPaint.setColor(Color.WHITE); mBgPaint.setStyle(Paint.Style.STROKE); } @Override protected void onDraw(Canvas canvas) { if (null == mTemplet) { //開啟旋轉(zhuǎn)動(dòng)畫 mTemplet = new RotateState(); } //傳遞Canvas對(duì)象 mTemplet.drawState(canvas); } /** * 繪制圓 * * @param canvas */ private void drawCircle(Canvas canvas) { //獲取每個(gè)小圓間隔的角度 float rotateAngle = (float) (2 * Math.PI / colors.length); for (int i = 0; i < colors.length; i++) { //每個(gè)小圓的實(shí)際角度 double angle = rotateAngle * i + mBigCircleRotateAngle; //這里加上大圓旋轉(zhuǎn)的角度是為了帶動(dòng)小圓一起旋轉(zhuǎn) //計(jì)算每個(gè)小圓的圓心坐標(biāo) float cx = (float) (mBigCircleRaduis * Math.cos(angle)) + mBigCenterPoint.x; float cy = (float) (mBigCircleRaduis * Math.sin(angle)) + mBigCenterPoint.y; //繪制6個(gè)小圓 mFgPaint.setColor(colors[i]); canvas.drawCircle(cx, cy, mSubCircleRadius, mFgPaint); } } /** * 繪制背景 * * @param canvas */ private void drawBackground(Canvas canvas) { if (mBgStrokeCircleRadius > 0f) { //不斷擴(kuò)散的空心圓,空心圓的半徑為屏幕對(duì)角線的一半,空心圓的線寬則從線寬一半到0 float strokeWidth = mDiagonalDist - mBgStrokeCircleRadius;//線寬從對(duì)角線的1/2 ~ 0 mBgPaint.setStrokeWidth(strokeWidth); float radius = mBgStrokeCircleRadius + strokeWidth / 2;//半徑從對(duì)角線的1/4 ~ 1/2 canvas.drawCircle(mBigCenterPoint.x, mBigCenterPoint.y,radius , mBgPaint); } else { //繪制白色背景 canvas.drawColor(Color.WHITE); } } private abstract class AnimatorTemplet { abstract void drawState(Canvas canvas); } /** * 繪制旋轉(zhuǎn)動(dòng)畫 */ private class RotateState extends AnimatorTemplet { ValueAnimator mValueAnimator; public RotateState() { //旋轉(zhuǎn)的過(guò)程,就是不斷的獲取大圓的角度,從0-2π mValueAnimator = ValueAnimator.ofFloat(0, (float) Math.PI * 2); mValueAnimator.setInterpolator(new LinearInterpolator());//勻速插值器 mValueAnimator.setDuration(1200); mValueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) { //獲取大圓旋轉(zhuǎn)的角度 mBigCircleRotateAngle = (float) animation.getAnimatedValue(); //重繪 invalidate(); } }); mValueAnimator.setRepeatCount(ValueAnimator.INFINITE);//無(wú)限循環(huán) mValueAnimator.start(); } /** * 停止旋轉(zhuǎn)動(dòng)畫,在數(shù)據(jù)加載完畢后供外部調(diào)用 */ public void stopRotate() { mValueAnimator.cancel(); } @Override void drawState(Canvas canvas) { drawBackground(canvas); drawCircle(canvas); } } /** * 繪制聚合動(dòng)畫 */ private class MergingState extends AnimatorTemplet { public MergingState() { //聚合的過(guò)程,就是不斷的改變大圓的半徑,從mBigCircleRaduis~0 ValueAnimator valueAnimator = ValueAnimator.ofFloat(mBigCircleRaduis, 0); valueAnimator.setInterpolator(new OvershootInterpolator(10f));//彈性插值器 valueAnimator.setDuration(600); valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) { //獲取大圓變化的半徑 mBigCircleRaduis = (float) animation.getAnimatedValue(); //重繪 invalidate(); } }); valueAnimator.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { //聚合執(zhí)行完后進(jìn)入下一個(gè)擴(kuò)散動(dòng)畫 mTemplet = new SpreadState(); } }); valueAnimator.start(); } @Override void drawState(Canvas canvas) { drawBackground(canvas); drawCircle(canvas); } } /** * 繪制擴(kuò)散動(dòng)畫 */ private class SpreadState extends AnimatorTemplet { public SpreadState() { //擴(kuò)散的過(guò)程,就是不斷的改變背景畫繪制空心圓的半徑,從0~mDiagonalDist ValueAnimator valueAnimator = ValueAnimator.ofFloat(0, mDiagonalDist); valueAnimator.setDuration(600); valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) { //獲取大圓變化的半徑 mBgStrokeCircleRadius = (float) animation.getAnimatedValue(); //重繪 invalidate(); } }); valueAnimator.start(); } @Override void drawState(Canvas canvas) { drawBackground(canvas); } } /** * 停止加載動(dòng)畫 */ public void stopLoading() { if (null != mTemplet && mTemplet instanceof RotateState) { ((RotateState) mTemplet).stopRotate(); //開啟下一個(gè)聚合動(dòng)畫 post(new Runnable() { @Override public void run() { mTemplet = new MergingState(); } }); } } }
測(cè)試的Activity
package blog.csdn.net.mchenys.myanimationloading; import android.os.Bundle; import android.os.Handler; import android.support.v7.app.AppCompatActivity; import android.widget.FrameLayout; import android.widget.ImageView; public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); FrameLayout content = new FrameLayout(this); content.setOnClickListener(null); ImageView bg = new ImageView(this); bg.setImageResource(R.drawable.fg); bg.setScaleType(ImageView.ScaleType.FIT_XY); content.addView(bg); final AnimationLoading loading = new AnimationLoading(this); content.addView(loading); setContentView(content); new Handler().postDelayed(new Runnable() { @Override public void run() { //3s后停止加載動(dòng)畫 loading.stopLoading(); } },3000); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
項(xiàng)目發(fā)布Debug和Release版的區(qū)別詳解
這篇文章主要為大家詳細(xì)介紹了項(xiàng)目發(fā)布Debug和Release版的區(qū)別,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10Android UI設(shè)計(jì)與開發(fā)之使用ViewPager實(shí)現(xiàn)歡迎引導(dǎo)頁(yè)面
這篇文章主要為大家詳細(xì)介紹了Android UI設(shè)計(jì)與開發(fā)之使用ViewPager實(shí)現(xiàn)歡迎引導(dǎo)頁(yè)面,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08android調(diào)用國(guó)家氣象局天氣預(yù)報(bào)接口json數(shù)據(jù)格式解釋
平時(shí)我們?cè)陂_發(fā)的過(guò)程中有時(shí)會(huì)要用到天氣預(yù)報(bào)的信息,國(guó)家氣象局為我們提供了天氣預(yù)報(bào)的接口,只需要我們?nèi)ソ馕鼍托辛?。很方便很好?/div> 2013-11-11解決VSCode調(diào)試react-native android項(xiàng)目錯(cuò)誤問(wèn)題
這篇文章主要介紹了VSCode調(diào)試react-native android項(xiàng)目錯(cuò)誤解決辦法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12Android上傳文件到服務(wù)端并顯示進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了Android上傳文件到服務(wù)端,并顯示進(jìn)度條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11android圖像繪制(四)自定義一個(gè)SurfaceView控件
自定義控件(類似按鈕等)的使用,自定義一個(gè)SurfaceView。如某一塊的動(dòng)態(tài)圖(自定義相應(yīng)),或者類似UC瀏覽器下面的工具欄,感興趣的朋友可以了解下2013-01-01android 仿QQ動(dòng)態(tài)背景、視頻背景的示例代碼
本篇文章主要介紹了android 仿QQ動(dòng)態(tài)背景、視頻背景的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03Android應(yīng)用啟動(dòng)另外一個(gè)apk應(yīng)用的方法
這篇文章主要介紹了Android應(yīng)用啟動(dòng)另外一個(gè)apk應(yīng)用的方法,涉及Android基于intent的package調(diào)用與管理技巧,需要的朋友可以參考下2016-02-02android實(shí)現(xiàn)動(dòng)態(tài)顯隱進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)動(dòng)態(tài)顯隱進(jìn)度條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07最新評(píng)論