Android实现指针刻度转盘
作者:cyfxiaochen 发布时间:2022-06-08 07:41:42
标签:Android,指针,转盘
本文实例为大家分享了Android实现指针刻度转盘的具体代码,供大家参考,具体内容如下
一. 先上个效果图,实现如图所示刻度转盘和2个文本的绘制,最后1个刻度绘制的比较长一些(后期会添加动画效果,未完待续…):
二. 话不多说,上代码,Timber可使用Log代替,也可根据自身需求将配置属性放到attrs.xml中去:
package com.landleaf.householdtype.widget;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;
import androidx.annotation.Nullable;
import timber.log.Timber;
public class PanelTempCircle extends View {
private static final String TAG = PanelTempCircle.class.getSimpleName();
//#EFEFEF
//#47C496
private Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
//画笔宽度,线段长度,最后一条大线条的长度 比其他线段长的长度
private int strokeWidth = 7, lineLength = 40, maxLineLength = 10;
//绘制文本距离圆
private int txtMargin = 10;
//中心点坐标
private int centerX, centerY;
//内圆半径,外圆半径
private int innerRadius, outRadius;
//绘制文本
private String leftText = "0", rightText = "30";
//绘制文本的字体大小
private int textSize = 25;
//背景 or 进度条颜色
private int colorBackground = Color.parseColor("#EFEFEF");
private int colorProgress = Color.parseColor("#18C8C7");
private int colorText = Color.parseColor("#999999");
float fullAngle = 180f;
float cutAngle = 90f;
//每个线段相隔的宽度
private static final int perAngle = 6;
private int startAngle = -12;
public PanelTempCircle(Context context) {
super(context);
initPaint(context, null);
}
public PanelTempCircle(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
initPaint(context, attrs);
}
private void initPaint(Context context, AttributeSet attrs) {
paint.setStrokeCap(Paint.Cap.ROUND);
paint.setTextSize(textSize);
paint.setStrokeWidth(strokeWidth);
paint.setTextAlign(Paint.Align.CENTER);
paint.setColor(colorBackground);
}
public PanelTempCircle(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
initPaint(context, attrs);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
drawCircle(startAngle, 60, canvas, paint);
}
private void drawCircle(float startAngle, float endAngle, Canvas canvas, Paint paint) {
for (float i = startAngle; i <= fullAngle - startAngle; i = i + perAngle) {
//-12.-6,0,6....180,186,192
//得出坐标
int startM, startN, endM, endN, startX, startY, endX, endY;
int startPaintRadius = innerRadius;
int endPaintRadius = outRadius;
float currentAngle = i;
if (i <= 0) {
currentAngle = Math.abs(i);
} else if (currentAngle > fullAngle) {
currentAngle = i - fullAngle;
}
//当前进度=结束进度
if (i == endAngle) {
startPaintRadius = innerRadius - maxLineLength;
endPaintRadius = outRadius + maxLineLength;
}
//起始点
double angleSin = Math.sin(Math.PI * (Math.abs(currentAngle) / fullAngle));
//起始点 高度 宽度
startM = (int) (angleSin * startPaintRadius);
startN = (int) Math.sqrt(Math.pow(startPaintRadius, 2) - Math.pow(startM, 2));
//结束点 高度 宽度
endM = (int) (angleSin * endPaintRadius);
endN = (int) Math.sqrt(Math.pow(endPaintRadius, 2) - Math.pow(endM, 2));
// Log.i(TAG, startM + "," + startN + "," + endM + "," + endN);
//获得起始点和结束点的坐标
if (i < 0) {
//第三象限
startX = centerX - startN;
endX = centerX - endN;
startY = centerY + startM;
endY = centerY + endM;
} else if (i > fullAngle) {
//第二象限
startX = centerX + startN;
endX = centerX + endN;
startY = centerY + startM;
endY = centerY + endM;
} else {
if (i < cutAngle) {
//第四象限
startX = centerX - startN;
endX = centerX - endN;
startY = centerY - startM;
endY = centerY - endM;
} else {
//第一象限
startX = centerX + startN;
endX = centerX + endN;
startY = centerY - startM;
endY = centerY - endM;
}
}
//设置线条绘制颜色
if (i <= endAngle) {
paint.setColor(colorProgress);
} else {
paint.setColor(colorBackground);
}
canvas.drawLine(startX, startY, endX, endY, paint);
//判断是否需要绘制文本
if (i == startAngle) {
int textWidth = getTextWidth(paint, leftText);
paint.setColor(colorText);
canvas.drawText(leftText, startX + textWidth + txtMargin, startY, paint);
Timber.tag(TAG).i("绘制左侧文本:" + (startX + textWidth + txtMargin) + "," + startY);
}
if (i == fullAngle - startAngle) {
int textWidth = getTextWidth(paint, rightText);
paint.setColor(colorText);
canvas.drawText(rightText, startX - textWidth - txtMargin, startY, paint);
Timber.tag(TAG).i("绘制右侧文本:" + (startX - textWidth - txtMargin) + "," + startY);
}
}
}
public int getTextWidth(Paint paint, String str) {
int iRet = 0;
if (str != null && str.length() > 0) {
int len = str.length();
float[] widths = new float[len];
paint.getTextWidths(str, widths);
for (int j = 0; j < len; j++) {
iRet += (int) Math.ceil(widths[j]);
}
}
return iRet;
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
innerRadius = (getMeasuredWidth() - lineLength * 2 - maxLineLength * 2) / 2;
outRadius = lineLength + innerRadius;
Timber.tag(TAG).i("内圈半径:" + innerRadius + ",外圈半径:" + outRadius);
centerX = outRadius + maxLineLength;
centerY = outRadius + maxLineLength;
Timber.tag(TAG).i("中心坐标:(x=" + centerX + ",y=" + centerY + ")");
int heightMode = MeasureSpec.getMode(heightMeasureSpec);
if (heightMode == MeasureSpec.AT_MOST) {
double angleSin = Math.sin(Math.PI * (Math.abs(startAngle) / fullAngle));
int endPaintRadius = outRadius + maxLineLength;
int height = (int) (endPaintRadius + angleSin * endPaintRadius);
setMeasuredDimension(widthMeasureSpec, height);
}
}
}
三. xml中使用方式:
说明:主要申明宽度即可,高度会在代码中进行计算;
<com.landleaf.householdtype.widget.PanelTempCircle
android:id="@+id/mptc_set_temp"
android:layout_width="270dp"
android:layout_height="wrap_content"
app:layout_constraintBottom_toTopOf="@+id/tvHumidity"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.3" />
来源:https://blog.csdn.net/cyfxiaochen/article/details/108016014


猜你喜欢
- 前些天,有一个需求要用SpringBoot的多环境,当时没有系统学习springboot ,所以在网上找来找去的找到了一个解决方案,并写了一
- mybatis-plus框架功能很强大,把很多功能都集成了,比如自动生成代码结构,mybatis crud封装,分页,动态数据源等等,附上官
- 多数据源配置首先是配置文件这里采用yml配置文件,其他类型配置文件同理我配置了两个数据源,一个名字叫ds1数据源,一个名字叫ds2数据源,如
- 不久之前,我面试了一些求职Java高级开发工程师的应聘者。我常常会面试他们说,“你能给我介绍一些Java中得弱引用吗?”,如果面试者这样说,
- 本文实例讲述了Android编程实现WebView添加进度条的方法。分享给大家供大家参考,具体如下:标准的XML界面<?xml ver
- 1.简介学了几周的Java,闲来无事,写个乞丐版的扫雷,加强一下Java基础知识。2.编写过程编写这个游戏,一共经历了三个阶段,编写了三个版
- 本文开始做一个网上商城的项目,首先从搭建环境开始,一步步
- 代码如下所示:<!-- 配置数据源 --> <bean id="dataSource" c
- 网易Java程序员两轮面试题,请作答。part 1:网易JAVA程序员一面1.volatile有什么用?2.Minor GC和Full GC
- 延迟赋值主要有两点:1.一个参数可能或可能没被赋值.2.一个参数在一个函数中每次使用时可能被赋值.如下面的这种情况:int Add(int
- 前言在我们公司里,不同的服务之间通过Feign进行远程调用,但是,我们在尝试使调用可重试时遇到了一个小问题,Feign框架本身可以配置的自己
- 工厂模式和简单工厂有什么区别。废话不多说,对比第一篇例子应该很清楚能看出来。优点: 工厂模式弥补了简单工厂模式中违背开放-封闭原则,又保持了
- 比如要获取打开摄像头的应用程序名称,只需要在frameworks/base/core/android/hardware/Camera.jav
- 在往常的 HTTP 调用中,一直都是使用的官方提供的 RestTemplate 来进行远程调用,该调用方式将组装代码冗余到正常业务代码中,不
- 以下摘自胖哥分享的 2022开工福利教程。在学习Spring Security的时候你有没有下面这两个疑问:Spring Security的
- 本文实例分析了C#中out保留字的用法,分享给大家供大家参考。具体用法分析如下:C#中的out保留字表示这个变量要回传值,最简单的应用是除法
- 代理模式也是一种非常常见的设计模式。了解Spring框架的都知道,Spring AOP 使用的就是 * 模式。今天就来系统的重温一遍代理模
- EasyCode 插件EasyCode 插件 是一款根据表结构生成代码的很方便的Idea插件, 强烈推荐. 并且可以自定义模板来控制生成的类
- 提出问题我做的是一个通讯录,用到了选项菜单,每一个菜单项左边我都添加了一个小图标,运行后发现没有显示出来。解决方案利用反射机制,根据对象来寻
- 1、需要引入依赖<dependency> &l