自定义View之仿支付宝v9.9芝麻信用分仪表盘效果

iOS的芝麻信用分截图.png

前言

  • 灵感来自几天前看到简书一位作者的仿芝麻信用自定义View的文章,很不错,所以我换了一种方式来进行实现,写了旧版和新版芝麻信用分仪表盘的效果.

  • Github地址:HotBitmapGG/CreditSesameRingView

截图

  • 这是我做的效果,还是有点差距的,嘿嘿.


正文

  • 9.9版本芝麻信用分的实现

首先初始化各种画笔,默认的size,padding,小圆点.
(因为实在找不到原版芝麻信用的带点模糊效果的小圆点,所以只好用这个代替)

//View的默认大小
defaultSize = dp2px(250);
//默认Padding大小
arcDistance = dp2px(14);

//外层圆环画笔
mMiddleArcPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mMiddleArcPaint.setStrokeWidth(8);
mMiddleArcPaint.setColor(Color.WHITE);
mMiddleArcPaint.setStyle(Paint.Style.STROKE);
mMiddleArcPaint.setAlpha(80);

//内层圆环画笔
mInnerArcPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mInnerArcPaint.setStrokeWidth(30);
mInnerArcPaint.setColor(Color.WHITE);
mInnerArcPaint.setAlpha(80);
mInnerArcPaint.setStyle(Paint.Style.STROKE);

//正中间字体画笔
mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mTextPaint.setColor(Color.WHITE);
mTextPaint.setTextAlign(Paint.Align.CENTER);

//圆环大刻度画笔
mCalibrationPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mCalibrationPaint.setStrokeWidth(4);
mCalibrationPaint.setStyle(Paint.Style.STROKE);
mCalibrationPaint.setColor(Color.WHITE);
mCalibrationPaint.setAlpha(120);

//圆环小刻度画笔
mSmallCalibrationPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mSmallCalibrationPaint.setStrokeWidth(1);
mSmallCalibrationPaint.setStyle(Paint.Style.STROKE);
mSmallCalibrationPaint.setColor(Color.WHITE);
mSmallCalibrationPaint.setAlpha(130);

//圆环刻度文本画笔
mCalibrationTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mCalibrationTextPaint.setTextSize(30);
mCalibrationTextPaint.setColor(Color.WHITE);

//外层进度画笔
mArcProgressPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mArcProgressPaint.setStrokeWidth(8);
mArcProgressPaint.setColor(Color.WHITE);
mArcProgressPaint.setStyle(Paint.Style.STROKE);
mArcProgressPaint.setStrokeCap(Paint.Cap.ROUND);

//外层圆环上小圆点Bitmap画笔
mBitmapPaint = new Paint();
mBitmapPaint.setStyle(Paint.Style.FILL);
mBitmapPaint.setAntiAlias(true);

//初始化小圆点图片
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.ic_circle);
//当前点的实际位置
pos = new float[2];
//当前点的tangent值
tan = new float[2];
matrix = new Matrix();

代码很简单,就是各种初始化,往下看.

View的测量,主要在给设置warp_content时候给定一个默认宽高值.

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec){    
setMeasuredDimension(resolveMeasure(widthMeasureSpec, defaultSize),    
        resolveMeasure(heightMeasureSpec, defaultSize));}

//根据传入的值进行测量
public int resolveMeasure(int measureSpec, int defaultSize){    
int result = 0;    
int specSize = MeasureSpec.getSize(measureSpec);    
switch (MeasureSpec.getMode(measureSpec))   
 {      
   case MeasureSpec.UNSPECIFIED:          
   result = defaultSize;           
   break;       
   case MeasureSpec.AT_MOST:          
   //设置warp_content时设置默认值           
   result = Math.min(specSize, defaultSize);          
    break;        
  case MeasureSpec.EXACTLY:           
  //设置math_parent 和设置了固定宽高值           
  break;        
  default:           
  result = defaultSize;  
 }   
 return result;}

然后确定View的宽高后的回调方法.

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh){   
 super.onSizeChanged(w, h, oldw, oldh);   
 width = w;   
 height = h;    
 radius = width / 2;    
//外层圆环矩形
 mMiddleRect = new RectF(defaultPadding, defaultPadding,width - defaultPadding, height - defaultPadding);    
//内层圆环矩形
 mInnerRect = new RectF(defaultPadding + arcDistance, defaultPadding + arcDistance,width - defaultPadding - arcDistance, height - defaultPadding - arcDistance);    
// 外层进度矩形
 mMiddleProgressRect = new RectF(defaultPadding, defaultPadding,width - defaultPadding, height - defaultPadding);
}

这里就是初始化圆弧所需要的矩形实现,下边开始进行重点,绘制,
绘制外层的圆弧,很简单, 圆弧的起始角度,角度.

private void drawMiddleArc(Canvas canvas){    
canvas.drawArc(mMiddleRect, mStartAngle, mEndAngle, false, mMiddleArcPaint);
}

绘制内层圆弧

private void drawInnerArc(Canvas canvas){   
 canvas.drawArc(mInnerRect, mStartAngle, mEndAngle, false, mInnerArcPaint);
}

绘制内层圆弧上的小刻度,画布旋转到圆弧左下角起点,计算出每条刻度线的起始点后,整个圆弧是210度,
每6角度绘制一条刻度线.

private void drawSmallCalibration(Canvas canvas){    
 //旋转画布    
 canvas.save();   
 canvas.rotate(-105, radius, radius);    
 //计算刻度线的起点结束点    
 int startDst = (int) (defaultPadding + arcDistance - mInnerArcPaint.getStrokeWidth() / 2 - 1);   
 int endDst = (int) (startDst + mInnerArcPaint.getStrokeWidth());    
 for (int i = 0; i <= 35; i++)    {        
 //每旋转6度绘制一个小刻度       
 canvas.drawLine(radius, startDst, radius, endDst, mSmallCalibrationPaint);        
 canvas.rotate(6, radius, radius);    
}    
canvas.restore();
}

绘制内层圆弧上的大刻度,350, 550, 600,650, 700, 950,对应的信用分值,
一样旋转画布,计算刻度线的起始点,计算出每次旋转的角度,每35度旋转一次,依次绘制对应的大刻度线,
然后绘制对应的文本内容,使用paint的measureText方法测量出文本的长度,依次绘制对应的文本内容.

private void drawCalibrationAndText(Canvas canvas){   
 //旋转画布进行绘制对应的刻度    
 canvas.save();   
 canvas.rotate(-105, radius, radius);    
 //计算刻度线的起点结束点    
 int startDst = (int) (defaultPadding + arcDistance - mInnerArcPaint.getStrokeWidth() / 2 - 1);    
 int endDst = (int) (startDst + mInnerArcPaint.getStrokeWidth());    
 //刻度旋转的角度    
 int rotateAngle = 210 / 10;    
 for (int i = 1; i < 12; i++)    {       
 if (i % 2 != 0)        
 {            
 canvas.drawLine(radius, startDst, radius, endDst, mCalibrationPaint);       
 }        
 // 测量文本的长度        
float textLen = mCalibrationTextPaint.measureText(sesameStr[i - 1]); 
canvas.drawText(sesameStr[i - 1], radius - textLen / 2, endDst + 40, mCalibrationTextPaint);        
canvas.rotate(rotateAngle, radius, radius);    
}    
canvas.restore();}

绘制中间的信用分值,信用等级,评估时间等文本,这个比较简单,直接drawText,依次高低排列绘制即可.

private void drawCenterText(Canvas canvas){    
 //绘制Logo    
 mTextPaint.setTextSize(30);    
 canvas.drawText("BETA", radius, radius - 130, mTextPaint);   
 //绘制信用分数    
 mTextPaint.setTextSize(200);   
 mTextPaint.setStyle(Paint.Style.STROKE);    
 canvas.drawText(String.valueOf(mMinNum), radius, radius + 70, mTextPaint);   
 //绘制信用级别    
 mTextPaint.setTextSize(80);    
 canvas.drawText(sesameLevel, radius, radius + 160, mTextPaint);    
 //绘制评估时间    
 mTextPaint.setTextSize(30);    
 canvas.drawText(evaluationTime, radius, radius + 205, mTextPaint);
}

绘制最外层的进度,这里使用的Path添加要绘制的圆弧,因为需要去不断的计算坐标点,主要用到了PathMeasure这个类,将绘制的圆弧加入到path中,

当前点的实际位置
private float[] pos;

当前的tangent值
private float[] tan;

获取路径的终点的正切值和坐标,然后根据坐标点绘制小圆点
PathMeasure pathMeasure = new PathMeasure(path, false);
pathMeasure.getPosTan(pathMeasure.getLength() * 1, pos, tan);

关于PathMeasure,推荐看GcsSloop/AndroidNote ,我也是跟着这个笔记学习的自定义控件.

private void drawRingProgress(Canvas canvas){   

 Path path = new Path();    
 path.addArc(mMiddleProgressRect, mStartAngle, mCurrentAngle);
 PathMeasure pathMeasure = new PathMeasure(path, false);
 pathMeasure.getPosTan(pathMeasure.getLength() * 1, pos, tan);    
 matrix.reset();    
 matrix.postTranslate(pos[0] - bitmap.getWidth() / 2, pos[1] - bitmap.getHeight() / 2);
 canvas.drawPath(path, mArcProgressPaint);    
 //起始角度不为0时候才进行绘制小圆点    
 if (mCurrentAngle == 0)       
        return;    
 canvas.drawBitmap(bitmap, matrix, mBitmapPaint);   
 mBitmapPaint.setColor(Color.WHITE);    
 canvas.drawCircle(pos[0], pos[1], 8, mBitmapPaint);
}

好了,到这里所有绘制完毕了,接下来让圆弧进度条动起来吧,使用ValueAnimator,
进度条动画定义了圆弧进度条的开始角度mCurrentAngle,圆弧角度mTotalAngle,
数值动画定义了初始化minNum=0,maxNum根据传入的数值进行计算.

public void startAnim(){    
 ValueAnimator mAngleAnim = ValueAnimator.ofFloat(mCurrentAngle, mTotalAngle); 
 mAngleAnim.setInterpolator(new AccelerateDecelerateInterpolator()); 
 mAngleAnim.setDuration(3000);    
 mAngleAnim.addUpdateListener(new ValueAnimator.AnimatorUpdateListener(){        
  @Override      
 public void onAnimationUpdate(ValueAnimator valueAnimator){           
   mCurrentAngle = (float) valueAnimator.getAnimatedValue();            
   postInvalidate();        

}    
});   
  mAngleAnim.start();    

  ValueAnimator mNumAnim = ValueAnimator.ofInt(mMinNum, mMaxNum);
  mNumAnim.setDuration(3000);    
  mNumAnim.setInterpolator(new LinearInterpolator());    
  mNumAnim.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { 
    @Override        
 public void onAnimationUpdate(ValueAnimator valueAnimator){            
  mMinNum = (int) valueAnimator.getAnimatedValue();           
  postInvalidate();       
 
}    
});   
 mNumAnim.start();}

最后根据传入的信用分值计算圆弧进度条所到的角度.

public void setSesameValues(int values){   
 if (values <= 350){       
 mMaxNum = values;        
 mTotalAngle = 0f;        
 sesameLevel = "信用较差";        
 evaluationTime = "评估时间:" + getCurrentTime();    
 } else if (values <= 550){        
 mMaxNum = values;        
 mTotalAngle = (values - 350) * 80 / 400f + 2;        
 sesameLevel = "信用较差";        
 evaluationTime = "评估时间:" + getCurrentTime();    
 } else if (values <= 700)
 {        
 mMaxNum = values;        
 if (values > 550 && values <= 600){           
  sesameLevel = "信用中等";        
 } else if (values > 600 && values <= 650){            
 sesameLevel = "信用良好";        
 } else {            
 sesameLevel = "信用优秀";        
 }        
 mTotalAngle = (values - 550) * 120 / 150f + 43;       
 evaluationTime = "评估时间:" + getCurrentTime();    
 } else if (values <= 950){        
 mMaxNum = values;        
 mTotalAngle = (values - 700) * 40 / 250f + 170;        
 sesameLevel = "信用极好";        
 evaluationTime = "评估时间:" + getCurrentTime();    
 } else{        
 mTotalAngle = 240f;    
 }    
 startAnim();
}

最后

这篇文章只分析了新版的实现过程,旧版的的实现思路也差不多,代码也不复杂,可以直接看源码实现,由于本人文章写的不多,所以感觉把代码的实现用语言来组织还真挺难的,所以写的不好的地方还请各位大大见谅,有问题可联系我,最后希望如果觉得还可以的,请给个star, 谢谢啦.

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

推荐阅读更多精彩内容