Android Espresso - 如果未选中,请单击复选框

Posted

技术标签:

【中文标题】Android Espresso - 如果未选中,请单击复选框【英文标题】:Android Espresso - Click checkbox if not checked 【发布时间】:2016-10-15 14:45:26 【问题描述】:

我有onView(withId(R.id.check_box)).perform(click()),但我只想在未选中复选框的情况下执行此操作。如何在浓缩咖啡中做到这一点?

【问题讨论】:

【参考方案1】:

你的代码中有这样的东西:

@Rule
public ActivityTestRule<ActivityMain> ActivityMainRule = new ActivityTestRule<>(ActivityMain.class);

试试

if (!ActivityMainRule.getActivity().findViewById(R.id.check_box).isChecked()) 
    onView(withId(R.id.check_box)).perform(click())    

【讨论】:

Google 不建议这样做,它违背了 Espresso 的工作原理并且非常危险youtu.be/isihPOY2vS4?t=4m12s【参考方案2】:

这似乎是您测试的一部分,需要选中复选框。

您可以通过以下方式检查:

onView(withId(R.id.checkbox)).check(matches(not(isChecked())));

如果失败,您的测试将失败,这对您来说可能是件好事。然后,您可以在这种情况匹配后执行您想要的点击。

如果这不是您想要的情况,您能否详细解释一下您在此 Espresso 测试中尝试做的事情?

【讨论】:

我想确保在进行测试之前选中该复选框。它可能未选中的原因是因为另一个测试未选中它。 Espresso 不应该能够设置任何视图状态或以任何方式操作视图。如果没有其他方法可以构建您的测试,那么之前的答案将起作用,但又违反了 espresso 约定。我建议以一种使它们更加模块化的方式来构建你的测试,这样它们就不必操纵任何视图。 @kyrax @kyrax 你能接受我的回答吗?因为应该知道在浓缩咖啡测试中使用活动是非常非常糟糕的。 @kryax 我不确定为什么这是公认的答案,因为这不能解决原始问题。如果您想在不通过测试的情况下操作视图,请查看我发布的解决方案。【参考方案3】:

我有同样的问题,我写了一个自己的 ViewAction 总是取消选中我的 SwitchCompat

class UncheckViewAction implements ViewAction

    @Override
    public Matcher<View> getConstraints() 
        return new Matcher<View>() 
            @Override
            public boolean matches(Object item) 
                return isA(SwitchCompat.class).matches(item);
            

            @Override
            public void describeMismatch(Object item, Description mismatchDescription) 

            

            @Override
            public void _dont_implement_Matcher___instead_extend_BaseMatcher_() 

            

            @Override
            public void describeTo(Description description) 

            
        ;
    

    @Override
    public String getDescription() 
        return null;
    

    @Override
    public void perform(UiController uiController, View view) 
        SwitchCompat scView = (SwitchCompat) view;
        scView.setChecked(false);
    

并像这样使用它:

onView(withId(R.id.check_box)).perform(new UncheckViewAction())

现在我可以确定找到的开关始终处于未选中状态,无论之前是选中还是未选中。

【讨论】:

dont_implement_Matcher___instead_extend_BaseMatcher【参考方案4】:

我还想根据之前的状态切换复选框/开关。起初,我尝试用这个来打开一个关闭的复选框:

onView(withId(R.id.checkbox)).check(matches(isNotChecked())).perform(scrollTo(), click());

...这将关闭一个已打开的复选框:

onView(withId(R.id.checkbox)).check(matches(isChecked())).perform(scrollTo(), click());

但是,这不起作用,因为 Espresso 会在执行操作之前寻找特定的切换状态。有时,您不知道它是打开还是关闭。

我的解决方案是使用自定义 ViewAction 来关闭/打开任何不依赖于先前状态的可检查对象(开关、复选框等)。因此,如果它已经打开,它将保持打开状态。如果它关闭,它将切换到 ON。这是 ViewAction:

public static ViewAction setChecked(final boolean checked) 
    return new ViewAction() 
        @Override
        public BaseMatcher<View> getConstraints() 
            return new BaseMatcher<View>() 
                @Override
                public boolean matches(Object item) 
                    return isA(Checkable.class).matches(item);
                

                @Override
                public void describeMismatch(Object item, Description mismatchDescription) 

                @Override
                public void describeTo(Description description) 
            ;
        

        @Override
        public String getDescription() 
            return null;
        

        @Override
        public void perform(UiController uiController, View view) 
            Checkable checkableView = (Checkable) view;
            checkableView.setChecked(checked);
        
    ;

下面是你如何使用它(在这个例子中,当你想要切换到 ON 时):

onView(withId(R.id.toggle)).perform(scrollTo(), setChecked(true));

【讨论】:

完美,谢谢@FrostRocket 我唯一需要更改的是删除“scrollTo()” ViewAction,因为我收到错误消息,提示我的复选框不支持 scrollTo:onView(withId(R.id.global_search)).perform(setChecked(globalSearch)); 就像你一样看,我想将复选框设置为提供的“globalSearch”变量的值 我更新了答案,添加了对复选框当前状态的检查,并仅在其状态不是必需时才更改它以避免不必要的事件:if (checkableView.isChecked() != checked) checkableView.setChecked(checked); setChecked 的内部实现已经考虑到了这一点。 这是一个绝佳的选择。但是,请注意,如果您在活动中基于被单击的复选框执行操作,则不会触发它。 @Stonz2 有一个好处——setChecked() 不会触发侦听器操作。要解决此问题,请改为单击视图:if (checkableView.isChecked() != checked) click().perform(uiController, view);【参考方案5】:

正如@FrostRocket 建议的那样,但用 Kotlin 编写。

我们定义了一个只能对可检查项目执行的自定义操作(在约束中指定)。所以我们安全地将视图转换为 Checkable 以访问 setCheckable 方法。

fun setChecked(checked: Boolean) = object : ViewAction 
    val checkableViewMatcher = object : BaseMatcher<View>() 
        override fun matches(item: Any?): Boolean = isA(Checkable::class.java).matches(item)
        override fun describeTo(description: Description?) 
            description?.appendText("is Checkable instance ")
        
    

    override fun getConstraints(): BaseMatcher<View> = checkableViewMatcher
    override fun getDescription(): String? = null
    override fun perform(uiController: UiController?, view: View) 
        val checkableView: Checkable = view as Checkable
        checkableView.isChecked = checked
    

【讨论】:

【参考方案6】:

这个简单的解决方案可能会对您有所帮助

onView(withId(R.id.checkBox_tea)).check(matches(isNotChecked())).perform(click()).check(matches(isChecked()));

非常适合我。

【讨论】:

我不明白这不是公认的答案......它是唯一没有任何操作/覆盖的答案,它只需要一行,不需要额外的类

以上是关于Android Espresso - 如果未选中,请单击复选框的主要内容,如果未能解决你的问题,请参考以下文章

Android/Gradle espresso 测试未开始活动

Android Espresso - 嵌套父母的组合视图匹配器

Android测试:Espresso 自动化测试

在 Espresso 测试中禁用自动更正功能

使用Espresso Test Recorder编写Android测试

Android自动化测试--Espresso框架使用