许多 Composable 内置支持点击,并包含一个onClick
lambda 表达式。例如,您可以创建一个可点击的Surface
,其中包含适用于与表面交互的所有 Material Design 行为。
Surface(onClick = { /* handle click */ }) { Text("Click me!", Modifier.padding(24.dp)) }
但是,点击不是用户与 Composable 交互的唯一方式。此页面重点介绍涉及单个指针的手势,其中该指针的位置对于处理该事件并不重要。下表列出了这些类型的手势。
手势 |
描述 |
点击 |
指针按下然后抬起 |
双击 |
指针按下,抬起,按下,抬起 |
长按 |
指针按下并保持较长时间 |
按下 |
指针按下 |
响应点击
clickable
是一个常用修饰符,使 Composable 对点击做出反应。此修饰符还添加了其他功能,例如对焦点、鼠标和触笔悬停的支持,以及按下时可自定义的视觉指示。此修饰符以最广泛的意义响应“点击”——不仅是鼠标或手指,还包括通过键盘输入或使用辅助功能服务时的点击事件。
想象一下一个图像网格,当用户点击图像时,图像会全屏显示。
您可以将clickable
修饰符添加到网格中的每个项目以实现此行为。
@Composable private fun ImageGrid(photos: List<Photo>) { var activePhotoId by rememberSaveable { mutableStateOf<Int?>(null) } LazyVerticalGrid(columns = GridCells.Adaptive(minSize = 128.dp)) { items(photos, { it.id }) { photo -> ImageItem( photo, Modifier.clickable { activePhotoId = photo.id } ) } } if (activePhotoId != null) { FullScreenImage( photo = photos.first { it.id == activePhotoId }, onDismiss = { activePhotoId = null } ) } }
clickable
修饰符还添加了其他行为。
interactionSource
和indication
,默认情况下,当用户点击 Composable 时,它们会绘制一个涟漪效果。了解如何在处理用户交互页面上自定义这些内容。- 允许辅助功能服务通过设置语义信息与元素交互。
- 通过允许焦点并按下
Enter
或方向键中心来支持键盘或操纵杆交互。 - 使元素可悬停,以便它对鼠标或触笔悬停在其上方做出响应。
长按以显示上下文菜单。
combinedClickable
允许您除了正常的点击行为外,还添加双击或长按行为。您可以使用combinedClickable
在用户触摸并按住网格图像时显示上下文菜单。
var contextMenuPhotoId by rememberSaveable { mutableStateOf<Int?>(null) } val haptics = LocalHapticFeedback.current LazyVerticalGrid(columns = GridCells.Adaptive(minSize = 128.dp)) { items(photos, { it.id }) { photo -> ImageItem( photo, Modifier .combinedClickable( onClick = { activePhotoId = photo.id }, onLongClick = { haptics.performHapticFeedback(HapticFeedbackType.LongPress) contextMenuPhotoId = photo.id }, onLongClickLabel = stringResource(R.string.open_context_menu) ) ) } } if (contextMenuPhotoId != null) { PhotoActionsSheet( photo = photos.first { it.id == contextMenuPhotoId }, onDismissSheet = { contextMenuPhotoId = null } ) }
最佳实践是在用户长按元素时包含触觉反馈,这就是代码片段包含performHapticFeedback
调用的原因。
通过点击遮罩来关闭 Composable
在上面的示例中,clickable
和combinedClickable
为您的 Composable 添加了有用的功能。它们在交互时显示视觉指示,响应悬停,并包括焦点、键盘和辅助功能支持。但是,这种额外的行为并不总是理想的。
让我们看一下图像详情屏幕。背景应该是半透明的,用户应该能够点击该背景以关闭详情屏幕。
在这种情况下,该背景不应该在交互时有任何视觉指示,不应该响应悬停,不应该可聚焦,并且它对键盘和辅助功能事件的响应与典型的 Composable 不同。与其尝试调整clickable
行为,不如降到较低的抽象级别,直接使用pointerInput
修饰符与detectTapGestures
方法结合使用。
@Composable private fun Scrim(onClose: () -> Unit, modifier: Modifier = Modifier) { val strClose = stringResource(R.string.close) Box( modifier // handle pointer input .pointerInput(onClose) { detectTapGestures { onClose() } } // handle accessibility services .semantics(mergeDescendants = true) { contentDescription = strClose onClick { onClose() true } } // handle physical keyboard input .onKeyEvent { if (it.key == Key.Escape) { onClose() true } else { false } } // draw scrim .background(Color.DarkGray.copy(alpha = 0.75f)) ) }
作为pointerInput
修饰符的关键,您传递onClose
lambda 表达式。这会自动重新执行 lambda 表达式,确保在用户点击遮罩时调用正确的回调。
双击缩放
有时clickable
和combinedClickable
包含的信息不足以以正确的方式响应交互。例如,Composable 可能需要访问交互发生在 Composable 边界内的位置。
让我们再次看一下图像详情屏幕。最佳实践是可以通过双击放大图像。
正如您在视频中看到的,缩放发生在点击事件的位置附近。当我们在图像的左侧放大与在右侧放大时,结果是不同的。我们可以使用pointerInput
修饰符与detectTapGestures
结合使用,将点击位置合并到我们的计算中。
var zoomed by remember { mutableStateOf(false) } var zoomOffset by remember { mutableStateOf(Offset.Zero) } Image( painter = rememberAsyncImagePainter(model = photo.highResUrl), contentDescription = null, modifier = modifier .pointerInput(Unit) { detectTapGestures( onDoubleTap = { tapOffset -> zoomOffset = if (zoomed) Offset.Zero else calculateOffset(tapOffset, size) zoomed = !zoomed } ) } .graphicsLayer { scaleX = if (zoomed) 2f else 1f scaleY = if (zoomed) 2f else 1f translationX = zoomOffset.x translationY = zoomOffset.y } )
为您推荐
- 注意:当 JavaScript 关闭时,会显示链接文本。
- 了解手势
- Compose 中的 Material Design 2
- Jetpack Compose 的 Kotlin