Android仿支付宝上芝麻信用分雷达图

一、首先看下支付宝上芝麻信用分的效果图:

二、思路

1、确定雷达图中心点坐标

2、绘制多边形及连接线

3、根据维度值绘制覆盖区域

4、绘制分数

5、绘制每个维度的标题文字和图标

三、实现

获取布局的中心坐标

onSizeChanged(int w, int h, int oldw, int oldh)方法里面,根据View的长宽,计算出雷达图的半径(这里取布局宽高最小值的四分之一,可以自定义),获取整个布局的中心坐标。

public class CreditScoreView extends View {

 //数据个数
 private int dataCount = 5;
 //每个角的弧度
 private float radian = (float) (Math.PI * 2 / dataCount);
 //雷达图半径
 private float radius;
 //中心X坐标
 private int centerX;
 //中心Y坐标
 private int centerY;
 //各维度标题
 private String[] titles = {"履约能力", "信用历史", "人脉关系", "行为偏好", "身份特质"};
 //各维度图标
 private int[] icons = {R.mipmap.ic_performance, R.mipmap.ic_history, R.mipmap.ic_contacts,
   R.mipmap.ic_predilection, R.mipmap.ic_identity};
 //各维度分值
 private float[] data = {170, 180, 160, 170, 180};
 //数据最大值
 private float maxValue = 190;
 //雷达图与标题的间距
 private int radarMargin = DensityUtils.dp2px(getContext(), 15);
 //雷达区画笔
 private Paint mainPaint;
 //数据区画笔
 private Paint valuePaint;
 //分数画笔
 private Paint scorePaint;
 //标题画笔
 private Paint titlePaint;
 //图标画笔
 private Paint iconPaint;
 //分数大小
 private int scoreSize = DensityUtils.dp2px(getContext(), 28);
 //标题文字大小
 private int titleSize = DensityUtils.dp2px(getContext(), 13);

 ...

 @Override
 protected void onSizeChanged(int w, int h, int oldw, int oldh) {
  //雷达图半径
  radius = Math.min(h, w) / 2 * 0.5f;
  //中心坐标
  centerX = w / 2;
  centerY = h / 2;
  postInvalidate();
  super.onSizeChanged(w, h, oldw, oldh);
 }

 ...
}

绘制多边形和连接线

主要看下getPoint方法,此方法封装了获取雷达图上各个点坐标的计算逻辑。

/**
 * 绘制多边形
 *
 * @param canvas 画布
 */
private void drawPolygon(Canvas canvas) {
 Path path = new Path();
 for (int i = 0; i < dataCount; i++) {
  if (i == 0) {
   path.moveTo(getPoint(i).x, getPoint(i).y);
  } else {
   path.lineTo(getPoint(i).x, getPoint(i).y);
  }
 }

 //闭合路径
 path.close();
 canvas.drawPath(path, mainPaint);
}

/**
 * 绘制连接线
 *
 * @param canvas 画布
 */
private void drawLines(Canvas canvas) {
 Path path = new Path();
 for (int i = 0; i < dataCount; i++) {
  path.reset();
  path.moveTo(centerX, centerY);
  path.lineTo(getPoint(i).x, getPoint(i).y);
  canvas.drawPath(path, mainPaint);
 }
}

getPoint方法,参数radarMarginpercent在此步骤赋予默认值。

/**
 * 获取雷达图上各个点的坐标
 *
 * @param position 坐标位置(右上角为0,顺时针递增)
 * @return 坐标
 */
private Point getPoint(int position) {
 return getPoint(position, 0, 1);
}

/**
 * 获取雷达图上各个点的坐标(包括维度标题与图标的坐标)
 *
 * @param position 坐标位置
 * @param radarMargin 雷达图与维度标题的间距
 * @param percent  覆盖区的的百分比
 * @return 坐标
 */
private Point getPoint(int position, int radarMargin, float percent) {
 int x = 0;
 int y = 0;

 if (position == 0) {
  x = (int) (centerX + (radius + radarMargin) * Math.sin(radian) * percent);
  y = (int) (centerY - (radius + radarMargin) * Math.cos(radian) * percent);

 } else if (position == 1) {
  x = (int) (centerX + (radius + radarMargin) * Math.sin(radian / 2) * percent);
  y = (int) (centerY + (radius + radarMargin) * Math.cos(radian / 2) * percent);

 } else if (position == 2) {
  x = (int) (centerX - (radius + radarMargin) * Math.sin(radian / 2) * percent);
  y = (int) (centerY + (radius + radarMargin) * Math.cos(radian / 2) * percent);

 } else if (position == 3) {
  x = (int) (centerX - (radius + radarMargin) * Math.sin(radian) * percent);
  y = (int) (centerY - (radius + radarMargin) * Math.cos(radian) * percent);

 } else if (position == 4) {
  x = centerX;
  y = (int) (centerY - (radius + radarMargin) * percent);
 }

 return new Point(x, y);
}


多边形和连接线

绘制覆盖区域

/**
 * 绘制覆盖区域
 *
 * @param canvas 画布
 */
private void drawRegion(Canvas canvas) {
 Path path = new Path();

 for (int i = 0; i < dataCount; i++) {
  //计算百分比
  float percent = data[i] / maxValue;
  int x = getPoint(i, 0, percent).x;
  int y = getPoint(i, 0, percent).y;
  if (i == 0) {
   path.moveTo(x, y);
  } else {
   path.lineTo(x, y);
  }
 }

 //绘制填充区域的边界
 path.close();
 valuePaint.setStyle(Paint.Style.STROKE);
 canvas.drawPath(path, valuePaint);

 //绘制填充区域
 valuePaint.setStyle(Paint.Style.FILL_AND_STROKE);
 canvas.drawPath(path, valuePaint);
}


覆盖区域

绘制分数

/**
 * 绘制分数
 *
 * @param canvas 画布
 */
private void drawScore(Canvas canvas) {
 int score = 0;
 //计算总分
 for (int i = 0; i < dataCount; i++) {
  score += data[i];
 }
 canvas.drawText(score + "", centerX, centerY + scoreSize / 2, scorePaint);
}


分数

绘制标题

/**
 * 绘制标题
 *
 * @param canvas 画布
 */
private void drawTitle(Canvas canvas) {
 for (int i = 0; i < dataCount; i++) {
  int x = getPoint(i, radarMargin, 1).x;
  int y = getPoint(i, radarMargin, 1).y;

  Bitmap bitmap = BitmapFactory.decodeResource(getResources(), icons[i]);
  int iconHeight = bitmap.getHeight();
  float titleWidth = titlePaint.measureText(titles[i]);

  //底下两个角的坐标需要向下移动半个图片的位置(1、2)
  if (i == 1) {
   y += (iconHeight / 2);
  } else if (i == 2) {
   x -= titleWidth;
   y += (iconHeight / 2);
  } else if (i == 3) {
   x -= titleWidth;
  } else if (i == 4) {
   x -= titleWidth / 2;
  }
  canvas.drawText(titles[i], x, y, titlePaint);
 }
}


标题

绘制图标

/**
 * 绘制图标
 *
 * @param canvas 画布
 */
private void drawIcon(Canvas canvas) {
 for (int i = 0; i < dataCount; i++) {
  int x = getPoint(i, radarMargin, 1).x;
  int y = getPoint(i, radarMargin, 1).y;

  Bitmap bitmap = BitmapFactory.decodeResource(getResources(), icons[i]);
  int iconWidth = bitmap.getWidth();
  int iconHeight = bitmap.getHeight();
  float titleWidth = titlePaint.measureText(titles[i]);

  //上面获取到的x、y坐标是标题左下角的坐标
  //需要将图标移动到标题上方居中位置
  if (i == 0) {
   x += (titleWidth - iconWidth) / 2;
   y -= (iconHeight + getTextHeight(titlePaint));
  } else if (i == 1) {
   x += (titleWidth - iconWidth) / 2;
   y -= (iconHeight / 2 + getTextHeight(titlePaint));
  } else if (i == 2) {
   x -= (iconWidth + (titleWidth - iconWidth) / 2);
   y -= (iconHeight / 2 + getTextHeight(titlePaint));
  } else if (i == 3) {
   x -= (iconWidth + (titleWidth - iconWidth) / 2);
   y -= (iconHeight + getTextHeight(titlePaint));
  } else if (i == 4) {
   x -= iconWidth / 2;
   y -= (iconHeight + getTextHeight(titlePaint));
  }

  canvas.drawBitmap(bitmap, x, y, titlePaint);
 }
}
/**
 * 获取文本的高度
 *
 * @param paint 文本绘制的画笔
 * @return 文本高度
 */
private int getTextHeight(Paint paint) {
 Paint.FontMetrics fontMetrics = paint.getFontMetrics();
 return (int) (fontMetrics.descent - fontMetrics.ascent);
}


图标

总结

好了,到这里主要的绘制工作就完成了,有些图标实在找不到,就用相似的代替了。希望这篇文章的内容对各位Android开发者们能有所帮助,如果有疑问大家可以留言交流。

(0)

相关推荐

  • Android雷达扫描动态界面制作

    先看看效果图: 源码: package com.zihao.radar; import android.app.Activity; import android.os.Bundle; import android.view.Window; import android.view.WindowManager; import com.zihao.radar.view.RadarView; public class MainActivity extends Activity { private Rad

  • 基于Android自定义控件实现雷达效果

    如何制作出类似雷达扫描的效果,具体方法如下 一.效果图 二.实现思路 1.自定义控件RadarView用来画雷达的效果图,可以自定义属性包括 backgroundColor:背景颜色 circleNum:圆的数量 startColor:开始颜色 endColor:结束颜色 lineColor:线的颜色 2.通过Handler循环发送消息到MessageQueue中,将mRotate加3,使Matrix旋转mRotate,重绘雷达扫描的圆. 3.通过梯度渐变扫描渲染器SweepGradient,在

  • Android仿微信雷达扫描效果的实现方法

    本文主要给大家介绍的是关于Android实现微信雷达扫描效果的相关内容,分享出来供大家参考学习,下面来看看详细的介绍: 废话不多说 先上图(用AS录制的 转换工具不是很好 所以看得效果不是很好) 效果图 示例代码 Activity 代码 public class ShapeDrawableActivity extends AppCompatActivity { private ImageView ivLightbeam; private ObjectAnimator radarScanAnim;

  • Android仿微信雷达辐射搜索好友(逻辑清晰实现简单)

    不知不觉这个春节也已经过完了,遗憾家里没网,没能及时给大家送上祝福,今天回到深圳,明天就要上班了,小伙伴们是不是和我一样呢?今天讲的是一个大家都见过的动画,雷达搜索好友嘛,原理也十分的简单,你看完我的分析,也会觉得很简单了,国际惯例,无图无真相,我们先看看效果图,对了,真 测试机送人了,所讲这段时间应该一直用模拟器显示吧! 仿微信雷达扫描,仿安卓微信.云播雷达扫描动画效果点击中间的黑色圆圈开始扫描动画,再次点击复位,需要这种效果的朋友可以自己下载看一下. 效果图如下所示: 这个界面相信大家都认识

  • Android Shader应用开发之雷达扫描效果

    本文实例为大家分享了Android雷达扫描效果的具体代码,供大家参考,具体内容如下 效果图 知识点提要 Shader 矩阵matrix 属性动画 ShaderView3 package com.example.apple.shaderdemo; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.

  • Android动画之雷达扫描效果

    我们首先看一下效果图,有个整体的印象 好了,为了便于理解,这里就按照动画所见内容依次展开来说 准备 这里决定采用canvas(画布)和paint(画笔)实现了这个简单动画控件. 由图片可以看到有两条交叉的十字线.几个圆圈和一些白点,那么首先定义一下所需的画笔,画布及一些数据 setBackgroundColor(Color.TRANSPARENT); //宽度=5,抗锯齿,描边效果的白色画笔 mPaintLine = new Paint(); mPaintLine.setStrokeWidth(

  • Android仿微信、QQ附近好友雷达扫描效果

    1.概述 最近一直到在带实习生,因为人比较多,所以很长一段时间没有更新博客了,今天更新一篇雷达扫描附近好友效果,以后尽量每周更新一篇,先看一下效果: 2.实现 1.效果分析 效果分为两个部分,一个是上半部分的自定义RadarView,还有就是下半部分的ViewPager,至于怎么做到缩放和背景虚化的效果大家可以去看看LazyViewPager这里不详细介绍,这里主要实现扫描效果部分. 2.扫描效果实现 2.1自定义RadarView在onDraw()方法中画六个圆圈,至于圆圈的半径是多少我们需要

  • Android编程简单实现雷达扫描效果

    本文实例讲述了Android编程简单实现雷达扫描效果.分享给大家供大家参考,具体如下: 在eoe看到有一篇关于雷达扫描的文章,然后看了下,很简单,但是觉得还有很多可以优化的地方,下面贴出来 package com.example.wave; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Matrix; im

  • Android自定义ViewGroup实现绚丽的仿支付宝咻一咻雷达脉冲效果

    去年春节的时候支付宝推行的集福娃活动着实火的不能再火了,更给力的是春晚又可以全民参与咻一咻集福娃活动,集齐五福就可平分亿元大红包,只可惜没有敬业福--那时候在家没事写了个咻一咻插件,只要到了咻一咻的时间点插件就可以自动的点击咻一咻来咻红包,当时只是纯粹练习这部分技术代码没有公开,后续计划写篇关于插件这方面的文章,扯远了(*^__^*) --我们知道在支付宝的咻一咻页面有个雷达扩散的动画效果,当时感觉动画效果非常棒,于是私下尝试着实现了类似的效果,后来在github发现有大神也写有类似效果,于是读

  • Android仿支付宝上芝麻信用分雷达图

    一.首先看下支付宝上芝麻信用分的效果图: 二.思路 1.确定雷达图中心点坐标 2.绘制多边形及连接线 3.根据维度值绘制覆盖区域 4.绘制分数 5.绘制每个维度的标题文字和图标 三.实现 获取布局的中心坐标 在onSizeChanged(int w, int h, int oldw, int oldh)方法里面,根据View的长宽,计算出雷达图的半径(这里取布局宽高最小值的四分之一,可以自定义),获取整个布局的中心坐标. public class CreditScoreView extends

  • Android自定义View仿支付宝芝麻信用分仪表盘

    先看下iOS的芝麻信用分截图 这是我做的效果,还是有点差距的 支付宝9.9版本芝麻信用分的实现 首先初始化各种画笔,默认的size,padding,小圆点. (因为实在找不到原版芝麻信用的带点模糊效果的小圆点,所以只好用这个代替) //View的默认大小 defaultSize = dp2px(250); //默认Padding大小 arcDistance = dp2px(14); //外层圆环画笔 mMiddleArcPaint = new Paint(Paint.ANTI_ALIAS_FLA

  • js canvas仿支付宝芝麻信用分仪表盘

    这是一个仿支付宝芝麻信用分的一个canvas,其实就是一个动画仪表盘. 首先, 上原图: 这个是在下支付宝上的截图,分低各位见笑了.然后看下我用canvas实现的效果图: <canvas id="canvas" width="400" height="700" data-score='724'></canvas> <!-- 设置data-score,分数区间[400, 900] --> 唉,总感觉不像.这个是G

  • Android仿支付宝笑脸刷新加载动画的实现代码

    看到支付宝的下拉刷新有一个笑脸的动画,因此自己也动手实现一下.效果图如下: 一.总体思路 1.静态部分的笑脸. 这一部分的笑脸就是一个半圆弧,加上两颗眼睛,这部分比较简单,用于一开始的展示. 2.动态笑脸的实现. 2.1.先是从底部有一个圆形在运动,运动在左眼位置时把左眼给绘制,同时圆形继续运动,运动到右眼位置时绘制右眼,圆形继续运动到最右边的位置. 2.2.当上面的圆形运动到最右边时候,开始不断绘制脸,从右向左,脸不断增长,这里脸设置为接近半个圆形的大小. 2.3.当脸画完的时候,开始让脸旋转

  • Android 仿支付宝中的余额宝收益进度条

    一. 看效果 二.上代码 package com.framework.widget; import android.app.Activity; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import androi

  • Android仿支付宝的头部伸缩动画效果

    Android5.0推出的MaterialDesign库包含了处理头部工具栏的多个控件,不但允许自定义顶部导航栏,而且导航栏高度是可以伸缩的.如此一来,一方面导航栏能够放得下更多控件,另一方面在用户想看具体内容时也能腾出更多的屏幕空间. 这么说可能比较抽象,那就先来看看两张导航栏的效果图,第一张是导航栏完全展开时的界面,此时页面头部的导航栏占据了较大部分的高度: 第二张是导航栏完全收缩时的界面,此时头部导航栏只剩矮矮的一个长条. 看起来很眼熟是不是,上面的截图正是仿支付宝首页的头部效果.如果你熟

  • Android仿支付宝自定义密码输入框及安全键盘(密码键盘)

     0.前言 之前做过的项目里有运用到一个支付场景:用户办理业务时需要输入交易密码,并且可根据平台下发的支付方式进行选择.这与支付宝的密码输入方式十分相似,如果使用Android系统或者第三方软件的键盘,会有密码泄露的风险.因此,大多数的应用软件使用的是自定义的密码输入框及安全键盘. 由于密码输入方式需要实现一个从底部弹出的效果,因此总体上决定采用BottomSheetDialog来进行封装,同时为了提高安全性,还应该随机生成键盘上的数字,界面如下图所示:   首先新建一个PasswordInpu

  • Android仿支付宝中余额宝的数字动画效果

    实现效果图: 下面是具体代码,可直接复制: package com.lcw.rabbit.widget; import android.animation.ObjectAnimator; import android.content.Context; import android.text.TextUtils; import android.util.AttributeSet; import android.view.animation.AccelerateDecelerateInterpola

  • Android 仿支付宝密码输入框效果

    模仿支付宝输入效果,实现很简单,就是画个矩形框和圆形,其他的通过组合view来实现所有功能,虽然简单但是封装起来,方便以后使用,也分享一下,希望对别人也有点帮助. 1.如何使用,可以设置自己的进入退出动画,不设置则没有动画效果,自己觉得封装之后还是非常用好的. private MyInputPwdUtil myInputPwdUtil; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(sa

  • Android仿支付宝微信支付密码界面弹窗封装dialog

    一,功能效果 二,实现过程 1,先写xml文件:dialog_keyboard.xml 注意事项 (1),密码部分用的是一个线性布局中6个TextView,并设置android:inputType="numberPassword",外框是用的一个有stroke属性的shape, (2),1-9数字是用的recycleview ,每个item的底部和右边有1dp的黑线,填充后形成分割线. (3),recycleview 要设置属性  android:overScrollMode=&quo

随机推荐