迁移到指示和涟漪 API

为了提高使用 Modifier.clickable 的交互式组件的合成性能,我们引入了新的 API。这些 API 允许更有效地实现 Indication,例如水波纹。

androidx.compose.foundation:foundation:1.7.0+androidx.compose.material:material-ripple:1.7.0+ 包含以下 API 更改

已弃用

替换

Indication#rememberUpdatedInstance

IndicationNodeFactory

rememberRipple()

Material 库中提供了新的 ripple() API。

注意:在此上下文中,“Material 库”指的是 androidx.compose.material:materialandroidx.compose.material3:material3androidx.wear.compose:compose-materialandroidx.wear.compose:compose-material3.

RippleTheme

或者

  • 使用 Material 库的 RippleConfiguration API,或者
  • 构建您自己的设计系统水波纹实现

此页面描述了行为更改的影响以及迁移到新 API 的说明。

行为更改

以下库版本包含水波纹行为更改

  • androidx.compose.material:material:1.7.0+
  • androidx.compose.material3:material3:1.3.0+
  • androidx.wear.compose:compose-material:1.4.0+

这些版本的 Material 库不再使用 rememberRipple();而是使用新的 ripple API。因此,它们不会查询 LocalRippleTheme。因此,如果您在应用程序中设置了 LocalRippleTheme,**Material 组件将不会使用这些值**。

下一节描述了如何在不迁移的情况下临时回退到旧行为;但是,我们建议迁移到新的 API。有关迁移说明,请参阅rememberRipple 迁移到 ripple 和后续部分。

升级 Material 库版本而不迁移

为了解除升级库版本的障碍,您可以使用临时的 LocalUseFallbackRippleImplementation CompositionLocal API 将 Material 组件配置为回退到旧行为

CompositionLocalProvider(LocalUseFallbackRippleImplementation provides true) {
    MaterialTheme {
        App()
    }
}

确保在 MaterialTheme外部提供此项,以便可以通过 LocalIndication 提供旧水波纹。

以下部分描述了如何迁移到新的 API。

rememberRipple 迁移到 ripple

使用 Material 库

如果您正在使用 Material 库,请直接将 rememberRipple() 替换为对来自相应库的 ripple() 的调用。此 API 使用从 Material 主题 API 派生的值创建水波纹。然后,将返回的对象传递给 Modifier.clickable 和/或其他组件。

例如,以下代码段使用了已弃用的 API

Box(
    Modifier.clickable(
        onClick = {},
        interactionSource = remember { MutableInteractionSource() },
        indication = rememberRipple()
    )
) {
    // ...
}

您应该将上述代码段修改为

@Composable
private fun RippleExample() {
    Box(
        Modifier.clickable(
            onClick = {},
            interactionSource = remember { MutableInteractionSource() },
            indication = ripple()
        )
    ) {
        // ...
    }
}

请注意,ripple() 不再是可组合函数,也不需要记住。它也可以在多个组件之间重复使用,类似于修饰符,因此请考虑将水波纹创建提取到顶级值以节省分配。

实现自定义设计系统

如果您正在实现自己的设计系统,并且之前正在使用 rememberRipple() 以及自定义 RippleTheme 来配置水波纹,则应改为提供自己的水波纹 API,该 API 代理到 material-ripple 中公开的水波纹节点 API。然后,您的组件可以使用直接使用主题值的自定义水波纹。有关更多信息,请参阅RippleTheme 迁移

RippleTheme 迁移

临时退出行为更改

Material 库有一个临时的 CompositionLocalLocalUseFallbackRippleImplementation,您可以使用它将所有 Material 组件配置为回退到使用 rememberRipple。这样,rememberRipple 将继续查询 LocalRippleTheme

以下代码段演示了如何使用 LocalUseFallbackRippleImplementation CompositionLocal API

CompositionLocalProvider(LocalUseFallbackRippleImplementation provides true) {
    MaterialTheme {
        App()
    }
}

如果您正在使用构建在 Material 之上的自定义应用程序主题,则可以安全地将组合本地作为应用程序主题的一部分提供

@OptIn(ExperimentalMaterialApi::class)
@Composable
fun MyAppTheme(content: @Composable () -> Unit) {
    CompositionLocalProvider(LocalUseFallbackRippleImplementation provides true) {
        MaterialTheme(content = content)
    }
}

有关更多信息,请参阅升级 Material 库版本而不迁移 部分。

使用 RippleTheme 为给定组件禁用水波纹

materialmaterial3 库公开了 RippleConfigurationLocalRippleConfiguration,它们允许您配置子树中水波纹的外观。请注意,RippleConfigurationLocalRippleConfiguration 处于实验阶段,仅用于每个组件的自定义。这些 API 不支持全局/主题范围的自定义;有关该用例的更多信息,请参阅使用 RippleTheme 全局更改应用程序中的所有水波纹

例如,以下代码段使用了已弃用的 API

private object DisabledRippleTheme : RippleTheme {

    @Composable
    override fun defaultColor(): Color = Color.Transparent

    @Composable
    override fun rippleAlpha(): RippleAlpha = RippleAlpha(0f, 0f, 0f, 0f)
}

// ...
    CompositionLocalProvider(LocalRippleTheme provides DisabledRippleTheme) {
        Button {
            // ...
        }
    }

您应该将上述代码段修改为

CompositionLocalProvider(LocalRippleConfiguration provides null) {
    Button {
        // ...
    }
}

使用 RippleTheme 更改给定组件的水波纹颜色/alpha

如上一节所述,RippleConfigurationLocalRippleConfiguration 是实验性 API,仅用于每个组件的自定义。

例如,以下代码段使用了已弃用的 API

private object DisabledRippleThemeColorAndAlpha : RippleTheme {

    @Composable
    override fun defaultColor(): Color = Color.Red

    @Composable
    override fun rippleAlpha(): RippleAlpha = MyRippleAlpha
}

// ...
    CompositionLocalProvider(LocalRippleTheme provides DisabledRippleThemeColorAndAlpha) {
        Button {
            // ...
        }
    }

您应该将上述代码段修改为

@OptIn(ExperimentalMaterialApi::class)
private val MyRippleConfiguration =
    RippleConfiguration(color = Color.Red, rippleAlpha = MyRippleAlpha)

// ...
    CompositionLocalProvider(LocalRippleConfiguration provides MyRippleConfiguration) {
        Button {
            // ...
        }
    }

使用 RippleTheme 全局更改应用程序中的所有水波纹

以前,您可以使用 LocalRippleTheme 在主题级别定义水波纹行为。这本质上是自定义设计系统组合本地和水波纹之间的集成点。现在,material-ripple 公开了 createRippleModifierNode() 函数,而不是公开通用的主题化原语。此函数允许设计系统库创建更高阶的 wrapper 实现,查询其主题值,然后将水波纹实现委托给此函数创建的节点。

这允许设计系统直接查询它们所需的内容,并在顶部公开任何所需的由用户配置的主题层,而无需符合 material-ripple 层提供的主题。此更改还使水波纹符合的主题/规范更加明确,因为是水波纹 API 本身定义了该契约,而不是隐式地从主题派生。

有关指导,请参阅 Material 库中的水波纹 API 实现,并根据需要替换对 Material 组合本地的调用以用于您自己的设计系统。

Indication 迁移到 IndicationNodeFactory

传递 Indication

如果您只是创建 Indication 以传递,例如创建要传递给 Modifier.clickableModifier.indication 的水波纹,则无需进行任何更改。IndicationNodeFactory 继承自 Indication,因此所有内容都将继续编译并工作。

创建 Indication

如果您正在创建自己的 Indication 实现,则在大多数情况下迁移应该很简单。例如,考虑一个在按下时应用缩放效果的 Indication

object ScaleIndication : Indication {
    @Composable
    override fun rememberUpdatedInstance(interactionSource: InteractionSource): IndicationInstance {
        // key the remember against interactionSource, so if it changes we create a new instance
        val instance = remember(interactionSource) { ScaleIndicationInstance() }

        LaunchedEffect(interactionSource) {
            interactionSource.interactions.collectLatest { interaction ->
                when (interaction) {
                    is PressInteraction.Press -> instance.animateToPressed(interaction.pressPosition)
                    is PressInteraction.Release -> instance.animateToResting()
                    is PressInteraction.Cancel -> instance.animateToResting()
                }
            }
        }

        return instance
    }
}

private class ScaleIndicationInstance : IndicationInstance {
    var currentPressPosition: Offset = Offset.Zero
    val animatedScalePercent = Animatable(1f)

    suspend fun animateToPressed(pressPosition: Offset) {
        currentPressPosition = pressPosition
        animatedScalePercent.animateTo(0.9f, spring())
    }

    suspend fun animateToResting() {
        animatedScalePercent.animateTo(1f, spring())
    }

    override fun ContentDrawScope.drawIndication() {
        scale(
            scale = animatedScalePercent.value,
            pivot = currentPressPosition
        ) {
            this@drawIndication.drawContent()
        }
    }
}

您可以分两步迁移此项

  1. ScaleIndicationInstance 迁移为 DrawModifierNodeDrawModifierNode 的 API 表面与 IndicationInstance 非常相似:它公开了 ContentDrawScope#draw() 函数,该函数在功能上等效于 IndicationInstance#drawContent()。您需要更改该函数,然后在节点内部直接实现 collectLatest 逻辑,而不是在 Indication 中。

    例如,以下代码段使用了已弃用的 API

    private class ScaleIndicationInstance : IndicationInstance {
        var currentPressPosition: Offset = Offset.Zero
        val animatedScalePercent = Animatable(1f)
    
        suspend fun animateToPressed(pressPosition: Offset) {
            currentPressPosition = pressPosition
            animatedScalePercent.animateTo(0.9f, spring())
        }
    
        suspend fun animateToResting() {
            animatedScalePercent.animateTo(1f, spring())
        }
    
        override fun ContentDrawScope.drawIndication() {
            scale(
                scale = animatedScalePercent.value,
                pivot = currentPressPosition
            ) {
                this@drawIndication.drawContent()
            }
        }
    }

    您应该将上述代码段修改为

    private class ScaleIndicationNode(
        private val interactionSource: InteractionSource
    ) : Modifier.Node(), DrawModifierNode {
        var currentPressPosition: Offset = Offset.Zero
        val animatedScalePercent = Animatable(1f)
    
        private suspend fun animateToPressed(pressPosition: Offset) {
            currentPressPosition = pressPosition
            animatedScalePercent.animateTo(0.9f, spring())
        }
    
        private suspend fun animateToResting() {
            animatedScalePercent.animateTo(1f, spring())
        }
    
        override fun onAttach() {
            coroutineScope.launch {
                interactionSource.interactions.collectLatest { interaction ->
                    when (interaction) {
                        is PressInteraction.Press -> animateToPressed(interaction.pressPosition)
                        is PressInteraction.Release -> animateToResting()
                        is PressInteraction.Cancel -> animateToResting()
                    }
                }
            }
        }
    
        override fun ContentDrawScope.draw() {
            scale(
                scale = animatedScalePercent.value,
                pivot = currentPressPosition
            ) {
                this@draw.drawContent()
            }
        }
    }

  2. ScaleIndication 迁移为实现 IndicationNodeFactory。由于收集逻辑现在已移至节点中,因此这是一个非常简单的工厂对象,其唯一职责是创建节点实例。

    例如,以下代码段使用了已弃用的 API

    object ScaleIndication : Indication {
        @Composable
        override fun rememberUpdatedInstance(interactionSource: InteractionSource): IndicationInstance {
            // key the remember against interactionSource, so if it changes we create a new instance
            val instance = remember(interactionSource) { ScaleIndicationInstance() }
    
            LaunchedEffect(interactionSource) {
                interactionSource.interactions.collectLatest { interaction ->
                    when (interaction) {
                        is PressInteraction.Press -> instance.animateToPressed(interaction.pressPosition)
                        is PressInteraction.Release -> instance.animateToResting()
                        is PressInteraction.Cancel -> instance.animateToResting()
                    }
                }
            }
    
            return instance
        }
    }

    您应该将上述代码段修改为

    object ScaleIndicationNodeFactory : IndicationNodeFactory {
        override fun create(interactionSource: InteractionSource): DelegatableNode {
            return ScaleIndicationNode(interactionSource)
        }
    
        override fun hashCode(): Int = -1
    
        override fun equals(other: Any?) = other === this
    }

使用 Indication 创建 IndicationInstance

在大多数情况下,您应该使用 Modifier.indication 来显示组件的 Indication。但是,在您使用 rememberUpdatedInstance 手动创建 IndicationInstance 的罕见情况下,您需要更新您的实现以检查 Indication 是否为 IndicationNodeFactory,以便您可以使用更轻量级的实现。例如,Modifier.indication 会在内部委托给创建的节点(如果它是 IndicationNodeFactory)。如果不是,它将使用 Modifier.composed 调用 rememberUpdatedInstance