Espresso - 检查 RecyclerView 项目是否订购正确

问题描述 投票:0回答:5

如何使用 Espresso 检查 RecyclerView 项目是否按正确顺序显示?我正在尝试测试它,通过每个元素的标题文本检查它。

当我尝试这段代码时,它可以单击该元素,但无法继续执行单击尝试断言该元素的文本

onView(withId(R.id.rv_metrics)).perform(actionOnItemAtPosition(0, click()));

当我尝试使用自定义匹配器时,我不断收到错误

Error performing 'load adapter data' on view 'with id: mypackage_name:id/rv_metrics'

我现在知道了

onData doesn't work for RecyclerView
但在此之前我尝试使用自定义匹配器来完成此任务。

 public static Matcher<Object> hasTitle(final String inputString) {
    return new BoundedMatcher<Object, Metric>(Metric.class) {
        @Override
        protected boolean matchesSafely(Metric metric) {

            return inputString.equals(metric.getMetric());

        }

        @Override
        public void describeTo(org.hamcrest.Description description) {
            description.appendText("with title: ");
        }
    };
}

我也尝试过类似的方法,但由于作为 actionOnItemAtPosition 方法的参数给出的类型,它显然不起作用,但我们是否有类似的东西可以工作?

onView(withId(R.id.rv_metrics)).check(actionOnItemAtPosition(0, ViewAssertions.matches(withText("Weight"))));

请问我在这里缺少什么? 非常感谢。

android android-recyclerview android-espresso
5个回答
10
投票

正如here所提到的,

RecyclerView
对象的工作方式与
AdapterView
对象不同,因此
onData()
不能用于与它们交互。

为了在

RecyclerView
的特定位置找到视图,您需要实现自定义
RecyclerViewMatcher
,如下所示:

public class RecyclerViewMatcher {
    private final int recyclerViewId;

    public RecyclerViewMatcher(int recyclerViewId) {
        this.recyclerViewId = recyclerViewId;
    }

    public Matcher<View> atPosition(final int position) {
        return atPositionOnView(position, -1);
    }

    public Matcher<View> atPositionOnView(final int position, final int targetViewId) {

        return new TypeSafeMatcher<View>() {
            Resources resources = null;
            View childView;

            public void describeTo(Description description) {
                String idDescription = Integer.toString(recyclerViewId);
                if (this.resources != null) {
                    try {
                        idDescription = this.resources.getResourceName(recyclerViewId);
                    } catch (Resources.NotFoundException var4) {
                        idDescription = String.format("%s (resource name not found)",
                                                      new Object[] { Integer.valueOf
                                                          (recyclerViewId) });
                    }
                }

                description.appendText("with id: " + idDescription);
            }

            public boolean matchesSafely(View view) {

                this.resources = view.getResources();

                if (childView == null) {
                    RecyclerView recyclerView =
                        (RecyclerView) view.getRootView().findViewById(recyclerViewId);
                    if (recyclerView != null && recyclerView.getId() == recyclerViewId) {
                        childView = recyclerView.findViewHolderForAdapterPosition(position).itemView;
                    }
                    else {
                        return false;
                    }
                }

                if (targetViewId == -1) {
                    return view == childView;
                } else {
                    View targetView = childView.findViewById(targetViewId);
                    return view == targetView;
                }

            }
        };
    }
}

然后以这种方式在您的测试用例中使用它:

@Test
void testCase() {
    onView(new RecyclerViewMatcher(R.id.rv_metrics)
        .atPositionOnView(0, R.id.txt_title))
        .check(matches(withText("Weight")))
        .perform(click());

    onView(new RecyclerViewMatcher(R.id.rv_metrics)
        .atPositionOnView(1, R.id.txt_title))
        .check(matches(withText("Height")))
        .perform(click());
}

4
投票

如果有人对 Kotlin 版本感兴趣,这里是

fun hasItemAtPosition(position: Int, matcher: Matcher<View>) : Matcher<View> {
    return object : BoundedMatcher<View, RecyclerView>(RecyclerView::class.java) {

        override fun describeTo(description: Description?) {
            description?.appendText("has item at position $position : ")
            matcher.describeTo(description)
        }

        override fun matchesSafely(item: RecyclerView?): Boolean {
            val viewHolder = item?.findViewHolderForAdapterPosition(position)
            return matcher.matches(viewHolder?.itemView)
        }
    }
}

用途:

onView(withId(R.id.recycler_view)).check(matches(hasItemAtPosition(0, hasDescendant(withText("item_1_title")))))
//or use hasDescendant(withId(R.id.item_1_title))

1
投票

我简化了一点 Mosius 的回答:

public static Matcher<View> hasItemAtPosition(final Matcher<View> matcher, final int position) {
    return new BoundedMatcher<View, RecyclerView>(RecyclerView.class) {
        @Override
        public void describeTo(Description description) {
            description.appendText("has item at position " + position + ": ");
            matcher.describeTo(description);
        }
        @Override
        protected boolean matchesSafely(RecyclerView recyclerView) {
            RecyclerView.ViewHolder viewHolder = recyclerView.findViewHolderForAdapterPosition(position);
            return matcher.matches(viewHolder.itemView);
        }
    };
}

我们将

Matcher
传递给函数,以便我们可以提供进一步的条件。用法示例:

onView(hasItemAtPosition(hasDescendant(withText("Item 1")), 0)).check(matches(isDisplayed()));
onView(hasItemAtPosition(hasDescendant(withText("Item 2")), 1)).check(matches(isDisplayed()));

1
投票

原来的问题已经解决,但我在这里发布一个答案,因为发现Barista库用一个单行代码解决了这个问题。

assertDisplayedAtPosition(R.id.rv_metrics, 0, R.id.tv_title, "weight");

它是在 Espresso 之上制作的,可以在 here

找到它的文档

希望这对某人有帮助。 :)


0
投票

如果您想在

RecyclerView
中的某个位置匹配匹配器,那么您可以尝试创建自定义
Matcher<View>
:

public static Matcher<View> hasItemAtPosition(int position, Matcher<View> matcher) {
    return new BoundedMatcher<View, RecyclerView>(RecyclerView.class) {

        @Override public void describeTo(Description description) {
            description.appendText("has item: ");
            matcher.describeTo(description);
            description.appendText(" at position: " + position);
        }

        @Override protected boolean matchesSafely(RecyclerView view) {
            RecyclerView.Adapter adapter = view.getAdapter();
            int type = adapter.getItemViewType(position);
            RecyclerView.ViewHolder holder = adapter.createViewHolder(view, type);
            adapter.onBindViewHolder(holder, position);
            return matcher.matches(holder.itemView);
        }
    };
}

您可以使用它,例如:

onView(withId(R.id.rv_metrics)).check(matches(0, hasDescendant(withText("Weight")))))
© www.soinside.com 2019 - 2024. All rights reserved.