android 自定义加载动画效果,Android自定义加载圈动画效果

本文实例为大家分享了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);

}

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果你想要在 Android 应用中使用自定义动画,可以通过创建自定义的布局和使用 Animation 类来实现。以下是一个简单的示例,演示如何在 Android 应用中自定义动画: 首先,在 res/layout 目录下创建一个新的布局文件,例如 custom_loading.xml,其中包含自定义动画布局,例如: ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="vertical" android:gravity="center"> <ImageView android:id="@+id/loading_image" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/custom_loading_animation" android:layout_marginBottom="16dp"/> <TextView android:id="@+id/loading_text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Loading..." android:textColor="@android:color/white"/> </LinearLayout> ``` 其中,loading_image 是一个自定义动画的 ImageView,loading_text 是一个文本视图,用于显示消息。 接下来,在 res/drawable 目录下创建一个新的动画文件,例如 custom_loading_animation.xml,其中包含自定义动画动画,例如: ```xml <?xml version="1.0" encoding="utf-8"?> <animation-list xmlns:android="http://schemas.android.com/apk/res/android" android:oneshot="false"> <item android:drawable="@drawable/custom_loading_frame1" android:duration="100" /> <item android:drawable="@drawable/custom_loading_frame2" android:duration="100" /> <item android:drawable="@drawable/custom_loading_frame3" android:duration="100" /> <item android:drawable="@drawable/custom_loading_frame4" android:duration="100" /> </animation-list> ``` 其中,custom_loading_frame1、custom_loading_frame2、custom_loading_frame3、custom_loading_frame4 是自定义动画的帧。 最后,在你的 Activity 中,使用 LayoutInflater 类将 custom_loading.xml 布局文件实例化为 View 对象,并使用 Animation 类将 custom_loading_animation.xml 动画文件到 ImageView 中,例如: ```java LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE); View view = inflater.inflate(R.layout.custom_loading, null); ImageView imageView = (ImageView) view.findViewById(R.id.loading_image); Animation animation = AnimationUtils.loadAnimation(this, R.drawable.custom_loading_animation); imageView.startAnimation(animation); ProgressDialog progressDialog = new ProgressDialog(this); progressDialog.setCancelable(false); progressDialog.show(); progressDialog.setContentView(view); ``` 其中,使用 LayoutInflater 类将 custom_loading.xml 布局文件实例化为 View 对象,并使用 findViewById() 方法获取 loading_image ImageView 对象。然后,使用 AnimationUtils.loadAnimation() 方法将 custom_loading_animation.xml 动画文件到 ImageView 中,并调用 startAnimation() 方法开始播放动画。最后,将 View 对象设置为 ProgressDialog 的内容视图,调用 show() 方法显示 ProgressDialog。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值