如何使用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"))));
我在这里想念什么? 非常感谢。
答案 0 :(得分: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)
}
}
}
答案 1 :(得分:3)
正如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());
}
答案 2 :(得分: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")))))
答案 3 :(得分:0)
我简化了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()));
答案 4 :(得分:0)
原来的问题已解决,但是我在这里发布了答案,发现 Barista 库用一条单行代码解决了该问题。
assertDisplayedAtPosition(R.id.rv_metrics, 0, R.id.tv_title, "weight");
它是在Espresso之上制成的,其文档可以在here
中找到希望这对某人可能有所帮助。 :)