本文实例为大家分享了Android自定义加载圈动画展示的具体代码,供大家参考,具体内容如下

实现如下效果:
该效果图主要有3个动画:
1.旋转动画
2.聚合动画
3.扩散动画
以上3个动画都是通过ValueAnimator来实现,配合自定义View的onDraw()方法实现不断的刷新和绘制界面.
具体代码如下:
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;//大圆的圆心坐标
private Paint mBgPaint;//绘制背景的画笔
private Paint mFgPaint;//绘制前景色的画笔
private AnimatorTemplet mTemplet;//动画模板
float mBigCircleRotateAngle;//大圆旋转的角度
float mDiagonalDist;//屏幕对角线一半的距离
float mBgStrokeCircleRadius;//用于作为绘制背景空心圆的半径
//6个小圆的颜色
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);
//确定大圆的圆心坐标
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) {
//开启旋转动画
mTemplet = new RotateState();
}
//传递Canvas对象
mTemplet.drawState(canvas);
}
/**
* 绘制圆
*
* @param canvas
*/
private void drawCircle(Canvas canvas) {
//获取每个小圆间隔的角度
float rotateAngle = (float) (2 * Math.PI / colors.length);
for (int i = 0; i < colors.length; i++) {
//每个小圆的实际角度
double angle = rotateAngle * i + mBigCircleRotateAngle; //这里加上大圆旋转的角度是为了带动小圆一起旋转
//计算每个小圆的圆心坐标
float cx = (float) (mBigCircleRaduis * Math.cos(angle)) + mBigCenterPoint.x;
float cy = (float) (mBigCircleRaduis * Math.sin(angle)) + mBigCenterPoint.y;
//绘制6个小圆
mFgPaint.setColor(colors[i]);
canvas.drawCircle(cx, cy, mSubCircleRadius, mFgPaint);
}
}
/**
* 绘制背景
*
* @param canvas
*/
private void drawBackground(Canvas canvas) {
if (mBgStrokeCircleRadius > 0f) {
//不断扩散的空心圆,空心圆的半径为屏幕对角线的一半,空心圆的线宽则从线宽一半到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);
}
/**
* 绘制旋转动画
*/
private class RotateState extends AnimatorTemplet {
ValueAnimator mValueAnimator;
public RotateState() {
//旋转的过程,就是不断的获取大圆的角度,从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) {
//获取大圆旋转的角度
mBigCircleRotateAngle = (float) animation.getAnimatedValue();
//重绘
invalidate();
}
});
mValueAnimator.setRepeatCount(ValueAnimator.INFINITE);//无限循环
mValueAnimator.start();
}
/**
* 停止旋转动画,在数据加载完毕后供外部调用
*/
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) {
//聚合执行完后进入下一个扩散动画
mTemplet = new SpreadState();
}
});
valueAnimator.start();
}
@Override
void drawState(Canvas canvas) {
drawBackground(canvas);
drawCircle(canvas);
}
}
/**
* 绘制扩散动画
*/
private class SpreadState extends AnimatorTemplet {
public SpreadState() {
//扩散的过程,就是不断的改变背景画绘制空心圆的半径,从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();
//开启下一个聚合动画
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);
}
}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# 加载圈
# Android实现加载圈
# Android实现仿iOS菊花加载圈动画效果
# Android绘制圆形百分比加载圈效果
# Android自定义加载圈的方法
# 小圆
# 加载
# 自定义
# 都是
# 插值
# 大家分享
# 来实现
# 完后
# 具体内容
# 大家多多
# 是为了
# 前景色
# 主要有
# 完毕后
# float
# mBigCircleRaduis
# mSubCircleRadius
# mBgPaint
# mFgPaint
# mBigCenterPoint
相关文章:
三星网站视频制作教程下载,三星w23网页如何全屏?
黑客如何利用漏洞与弱口令入侵网站服务器?
如何配置IIS站点权限与局域网访问?
,怎么用自己头像做动态表情包?
公司网站的制作公司,企业网站制作基本流程有哪些?
微信推文制作网站有哪些,怎么做微信推文,急?
智能起名网站制作软件有哪些,制作logo的软件?
制作国外网站的软件,国外有哪些比较优质的网站推荐?
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略
如何破解联通资金短缺导致的基站建设难题?
简历在线制作网站免费版,如何创建个人简历?
动图在线制作网站有哪些,滑动动图图集怎么做?
建站之星ASP如何实现CMS高效搭建与安全管理?
独立制作一个网站多少钱,建立网站需要花多少钱?
制作企业网站建设方案,怎样建设一个公司网站?
电商网站制作价格怎么算,网上拍卖流程以及规则?
,石家庄四十八中学官网?
建站之星如何实现五合一智能建站与营销推广?
学校建站服务器如何选型才能满足性能需求?
深圳网站制作平台,深圳市做网站好的公司有哪些?
网站制作话术技巧,网站推广做的好怎么话术?
如何用虚拟主机快速搭建网站?详细步骤解析
seo网站制作优化,网站SEO优化步骤有哪些?
如何处理“XML格式不正确”错误 常见XML well-formed问题解决方法
大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
兔展官网 在线制作,怎样制作微信请帖?
南京网站制作费用,南京远驱官方网站?
打鱼网站制作软件,波克捕鱼官方号怎么注册?
免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?
如何获取PHP WAP自助建站系统源码?
重庆网站制作公司哪家好,重庆中考招生办官方网站?
如何获取开源自助建站系统免费下载链接?
网站制作公司广州有几家,广州尚艺美发学校网站是多少?
安徽网站建设与外贸建站服务专业定制方案
建站VPS推荐:2025年高性能服务器配置指南
实现虚拟支付需哪些建站技术支撑?
网站设计制作企业有哪些,抖音官网主页怎么设置?
如何在万网开始建站?分步指南解析
网页设计与网站制作内容,怎样注册网站?
如何通过IIS搭建网站并配置访问权限?
寿县云建站:智能SEO优化与多行业模板快速上线指南
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
如何在IIS管理器中快速创建并配置网站?
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
建站之星如何助力网站排名飙升?揭秘高效技巧
公司网站制作价格怎么算,公司办个官网需要多少钱?
番禺网站制作公司哪家值得合作,番禺图书馆新馆开放了吗?
佛山网站制作系统,佛山企业变更地址网上办理步骤?
*请认真填写需求信息,我们会在24小时内与您取得联系。