android自定义刻度盘 表盘 速度表

2023-11-05 12:10

本文主要是介绍android自定义刻度盘 表盘 速度表,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

android自定义刻度盘 表盘 速度表

效果图
这里写图片描述
这里写图片描述
直接在xml文件中添加即可,在代码中调用setProgress(int) 即可。代码注释写的很清楚,有兴趣的随便改改。

InstrumentView.java

package com.xk.testdemo.view;import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.Rect;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;import com.xk.testdemo.config.ConfigColor;import static android.R.attr.angle;
import static android.R.attr.baseline;
import static android.R.attr.dial;
import static android.R.attr.end;
import static android.R.attr.path;
import static android.R.attr.radius;
import static android.R.attr.width;
import static android.R.attr.x;
import static android.R.attr.y;/*** Created by xuekai on 2016/10/26.*/public class InstrumentView extends View {private String color_outcircle = "#DEDEDE";private String color_bg_outcircle = "#2690F8";private String color_bg_incircle = "#58ADE4";private String color_progress = "#87CEEB";private String color_smart_circle = "#C2B9B0";private String color_indicator_left = "#E1DCD6";private String color_indicator_right = "#F4EFE9";/*** 当前进度*/private int progress = 50;/*** 要画的内容的实际宽度*/private int contentWidth;/*** view的实际宽度*/private int viewWidth;/*** view的实际高度*/private int viewHeight;/*** 外环线的宽度*/private int outCircleWidth = 1;/*** 外环的半径*/private int outCircleRadius = 0;/*** 内环的半径*/private int inCircleRedius = 0;/*** 内环与外环的距离*/private int outAndInDistance = 0;/*** 内环的宽度*/private int inCircleWidth = 0;/*** 刻度盘距离它外面的圆的距离*/private int dialOutCircleDistance = 0;/*** 内容中心的坐标*/private int[] centerPoint = new int[2];/*** 刻度线的数量*/private int dialCount = 0;/*** 每隔几次出现一个长线*/private int dialPer = 0;/*** 长线的长度*/private int dialLongLength = 0;/*** 短线的长度*/private int dialShortLength = 0;/*** 刻度线距离圆心最远的距离*/private int dialRadius = 0;/*** 圆弧开始的角度*/private int startAngle = 0;/*** 圆弧划过的角度*/private int allAngle = 0;private Paint mPaint;/*** 刻度盘上数字的数量*/private int figureCount = 6;public InstrumentView(Context context) {this(context, null);}public InstrumentView(Context context, AttributeSet attrs) {this(context, attrs, 0);}public InstrumentView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}private void init() {mPaint = new Paint();mPaint.setStyle(Paint.Style.STROKE);mPaint.setAntiAlias(true);}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);initValues();}/*** 初始化尺寸*/private void initValues() {viewWidth = getMeasuredWidth();viewHeight = getMeasuredHeight();contentWidth = viewWidth > viewHeight ? viewHeight : viewWidth;outCircleRadius = contentWidth / 2 - outCircleWidth;outAndInDistance = (int) (contentWidth / 26.5);inCircleWidth = (int) (contentWidth / 18.7);centerPoint[0] = viewWidth / 2;centerPoint[1] = viewHeight / 2;inCircleRedius = outCircleRadius - outAndInDistance - inCircleWidth / 2;startAngle = 150;allAngle = 240;dialOutCircleDistance = inCircleWidth;dialCount = 50;dialPer = 5;dialLongLength = (int) (dialOutCircleDistance / 1.2);dialShortLength = (int) (dialLongLength / 1.8);dialRadius = inCircleRedius - dialOutCircleDistance;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);drawStatic(canvas);drawDynamic(canvas);}/*** 绘制静态的部分** @param canvas*/private void drawStatic(Canvas canvas) {drawOutCircle(canvas);drawCircleWithRound(startAngle, allAngle, inCircleWidth, inCircleRedius, color_outcircle, canvas);drawDial(startAngle, allAngle, dialCount, dialPer, dialLongLength, dialShortLength, dialRadius, canvas);drawBackGround(canvas);drawFigure(canvas, figureCount);}private void drawFigure(Canvas canvas, int count) {int figure = 0;int angle;for (int i = 0; i < count; i++) {figure = (int) (100 / (1f * count-1) * i);angle = (int) ((allAngle) / ((count-1) * 1f) * i) + startAngle;int[] pointFromAngleAndRadius = getPointFromAngleAndRadius(angle, dialRadius - dialLongLength * 2 );mPaint.setTextSize(15);mPaint.setTextAlign(Paint.Align.CENTER);canvas.save();canvas.rotate(angle+90,pointFromAngleAndRadius[0],pointFromAngleAndRadius[1]);canvas.drawText(figure+"%",pointFromAngleAndRadius[0],pointFromAngleAndRadius[1],mPaint);canvas.restore();}}/*** 画内层背景** @param canvas*/private void drawBackGround(Canvas canvas) {mPaint.setColor(Color.parseColor(color_bg_outcircle));mPaint.setStyle(Paint.Style.STROKE);mPaint.setStrokeWidth(outCircleRadius / 3 / 2);canvas.drawCircle(centerPoint[0], centerPoint[1], outCircleRadius / 3, mPaint);mPaint.setColor(Color.parseColor(color_bg_incircle));mPaint.setStyle(Paint.Style.FILL);canvas.drawCircle(centerPoint[0], centerPoint[1], (outCircleRadius / 3f / 2), mPaint);}/*** 画刻度盘** @param startAngle  开始画的角度* @param allAngle    总共划过的角度* @param dialCount   总共的线的数量* @param per         每隔几个出现一次长线* @param longLength  长仙女的长度* @param shortLength 短线的长度* @param radius      距离圆心最远的地方的半径*/private void drawDial(int startAngle, int allAngle, int dialCount, int per, int longLength, int shortLength, int radius, Canvas canvas) {int length;int angle;for (int i = 0; i <= dialCount; i++) {angle = (int) ((allAngle) / (dialCount * 1f) * i) + startAngle;if (i % 5 == 0) {length = longLength;} else {length = shortLength;}drawSingleDial(angle, length, radius, canvas);}}/*** 画刻度中的一条线** @param angle  所处的角度* @param length 线的长度* @param radius 距离圆心最远的地方的半径*/private void drawSingleDial(int angle, int length, int radius, Canvas canvas) {int[] startP = getPointFromAngleAndRadius(angle, radius);int[] endP = getPointFromAngleAndRadius(angle, radius - length);canvas.drawLine(startP[0], startP[1], endP[0], endP[1], mPaint);}/*** 画最外层的圆** @param canvas*/private void drawOutCircle(Canvas canvas) {mPaint.setStrokeWidth(outCircleWidth);mPaint.setStyle(Paint.Style.STROKE);mPaint.setColor(Color.parseColor(color_outcircle));canvas.drawCircle(centerPoint[0], centerPoint[1], outCircleRadius, mPaint);}/*** 绘制动态的部分** @param canvas*/private void drawDynamic(Canvas canvas) {drawProgress(progress, canvas);drawIndicator(progress, canvas);drawCurrentProgressTv(progress, canvas);}/*** 绘制当前进度是文字** @param progress* @param canvas*/private void drawCurrentProgressTv(int progress, Canvas canvas) {
//        canvas.drawText("当前进度:"+progress+"%",);mPaint.setTextSize(25);mPaint.setTextAlign(Paint.Align.CENTER);Paint.FontMetrics fontMetrics = mPaint.getFontMetrics();float baseLine1 = centerPoint[1] + (outCircleRadius / 20f * 11 - fontMetrics.top - fontMetrics.bottom);canvas.drawText("当前进度", centerPoint[0], baseLine1, mPaint);
//drawText的第二个参数的值=要让文字的中心放在哪-(fontMetrics.top+fontMetrics.bottom)/2
//此时求出来的baseline可以使文字竖直居中float baseLine2 = outCircleRadius / 20f * 11 - 3 * (fontMetrics.bottom + fontMetrics.top) + centerPoint[1];canvas.drawText(progress + "%", centerPoint[0], baseLine2, mPaint);}/*** 画指针以及他的背景** @param progress* @param canvas*/private void drawIndicator(int progress, Canvas canvas) {drawPointer(canvas);drawIndicatorBg(canvas);}/*** 指针的最远处的半径和刻度线的一样*/private void drawPointer(Canvas canvas) {RectF rectF = new RectF(centerPoint[0] - (int) (outCircleRadius / 3f / 2 / 2),centerPoint[1] - (int) (outCircleRadius / 3f / 2 / 2), centerPoint[0] + (int) (outCircleRadius / 3f / 2 / 2), centerPoint[1] + (int) (outCircleRadius / 3f / 2 / 2));int angle = (int) ((allAngle) / (100 * 1f) * progress) + startAngle;//指针的定点坐标int[] peakPoint = getPointFromAngleAndRadius(angle, dialRadius);//顶点朝上,左侧的底部点的坐标int[] bottomLeft = getPointFromAngleAndRadius(angle - 90, (int) (outCircleRadius / 3f / 2 / 2));//顶点朝上,右侧的底部点的坐标int[] bottomRight = getPointFromAngleAndRadius(angle + 90, (int) (outCircleRadius / 3f / 2 / 2));Path path = new Path();mPaint.setColor(Color.parseColor(color_indicator_left));path.moveTo(centerPoint[0], centerPoint[1]);path.lineTo(peakPoint[0], peakPoint[1]);path.lineTo(bottomLeft[0], bottomLeft[1]);path.close();canvas.drawPath(path, mPaint);canvas.drawArc(rectF, angle - 180, 100, true, mPaint);Log.e("InstrumentView", "drawPointer" + angle);mPaint.setColor(Color.parseColor(color_indicator_right));path.reset();path.moveTo(centerPoint[0], centerPoint[1]);path.lineTo(peakPoint[0], peakPoint[1]);path.lineTo(bottomRight[0], bottomRight[1]);path.close();canvas.drawPath(path, mPaint);canvas.drawArc(rectF, angle + 80, 100, true, mPaint);}private void drawIndicatorBg(Canvas canvas) {mPaint.setColor(Color.parseColor(color_smart_circle));mPaint.setStyle(Paint.Style.FILL);canvas.drawCircle(centerPoint[0], centerPoint[1], (outCircleRadius / 3f / 2 / 4), mPaint);}/*** 根据进度画进度条** @param progress 最大进度为100.最小为0*/private void drawProgress(int progress, Canvas canvas) {float ratio = progress / 100f;int angle = (int) (allAngle * ratio);drawCircleWithRound(startAngle, angle, inCircleWidth, inCircleRedius, color_progress, canvas);}public int getProgress() {return progress;}public void setProgress(int progress) {this.progress = progress;invalidate();}/*** 画一个两端为圆弧的圆形曲线** @param startAngle 曲线开始的角度* @param allAngle   曲线走过的角度* @param radius     曲线的半径* @param width      曲线的厚度*/private void drawCircleWithRound(int startAngle, int allAngle, int width, int radius, String color, Canvas canvas) {mPaint.setStrokeWidth(width);mPaint.setStyle(Paint.Style.STROKE);mPaint.setColor(Color.parseColor(color));RectF rectF = new RectF(centerPoint[0] - radius, centerPoint[1] - radius, centerPoint[0] + radius, centerPoint[1] + radius);canvas.drawArc(rectF, startAngle, allAngle, false, mPaint);drawArcRoune(radius, startAngle, width, canvas);drawArcRoune(radius, startAngle + allAngle, width, canvas);}/*** 绘制圆弧两端的圆** @param radius 圆弧的半径* @param angle  所处于圆弧的多少度的位置* @param width  圆弧的宽度*/private void drawArcRoune(int radius, int angle, int width, Canvas canvas) {int[] point = getPointFromAngleAndRadius(angle, radius);mPaint.setStrokeWidth(0);mPaint.setStyle(Paint.Style.FILL);mPaint.setStyle(Paint.Style.FILL);canvas.drawCircle(point[0], point[1], width / 2, mPaint);}/*** 根据角度和半径,求一个点的坐标** @param angle* @param radius* @return*/private int[] getPointFromAngleAndRadius(int angle, int radius) {double x = radius * Math.cos(angle * Math.PI / 180) + centerPoint[0];double y = radius * Math.sin(angle * Math.PI / 180) + centerPoint[1];return new int[]{(int) x, (int) y};}}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:paddingBottom="@dimen/activity_vertical_margin"android:paddingLeft="@dimen/activity_horizontal_margin"android:paddingRight="@dimen/activity_horizontal_margin"android:paddingTop="@dimen/activity_vertical_margin"tools:context="com.xk.testdemo.MainActivity"><com.xk.testdemo.view.InstrumentView
        android:layout_width="wrap_content"android:layout_height="500px"android:layout_alignParentTop="true"android:layout_alignParentLeft="true"android:layout_alignParentStart="true"android:id="@+id/instrumentView" />
<SeekBar
    android:layout_alignParentBottom="true"android:layout_width="match_parent"android:layout_height="wrap_content"android:id="@+id/sb"/></RelativeLayout>

MainActivity.java

package com.xk.testdemo;import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.SeekBar;import com.xk.testdemo.view.IndicatorContainer;
import com.xk.testdemo.view.InstrumentView;public class MainActivity extends AppCompatActivity {private IndicatorContainer myView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);final InstrumentView viewById = (InstrumentView) findViewById(R.id.instrumentView);SeekBar viewById1 = (SeekBar) findViewById(R.id.sb);viewById1.setMax(100);viewById1.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {@Overridepublic void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {viewById.setProgress(progress);}@Overridepublic void onStartTrackingTouch(SeekBar seekBar) {}@Overridepublic void onStopTrackingTouch(SeekBar seekBar) {}});}
}

这篇关于android自定义刻度盘 表盘 速度表的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/349757

相关文章

Vite 打包目录结构自定义配置小结

《Vite打包目录结构自定义配置小结》在Vite工程开发中,默认打包后的dist目录资源常集中在asset目录下,不利于资源管理,本文基于Rollup配置原理,本文就来介绍一下通过Vite配置自定义... 目录一、实现原理二、具体配置步骤1. 基础配置文件2. 配置说明(1)js 资源分离(2)非 JS 资

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

聊聊springboot中如何自定义消息转换器

《聊聊springboot中如何自定义消息转换器》SpringBoot通过HttpMessageConverter处理HTTP数据转换,支持多种媒体类型,接下来通过本文给大家介绍springboot中... 目录核心接口springboot默认提供的转换器如何自定义消息转换器Spring Boot 中的消息

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Python自定义异常的全面指南(入门到实践)

《Python自定义异常的全面指南(入门到实践)》想象你正在开发一个银行系统,用户转账时余额不足,如果直接抛出ValueError,调用方很难区分是金额格式错误还是余额不足,这正是Python自定义异... 目录引言:为什么需要自定义异常一、异常基础:先搞懂python的异常体系1.1 异常是什么?1.2

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

C语言自定义类型之联合和枚举解读

《C语言自定义类型之联合和枚举解读》联合体共享内存,大小由最大成员决定,遵循对齐规则;枚举类型列举可能值,提升可读性和类型安全性,两者在C语言中用于优化内存和程序效率... 目录一、联合体1.1 联合体类型的声明1.2 联合体的特点1.2.1 特点11.2.2 特点21.2.3 特点31.3 联合体的大小1

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束