Android——自定义对号动画View练习

公司拿给客户演示用的App里当一笔交易完成时,有一个这样的动画,之前的代码里是用了19个图片,然后使用帧动画做的。要换颜色,就需要重新将19个图片的颜色进行调整,麻烦UI小姐姐,比较费事。就尝试使用一个自定义View来实现下,主要是对Path的练习

对号动画

1. 使用

布局代码

 <com.example.gcc.retrofitl.checkmark.RightMarkView
     android:id="@+id/activity_right_mark_rmv"
     android:layout_width="150dp"
     android:layout_height="150dp" />

直接引用就可以,然后主要就是宽和高


Activity代码

public class RightMarkViewActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_right_mark_view);
        initView();
    }

    private void initView() {
        final RightMarkView markView = 
                    (RightMarkView) findViewById(R.id.activity_right_mark_rmv);
        // 设置开始和结束两种颜色
        markView.setColor(Color.parseColor("#FF4081"), Color.YELLOW);
        // 设置画笔粗细
        markView.setStrokeWidth(10f);

        Button bt = (Button) findViewById(R.id.activity_right_mark_bt_start);
        bt.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                markView.startAnimator();
            }
        });

    }
}

可以设置颜色和绘制的粗细


2 RightMarkView

public class RightMarkView extends View {
    private static final String TAG = RightMarkView.class.getSimpleName();
    private Paint mPaint;
    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.RED;
    private int mEndColor = Color.YELLOW;

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

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

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

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

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

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

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

        // 初始化动画
        initCircleAnimator();

        // 利用 post 获取 View 的宽高
        // post 内任务,会在第2次执行 onMeasure() 方法后执行
        post(() -> {
            // 初始化圆环 Path
            initCirclePath();

            // 初始化线性渐变
            // 由于要使用 mRealSize ,放 post 内
            initShader();
        });
    }


    /**
     * 开启动画
     */
    public void startAnimator() {
        mPaint.setStrokeWidth(mStrokeWidth);
        mPaint.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, mPaint);
    }

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

        if (isCircleNeedCancel) {
            log("圆形动画取消");
            mCircleValueAnimator.cancel();
        }

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


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

    /**
     * 画笔
     */
    private void initPaint() {
        mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        mPaint.setStyle(Paint.Style.STROKE);
        mPaint.setStrokeJoin(Paint.Join.ROUND);
    }

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

        // 添加圆环路径
        mCirclePath = new Path();
        float x = mRealSize / 2;
        float y = mRealSize / 2;
        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(ANIMATOR_TIME);

        // 插值器
        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());
    }


    /**
     * 关联对号 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();
    }

    private void log(String s) {
        Log.e(TAG, " ---> { " + s + " }");
    }
}

待优化的地方

LinearGradient shader = 
      new LinearGradient(0, 0, mRelativeLength, mRelativeLength,
                        mStartColor, mEndColor, Shader.TileMode.REPEAT)

颜色渐变这里,偷了懒,直接从整个View的左上角到右下角做了线性渐变,开始渐变和结束渐变的坐标需要再做更精确的计算,否则渐变的颜色过渡可能和UI小姐姐们给的效果图不匹配

initRightMarkPath()方法中,初始化对号Path的比例,也需要调整,需要找UI姐姐帮忙定下比例,由于只是练习,我并不想去麻烦小姐姐,比例也就自己试着定了下


3. 最后

这种效果应该有更方便简单的实现思路,我这样实现也是做了个小练习

有错误,请指出

共勉 : )

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容