当前位置: 首页 > news >正文

Android Compose 自定义滑动进度条

androidx.compose.material3 我用1.2.0版本的Slider非常好用,但是升级到1.3.2后滑轨后面多了个点,滑块背景也移除不掉而且默认透明了我暂时没有找到让他不透明。所以自定义了一个先用着。

@Composable
fun CustomSlider(state: CustomSliderState,modifier: Modifier = Modifier,trackHeight: Dp = 4.dp,thumbSize: Dp = 24.dp,trackColor: Color = Color.Gray,progressColor: Color = Color.Blue,thumbColor: Color = Color.White,onValueChange: ((Float) -> Unit)? = null,onValueChangeFinished: ((Float) -> Unit)? = null
) {var sliderWidth by remember { mutableFloatStateOf(0f) }Box(modifier = modifier.height(thumbSize + 16.dp) // 确保足够高度容纳滑块.pointerInput(Unit) {awaitEachGesture {val event = awaitPointerEvent(PointerEventPass.Initial)if (event.changes.firstOrNull()?.changedToDown() == true) {val down = awaitFirstDown(requireUnconsumed = true)state.setPressed(true)val newPosition = (event.changes.first().position.x / sliderWidth).coerceIn(0f, 1f)val newValue = state.range.start + newPosition * (state.range.endInclusive - state.range.start)state.updateValue(newValue)onValueChange?.invoke(state.value)var drag: PointerInputChange?do {drag = awaitTouchSlopOrCancellation(down.id) { change, _ ->if (change.positionChange() != Offset.Zero) change.consume()}} while (drag != null && !drag.isConsumed)if (drag != null) {val dragResult = drag(drag.id) {val newPosition = (it.position.x / sliderWidth).coerceIn(0f, 1f)val newValue = state.range.start + newPosition * (state.range.endInclusive - state.range.start)state.updateValue(newValue)onValueChange?.invoke(state.value)}// 确保完成回调被触发if (!dragResult || drag.isConsumed) {onValueChangeFinished?.invoke(state.value)}} else {onValueChangeFinished?.invoke(state.value)}}}},contentAlignment = Alignment.CenterStart) {// 绘制轨道(Track)Canvas(modifier = Modifier.fillMaxWidth().clip(CircleShape).height(trackHeight).align(Alignment.CenterStart).onSizeChanged { size ->sliderWidth = size.width.toFloat()}) {// 背景轨道drawRect(color = trackColor,size = Size(size.width.toFloat(), trackHeight.toPx()))// 进度轨道drawRect(color = progressColor,size = Size(width = (state.value - state.range.start) / (state.range.endInclusive - state.range.start) * size.width,height = trackHeight.toPx()))}// 滑块(Thumb)根据你自己需求替换Box(modifier = Modifier.offset {IntOffset(x = ((state.value - state.range.start) / (state.range.endInclusive - state.range.start) * sliderWidth - thumbSize.toPx()/2).toInt(),y = 0)}.shadow(4.dp, CircleShape).size(thumbSize).background(thumbColor, CircleShape))}
}class CustomSliderState(initialValue: Float,internal val range: ClosedFloatingPointRange<Float> = 0f..1f
) {private val _value = mutableFloatStateOf(initialValue.coerceIn(range))private val _isPressed = mutableStateOf(false) // 新增按下状态val value: Floatget() = _value.floatValueval isPressed: Boolean // 暴露给外部get() = _isPressed.valuefun updateValue(newValue: Float) {_value.floatValue = newValue.coerceIn(range)}fun setPressed(pressed: Boolean) { // 更新按下状态_isPressed.value = pressed}
}@Composable
fun rememberCustomSliderState(initialValue: Float = 0f,range: ClosedFloatingPointRange<Float> = 0f..1f
): CustomSliderState {return remember { CustomSliderState(initialValue, range) }
}

 使用

 val sliderState = rememberCustomSliderState(initialValue = 0.5f,range = 0f..100f)CustomSlider(state = sliderState,modifier = Modifier.fillMaxWidth().padding(horizontal = 36.dp),trackHeight = 12.dp,thumbSize = 24.dp,trackColor = Color.LightGray,progressColor = Color(0xFF6200EE),thumbColor = Color.White,onValueChange = {Log.e("onValueChange","float---$it")},onValueChangeFinished = {Log.e("onValueChangeFinished","float---$it")})

相关文章:

  • JAVA CAS 详解
  • 【学习笔记】RTSP-Ovnif-GB28181
  • 5.Caffe
  • 如何抓取DP_AUX辅助通道数据
  • 简单实现shardingSphere + MybatisPlus分库分表2025
  • 【期末速成】软件项目管理
  • java使用WebMagic架构写个分布式爬虫
  • MySQL中触发器详解 触发器在自动化任务中的应用场景
  • 今日行情明日机会——20250611
  • 缓冲区(C语言缓冲区+内核缓冲区)一个例子解释他们的关系和作用!!!
  • TF-IDF算法的代码实践应用——关键词提取、文本分类、信息检索
  • AI时代的弯道超车之第二十五章:《生命3.0》未来AI有生命了怎么办?
  • Vuex 中Mutation 和Action介绍
  • Python环境搭建竞赛技术
  • wordpress搬家 数据库备份迁移
  • 大模型Transformer触顶带来的“热潮退去”,稀疏注意力架构创新或是未来
  • STM32外设学习之ADC
  • HNCTF2025 - Misc、Osint、Crypto WriteUp
  • 日语学习-日语知识点小记-进阶-JLPT-真题训练-N2阶段(1):单词部分练习
  • Linux操作系统基线检查与安全加固概述
  • 政府网站建设评语/佛山seo技术
  • 网站是否需要备案/百度的推广广告
  • 东莞wordpress/seo在线论坛
  • 移动路由器做网站服务器/爱站网关键词密度
  • 四川网络科技有限公司/seo关键词分类
  • 做网站没有高清图片怎么办/建站软件