Android自定义一个带背景的圆环形进度条(Kotlin)

news2025/1/19 14:13:20

前言

在Android开发过程中,难免遇到一些复杂的UI组件需要我们自定义

当然使用系统原生组件拼凑也能完成,但是UI复杂度增加了不说,在更新UI状态的时候还不好管理,最重要的是复用的价值不大,上述的操作很容易引增加码冗余度和阅读难度,为此自定义UI成了一个非常不错的选择。

实现一个带进度条的播放按钮,类似于QQ音乐底部控住组件中的播放按钮。

1.ProgressBar组件

ProgressBar是Android开发中用于显示进度的一个UI组件。它通常用于向用户展示某个操作的进度情况,比如文件下载、数据加载等场景。ProgressBar可以以不同的形式展现,常见的有圆形进度条和水平进度条。

主要属性

  • max:进度条的最大值,默认为100。
  • progress:当前进度值。
  • progressDrawable:自定义进度条的Drawable。
  • indeterminate:是否设置为不确定模式,不确定模式的进度条会不断循环动画,而不会停止在一个特定的进度。
  • indeterminateDrawable:自定义不确定模式进度条的Drawable。
  • secondaryProgress:次要进度值,可以用于显示中间进度,比如缓冲进度。

使用场景

  • 下载文件时显示下载进度。
  • 加载数据时显示加载进度。
  • 视频播放时显示缓冲进度。
<ProgressBar  
    android:id="@+id/progressBar"  
    style="?android:attr/progressBarStyleHorizontal"  
    android:layout_width="match_parent"  
    android:layout_height="wrap_content"  
    android:max="100"  
    android:progress="50" />

2.自定义组件

继承View,在onDraw中使用Canvas绘制

背景

使用drawBitmap绘制背景

canvas.drawBitmap(it, null, rectF, null)

圆环

drawCircle绘制圆,其中Paint的style设置为STROKE可以绘制环形

strokeWidth就是圆环的半径

private val backgroundPaint = Paint()

backgroundPaint.apply {
    color = backgroundColor
    strokeWidth = 10f
    style = Paint.Style.STROKE
    isAntiAlias = false
}


canvas.drawCircle(w, h, r, backgroundPaint)

进度条

drawArc实现环形进度条的绘制

rectF是整个环形的内切正方形坐标,这个正方形坐标需要和背景所在的圆环对上,否则可能出现进度条和背景对不上的问题

private val progressPaint = Paint()

progressPaint.apply {
    color = progressColor
    strokeWidth = 10f
    style = Paint.Style.STROKE
    isAntiAlias = false
}


canvas.drawArc(rectF, 270f, progress.toFloat() / max.toFloat() * 360f, false, progressPaint)

 完整代码

class CircleBar : View {
    private val backgroundPaint = Paint()
    private val progressPaint = Paint()
    private var max = 100
    private var progress = 0
    private var backgroundColor: Int = Color.WHITE
    private var progressColor: Int = Color.BLACK
    private var backgroundBitmap: Bitmap? = null
    private val rectF = RectF()
    private val offset = 5f

    constructor(context: Context) : super(context)

    constructor(context: Context, attributeSet: AttributeSet) : super(context, attributeSet) {
        init(context, attributeSet)
    }

    constructor(context: Context, attributeSet: AttributeSet, defStyleAttr: Int) : super(
        context,
        attributeSet,
        defStyleAttr
    ) {
        init(context, attributeSet)
    }

    private fun init(context: Context, attributeSet: AttributeSet) {
        val typedArray = context.obtainStyledAttributes(attributeSet, R.styleable.CircleBar)
        max = typedArray.getInteger(R.styleable.CircleBar_max, 100)
        progress = typedArray.getInteger(R.styleable.CircleBar_progress, 0)
        backgroundColor = typedArray.getColor(R.styleable.CircleBar_backgroundColor, Color.WHITE)
        progressColor = typedArray.getColor(R.styleable.CircleBar_progressColor, Color.BLACK)
        val background = typedArray.getDrawable(R.styleable.CircleBar_background)
        if (background != null) {
            backgroundBitmap = (background as BitmapDrawable).bitmap
        }
        typedArray.recycle()

        backgroundPaint.apply {
            color = backgroundColor
            strokeWidth = 10f
            style = Paint.Style.STROKE
            isAntiAlias = false
        }
        progressPaint.apply {
            color = progressColor
            strokeWidth = 10f
            style = Paint.Style.STROKE
            isAntiAlias = false
        }
    }

    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)
        val w = width / 2f
        val h = height / 2f
        val r = w - offset
        rectF.set(offset, h - r, w + r, h + r)
        backgroundBitmap?.let { canvas.drawBitmap(it, null, rectF, null) }
        canvas.drawCircle(w, h, r, backgroundPaint)
        canvas.drawArc(rectF, 270f, progress.toFloat() / max.toFloat() * 360f, false, progressPaint)
    }

    fun setMax(max: Int) {
        this.max = max
        invalidate()
    }

    fun setProgress(progress: Int) {
        this.progress = progress
        invalidate()
    }

    fun getMax(): Int {
        return max
    }

}

组件主题样式

<declare-styleable name="CircleBar">
    <attr name="max" format="integer" />
    <attr name="progress" format="integer" />
    <attr name="backgroundColor" format="color" />
    <attr name="progressColor" format="color" />
    <attr name="background" format="reference" />
</declare-styleable>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2067269.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

温湿度传感器---DHT11

温湿度传感器---DHT11 一、DHT11介绍二、DHT11原理图三、DHT11工作原理和时序3.1 DHT11工作时序 四、DHT11代码 一、DHT11介绍 DHT11数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器&#xff0c;传感器内部包括一个8位单片机控制一个电阻式感湿元件和一个NTC…

如何用Java SpringBoot+Vue打造“花开富贵”花园管理系统【实战教程】

✍✍计算机编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java实战 |…

学习【正点原子】新建寄存器版本MDK工程过程中错误总结

参考视频&#xff1a; 第21讲 基础篇-新建寄存器版本MDK工程2_哔哩哔哩_bilibili 参考文档&#xff1a;F:\STM32学习\【正点原子】探索者STM32F407开发板V3 资料盘(A盘)STM32F407 探索者开发指南V1.2 准备 使用的是正点原子探索者【STM32F407ZG】&#xff0c;编译软件版本是…

读论文《Behavior Pattern Mining-based Multi-Behavior Recommendation》

论文地址&#xff1a;arxiv.org/pdf/2408.12152v1 项目地址&#xff1a;GitHub - rookitkitlee/BPMR 基于行为模式挖掘的多行为推荐&#xff1a;论文提出了一种新颖的多行为推荐算法&#xff08;BPMR&#xff09;&#xff0c;旨在通过分析用户和项目之间的复杂交互模式来提高…

【数据分析】数据的计量尺度、数据集中趋势

一、数据的计量尺度 数据的计量尺度分为四类&#xff1a;定类尺度、定序尺度、定距尺度、定比尺度。 数据的计量尺度特点逻辑与数学运算常见的例子数据类型定类尺度无等级次序是否相等 性别&#xff1a;男、女... 民族&#xff1a;汉族、满族... 职业&#xff1a;医生、消防员…

文心快码Baidu Comate 帮你解大厂面试题:在8g内存的机器,能否启动一个7G堆大小的java进程?

&#x1f50d;【大厂面试真题】系列&#xff0c;带你攻克大厂面试真题&#xff0c;秒变offer收割机&#xff01; ❓今日问题&#xff1a;在8g内存的机器&#xff0c;能否启动一个7G堆大小的java进程&#xff1f; ❤️一起看看文心快码Baidu Comate给出的答案吧&#xff01;如…

网络安全 day2 --- 宝塔搭建网站、phpstudy、IIS搭建网站的区别、docker、建站分配站、前后端分离

宝塔建站 以下步骤目前在VPS上操作 直接网上搜索宝塔官网进行下载window面板&#xff0c;然后安装宝塔面板 https://download.bt.cn/win/panel/BtSoft.zip 也可以直接用我这个下载链接 等待安装成功&#xff0c;安装成功后直接进入宝塔面板 登陆宝塔面板之后我们需要注册…

RAG的数据清洗和拆分

在大模型实际落地的时候&#xff0c;存在一些问题&#xff0c;主要集中在以下方面&#xff1a; 缺少垂直领域知识&#xff1a;虽然大模型压缩了大量的人类知识&#xff0c;但在垂直场景上明显存在短板&#xff0c;需要专业化的服务去解决特定问题。 存在幻觉、应用有一定门槛&…

解决 Navicat 删除唯一键(unique)后保存失败的问题:1-near “)“:syntax error

1、问题描述 我按照以下步骤删除如下图所示的 studentId 唯一键&#xff1a; 可以看到唯一键已经被删除了&#xff0c;但是此时我点击保存按钮时报错&#xff0c;保存失败&#xff1a; 2、解决方法 还需要点击如下图所示的删除唯一键按钮&#xff0c;才算是真正删除成功&…

作为一个电子工程师,在选择合适的网络变压器

作为一个电子工程师&#xff0c;在选择合适的网络变压器&#xff08;Network Transformer&#xff09;设计PCB时&#xff0c;应该遵循一系列的准则和步骤&#xff0c;确保所设计的电路既满足功能需求&#xff0c;又具备良好的性能和可靠性。以下是选择和设计过程中应该考虑的关…

坚持绿色发展的上海智算中心,稳步推进中

自今年年初正式封顶以来&#xff0c;云端股份上海智算中心在外墙及内部的建设进展顺利。这座智算中心地理位置优越&#xff0c;正逐步成为推动数字经济发展的重要力量。 位置优势 云端股份上海智算中心毗邻智慧岛数据产业园&#xff0c;是崇明区目前建设的唯一一座智算中心&am…

Flutter ListView控件

ListView是flutter中线性排列的可滚动的列表部件。ListView 是最常用的滚动小部件。它在滚动方向上一个接一个地显示其子项。在交叉轴上&#xff0c;子项需要填满 ListView。 如果非空&#xff0c;则 itemExtent 会强制子项在滚动方向上具有给定的范围。 如果非空&#xff0c;…

@Transactional 注解 this调用注解不生效

如果在一个方法上标注Transactional,要确定这个方法的调用方是容器中的代理对象还是普通的java对象. 如果是代理对象事务会开启生效 如果是普通的java对象事务会失效 如果是这样调用,那 grantDayBonus 方法肯定是容器中的代理对象调起的.事务会生效

【机器学习练习】糖尿病预测

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 一、数据预处理 1. 数据导入 import numpy as np import pandas as pd import seaborn as sns from sklearn.model_selec…

第二十九节、场景管理和切换

使用这个加载场景&#xff0c;实现场景的切换 是一个打包工具&#xff0c;使包体变小&#xff0c;不会重复打包包体 这个也可以远程进行打包&#xff0c;将代码上传到网上的库中&#xff0c;实现热更新 勾选后就会默认将当前场景加载到这里 简化名字 当物体需要跨场景&#xf…

观测云对接 Pinpoint 最佳实践

简介 Pinpoint 是一个开源的分布式应用性能监控工具&#xff0c;由韩国搜索引擎公司 Naver 开发。它主要用于跟踪和监控分布式应用程序和微服务架构中的性能问题。Pinpoint 的核心功能包括对分布式链路的追踪和监控&#xff0c;通过收集和分析数据来帮助开发者定位问题所在。P…

node版本管理工具Node Version Manager(nvm)的安装及使用

文章目录 一、nvm安装1.1,下载nvm安装包程序&#xff08;选最高版本安装即可&#xff0c;安装前先把本机电脑node卸载&#xff0c;不然管理不到&#xff09;双击安装包进行安装&#xff08;傻瓜式下一步即可&#xff09; 二、配置镜像2.1、使用管理员运行命令提示符2.2、检查nv…

健康减调攻略:1月轻松掉十斤

减调减调&#xff0c;有减才有调&#xff0c;饮食为调、运动重在减&#xff0c;他们是相互作用的&#xff0c;要瘦&#xff0c;只有饮食与运动结合才能事半功倍。 饮食调减&#xff1a; 不需要花多大钱&#xff0c;也不用费精力和时间自己做饭&#xff0c;或狂抠饮食量和食物…

6款ai伪原创软件app,自动生成文章效率更高

在当今信息爆炸的时代&#xff0c;内容创作的需求日益增长。无论是专业的写手、自媒体从业者&#xff0c;还是企业的营销人员&#xff0c;都在不断寻求提高创作效率的方法。而ai伪原创软件app的出现&#xff0c;为创作者们带来了新的解决方案。下面就为大家介绍6个强大的ai伪原…

6款大学生电脑里的必装软件,装进电脑慢慢用

分享6款大学生爱用的windows软件&#xff0c;个个功能强大&#xff0c;装进电脑能提升效率&#xff0c;让大学生活更轻松&#xff01; 1、Everything 大学四年怎么都得积攒下一堆PPT和文档&#xff0c;日子久了存放的乱七八糟&#xff0c;想找的时候一个头两个大。这时候Every…