Android画对号动画

把mCircleValueAnimator的duration设置大于零可以先画背景再画对号,修改画笔样式可以把背景化成圈或者纯色背景。

package com.lianzhuo.qukanba.widget;

import android.animation.Animator;
import android.animation.AnimatorListenerAdapter;
import android.animation.ValueAnimator;
import android.content.Context;
import android.graphics.*;
import android.util.AttributeSet;
import android.view.View;
import android.view.animation.AccelerateDecelerateInterpolator;
import androidx.annotation.ColorInt;
import androidx.annotation.Nullable;

/**
 * create by chenyingjie on 2019/10/29
 * desc
 */
public class MarkView extends View {

    private Paint mPaint;
    private Paint hootPaint;
    private PathMeasure mPathMeasure;

    /**
     * 圆环路径
     */
    private Path mCirclePath;

    /**
     * 截取的路径
     */
    private Path mDstPath;

    /**
     * Path 长度
     */
    private float mPathLength;

    /**
     * 动画估值
     */
    private float mAnimatorValue;

    /**
     * 圆环是否已经加载过
     */
    private boolean mIsHasCircle = false;

    /**
     * View 是个正方形,宽高中小的一个值,根据小的值来定位绘制
     */
    private int mRealSize;

    /**
     * 颜色
     */
    private int mStartColor = Color.parseColor("#d0021b");
    private int mEndColor = Color.parseColor("#d0021b");

    /**
     * 画笔宽度
     */
    private float mStrokeWidth = 4f;

    /**
     * 圆形动画
     */
    private ValueAnimator mCircleValueAnimator;

    /**
     * 对号
     */
    private ValueAnimator mRightMarkValueAnimator;

    /**
     * 默认大小
     */
    private static final int DEFAULT_SIZE = 15;

    /**
     * 动画执行时间
     */
    public static final int ANIMATOR_TIME = 1000;

    public MarkView(Context context) {
        this(context, null);
    }

    public MarkView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        // 初始化画笔
        initPaint();

        // 初始化动画
        initCircleAnimator();

        // 利用 post 获取 View 的宽高
        // post 内任务,会在第2次执行 onMeasure() 方法后执行
        post(() -> {
            // 初始化圆环 Path
            initCirclePath();
            // 初始化线性渐变
            // 由于要使用 mRealSize ,放 post 内
            initShader();
        });
    }

    /**
     * 开启动画
     */
    public void startAnimator() {
        setVisibility(VISIBLE);
        hootPaint.setStrokeWidth(mStrokeWidth);
        hootPaint.setColor(mStartColor);
        mCircleValueAnimator.start();
    }

    /**
     * 设置颜色
     */
    public void setColor(@ColorInt int startColor, @ColorInt int endColor) {
        this.mStartColor = startColor;
        this.mEndColor = endColor;
    }

    /**
     * 设置画笔粗细
     */
    public void setStrokeWidth(float strokeWidth) {
        this.mStrokeWidth = strokeWidth;
    }

    /**
     * 测量,强制将 View 设置为正方形
     * 当宽和高有一个为 wrap_content 时,就将宽高都定为 150 px
     * 当宽或者高有一个小于 150 px 时,都设置为 150 px
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int wSpecMode = MeasureSpec.getMode(widthMeasureSpec);
        int wSpecSize = MeasureSpec.getSize(widthMeasureSpec);
        int hSpecMode = MeasureSpec.getMode(heightMeasureSpec);
        int hSpecSize = MeasureSpec.getSize(heightMeasureSpec);

        // 取宽高中小的,强制设置成为正方形
        int realSize = Math.min(wSpecSize, hSpecSize);
        // 宽高模式是否有一个为 AT_MOST
        boolean isAnyOneAtMost = (wSpecMode == MeasureSpec.AT_MOST || hSpecMode == MeasureSpec.AT_MOST);
        if (!isAnyOneAtMost) {
            // 将宽高中小的值 realSize 与 150px 比较,取大的值
            realSize = Math.max(realSize, DEFAULT_SIZE);
            setMeasuredDimension(realSize, realSize);
        } else {
            setMeasuredDimension(DEFAULT_SIZE, DEFAULT_SIZE);
        }
    }

    /**
     * 绘制
     * @param canvas 画布
     */
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        if (mDstPath == null) {
            return;
        }
        // 绘制已经记录过的圆圈 Path
        if (mIsHasCircle) {
            canvas.drawPath(mCirclePath, mPaint);
        }

        // 刷新当前截取 Path
        mDstPath.reset();

        // 避免硬件加速的Bug
        mDstPath.lineTo(0, 0);

        // 截取片段
        float stop = mPathLength * mAnimatorValue;
        mPathMeasure.getSegment(0, stop, mDstPath, true);
        // 绘制截取的片段
        canvas.drawPath(mDstPath, hootPaint);
    }

    /**
     * 当View从屏幕消失时,关闭可能在执行的动画,以免可能出现内存泄漏
     */
    @Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        // 取消圆形动画
        boolean isCircleNeedCancel = (mCircleValueAnimator != null && mCircleValueAnimator.isRunning());
        if (isCircleNeedCancel) {
            mCircleValueAnimator.cancel();
        }

        // 取消对号动画
        boolean isRightMarkNeedCancel = (mRightMarkValueAnimator != null && mRightMarkValueAnimator.isRunning());
        if (isRightMarkNeedCancel) {
            mRightMarkValueAnimator.cancel();
        }
    }


    /**
     * 线性渐变
     */
    private void initShader() {
        // 使用线性渐变
        LinearGradient shader = new LinearGradient(0, 0, mRealSize, mRealSize, mStartColor, mEndColor, Shader.TileMode.REPEAT);
        hootPaint.setShader(shader);
    }

    /**
     * 画笔
     */
    private void initPaint() {
        mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        mPaint.setColor(Color.parseColor("#b3000000"));
        mPaint.setStyle(Paint.Style.FILL);
        mPaint.setStrokeJoin(Paint.Join.ROUND);

        hootPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        hootPaint.setStyle(Paint.Style.STROKE);
        hootPaint.setStrokeJoin(Paint.Join.ROUND);
    }

    /**
     * 绘制路径
     */
    private void initCirclePath() {
        // 获取View 的宽
        mRealSize = getWidth();

        // 添加圆环路径
        mCirclePath = new Path();
        float x = mRealSize / 2f;
        float y = mRealSize / 2f;
        float radius = x / 3 * 2;
        mCirclePath.addCircle(x, y, radius, Path.Direction.CW);

        // PathMeasure
        mPathMeasure = new PathMeasure();
        mPathMeasure.setPath(mCirclePath, false);

        // 此时为圆的周长
        mPathLength = mPathMeasure.getLength();

        // Path dst 用来存储截取的Path片段
        mDstPath = new Path();
    }

    /**
     * 初始化圆形动画
     */
    private void initCircleAnimator() {
        // 圆环动画
        mCircleValueAnimator = ValueAnimator.ofFloat(0, 1);

        // 动画过程
        mCircleValueAnimator.addUpdateListener(animation -> {
            mAnimatorValue = (float) animation.getAnimatedValue();
            invalidate();
        });

        // 动画时间
        mCircleValueAnimator.setDuration(0);

        // 插值器
        mCircleValueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());

        // 圆环结束后,开启对号的动画
        mCircleValueAnimator.addListener(new AnimatorListenerAdapter() {
            @Override
            public void onAnimationEnd(Animator animation) {
                super.onAnimationEnd(animation);
                mIsHasCircle = true;
                initMarkAnimator();
                initRightMarkPath();
                mRightMarkValueAnimator.start();
            }
        });
    }

    /**
     * 初始化对号动画
     */
    private void initMarkAnimator() {
        mRightMarkValueAnimator = ValueAnimator.ofFloat(0, 1);
        // 动画过程
        mRightMarkValueAnimator.addUpdateListener(animation -> {
            mAnimatorValue = (float) animation.getAnimatedValue();
            invalidate();
        });

        // 动画时间
        mRightMarkValueAnimator.setDuration(ANIMATOR_TIME);

        // 插值器
        mRightMarkValueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());

        mRightMarkValueAnimator.addListener(new AnimatorListenerAdapter() {
            @Override
            public void onAnimationEnd(Animator animation) {
                super.onAnimationEnd(animation);
                setVisibility(INVISIBLE);
            }
        });
    }

    /**
     * 关联对号 Path
     */
    private void initRightMarkPath() {
        Path path = new Path();
        // 对号起点
        float startX = (float) (0.3 * mRealSize);
        float startY = (float) (0.5 * mRealSize);
        path.moveTo(startX, startY);

        // 对号拐角点
        float cornerX = (float) (0.43 * mRealSize);
        float cornerY = (float) (0.66 * mRealSize);
        path.lineTo(cornerX, cornerY);

        // 对号终点
        float endX = (float) (0.75 * mRealSize);
        float endY = (float) (0.4 * mRealSize);
        path.lineTo(endX, endY);

        // 重新关联Path
        mPathMeasure.setPath(path, false);

        // 此时为对号 Path 的长度
        mPathLength = mPathMeasure.getLength();
    }
}

发布了30 篇原创文章 · 获赞 18 · 访问量 2万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章