Android Compose:如何在单元测试中验证开关是打开还是关闭

Mis*_*ith 6 android unit-testing android-jetpack-compose switch-component

我在可组合项中有一个 Switch:

Switch(
    checked = false,
    modifier = Modifier.testTag("mySwitch")
)
Run Code Online (Sandbox Code Playgroud)

我正在尝试通过组合单元测试来验证它:

composeTestRule.onAllNodesWithTag("mySwitch")
    .assertAll(isToggleable() and isOff())
Run Code Online (Sandbox Code Playgroud)

但是,它会失败,但出现以下异常:

java.lang.AssertionError: Failed to assertAll((ToggleableState is defined) && (ToggleableState = 'Off'))
Found '1' nodes not matching:
1) Node #8 at (l=955.0, t=387.0, r=1054.0, b=450.0)px, Tag: 'switch'
Has 4 siblings
Selector used: 'TestTag = 'mySwitch''
Run Code Online (Sandbox Code Playgroud)

显然,该开关既不能切换也不能“开/关”。我还分别检查了assertIsToggleableassertIsOff,但都失败了。

我已经验证 Switch 对于测试中使用的 UI 状态是可见的。

为什么我的测试失败?应该可以轻松测试一个该死的 Switch。开关正是“可切换”的定义。那么我该如何测试它,我应该使用自定义语义属性吗?


免责声明:这个问题与这个问题不同。我想验证 Switch 状态,而不是单击它(我稍后会尝试)

Mis*_*ith 4

我的测试代码毕竟是正确的。assertIs[Not]ToggleableassertIsOffassertIsOn、 和assertAll(isToggleable())应该可以在 Switch 上使用。但由于 Compose 1.2.1 中的错误(或奇怪的未记录功能),除非您将回调作为参数传递,否则 Switch 不会标记为“可切换”。

我通过检查 的源代码发现了这一点Switch.kt。这是导致问题的部分:

@Composable
    @OptIn(ExperimentalMaterialApi::class)
    fun Switch(
        checked: Boolean,
        onCheckedChange: ((Boolean) -> Unit)?,
        ...
    ) {
        ...
        val toggleableModifier =
            if (onCheckedChange != null) {
                Modifier.toggleable(
                    value = checked,
                    onValueChange = onCheckedChange,
                    enabled = enabled,
                    role = Role.Switch,
                    interactionSource = interactionSource,
                    indication = null
                )
            } else {
                Modifier
            }
Run Code Online (Sandbox Code Playgroud)

正如您所看到的,除非您传递可选参数onCheckedChange,否则 Switch 不会变为可切换。这反过来会导致所有isToggleable-like 断言抛出异常,并且isOnisOff断言也会发生同样的情况,因为它们还要求节点是“可切换的”。

因此,将虚拟onCheckedChangelambda 传递给测试代码中的开关解决了问题:

Switch(checked=True,onCheckedChange={})
Run Code Online (Sandbox Code Playgroud)