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

寻找项目做的网站装饰工程经营范围有哪些

寻找项目做的网站,装饰工程经营范围有哪些,wordpress最新评论样式,网络营销策略分析案例想要实现长按按钮开始录音,松开发送的功能。发现 Button 这个控件如果去监听这些按下,松开,长按等事件,发现是不会触发的,究其原因是 Button 已经提前消耗了这些事件所以导致,这些监听无法被触发。因此为了…

想要实现长按按钮开始录音,松开发送的功能。发现 Button 这个控件如果去监听这些按下,松开,长按等事件,发现是不会触发的,究其原因是 Button 已经提前消耗了这些事件所以导致,这些监听无法被触发。因此为了实现这些功能就需要自己写一个 Button 来解决问题。

Button 实现原理

在 Jetpack Compose 中,Button 是一个高度封装的可组合函数(Composable),其底层是由多个 UI 组件组合而成,关键组成包括:Surface、Text、Row、InteractionSource 等

  • 源码
@Composable
fun Button(onClick: () -> Unit,modifier: Modifier = Modifier,enabled: Boolean = true,shape: Shape = ButtonDefaults.shape,colors: ButtonColors = ButtonDefaults.buttonColors(),elevation: ButtonElevation? = ButtonDefaults.buttonElevation(),border: BorderStroke? = null,contentPadding: PaddingValues = ButtonDefaults.ContentPadding,interactionSource: MutableInteractionSource? = null,content: @Composable RowScope.() -> Unit
) {@Suppress("NAME_SHADOWING")val interactionSource = interactionSource ?: remember { MutableInteractionSource() }val containerColor = colors.containerColor(enabled)val contentColor = colors.contentColor(enabled)val shadowElevation = elevation?.shadowElevation(enabled, interactionSource)?.value ?: 0.dpSurface(onClick = onClick,modifier = modifier.semantics { role = Role.Button },enabled = enabled,shape = shape,color = containerColor,contentColor = contentColor,shadowElevation = shadowElevation,border = border,interactionSource = interactionSource) {ProvideContentColorTextStyle(contentColor = contentColor,textStyle = MaterialTheme.typography.labelLarge) {Row(Modifier.defaultMinSize(minWidth = ButtonDefaults.MinWidth,minHeight = ButtonDefaults.MinHeight).padding(contentPadding),horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,content = content)}}
}

1. Surface 的作用(关键)

Surface 是 Compose 中的通用容器,负责:

  • 提供背景颜色(来自 ButtonColors)
  • 提供 elevation(阴影)
  • 提供点击行为(通过 onClick)
  • 提供 shape(圆角、裁剪等)
  • 提供 ripple 效果(内部自动通过 indication 使用 rememberRipple())
  • 使用 Modifier.clickable 实现交互响应

注意:几乎所有 Material 组件都会使用 Surface 来包裹内容,统一管理视觉表现。

2. InteractionSource

  • InteractionSource 是 Compose 中管理用户交互状态的机制(如 pressedhoveredfocused
  • Button 将其传入 Surface,用于响应和处理 ripple 动画等
  • MutableInteractionSource 配合,可以观察组件的状态变化

3. ButtonDefaults

ButtonDefaults 是提供默认值的工具类,包含:

  • elevation():返回 ButtonElevation 对象,用于设置不同状态下的阴影高度
  • buttonColors():返回 ButtonColors 对象,用于设置正常 / 禁用状态下的背景与文字颜色
  • ContentPadding:内容的默认内边距

4. Content Slot(RowScope.() -> Unit

Buttoncontent 是一个 RowScope 的 lambda,允许你自由组合子组件,如:

Button(onClick = { }) {Icon(imageVector = Icons.Default.Add, contentDescription = null)Spacer(modifier = Modifier.width(4.dp))Text("添加")
}

因为是在 RowScope 中,所以能用 Spacer 等布局函数在水平排列子项。


关键点说明
Surface提供背景、阴影、圆角、点击、ripple 效果的统一封装
InteractionSource用于收集用户交互状态(点击、悬停等)
ButtonDefaults提供默认颜色、阴影、Padding 等参数
Row + Text内容布局,允许图标 + 文本灵活组合
Modifier控制尺寸、形状、边距、点击响应等

如果想自定义 Button 的样式,也可以直接使用 Surface + Row 自己实现一个“按钮”,只需照着官方的做法组装即可。

@Suppress("DEPRECATION_ERROR")
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun Button(onClick: () -> Unit = {},onLongPress: () -> Unit = {},onPressed: () -> Unit = {},onReleased: () -> Unit = {},modifier: Modifier = Modifier,enabled: Boolean = true,shape: Shape = ButtonDefaults.shape,colors: ButtonColors = ButtonDefaults.buttonColors(),border: BorderStroke? = null,shadowElevation: Dp = 0.dp,contentPadding: PaddingValues = ButtonDefaults.ContentPadding,content: @Composable RowScope.() -> Unit = { Text("LongButton") }
) {val containerColor = colors.containerColorval contentColor = colors.contentColorSurface(modifier = modifier.minimumInteractiveComponentSize().pointerInput(enabled) {detectTapGestures(onPress = { offset ->onPressed()tryAwaitRelease()onReleased()},onTap = { onClick() },onLongPress = { onLongPress() })}.semantics { role = Role.Button },shape = shape,color = containerColor,contentColor = contentColor,shadowElevation = shadowElevation,border = border,) {CompositionLocalProvider(LocalContentColor provides contentColor,LocalTextStyle provides LocalTextStyle.current.merge(MaterialTheme.typography.labelLarge),) {Row(Modifier.defaultMinSize(ButtonDefaults.MinWidth, ButtonDefaults.MinHeight).padding(contentPadding),horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,content = content)}}
}

Button 的动画实现

为了让按钮在按下时提供自然的视觉反馈,Compose 通常会使用状态驱动的动画。最常见的方式是通过 animateColorAsState 来实现颜色的平滑过渡,比如按钮被按下时背景色或文字颜色稍微变暗,松开时再恢复。

这个动画实现的关键点在于:

  1. 交互状态:比如是否按下、是否禁用,可以通过 InteractionSource 结合 collectIsPressedAsState() 实时监听当前状态。
  2. 根据状态决定目标颜色:当状态变化时(如按下 -> 松开),我们会设置新的目标颜色。
  3. 使用动画驱动状态变化:通过 animateColorAsState() 把颜色变化变成带过渡效果的状态变化,而不是突变。

这种方式符合 Compose 的声明式编程模型,不需要手动写动画过程,而是让状态驱动 UI 动画。

下面是按钮颜色动画部分的代码片段,只展示相关的状态监听和动画逻辑,具体如何应用在 UI 中将在后续实现:

@Composable
fun AnimatedButtonColors(enabled: Boolean,interactionSource: InteractionSource,defaultContainerColor: Color,pressedContainerColor: Color,disabledContainerColor: Color
): State<Color> {val isPressed by interactionSource.collectIsPressedAsState()val targetColor = when {!enabled -> disabledContainerColorisPressed -> pressedContainerColorelse -> defaultContainerColor}// 返回一个状态驱动的动画颜色val animatedColor by animateColorAsState(targetColor, label = "containerColorAnimation")return rememberUpdatedState(animatedColor)
}

值得一提的是,Button 使用的动画类型为 ripple (涟漪效果)

这段代码仅负责计算当前的按钮背景色,并通过动画使其平滑过渡。它不会直接控制按钮的点击或布局逻辑,而是为最终的 UI 提供一个可动画的颜色状态。

后续可以将这个 animatedColor 应用于 Surface 或背景 Modifier 上,完成整体的按钮外观动画。

完整动画代码

// 1. 确保 interactionSource 不为空
val interaction = interactionSource ?: remember { MutableInteractionSource() }
// 2. 监听按下状态
val isPressed by interaction.collectIsPressedAsState()// 4. 按状态选 target 值
val defaultContainerColor = colors.containerColor
val disabledContainerColor = colors.disabledContainerColor
val defaultContentColor = colors.contentColor
val disabledContentColor = colors.disabledContentColorval targetContainerColor = when {!enabled -> disabledContainerColorisPressed -> defaultContainerColor.copy(alpha = 0.85f)else -> defaultContainerColor
}
val targetContentColor = when {!enabled -> disabledContentColorisPressed -> defaultContentColor.copy(alpha = 0.9f)else -> defaultContentColor
}// 5. 动画
val containerColorAni by animateColorAsState(targetContainerColor, label = "containerColor")
val contentColorAni by animateColorAsState(targetContentColor, label = "contentColor")// 涟漪效果
// 根据当前环境选择是否使用新版 Material3 的 ripple(),还是退回到老版的 rememberRipple() 实现
val ripple = if (LocalUseFallbackRippleImplementation.current) {rememberRipple(true, Dp.Unspecified, Color.Unspecified)
} else {ripple(true, Dp.Unspecified, Color.Unspecified)
}// 6. Surface + 手动发 PressInteraction
Surface(modifier = modifier.minimumInteractiveComponentSize().pointerInput(enabled) {detectTapGestures(onPress = { offset ->// 发起 PressInteraction,供 collectIsPressedAsState 监听val press = PressInteraction.Press(offset)val scope = CoroutineScope(coroutineContext)scope.launch {interaction.emit(press)}// 用户 onPressedonPressed()// 等待手指抬起或取消tryAwaitRelease()// 发 ReleaseInteractionscope.launch {interaction.emit(PressInteraction.Release(press))}// 用户 onReleasedonReleased()},onTap = { onClick() },onLongPress = { onLongPress() })}.indication(interaction, ripple).semantics { role = Role.Button },shape = shape,color = containerColorAni,contentColor = contentColorAni,shadowElevation = shadowElevation,border = border,
) {...}

这个 Button 的动画部分主要体现在按下状态下的颜色过渡。它通过 animateColorAsState 来实现背景色和文字颜色的动态变化。

当按钮被按下时,会使用 interaction.collectIsPressedAsState() 实时监听是否处于 Pressed 状态,进而动态计算目标颜色(targetContainerColortargetContentColor)。按下状态下颜色会降低透明度(背景 alpha = 0.85,文字 alpha = 0.9),形成按压视觉反馈。

颜色的渐变不是突变的,而是带有过渡动画,由 animateColorAsState 自动驱动。它会在目标颜色发生变化时,通过内部的动画插值器平滑过渡到目标值,用户无需手动控制动画过程。

使用 by animateColorAsState(...) 得到的是 State<Color> 类型的值,它会在颜色变化时自动重组,使整个按钮在交互中呈现更自然的过渡效果。

这种方式相比传统手动实现动画更简洁、声明性更强,也更容易和 Compose 的状态系统集成。

完整代码

// androidx.compose.material3: 1.3.0
import androidx.compose.animation.animateColorAsState
import androidx.compose.foundation.BorderStroke
import androidx.compose.foundation.gestures.detectTapGestures
import androidx.compose.foundation.indication
import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.interaction.PressInteraction
import androidx.compose.foundation.interaction.collectIsPressedAsState
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.RowScope
import androidx.compose.foundation.layout.defaultMinSize
import androidx.compose.foundation.layout.padding
import androidx.compose.material.ripple.rememberRipple
import androidx.compose.material3.Button
import androidx.compose.material3.ButtonColors
import androidx.compose.material3.ButtonDefaults
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.LocalContentColor
import androidx.compose.material3.LocalTextStyle
import androidx.compose.material3.LocalUseFallbackRippleImplementation
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.material3.minimumInteractiveComponentSize
import androidx.compose.material3.ripple
import androidx.compose.runtime.Composable
import androidx.compose.runtime.CompositionLocalProvider
import androidx.compose.runtime.getValue
import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.Shape
import androidx.compose.ui.input.pointer.pointerInput
import androidx.compose.ui.semantics.Role
import androidx.compose.ui.semantics.role
import androidx.compose.ui.semantics.semantics
import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.dp
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import kotlin.coroutines.coroutineContext@Suppress("DEPRECATION_ERROR")
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun Button(onClick: () -> Unit = {},onLongPress: () -> Unit = {},onPressed: () -> Unit = {},onReleased: () -> Unit = {},modifier: Modifier = Modifier,enabled: Boolean = true,shape: Shape = ButtonDefaults.shape,colors: ButtonColors = ButtonDefaults.buttonColors(),border: BorderStroke? = null,shadowElevation: Dp = 0.dp,contentPadding: PaddingValues = ButtonDefaults.ContentPadding,interactionSource: MutableInteractionSource? = null,content: @Composable RowScope.() -> Unit = { Text("LongButton") }
) {// 1. 确保 interactionSource 不为空val interaction = interactionSource ?: remember { MutableInteractionSource() }// 2. 监听按下状态val isPressed by interaction.collectIsPressedAsState()// 4. 按状态选 target 值val defaultContainerColor = colors.containerColorval disabledContainerColor = colors.disabledContainerColorval defaultContentColor = colors.contentColorval disabledContentColor = colors.disabledContentColorval targetContainerColor = when {!enabled -> disabledContainerColorisPressed -> defaultContainerColor.copy(alpha = 0.85f)else -> defaultContainerColor}val targetContentColor = when {!enabled -> disabledContentColorisPressed -> defaultContentColor.copy(alpha = 0.9f)else -> defaultContentColor}// 5. 动画val containerColorAni by animateColorAsState(targetContainerColor, label = "containerColor")val contentColorAni by animateColorAsState(targetContentColor, label = "contentColor")// 涟漪效果// 根据当前环境选择是否使用新版 Material3 的 ripple(),还是退回到老版的 rememberRipple() 实现val ripple = if (LocalUseFallbackRippleImplementation.current) {rememberRipple(true, Dp.Unspecified, Color.Unspecified)} else {ripple(true, Dp.Unspecified, Color.Unspecified)}// 6. Surface + 手动发 PressInteractionSurface(modifier = modifier.minimumInteractiveComponentSize().pointerInput(enabled) {detectTapGestures(onPress = { offset ->// 发起 PressInteraction,供 collectIsPressedAsState 监听val press = PressInteraction.Press(offset)val scope = CoroutineScope(coroutineContext)scope.launch {interaction.emit(press)}// 用户 onPressedonPressed()// 等待手指抬起或取消tryAwaitRelease()// 发 ReleaseInteractionscope.launch {interaction.emit(PressInteraction.Release(press))}// 用户 onReleasedonReleased()},onTap = { onClick() },onLongPress = { onLongPress() })}.indication(interaction, ripple).semantics { role = Role.Button },shape = shape,color = containerColorAni,contentColor = contentColorAni,shadowElevation = shadowElevation,border = border,) {CompositionLocalProvider(LocalContentColor provides contentColorAni,LocalTextStyle provides LocalTextStyle.current.merge(MaterialTheme.typography.labelLarge),) {Row(Modifier.defaultMinSize(ButtonDefaults.MinWidth, ButtonDefaults.MinHeight).padding(contentPadding),horizontalArrangement = Arrangement.Center,verticalAlignment = Alignment.CenterVertically,content = content)}}
}

文章转载自:

http://Xgy5johP.xtrnx.cn
http://tqi7UnN5.xtrnx.cn
http://hLJd1dhM.xtrnx.cn
http://4SR36LKu.xtrnx.cn
http://DPx607FQ.xtrnx.cn
http://wjky9jiV.xtrnx.cn
http://goRVEAK6.xtrnx.cn
http://jmCfdSld.xtrnx.cn
http://NbItk9Jv.xtrnx.cn
http://CtYApnQN.xtrnx.cn
http://WSWiXmzk.xtrnx.cn
http://GaYDumvk.xtrnx.cn
http://eimgkwIZ.xtrnx.cn
http://J2foTFUt.xtrnx.cn
http://kaVI9HJT.xtrnx.cn
http://WOncABWY.xtrnx.cn
http://ilyGFYcC.xtrnx.cn
http://tmGrZw5i.xtrnx.cn
http://fSYdh6p8.xtrnx.cn
http://OyEwjwbw.xtrnx.cn
http://Qtzn9Knw.xtrnx.cn
http://eE1SQIDs.xtrnx.cn
http://pVi5Ly0D.xtrnx.cn
http://V8oZdlHH.xtrnx.cn
http://ndJri81p.xtrnx.cn
http://1N663E7D.xtrnx.cn
http://bZxYdCCX.xtrnx.cn
http://PhgKY5gi.xtrnx.cn
http://HQh3zcvy.xtrnx.cn
http://mLze6fcf.xtrnx.cn
http://www.dtcms.com/wzjs/658328.html

相关文章:

  • 国内手机网站建设美术培训学校网站模板
  • element ui设计网站中企动力公司网站价格
  • 装饰网站设计模板wordpress新建页面显示数据
  • 网站代码在哪里看智能城市 电子商务网站建设
  • 做网站设计的价格wordpress音乐主题
  • 西安做企业网站秦皇岛哪有网站优化公司
  • 贵州建设厅报名登录网站宁波网站建设seo
  • 社交网站 ui兴安盟住房和城乡建设部网站
  • 网络设计网站网站刚通过备案
  • 怎么查看一个网站的后台淄博市住房和城乡建设厅网站
  • 网站使用什么数据库搭建专业网站服务器
  • 贵州做网站的如何对网站做优化
  • 生成logo的网站山东华建建设有限公司网站
  • 网站目录生成如何给网站做dns解析
  • 国外木屋建设网站网页界面设计总结与体会
  • 山东平台网站建设多少钱辽宁大连最新发布
  • 做初中试卷的网站wordpress 主题骨架
  • C语言网站开发pdf重庆市建设工程管理协会网站
  • 高明网站建设报价企业做网站都需要准备哪些材料
  • 网站建设拷贝软件福州网站建设信息
  • 网站开发的教学视频四川新冠感染最新消息
  • 深圳福田区住房和建设局官方网站全屋装修设计定制整装
  • 温州地区做网站杭州seo公司排名
  • 如何创建问卷网站seo推广一年要多少钱
  • 北京高端网站制作公司做箱包外贸哪个网站好
  • 重庆专业网站推广时间网站建设服务费进入什么科目
  • 婚庆摄影网站模板网页图片不能保存怎么破解
  • 沈阳黑酷做网站建设优化公司怎么样类似站酷的网站建站
  • 京东网站开发技术2核512内存装wordpress
  • 网站语言智慧团建入口登录网站