Android实现带数字的圆形进度条(自定义进度条)
发布时间 - 2026-01-10 22:53:21 点击率:次开发

设计搞了一个带圆形进度的进度条,在GitHub上逛了一圈,发现没有,自己撸吧。
先看界面效果:
主要思路是写一个继承ProgressBar的自定义View,不废话,直接上代码:
package com.fun.progressbarwithnumber;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.widget.ProgressBar;
public class HorizontalProgressBarWithNumber extends ProgressBar {
private static final int DEFAULT_TEXT_SIZE = 10;
private static final int DEFAULT_TEXT_COLOR = 0XFFFC00D1;
private static final int DEFAULT_COLOR_UNREACHED_COLOR = 0xFFd3d6da;
private static final int DEFAULT_HEIGHT_REACHED_PROGRESS_BAR = 2;
private static final int DEFAULT_HEIGHT_UNREACHED_PROGRESS_BAR = 2;
private static final int DEFAULT_CIRCLE_COLOR = 0XFF3F51B5;
protected Paint mPaint = new Paint();
// 字体颜色
protected int mTextColor = DEFAULT_TEXT_COLOR;
// 字体大小
protected int mTextSize = sp2px(DEFAULT_TEXT_SIZE);
// 覆盖进度高度
protected int mReachedProgressBarHeight = dp2px(DEFAULT_HEIGHT_REACHED_PROGRESS_BAR);
// 覆盖进度颜色
protected int mReachedBarColor = DEFAULT_TEXT_COLOR;
// 未覆盖进度高度
protected int mUnReachedProgressBarHeight = dp2px(DEFAULT_HEIGHT_UNREACHED_PROGRESS_BAR);
// 未覆盖进度颜色
protected int mUnReachedBarColor = DEFAULT_COLOR_UNREACHED_COLOR;
// 圆的颜色
protected int mCircleColor = DEFAULT_CIRCLE_COLOR;
protected int mRealWidth;
protected boolean mIfDrawText = true;
protected boolean mIfDrawCircle = true;
protected static final int VISIBLE = 0;
public HorizontalProgressBarWithNumber(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public HorizontalProgressBarWithNumber(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
obtainStyledAttributes(attrs);
mPaint.setTextSize(mTextSize);
mPaint.setColor(mTextColor);
mPaint.setAntiAlias(true);
}
private void obtainStyledAttributes(AttributeSet attrs) {
// 获取自定义属性
final TypedArray attributes = getContext().obtainStyledAttributes(attrs, R.styleable.HorizontalProgressBarWithNumber);
mTextColor = attributes.getColor(R.styleable.HorizontalProgressBarWithNumber_progress_text_color, DEFAULT_TEXT_COLOR);
mTextSize = (int) attributes.getDimension(R.styleable.HorizontalProgressBarWithNumber_progress_text_size, mTextSize);
mCircleColor = attributes.getColor(R.styleable.HorizontalProgressBarWithNumber_progress_circle_color, DEFAULT_CIRCLE_COLOR);
mReachedBarColor = attributes.getColor(R.styleable.HorizontalProgressBarWithNumber_progress_reached_color, mTextColor);
mUnReachedBarColor = attributes.getColor(R.styleable.HorizontalProgressBarWithNumber_progress_unreached_color, DEFAULT_COLOR_UNREACHED_COLOR);
mReachedProgressBarHeight = (int) attributes.getDimension(R.styleable.HorizontalProgressBarWithNumber_progress_reached_bar_height, mReachedProgressBarHeight);
mUnReachedProgressBarHeight = (int) attributes.getDimension(R.styleable.HorizontalProgressBarWithNumber_progress_unreached_bar_height, mUnReachedProgressBarHeight);
int textVisible = attributes.getInt(R.styleable.HorizontalProgressBarWithNumber_progress_text_visibility, VISIBLE);
if (textVisible != VISIBLE) {
mIfDrawText = false;
}
attributes.recycle();
int left = (int) (mReachedProgressBarHeight * 0.8), right = (int) (mReachedProgressBarHeight * 0.8);
int top = (int) (mReachedProgressBarHeight * 0.3 + dp2px(1)), bottom = (int) (mReachedProgressBarHeight * 0.3 + dp2px(1));
setPadding(left, top, right, bottom);
}
@Override
protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int width = MeasureSpec.getSize(widthMeasureSpec);
int height = measureHeight(heightMeasureSpec);
setMeasuredDimension(width, height);
mRealWidth = getMeasuredWidth() - getPaddingRight() - getPaddingLeft();
}
private int measureHeight(int measureSpec) {
int result;
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
if (specMode == MeasureSpec.EXACTLY) {
result = specSize;
} else {
float textHeight = (mPaint.descent() - mPaint.ascent());
result = (int) (getPaddingTop() + getPaddingBottom() + Math.max(
Math.max(mReachedProgressBarHeight, mUnReachedProgressBarHeight), Math.abs(textHeight)));
if (specMode == MeasureSpec.AT_MOST) {
result = Math.min(result, specSize);
}
}
return result;
}
@Override
protected synchronized void onDraw(Canvas canvas) {
canvas.save();
canvas.translate(getPaddingLeft(), getHeight() / 2);
boolean noNeedBg = false;
float radio = getProgress() * 1.0f / getMax();
float progressPosX = (int) (mRealWidth * radio);
String text = getProgress() + "%";
float textWidth = mPaint.measureText(text);
float textHeight = (mPaint.descent() + mPaint.ascent()) / 2;
float radius = (mReachedProgressBarHeight + getPaddingBottom() + getPaddingTop()) / 2;
// 覆盖的进度
float endX = progressPosX;
if (endX > -1) {
mPaint.setColor(mReachedBarColor);
RectF rectF = new RectF(0, 0 - getPaddingTop() - getPaddingBottom(),
endX, mReachedProgressBarHeight - getPaddingBottom());
canvas.drawRoundRect(rectF, 25, 25, mPaint);
}
// 未覆盖的进度
if (!noNeedBg) {
float start = progressPosX;
mPaint.setColor(mUnReachedBarColor);
RectF rectF = new RectF(start, 0 - getPaddingTop() - getPaddingBottom(),
mRealWidth + getPaddingRight() - radius, mReachedProgressBarHeight - getPaddingBottom());
canvas.drawRoundRect(rectF, 25, 25, mPaint);
}
// 圆
if (mIfDrawCircle) {
mPaint.setColor(mCircleColor);
canvas.drawCircle(progressPosX, 0, radius, mPaint);
}
// 文本
if (mIfDrawText) {
mPaint.setColor(mTextColor);
canvas.drawText(text, progressPosX - textWidth / 2, -textHeight, mPaint);
}
canvas.restore();
}
/**
* dp 2 px
*/
protected int dp2px(int dpVal) {
return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dpVal, getResources().getDisplayMetrics());
}
/**
* sp 2 px
*/
protected int sp2px(int spVal) {
return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, spVal, getResources().getDisplayMetrics());
}
}
使用
在布局文件中加入:
<com.fun.progressbarwithnumber.HorizontalProgressBarWithNumber
android:id="@+id/hpbwn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="10dp"
fun:progress_circle_color="#ff000000"
fun:progress_reached_bar_height="20dp"
fun:progress_reached_color="#FFFF4081"
fun:progress_text_color="#ffffffff"
fun:progress_text_size="14sp"
fun:progress_unreached_bar_height="20dp"
fun:progress_unreached_color="#ffBCB4E8" />
progress_reached_bar_height:当前进度的高度
progress_unreached_bar_height:剩余进度的高度
progress_text_size:圆圈内文字的大小
注意:
当前进度和剩余进度的高度要一致,圆圈大小和圆圈内文字的大小要配合Java代码调整。
项目源码:
https://github.com/hfrommane/ProgressBarWithNumber
以上所述是小编给大家介绍的Android实现带数字的圆形进度条(自定义进度条),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
# android
# 圆形进度条
# android自定义进度条
# android进度条
# Android动态自定义圆形进度条
# Android编程之ProgressBar圆形进度条颜色设置方法
# Android带圆形数字进度的自定义进度条示例
# Android编程实现WebView添加进度条的方法
# Android自定义控件实现圆形进度条
# Android自定义Material进度条效果
# Android自定义view实现进度条指示效果
# android自定义view制作圆形进度条效果
# Android编程基于自定义View实现绚丽的圆形进度条功能示例
# Android实现环形进度条的实例
# Android编程实现类似于圆形ProgressBar的进度条效果
# 自定义
# 进度条
# 小编
# 在此
# 给大家
# 所述
# 先看
# 给我留言
# 搞了
# 感谢大家
# 逛了
# 疑问请
# 有任何
# true
# mIfDrawText
# mRealWidth
# boolean
# mCircleColor
# attrs
# defStyle
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
免费制作统计图的网站有哪些,如何看待现如今年轻人买房难的情况?
如何确保FTP站点访问权限与数据传输安全?
Laravel中间件如何使用_Laravel自定义中间件实现权限控制
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
中国移动官方网站首页入口 中国移动官网网页登录
Laravel如何使用Service Container和依赖注入?(代码示例)
如何在景安云服务器上绑定域名并配置虚拟主机?
Laravel distinct去重查询_Laravel Eloquent去重方法
英语简历制作免费网站推荐,如何将简历翻译成英文?
如何在阿里云通过域名搭建网站?
如何快速上传自定义模板至建站之星?
android nfc常用标签读取总结
Laravel队列由Redis驱动怎么配置_Laravel Redis队列使用教程
Laravel怎么上传文件_Laravel图片上传及存储配置
Linux网络带宽限制_tc配置实践解析【教程】
Laravel怎么集成Log日志记录_Laravel单文件与每日日志配置及自定义通道【详解】
Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
如何用PHP工具快速搭建高效网站?
HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】
Win11怎么开启自动HDR画质_Windows11显示设置HDR选项
,交易猫的商品怎么发布到网站上去?
用v-html解决Vue.js渲染中html标签不被解析的问题
零基础网站服务器架设实战:轻量应用与域名解析配置指南
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
网站制作壁纸教程视频,电脑壁纸网站?
个人摄影网站制作流程,摄影爱好者都去什么网站?
香港网站服务器数量如何影响SEO优化效果?
如何在搬瓦工VPS快速搭建网站?
nginx修改上传文件大小限制的方法
北京网站制作的公司有哪些,北京白云观官方网站?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】
如何用ChatGPT准备面试 模拟面试问答与职场话术练习教程
Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】
如何破解联通资金短缺导致的基站建设难题?
Laravel如何发送系统通知?(Notification渠道示例)
长沙企业网站制作哪家好,长沙水业集团官方网站?
如何获取PHP WAP自助建站系统源码?
如何快速打造个性化非模板自助建站?
如何彻底卸载建站之星软件?
Laravel如何使用Vite进行前端资源打包?(配置示例)
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
移动端脚本框架Hammer.js
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
怎样使用JSON进行数据交换_它有什么限制
Python自然语言搜索引擎项目教程_倒排索引查询优化案例
Laravel Debugbar怎么安装_Laravel调试工具栏配置指南
Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程

