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

网站后期维护怎么做中国500强公司排名名单

网站后期维护怎么做,中国500强公司排名名单,深圳网上创建公司,最新新闻热点国家大事想要实现长按按钮开始录音,松开发送的功能。发现 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://www.dtcms.com/wzjs/547155.html

相关文章:

  • 江门网站建设推广平台站台建筑
  • 蓟县网站制作wordpress 多媒体管理
  • 网站制作九江网站模板在线预览
  • 网站平台需要做无形资产吗 怎么做wordpress时间格式
  • 网站在谷歌怎么做排名好网页设计培训
  • 网站推广该怎么做做家乡网站
  • 网站数据库5g深圳外贸网站建设
  • 珠海网站设计公司西部数码网站管理助手 mysql保存路径
  • 知名网站建设加工推广策略论文
  • 许昌企业网站建设大连网站平台研发
  • 好的平面设计网站有哪些html是静态网站
  • 淘宝客优惠券网站建设加盟官网重庆市两江新区建设管理局网站
  • 网站定制公司哪家好网站建设哪家技术好
  • iis7发布php网站wordpress 自学网
  • 制作哪个网站好北京金山办公软件股份有限公司官网
  • 手机网站建设的方法做调查问卷换赏金的网站
  • wordpress拿站网站建设怎么建好
  • 网站优化基础设计公司网站需要多少钱
  • 网站建设与管理 需求分析义乌营销型网站建设
  • 做漫画的网站有哪些公司的门户网站模版
  • 建设官方网站需要注意什么搜索引擎优化排名技巧
  • 电子商务网站建设需要什么网站价值评估怎么做
  • 整站seo定制wordpress页面发布不
  • 怎样解析网站域名有哪些好的做兼职的网站
  • 上海贸易网站建设茗匠智能门店管理系统
  • 网站开发分类wordpress 全部函数
  • 动漫做h免费网站发布设计任务的网站
  • 网站备案怎样提交管局ps网页排版设计
  • 国外的自建网站怎么做化妆品网站建设原因
  • 宝塔服务器搭建网站教程贵州省住房和城乡建设厅网站报名网