Android自定义view仿支付宝芝麻信用表盘

演示效果

Github代码地址

实现步骤:

1.画不同宽度和半径的内外圆弧
2.通过循环旋转canvas,在固定位置绘制短线刻度,长线刻度,刻度文字
3.绘制view中心几个文本,并调整位置
4.实时更新当前旋转角度刷新小圆点位置;
5.判断分数应该坐落的区间,再根据该区间的角度比例获取实际的角度。
6.用ObjectAnimator实现滚动文字

步骤代码讲解:

  • 这里是声明的变量及常量
    /** 圆环paint */
    private Paint ringPaint;
    /** 信用分文字paint */
    private Paint scoreTextPaint;
    /** 指示器paint */
    private Paint dotPaint;
    /** 外圆环宽度 */
    private int outerRingWidth;
    /** 内圆环宽度 */
    private int innerRingWidth;
    /** 表盘圆环总度数 */
    private final int totalAngle = 210;
    /** 圆弧上总共刻度数 */
    private final int totalDegreeScale = 30;
    /** 各分数刻度文本 */
    private String[] scores = new String[] {"350", "较差", "550", "中等", "600", "良好", "650", "优秀", "700", "极好", "950"};
    /** 信用分数 */
    private int score = 715;
    /** 当前扫过的数值角度 */
    private float curProgressAngle;
    /** 动画时长 */
    private final long ANIM_DURATION = 2000;
    /** 指示器结束时的角度 */
    private float stopAngle;
    /** 信用评级文字 */
    private String creditStr;

绘制外层两个圆弧,以12点钟为圆弧中点,左右各有105°

    private void drawRing (Canvas canvas) {
        ringPaint.setAlpha(80);
        int startAngle = -90-totalAngle/2;  //圆环起始角度,12点钟位置为-90°

        //绘制外圆环
        ringPaint.setStrokeWidth(outerRingWidth);
        RectF rectF = new RectF(outerRingWidth, outerRingWidth, getWidth()-outerRingWidth, getHeight()-outerRingWidth);
        canvas.drawArc(rectF, startAngle, totalAngle, false, ringPaint);

        //绘制内圆环
        ringPaint.setStrokeWidth(innerRingWidth);
        int padding = outerRingWidth + dp2px(getContext(), 12);
        RectF innerRectF = new RectF(padding, padding, getWidth()-padding, getHeight()-padding);
        canvas.drawArc(innerRectF, startAngle, totalAngle, false, ringPaint);
    }

圆弧上总共有30个小刻度,遍历每隔小刻度,每个小刻度绘制较细的线,每3个刻度绘制一次文本,每6个刻度绘制一次较粗的线,通过旋转画布来实现,以使在同一位置绘制刻度和文本得以实现,最后要恢复画布。

    private void drawDegreeScale(Canvas canvas) {
        canvas.save();

        int padding = dp2px(getContext(), 10);

        canvas.rotate(-totalAngle/2, getWidth()/2, getHeight()/2); //将画布逆时间旋转一半弧度,使以左端点为刻度起点

        for (int i=0;i<=totalDegreeScale;i++) {
            ringPaint.setAlpha(80);
            ringPaint.setStrokeWidth(2);
            //每一格绘制一个浅色刻度
            canvas.drawLine(getWidth()/2, padding, getWidth()/2, padding+innerRingWidth, ringPaint);

            ringPaint.setAlpha(100);
            ringPaint.setStrokeWidth(3);
            //每6格刻度绘制一个深色刻度,即大刻度
            if (i%6==0) {
                canvas.drawLine(getWidth()/2, padding, getWidth()/2, padding+innerRingWidth+5, ringPaint);
            }

            //每三格刻度绘制一个文字
            if (i%3==0) {
                scoreTextPaint.setAlpha(180);
                float textWidth = scoreTextPaint.measureText(scores[i/3]);  //测量该文本宽度,需向左移动半个文本宽度以对齐
                canvas.drawText(scores[i/3], getWidth()/2-textWidth/2, padding+innerRingWidth+dp2px(getContext(), 12), scoreTextPaint);
            }

            canvas.rotate(totalAngle/totalDegreeScale, getWidth()/2, getHeight()/2); //每次画完从中心开始旋转画布单位刻度的弧度
        }

        canvas.restore(); //恢复角度
    }

绘制中心文本,包括beta,分数,信用评级,评估时间,这个很简单,就是需要获取文本的宽高调节依次垂直居中。

    private void drawCenterText(Canvas canvas) {
        //绘制当前分数
        scoreTextPaint.setAlpha(255);
        scoreTextPaint.setTextSize(170);

        String curScore = String.valueOf(score);
        Rect scoreRect = new Rect();
        scoreTextPaint.getTextBounds(curScore, 0, curScore.length(), scoreRect);  //需左移文字宽度以居中
        canvas.drawText(curScore, getWidth()/2-scoreRect.width()/2, getHeight()/2, scoreTextPaint);

        //绘制BETA文字
        scoreTextPaint.setAlpha(150);
        scoreTextPaint.setTextSize(35);
        Rect betaRect= new Rect();
        String betaStr = "BETA";
        scoreTextPaint.getTextBounds(betaStr, 0, betaStr.length(), betaRect);  //beta需向坐上移动
        canvas.drawText(betaStr, getWidth()/2-betaRect.width()/2, getHeight()/2-scoreRect.height()-betaRect.height()/2, scoreTextPaint);

        //绘制信用等级文本
        scoreTextPaint.setAlpha(200);
        scoreTextPaint.setTextSize(75);
        Rect creditRect = new Rect();
        scoreTextPaint.getTextBounds(creditStr, 0, creditStr.length(), creditRect);
        canvas.drawText(creditStr, getWidth()/2-creditRect.width()/2, getHeight()/2+scoreRect.height()/2+20, scoreTextPaint);

        //绘制评估时间
        scoreTextPaint.setAlpha(150);
        scoreTextPaint.setTextSize(35);
        float timeStrWidth = scoreTextPaint.measureText("评估时间:2020.07.27");
        canvas.drawText("评估时间:2020.07.27", getWidth()/2-timeStrWidth/2, getHeight()/2+scoreRect.height()+10, scoreTextPaint);
    }

绘制进度动画小圆点:根据属性curProgressAngle的值来绘制指示器小圆点,圆心为中心,小圆点轨迹为圆,由三角函数知识得到坐标轨迹方程。
同时,使用Paint的BlurMaskFilter实现小圆点高斯模糊发光效果,发光模式为Blur.SOLID,内外发光。

    private void drawDot(Canvas canvas) {
        scoreTextPaint.setAlpha(230);
        float x = (float) (getWidth()/2 + (getWidth()/2-outerRingWidth)*Math.sin(Math.toRadians(curProgressAngle)));
        float y = (float) (getHeight()/2 - (getWidth()/2-outerRingWidth)*Math.cos(Math.toRadians(curProgressAngle)));
        canvas.drawCircle(x, y, outerRingWidth, dotPaint);
    }

使用ValueAnimator动画实时更新curProgressAngle并重绘,以实现小圆点实时更新位置实现进度动画。

    private void startIndicatorAnim() {
        ValueAnimator valueAnimator = ValueAnimator.ofFloat(-105, stopAngle);
        valueAnimator.setDuration(ANIM_DURATION);
        valueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());
        valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                curProgressAngle = (float) animation.getAnimatedValue();

                postInvalidate();  //数值改变实时更新绘制
            }
        });
        valueAnimator.start();
    }

根据分数计算小圆点结束时的实际角度:在当前大刻度范围的结束角度,为之前刻度角度加上在该区间按比例得出的角度,30个小刻度共210°,每个小刻度perLevelAngle角度为7°

    private String showCreditLevel() {
        int startAngle = -105;
        int perLevelAngle = totalAngle/5;  //有5段大刻度
        String creditLevelStr = null;
        if (score < 350) {
            creditLevelStr = "信用较差";
            stopAngle = startAngle;
        } else if (score >= 350 && score < 550) {
            creditLevelStr = "信用较差";
            stopAngle = startAngle + (float)(score-350)/(550-350)*perLevelAngle;
        } else if (score >= 550 && score < 600) {
            creditLevelStr = "信用中等";
            stopAngle = startAngle + perLevelAngle + (float)(score-550)/(600-550)*perLevelAngle;
        } else if (score >= 600 && score < 650) {
            creditLevelStr = "信用良好";
            stopAngle = startAngle + perLevelAngle*2 + (float)(score-600)/(650-600)*perLevelAngle;
        } else if (score >= 650 && score < 700) {
            creditLevelStr = "信用优秀";
            stopAngle = startAngle + perLevelAngle*3 + (float)(score-650)/(700-650)*perLevelAngle;
        } else if (score >= 700 && score < 950) {
            creditLevelStr = "信用极好";
            stopAngle = startAngle + perLevelAngle*4 + (float)(score-700)/(950-700)*perLevelAngle;
        }
        return creditLevelStr;
    }

最后,通过ObjectAnimator实时更新分数。

    public void runWithAnimation(int number){
        ObjectAnimator objectAnimator = ObjectAnimator.ofInt(this, "score", 0, number);
        objectAnimator.setDuration(ANIM_DURATION);
        objectAnimator.setInterpolator(new AccelerateDecelerateInterpolator());

        objectAnimator.start();
    }

最后附上完整代码,喜欢的给个❤️吧。

/**
 * create by libo
 * create on 2020/7/27
 * description 支付宝芝麻信用自定义view
 */
public class SesameCreditView extends View {
    /** 圆环paint */
    private Paint ringPaint;
    /** 信用分文字paint */
    private Paint scoreTextPaint;
    /** 指示器paint */
    private Paint dotPaint;
    /** 外圆环宽度 */
    private int outerRingWidth;
    /** 内圆环宽度 */
    private int innerRingWidth;
    /** 表盘圆环总度数 */
    private final int totalAngle = 210;
    /** 圆弧上总共刻度数 */
    private final int totalDegreeScale = 30;
    /** 各分数刻度文本 */
    private String[] scores = new String[] {"350", "较差", "550", "中等", "600", "良好", "650", "优秀", "700", "极好", "950"};
    /** 信用分数 */
    private int score = 715;
    /** 当前扫过的数值角度 */
    private float curProgressAngle;
    /** 动画时长 */
    private final long ANIM_DURATION = 2000;
    /** 指示器结束时的角度 */
    private float stopAngle;
    /** 信用评级文字 */
    private String creditStr;

    public SesameCreditView(Context context) {
        super(context);
        init();
    }

    public SesameCreditView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    private void init() {
        outerRingWidth = dp2px(getContext(), 3);
        innerRingWidth = dp2px(getContext(), 10);

        ringPaint = new Paint();
        ringPaint.setColor(getResources().getColor(R.color.white));
        ringPaint.setAntiAlias(true);
        ringPaint.setStyle(Paint.Style.STROKE);

        scoreTextPaint = new Paint();
        scoreTextPaint.setColor(getResources().getColor(R.color.white));
        scoreTextPaint.setAntiAlias(true);
        scoreTextPaint.setTextSize(32);

        dotPaint = new Paint();
        dotPaint.setColor(getResources().getColor(R.color.white));
        dotPaint.setAntiAlias(true);
        dotPaint.setMaskFilter(new BlurMaskFilter(outerRingWidth, BlurMaskFilter.Blur.SOLID));  //设置指示器发光

        creditStr = showCreditLevel();

        startIndicatorAnim();
        runWithAnimation(score);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        drawRing(canvas);
        drawDegreeScale(canvas);
        drawCenterText(canvas);
        drawDot(canvas);
    }

    /**
     * 绘制圆弧
     * @param canvas
     */
    private void drawRing (Canvas canvas) {
        ringPaint.setAlpha(80);
        int startAngle = -90-totalAngle/2;  //圆环起始角度,12点钟位置为-90°

        //绘制外圆环
        ringPaint.setStrokeWidth(outerRingWidth);
        RectF rectF = new RectF(outerRingWidth, outerRingWidth, getWidth()-outerRingWidth, getHeight()-outerRingWidth);
        canvas.drawArc(rectF, startAngle, totalAngle, false, ringPaint);

        //绘制内圆环
        ringPaint.setStrokeWidth(innerRingWidth);
        int padding = outerRingWidth + dp2px(getContext(), 12);
        RectF innerRectF = new RectF(padding, padding, getWidth()-padding, getHeight()-padding);
        canvas.drawArc(innerRectF, startAngle, totalAngle, false, ringPaint);
    }

    /**
     * 绘制圆环刻度,分数文字
     */
    private void drawDegreeScale(Canvas canvas) {
        canvas.save();

        int padding = dp2px(getContext(), 10);

        canvas.rotate(-totalAngle/2, getWidth()/2, getHeight()/2); //将画布逆时间旋转一半弧度,使以左端点为刻度起点

        for (int i=0;i<=totalDegreeScale;i++) {
            ringPaint.setAlpha(80);
            ringPaint.setStrokeWidth(2);
            //每一格绘制一个浅色刻度
            canvas.drawLine(getWidth()/2, padding, getWidth()/2, padding+innerRingWidth, ringPaint);

            ringPaint.setAlpha(100);
            ringPaint.setStrokeWidth(3);
            //每6格刻度绘制一个深色刻度,即大刻度
            if (i%6==0) {
                canvas.drawLine(getWidth()/2, padding, getWidth()/2, padding+innerRingWidth+5, ringPaint);
            }

            //每三格刻度绘制一个文字
            if (i%3==0) {
                scoreTextPaint.setAlpha(180);
                float textWidth = scoreTextPaint.measureText(scores[i/3]);  //测量该文本宽度,需向左移动半个文本宽度以对齐
                canvas.drawText(scores[i/3], getWidth()/2-textWidth/2, padding+innerRingWidth+dp2px(getContext(), 12), scoreTextPaint);
            }

            canvas.rotate(totalAngle/totalDegreeScale, getWidth()/2, getHeight()/2); //每次画完从中心开始旋转画布单位刻度的弧度
        }

        canvas.restore(); //恢复角度
    }

    /**
     * 绘制中心文本
     */
    private void drawCenterText(Canvas canvas) {
        //绘制当前分数
        scoreTextPaint.setAlpha(255);
        scoreTextPaint.setTextSize(170);

        String curScore = String.valueOf(score);
        Rect scoreRect = new Rect();
        scoreTextPaint.getTextBounds(curScore, 0, curScore.length(), scoreRect);  //需左移文字宽度以居中
        canvas.drawText(curScore, getWidth()/2-scoreRect.width()/2, getHeight()/2, scoreTextPaint);

        //绘制BETA文字
        scoreTextPaint.setAlpha(150);
        scoreTextPaint.setTextSize(35);
        Rect betaRect= new Rect();
        String betaStr = "BETA";
        scoreTextPaint.getTextBounds(betaStr, 0, betaStr.length(), betaRect);  //beta需向坐上移动
        canvas.drawText(betaStr, getWidth()/2-betaRect.width()/2, getHeight()/2-scoreRect.height()-betaRect.height()/2, scoreTextPaint);

        //绘制信用等级文本
        scoreTextPaint.setAlpha(200);
        scoreTextPaint.setTextSize(75);
        Rect creditRect = new Rect();
        scoreTextPaint.getTextBounds(creditStr, 0, creditStr.length(), creditRect);
        canvas.drawText(creditStr, getWidth()/2-creditRect.width()/2, getHeight()/2+scoreRect.height()/2+20, scoreTextPaint);

        //绘制评估时间
        scoreTextPaint.setAlpha(150);
        scoreTextPaint.setTextSize(35);
        float timeStrWidth = scoreTextPaint.measureText("评估时间:2020.07.27");
        canvas.drawText("评估时间:2020.07.27", getWidth()/2-timeStrWidth/2, getHeight()/2+scoreRect.height()+10, scoreTextPaint);
    }

    /**
     * 绘制进度动画小圆点
     */
    private void drawDot(Canvas canvas) {
        scoreTextPaint.setAlpha(230);
        float x = (float) (getWidth()/2 + (getWidth()/2-outerRingWidth)*Math.sin(Math.toRadians(curProgressAngle)));
        float y = (float) (getHeight()/2 - (getWidth()/2-outerRingWidth)*Math.cos(Math.toRadians(curProgressAngle)));
        canvas.drawCircle(x, y, outerRingWidth, dotPaint);
    }

    /**
     * 启动指示器加载动画
     */
    private void startIndicatorAnim() {
        ValueAnimator valueAnimator = ValueAnimator.ofFloat(-105, stopAngle);
        valueAnimator.setDuration(ANIM_DURATION);
        valueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());
        valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                curProgressAngle = (float) animation.getAnimatedValue();

                postInvalidate();  //数值改变实时更新绘制
            }
        });
        valueAnimator.start();
    }

    /**
     * 设置信用水平,每一刻小刻度是7°
     * 在当前大刻度范围的结束角度,为之前刻度角度加上在该区间按比例得出的角度
     */
    private String showCreditLevel() {
        int startAngle = -105;
        int perLevelAngle = totalAngle/5;  //有5段大刻度
        String creditLevelStr = null;
        if (score < 350) {
            creditLevelStr = "信用较差";
            stopAngle = startAngle;
        } else if (score >= 350 && score < 550) {
            creditLevelStr = "信用较差";
            stopAngle = startAngle + (float)(score-350)/(550-350)*perLevelAngle;
        } else if (score >= 550 && score < 600) {
            creditLevelStr = "信用中等";
            stopAngle = startAngle + perLevelAngle + (float)(score-550)/(600-550)*perLevelAngle;
        } else if (score >= 600 && score < 650) {
            creditLevelStr = "信用良好";
            stopAngle = startAngle + perLevelAngle*2 + (float)(score-600)/(650-600)*perLevelAngle;
        } else if (score >= 650 && score < 700) {
            creditLevelStr = "信用优秀";
            stopAngle = startAngle + perLevelAngle*3 + (float)(score-650)/(700-650)*perLevelAngle;
        } else if (score >= 700 && score < 950) {
            creditLevelStr = "信用极好";
            stopAngle = startAngle + perLevelAngle*4 + (float)(score-700)/(950-700)*perLevelAngle;
        }
        return creditLevelStr;
    }

    public void runWithAnimation(int number){
        ObjectAnimator objectAnimator = ObjectAnimator.ofInt(this, "score", 0, number);
        objectAnimator.setDuration(ANIM_DURATION);
        objectAnimator.setInterpolator(new AccelerateDecelerateInterpolator());

        objectAnimator.start();
    }

    public int getScore() {
        return score;
    }

    public void setScore(int score) {
        this.score = score;
    }

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