自定义View实战之渐变,可拨动,带动画圆环控件实现

2023-10-08 12:38

本文主要是介绍自定义View实战之渐变,可拨动,带动画圆环控件实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

效果图预览
这里写图片描述

1. 分析

1. 绘制中间数字
2. 绘制带刻度的圆环  考虑分成若干等份
3. 绘制渐变圆环 需要用到渐变相关属性
4. 动画处理的同时需要考虑时时计算角度
5. 圆环开关控制按钮的波动范围处理

2. 实现原理

1. 绘制文字这个简单
2. 绘制刻度圆环 分等份 用canvas.drawArc画若干圆环
3. 绘制渐变圆环,需要用到Paint的setShader方法设置渐变颜色
4. 属性动画处理

3. 初始化一些东西 初始化一般我放在onSizeChanged方法中

 @Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);mCenterX = w / 2;mCenterY = h / 2;int[] colors = {Color.RED,Color.GREEN,Color.YELLOW};mSweepGradient = new SweepGradient(mCenterX,mCenterY, colors,null);mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.ring_dot);mBitmap = conversionBitmap(mBitmap,dp2px(30),dp2px(30));mMinValidateTouchArcRadius = (int) (mCircleRadius - mBitmap.getWidth() / 2 * 1.5);mMaxValidateTouchArcRadius = (int) mCircleRadius;
}

4. 绘制中间数字和带刻度线的圆环

 /*** 画带刻度线的圆环*/
private void drawScaleRing(Canvas canvas) {float margin = mRingRadius - dp20;float left = mCenterX - margin;float top = mCenterY - margin;float right = left + 2 * margin;float bottom = top + 2 * margin;mScaleReacF.set(left,top,right,bottom);// 等分成360 / 6 = 60份 保证间距一样的for (int i = 0; i < sweepAngle / 6; i++) {canvas.drawArc(mScaleReacF, startAngle + i * 6, 1f, false, mScaleRingPaint);}
}/*** 画中间的数字*/
private void drawMidNum(Canvas canvas) {String value = String.valueOf(mStepNum);//测量文字的宽高mTextPaint.getTextBounds(value, 0, value.length(), mTextRect);int width = mTextRect.width();//文字宽int height = mTextRect.height();//文字高canvas.drawText(value, mCenterX - width / 2, mCenterY + height / 2, mTextPaint);
}

5. 绘制带渐变色的进度圆环

 /*** 画进度圆环*/
private void drawRingProgress(Canvas canvas) {canvas.save();//canvas.rotate(-90);float left = mCenterX - mRingRadius;float top = mCenterY - mRingRadius;float right = left + 2 * mRingRadius;float bottom = top + 2 * mRingRadius;mProgressReacF.set(left,top,right,bottom);//设置渐变颜色mProgressPaint.setShader(mSweepGradient);float sweepAngle = mCurrentAngle;//绘制圆环 startAngle开始角度 sweepAngle扫瞄的角度canvas.drawArc(mProgressReacF, startAngle, sweepAngle, false, mProgressPaint);canvas.restore();
}

6. 进度条动画和数字处理

//加载进度条动画
public void setProgressAnimation(float last, float currentProgress, long duration) {mCurrentAngle = (currentProgress / mMaxProgress) * sweepAngle;ValueAnimator progressAnimator = ValueAnimator.ofFloat(last, mCurrentAngle);progressAnimator.setDuration(duration);progressAnimator.setTarget(mCurrentAngle);progressAnimator.setInterpolator(new AccelerateInterpolator());progressAnimator.start();progressAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@Overridepublic void onAnimationUpdate(ValueAnimator animation) {/**每次要绘制的圆弧角度**/mCurrentAngle = (float) animation.getAnimatedValue();//根据当前进度和总进度计算当前步数mStepNum = (int) (mCurrentAngle / sweepAngle * mMaxStepNum);postInvalidate();}});
}

7. 触摸事件处理

@Override
public boolean onTouchEvent(MotionEvent event) {int x = (int) event.getX();int y = (int) event.getY();if (mIsShowControlIcon && (event.getAction() == MotionEvent.ACTION_MOVE || isTouchArc(x, y))) {// 通过当前触摸点搞到cos角度值float cos = computeCos(x, y);// 通过反三角函数获得角度值double angle;if (x < mCenterX) { // 滑动超过180度angle = Math.PI * RADIAN + Math.acos(cos) * RADIAN;} else { // 没有超过180度angle = Math.PI * RADIAN - Math.acos(cos) * RADIAN;}if (mCurrentAngle > 270 && angle < 90) {mCurrentAngle = 360;cos = -1;} else if (mCurrentAngle < 90 && angle > 270) {mCurrentAngle = 0;cos = -1;} else {mCurrentAngle = (float) angle;}mCurrentProgress = getSelectedValue();invalidate();return true;} else {return super.onTouchEvent(event);}
}private int getSelectedValue() {return Math.round(mMaxProgress * (mCurrentAngle / sweepAngle));
}/*** 按下时判断按下的点是否按在圆边范围内* @param x* @param y*/
private boolean isTouchArc(int x, int y) {double d = getTouchRadius(x, y);return d >= mMinValidateTouchArcRadius && d <= mMaxValidateTouchArcRadius;
}/*** 计算某点到圆点的距离* @param x* @param y*/
private double getTouchRadius(int x, int y) {int cx = x - mCenterX;int cy = y - mCenterY;return Math.hypot(cx, cy);
}/*** 拿到倾斜的cos值*/
private float computeCos(float x, float y) {float width = x - mCenterX;float height = y - mCenterY;float slope = (float) Math.sqrt(width * width + height * height);return height / slope;
}

7. 项目源代码下载

后面统一提供源代码下载链接

8. 联系方式

QQ:1509815887

这篇关于自定义View实战之渐变,可拨动,带动画圆环控件实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python并行处理实战之如何使用ProcessPoolExecutor加速计算

《Python并行处理实战之如何使用ProcessPoolExecutor加速计算》Python提供了多种并行处理的方式,其中concurrent.futures模块的ProcessPoolExecu... 目录简介完整代码示例代码解释1. 导入必要的模块2. 定义处理函数3. 主函数4. 生成数字列表5.

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

canal实现mysql数据同步的详细过程

《canal实现mysql数据同步的详细过程》:本文主要介绍canal实现mysql数据同步的详细过程,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的... 目录1、canal下载2、mysql同步用户创建和授权3、canal admin安装和启动4、canal

Nexus安装和启动的实现教程

《Nexus安装和启动的实现教程》:本文主要介绍Nexus安装和启动的实现教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、Nexus下载二、Nexus安装和启动三、关闭Nexus总结一、Nexus下载官方下载链接:DownloadWindows系统根

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

MySQL 横向衍生表(Lateral Derived Tables)的实现

《MySQL横向衍生表(LateralDerivedTables)的实现》横向衍生表适用于在需要通过子查询获取中间结果集的场景,相对于普通衍生表,横向衍生表可以引用在其之前出现过的表名,本文就来... 目录一、横向衍生表用法示例1.1 用法示例1.2 使用建议前面我们介绍过mysql中的衍生表(From子句

Mybatis的分页实现方式

《Mybatis的分页实现方式》MyBatis的分页实现方式主要有以下几种,每种方式适用于不同的场景,且在性能、灵活性和代码侵入性上有所差异,对Mybatis的分页实现方式感兴趣的朋友一起看看吧... 目录​1. 原生 SQL 分页(物理分页)​​2. RowBounds 分页(逻辑分页)​​3. Page

Python基于微信OCR引擎实现高效图片文字识别

《Python基于微信OCR引擎实现高效图片文字识别》这篇文章主要为大家详细介绍了一款基于微信OCR引擎的图片文字识别桌面应用开发全过程,可以实现从图片拖拽识别到文字提取,感兴趣的小伙伴可以跟随小编一... 目录一、项目概述1.1 开发背景1.2 技术选型1.3 核心优势二、功能详解2.1 核心功能模块2.