Использование Espresso для просмотра клика внутри элемента RecyclerView
Как я могу использовать Espresso для выбора определенного вида внутри элемента RecyclerView? Я знаю, что могу щелкнуть элемент в позиции 0, используя:
onView(withId(R.id.recyclerView))
.perform(RecyclerViewActions.actionOnItemAtPosition(0, click()));
Но мне нужно щелкнуть по определенному виду внутри этого элемента, а не по самому элементу.
Спасибо заранее.
- изменить -
Точнее: у меня есть RecyclerView (R.id.recycler_view
), элементы которого CardView (R.id.card_view
). Внутри каждого CardView у меня есть четыре кнопки (между прочим), и я хочу нажать на определенную кнопку (R.id.bt_deliver
).
Я хотел бы использовать новые функции Espresso 2.0, но я не уверен, что это возможно.
Если это невозможно, я хочу использовать что-то вроде этого (используя код Томаса Келлера):
onRecyclerItemView(R.id.card_view, ???, withId(R.id.bt_deliver)).perform(click());
но я не знаю, на что поставить вопросительные знаки.
Ответы
Ответ 1
Вы можете сделать это с настройкой действия вида.
public class MyViewAction {
public static ViewAction clickChildViewWithId(final int id) {
return new ViewAction() {
@Override
public Matcher<View> getConstraints() {
return null;
}
@Override
public String getDescription() {
return "Click on a child view with specified id.";
}
@Override
public void perform(UiController uiController, View view) {
View v = view.findViewById(id);
v.performClick();
}
};
}
}
Затем вы можете щелкнуть по нему с помощью
onView(withId(R.id.rv_conference_list)).perform(
RecyclerViewActions.actionOnItemAtPosition(0, MyViewAction.clickChildViewWithId(R.id. bt_deliver)));
Ответ 2
Теперь с помощью android.support.test.espresso.contrib стало проще:
1) Добавить тестовую зависимость
androidTestCompile('com.android.support.test.espresso:espresso-contrib:2.0') {
exclude group: 'com.android.support', module: 'appcompat'
exclude group: 'com.android.support', module: 'support-v4'
exclude module: 'recyclerview-v7'
}
* исключить 3 модуля, потому что, скорее всего, у вас уже есть это
2) Затем сделайте что-то вроде
onView(withId(R.id.recycler_grid))
.perform(RecyclerViewActions.actionOnItemAtPosition(0, click()));
или
onView(withId(R.id.recyclerView))
.perform(RecyclerViewActions.actionOnItem(
hasDescendant(withText("whatever")), click()));
или
onView(withId(R.id.recycler_linear))
.check(matches(hasDescendant(withText("whatever"))));
Ответ 3
Попробуйте следующий подход:
onView(withRecyclerView(R.id.recyclerView)
.atPositionOnView(position, R.id.bt_deliver))
.perform(click());
public static RecyclerViewMatcher withRecyclerView(final int recyclerViewId) {
return new RecyclerViewMatcher(recyclerViewId);
}
public class RecyclerViewMatcher {
final int mRecyclerViewId;
public RecyclerViewMatcher(int recyclerViewId) {
this.mRecyclerViewId = 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) {
int id = targetViewId == -1 ? mRecyclerViewId : targetViewId;
String idDescription = Integer.toString(id);
if (this.resources != null) {
try {
idDescription = this.resources.getResourceName(id);
} catch (Resources.NotFoundException var4) {
idDescription = String.format("%s (resource name not found)", id);
}
}
description.appendText("with id: " + idDescription);
}
public boolean matchesSafely(View view) {
this.resources = view.getResources();
if (childView == null) {
RecyclerView recyclerView =
(RecyclerView) view.getRootView().findViewById(mRecyclerViewId);
if (recyclerView != null) {
childView = recyclerView.findViewHolderForAdapterPosition(position).itemView;
}
else {
return false;
}
}
if (targetViewId == -1) {
return view == childView;
} else {
View targetView = childView.findViewById(targetViewId);
return view == targetView;
}
}
};
}
}
Ответ 4
Вот, как я решил вопрос в котлине:
fun clickOnViewChild(viewId: Int) = object : ViewAction {
override fun getConstraints() = null
override fun getDescription() = "Click on a child view with specified id."
override fun perform(uiController: UiController, view: View) = click().perform(uiController, view.findViewById<View>(viewId))
}
а потом
onView(withId(R.id.recyclerView)).perform(RecyclerViewActions.actionOnItemAtPosition<RecyclerView.ViewHolder>(position, clickOnViewChild(R.id.viewToClickInTheRow)))
Ответ 5
Сначала дайте своим кнопкам уникальные описания контента, то есть "строка 5 строки поставки".
<button android:contentDescription=".." />
Затем прокрутите до строки:
onView(withId(R.id.recycler_view)).perform(RecyclerViewActions.scrollToPosition(5));
Затем выберите представление, основанное на описании контента.
onView(withContentDescription("delivery button row 5")).perform(click());
Описание контента - отличный способ использовать Espresso onView и сделать ваше приложение более доступным.