Android自定義加載圈動畫效果
本文實(shí)例為大家分享了Android自定義加載圈動畫展示的具體代碼,供大家參考,具體內(nèi)容如下
實(shí)現(xiàn)如下效果:

該效果圖主要有3個(gè)動畫:
1.旋轉(zhuǎn)動畫
2.聚合動畫
3.擴(kuò)散動畫
以上3個(gè)動畫都是通過ValueAnimator來實(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;//動畫模板
float mBigCircleRotateAngle;//大圓旋轉(zhuǎn)的角度
float mDiagonalDist;//屏幕對角線一半的距離
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;
//屏幕對角線的一半
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)動畫
mTemplet = new RotateState();
}
//傳遞Canvas對象
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)的角度是為了帶動小圓一起旋轉(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ò)散的空心圓,空心圓的半徑為屏幕對角線的一半,空心圓的線寬則從線寬一半到0
float strokeWidth = mDiagonalDist - mBgStrokeCircleRadius;//線寬從對角線的1/2 ~ 0
mBgPaint.setStrokeWidth(strokeWidth);
float radius = mBgStrokeCircleRadius + strokeWidth / 2;//半徑從對角線的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)動畫
*/
private class RotateState extends AnimatorTemplet {
ValueAnimator mValueAnimator;
public RotateState() {
//旋轉(zhuǎn)的過程,就是不斷的獲取大圓的角度,從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);//無限循環(huán)
mValueAnimator.start();
}
/**
* 停止旋轉(zhuǎn)動畫,在數(shù)據(jù)加載完畢后供外部調(diào)用
*/
public void stopRotate() {
mValueAnimator.cancel();
}
@Override
void drawState(Canvas canvas) {
drawBackground(canvas);
drawCircle(canvas);
}
}
/**
* 繪制聚合動畫
*/
private class MergingState extends AnimatorTemplet {
public MergingState() {
//聚合的過程,就是不斷的改變大圓的半徑,從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ò)散動畫
mTemplet = new SpreadState();
}
});
valueAnimator.start();
}
@Override
void drawState(Canvas canvas) {
drawBackground(canvas);
drawCircle(canvas);
}
}
/**
* 繪制擴(kuò)散動畫
*/
private class SpreadState extends AnimatorTemplet {
public SpreadState() {
//擴(kuò)散的過程,就是不斷的改變背景畫繪制空心圓的半徑,從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);
}
}
/**
* 停止加載動畫
*/
public void stopLoading() {
if (null != mTemplet && mTemplet instanceof RotateState) {
((RotateState) mTemplet).stopRotate();
//開啟下一個(gè)聚合動畫
post(new Runnable() {
@Override
public void run() {
mTemplet = new MergingState();
}
});
}
}
}
測試的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后停止加載動畫
loading.stopLoading();
}
},3000);
}
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android編程操作嵌入式關(guān)系型SQLite數(shù)據(jù)庫實(shí)例詳解
這篇文章主要介紹了Android編程操作嵌入式關(guān)系型SQLite數(shù)據(jù)庫的方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Android操作SQLite數(shù)據(jù)庫的基本技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2016-01-01
Android入門之在Activity之間穿梭的Intent
Intent可以用來啟動Activity(startActivity(Intent))、Serveice(startService(Intent))等組件,可以用來綁定Activity和Service以建立它們之間的通信(bindServiceConnaction(Intent,ServiceConnection,int)),可以作為Broadcast Intent發(fā)送給廣播接收器2021-10-10
Android 使用 ViewPager循環(huán)廣告位的實(shí)現(xiàn)
本文給大家分享android使用 ViewPager循環(huán)廣告位的實(shí)現(xiàn),感興趣的朋友一起學(xué)習(xí)吧2015-11-11
Android實(shí)現(xiàn)3D翻轉(zhuǎn)動畫效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)3D翻轉(zhuǎn)動畫效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
Android畫圖之抗鋸齒paint和Canvas兩種方式實(shí)例
本篇文章主要介紹了Android畫圖之抗鋸齒paint和Canvas兩種方式實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-04-04
點(diǎn)擊圖標(biāo)進(jìn)入指定瀏覽器將首頁設(shè)置全透明解決一閃而過問題
進(jìn)入瀏覽器之前有一個(gè)頁面閃了一下,那是因?yàn)閺哪莻€(gè)空白的首頁跳過去的。解決的辦法是把他變成透明的就好了2014-08-08
Android自定義View實(shí)現(xiàn)箭頭沿圓轉(zhuǎn)動實(shí)例代碼
這篇文章主要介紹了Android自定義View實(shí)現(xiàn)箭頭沿圓轉(zhuǎn)動實(shí)例代碼,需要的朋友可以參考下2017-09-09
Kotlin中空判斷與問號和感嘆號標(biāo)識符使用方法
最近使用kotlin重構(gòu)項(xiàng)目,遇到了一個(gè)小問題,在Java中,可能會遇到判斷某個(gè)對象是否為空,為空執(zhí)行一段邏輯,不為空執(zhí)行另外一段邏輯,下面這篇文章主要給大家介紹了關(guān)于Kotlin中空判斷與問號和感嘆號標(biāo)識符處理操作的相關(guān)資料,需要的朋友可以參考下2022-12-12

