<com...SignChartView
android:id="@+id/signChart"
android:layout_width="265dp"
android:layout_height="265dp"
attrview:centerTextColor="@color/colorWhite"
attrview:centerTextSize="@dimen/font40"
attrview:circleWidth="20dp"
attrview:dataTextColor="@color/colorWhite"
attrview:dataTextSize="@dimen/font14" />
package com.signin;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.view.View; import java.util.Random; /**
* Created by xxxx on 2018/8/29.
*/ public class SignChartView extends View { /**
* 使用wrap_content时默认的尺寸
*/
private static final int DEFAULT_WIDTH = 220;
private static final int DEFAULT_HEIGHT = 220; /**
* 中心坐标
*/
private int centerX;
private int centerY; /**
* 半径
*/
private float radius; /**
* 弧形外接矩形
*/
private RectF rectF; /**
* 中间文本的大小
*/
private Rect centerTextBound = new Rect(); /**
* 数据文本的大小
*/
private Rect dataTextBound = new Rect(); /**
* 扇形画笔
*/
private Paint mArcPaint; /**
* 中心文本画笔
*/
private Paint centerTextPaint; /**
* 数据画笔
*/
private Paint dataPaint; /**
* 数据源数字数组
*/
private int[] numbers; /**
* 数据源名称数组
*/
private String[] names; /**
* 数据源总和
*/
private int sum; /**
* 颜色数组
*/
private int[] colors; private Random random = new Random(); //自定义属性 Start /**
* 中间字体大小
*/
private float centerTextSize = 80; /**
* 数据字体大小
*/
private float dataTextSize = 30; /**
* 中间字体颜色
*/
private int centerTextColor = Color.BLACK; /**
* 数据字体颜色
*/
private int dataTextColor = Color.BLACK; /**
* 圆圈的宽度
*/
private float circleWidth = 100; //自定义属性 End public SignChartView(Context context) {
super(context);
init();
} public SignChartView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
} public SignChartView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.PieView);
centerTextSize = typedArray.getDimension(R.styleable.PieView_centerTextSize, centerTextSize);
dataTextSize = typedArray.getDimension(R.styleable.PieView_dataTextSize, dataTextSize);
circleWidth = typedArray.getDimension(R.styleable.PieView_circleWidth, circleWidth);
centerTextColor = typedArray.getColor(R.styleable.PieView_centerTextColor, centerTextColor);
dataTextColor = typedArray.getColor(R.styleable.PieView_dataTextColor, dataTextColor);
typedArray.recycle();
init();
} /**
* 初始化
*/
private void init() {
mArcPaint = new Paint();
mArcPaint.setStrokeWidth(circleWidth);
mArcPaint.setAntiAlias(true);
mArcPaint.setStyle(Paint.Style.STROKE); centerTextPaint = new Paint();
centerTextPaint.setTextSize(centerTextSize);
centerTextPaint.setAntiAlias(true);
centerTextPaint.setTextAlign(Paint.Align.CENTER);
centerTextPaint.setColor(centerTextColor); dataPaint = new Paint();
dataPaint.setStrokeWidth(2);
dataPaint.setTextSize(dataTextSize);
dataPaint.setAntiAlias(true);
dataPaint.setColor(dataTextColor);
} @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int measureWidthSize = MeasureSpec.getSize(widthMeasureSpec);
int measureHeightSize = MeasureSpec.getSize(heightMeasureSpec);
int measureWidthMode = MeasureSpec.getMode(widthMeasureSpec);
int measureHeightMode = MeasureSpec.getMode(heightMeasureSpec);
if (measureWidthMode == MeasureSpec.AT_MOST
&& measureHeightMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(DEFAULT_WIDTH, DEFAULT_HEIGHT);
} else if (measureWidthMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(DEFAULT_WIDTH, measureHeightSize);
} else if (measureHeightMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(measureWidthSize, DEFAULT_HEIGHT);
}
} @Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
centerX = getMeasuredWidth() / 2;
centerY = getMeasuredHeight() / 2;
//设置半径为宽高最小值的1/4
radius = Math.min(getMeasuredWidth(), getMeasuredHeight()) / 4;
//设置扇形外接矩形
rectF = new RectF(centerX - radius,
centerY - radius,
centerX + radius,
centerY + radius);
} @Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
calculateAndDraw(canvas);
} /**
* 计算比例并且绘制扇形和数据
*/
private void calculateAndDraw(Canvas canvas) {
if (numbers == null || numbers.length == 0) {
return;
}
//扇形开始度数
int startAngle = 0;
//所占百分比
float percent;
//所占度数
float angle;
for (int i = 0; i < numbers.length; i++) {
percent = numbers[i] / (float) sum;
//获取百分比在360中所占度数
if (i == numbers.length - 1) {//保证所有度数加起来等于360
angle = 360 - startAngle;
} else {
angle = (float) Math.ceil(percent * 360);
}
//绘制第i段扇形
drawArc(canvas, startAngle, angle, colors[i]);
startAngle += angle; //绘制数据
if (numbers[i] <= 0) {
continue;
}
//当前弧线中心点相对于纵轴的夹角度数,由于扇形的绘制是从三点钟方向开始,所以加90
float arcCenterDegree = 90 + startAngle - angle / 2;
drawData(canvas, arcCenterDegree, i, percent);
}
//绘制中心数字总和
canvas.drawText(sum + "", centerX, centerY + centerTextBound.height() / 2, centerTextPaint);
} /**
* 计算每段弧度的中心坐标
*
* @param degree 当前扇形中心度数
*/
private float[] calculatePosition(float degree) {
//由于Math.sin(double a)中参数a不是度数而是弧度,所以需要将度数转化为弧度
//而Math.toRadians(degree)的作用就是将度数转化为弧度
//sin 一二正,三四负 sin(180-a)=sin(a)
//扇形弧线中心点距离圆心的x坐标
float x = (float) (Math.sin(Math.toRadians(degree)) * radius);
//cos 一四正,二三负
//扇形弧线中心点距离圆心的y坐标
float y = (float) (Math.cos(Math.toRadians(degree)) * radius); //每段弧度的中心坐标(扇形弧线中心点相对于view的坐标)
float startX = centerX + x;
float startY = centerY - y; float[] position = new float[2];
position[0] = startX;
position[1] = startY;
return position;
} /**
* 绘制数据
*
* @param canvas 画布
* @param degree 第i段弧线中心点相对于纵轴的夹角度数
* @param i 第i段弧线
* @param percent 数据百分比
*/
private void drawData(Canvas canvas, float degree, int i, float percent) {
//弧度中心坐标
float startX = calculatePosition(degree)[0];
float startY = calculatePosition(degree)[1]; //获取名称文本大小
dataPaint.getTextBounds(names[i], 0, names[i].length(), dataTextBound);
// //绘制名称数据,20为纵坐标偏移量
// canvas.drawText(names[i],
// startX - dataTextBound.width() / 2,
// startY + dataTextBound.height() / 2 - 20,
// dataPaint);
//
//
// //拼接百分比并获取文本大小
// DecimalFormat df = new DecimalFormat("0.0");
// String percentString = df.format(percent * 100) + "%";
// dataPaint.getTextBounds(percentString, 0, percentString.length(), dataTextBound);
//
// //绘制百分比数据,20为纵坐标偏移量
// canvas.drawText(percentString,
// startX - dataTextBound.width() / 2,
// startY + dataTextBound.height() * 2 - 20,
// dataPaint);
} /**
* 绘制扇形
*
* @param canvas 画布
* @param startAngle 开始度数
* @param angle 扇形的度数
* @param color 颜色
*/
private void drawArc(Canvas canvas, float startAngle, float angle, int color) {
mArcPaint.setColor(color);
//-0.5和+0.5是为了让每个扇形之间没有间隙
canvas.drawArc(rectF, startAngle - 0.5f, angle + 0.5f, false, mArcPaint);
} /**
* 生成随机颜色
*/
private int randomColor() {
int red = random.nextInt(256);
int green = random.nextInt(256);
int blue = random.nextInt(256);
return Color.rgb(red, green, blue);
} /**
* 设置数据
*
* @param numbers 数字数组
* @param names 名称数组
*/
public void setData(int[] numbers, String[] names) {
sum = 0;
if (numbers == null || numbers.length == 0 || names == null || names.length == 0) {
return;
}
if (numbers.length != names.length) {
//名称个数与数字个数不相等
return;
}
this.numbers = numbers;
this.names = names;
colors = new int[numbers.length];
for (int i = 0; i < this.numbers.length; i++) {
//计算总和
sum += numbers[i];
//随机颜色
if(numbers.length == 3){
colors[0] = Color.rgb(43,218,97);
colors[1] = Color.rgb(43,218,204);
colors[2] = Color.rgb(228,91,73);
}else{
colors[i] = randomColor();
} }
//计算总和数字的宽高
centerTextPaint.getTextBounds(sum + "", 0, (sum + "").length(), centerTextBound);
invalidate();
}
}

  

Android 自定义圆形图表的更多相关文章

  1. android 自定义动画

    android自定义动画注意是继承Animation,重写里面的initialize和applyTransformation,在initialize方法做一些初始化的工作,在applyTransfor ...

  2. Android自定义View 画弧形,文字,并增加动画效果

    一个简单的Android自定义View的demo,画弧形,文字,开启一个多线程更新ui界面,在子线程更新ui是不允许的,但是View提供了方法,让我们来了解下吧. 1.封装一个抽象的View类   B ...

  3. Android自定义View4——统计图View

    1.介绍 周末在逛慕课网的时候,看到了一张学习计划报告图,详细记录了自己一周的学习情况,天天都是0节课啊!正好在学习Android自定义View,于是就想着自己去写了一个,这里先给出一张慕课网的图,和 ...

  4. (转)[原] Android 自定义View 密码框 例子

    遵从准则 暴露您view中所有影响可见外观的属性或者行为. 通过XML添加和设置样式 通过元素的属性来控制其外观和行为,支持和重要事件交流的事件监听器 详细步骤见:Android 自定义View步骤 ...

  5. Android 自定义View合集

    自定义控件学习 https://github.com/GcsSloop/AndroidNote/tree/master/CustomView 小良自定义控件合集 https://github.com/ ...

  6. Android 自定义View (五)——实践

    前言: 前面已经介绍了<Android 自定义 view(四)-- onMeasure 方法理解>,那么这次我们就来小实践下吧 任务: 公司现有两个任务需要我完成 (1)监测液化天然气液压 ...

  7. Android 自定义 view(四)—— onMeasure 方法理解

    前言: 前面我们已经学过<Android 自定义 view(三)-- onDraw 方法理解>,那么接下我们还需要继续去理解自定义view里面的onMeasure 方法 推荐文章: htt ...

  8. Android 自定义 view(三)—— onDraw 方法理解

    前言: 上一篇已经介绍了用自己定义的属性怎么简单定义一个view<Android 自定义view(二) -- attr 使用>,那么接下来我们继续深究自定义view,下一步将要去简单理解自 ...

  9. Android 自定义view(二) —— attr 使用

    前言: attr 在前一篇文章<Android 自定义view -- attr理解>已经简单的进行了介绍和创建,那么这篇文章就来一步步说说attr的简单使用吧 自定义view简单实现步骤 ...

随机推荐

  1. 20165224 陆艺杰 Exp3 免杀原理与实践

    杀软是如何检测出恶意代码的? 识别代码特征码 监测像后门的行为 (2)免杀是做什么? 让后面程序不被安全软件发现 (3)免杀的基本方法有哪些? 多方式编码 半手工shellcode编程 完全自己写没有 ...

  2. 30 个 Java 集合面试问题及答案

    30 个 Java 集合面试问题及答案 Java集合框架为Java编程语言的基础,也是Java面试中很重要的一个知识点.这里,我列出了一些关于Java集合的重要问题和答案. 1.Java集合框架是什么 ...

  3. python学习之路---day09

    函数案例: return 可以终止函数后面的调用 def abc() print("1") print("2") print("3") pr ...

  4. Django-admin 的使用

    admin 组件的使用 Django 提供功能十分强大的后台管理组件 admin 来实现自动管理. admin 是一个组件,与 APP 一样,项目启动一开始就加载了.在 setting.py 中的 I ...

  5. 创建一个流(Stream)可以让Bitmap或Image保存到流里面

    创建一个流(Stream)可以让Bitmap或Image保存到流里面   http://blog.csdn.net/angxiao/article/details/7481465 写文件流       ...

  6. Tyvj - 1305 单调队列优化dp

    今天有点头痛就不写具体细节了,贴完走人 #include<iostream> #include<algorithm> #include<cstdio> #inclu ...

  7. c#和java的深拷贝CloneObject

    public static T Clone<T>(this T source) { if (!typeof(T).IsSerializable) { throw new ArgumentE ...

  8. Oracle列转行函数Listagg以及pivot查询示例

    简单的Oracle列转行函数Listagg示例: CREATE TABLE tbl_test (catalog VARCHAR(1),product VARCHAR(2),amount NUMBER) ...

  9. windos下安装mongodb

    一.下载 1. www.mongodb.com/download-center#community2.根据电脑系统下载对应的版本,3.选择zip(解压版本)/msi安装版本4.本文是用的msi(安装版 ...

  10. 2019.3.13 Java实现分数的四则运算

    Java实现分数的四则运算 自己的写法 markdown太久没写格式不会用了--将就着看 Fraction.java package com.lanou; public class Fraction ...