Avoir gridView qui a quelques images. La cellule de gridView provient de la même disposition prédéfinie, qui a les mêmes id et desc.
R.id.item_image == 2131493330
onView(withId(is(R.id.item_image))).perform(click());
Etant donné que toutes les cellules de la grille ont le même identifiant, AmbiguousViewMatcherException
. Comment choisir simplement le premier ou l’un d’eux?? Merci.!
Android.support.test.espresso.AmbiguousViewMatcherException: 'avec l'ID: est <2131493330>' correspond à plusieurs vues dans la hiérarchie . Les vues de problème sont marquées avec '**** MATCHES ****' ci-dessous.
+ -------------> ImageView {id = 2131493330, nom-objet = image_article, desc = image, visibilité = visible, largeur = 262, hauteur = 262, a-focus = false, a -focusable = false, has-window-focus = true, is-clickable = false, est-enabled = true, est-focus = false, est-focusable = false, is-layout-demandé = false, est sélectionné , root-is-layout-wanted = false, has-input-connection = false, x = 0.0, y = 0.0} **** MATCHES ****
+ -------------> ImageView {id = 2131493330, nom-objet = image_article, desc = image, visibilité = visible, largeur = 262, hauteur = 262, a-focus = false, a -focusable = false, has-window-focus = true, is-clickable = false, est-enabled = true, est-focus = false, est-focusable = false, is-layout-demandé = false, est sélectionné , root-is-layout-wanted = false, has-input-connection = false, x = 0.0, y = 0.0} MATCHES **** |
Vous devez utiliser onData()
pour opérer sur GridView
:
onData(withId(R.id.item_image))
.inAdapterView(withId(R.id.grid_adapter_id))
.atPosition(0)
.perform(click());
Ce code va cliquer sur l'image à l'intérieur du premier élément dans GridView
J'étais étonné de ne pouvoir trouver de solution en fournissant simplement un index avec un indicateur (c'est-à-dire avec Text, avec Id). La réponse acceptée ne résout que le problème lorsque vous utilisez onData et ListViews.
Si vous avez plus d'une vue à l'écran avec le même resID/text/contentDesc, vous pouvez choisir celle que vous voulez sans provoquer d'exception AmbiguousViewMatcherException à l'aide de ce matcher personnalisé:
public static Matcher<View> withIndex(final Matcher<View> matcher, final int index) {
return new TypeSafeMatcher<View>() {
int currentIndex = 0;
@Override
public void describeTo(Description description) {
description.appendText("with index: ");
description.appendValue(index);
matcher.describeTo(description);
}
@Override
public boolean matchesSafely(View view) {
return matcher.matches(view) && currentIndex++ == index;
}
};
}
Par exemple:
onView(withIndex(withId(R.id.my_view), 2)).perform(click());
effectuera une action de clic sur la troisième instance de R.id.my_view.
Pas complètement lié à la situation de la vue en grille mais vous pouvez utiliser des correspondeurs hamcrest allOf
pour combiner plusieurs conditions:
import static org.hamcrest.CoreMatchers.allOf;
onView(allOf(withId(R.id.login_password),
withEffectiveVisibility(ViewMatchers.Visibility.VISIBLE)))
.check(matches(isCompletelyDisplayed()))
.check(matches(withHint(R.string.password_placeholder)));
J'ai créé un ViewMatcher qui correspond à la première vue trouvée. Peut-être est-ce utile pour quelqu'un .. E.g. lorsque vous ne disposez pas d'un AdapterView pour utiliser onData ().
/**
* Created by stost on 15.05.14.
* Matches any view. But only on first match()-call.
*/
public class FirstViewMatcher extends BaseMatcher<View> {
public static boolean matchedBefore = false;
public FirstViewMatcher() {
matchedBefore = false;
}
@Override
public boolean matches(Object o) {
if (matchedBefore) {
return false;
} else {
matchedBefore = true;
return true;
}
}
@Override
public void describeTo(Description description) {
description.appendText(" is the first view that comes along ");
}
@Factory
public static <T> Matcher<View> firstView() {
return new FirstViewMatcher();
}
}
Utilisez-le comme ceci:
onView(FirstViewMatcher.firstView()).perform(click());
J'ai essayé de répondre à @FrostRocket, qui semblait la plus prometteuse, mais qui nécessitait quelques personnalisations:
public static Matcher<View> withIndex(final Matcher<View> matcher, final int index) {
return new TypeSafeMatcher<View>() {
int currentIndex;
int viewObjHash;
@SuppressLint("DefaultLocale") @Override
public void describeTo(Description description) {
description.appendText(String.format("with index: %d ", index));
matcher.describeTo(description);
}
@Override
public boolean matchesSafely(View view) {
if (matcher.matches(view) && currentIndex++ == index) {
viewObjHash = view.hashCode();
}
return view.hashCode() == viewObjHash;
}
};
}
Cas:
onView( withId( R.id.songListView ) ).perform( RealmRecyclerViewActions.scrollTo( Matchers.first(Matchers.withTextLabeled( "Love Song"))) );
onView( Matchers.first(withText( "Love Song")) ).perform( click() );
dans mon Matchers.class
public static Matcher<View> first(Matcher<View> expected ){
return new TypeSafeMatcher<View>() {
private boolean first = false;
@Override
protected boolean matchesSafely(View item) {
if( expected.matches(item) && !first ){
return first = true;
}
return false;
}
@Override
public void describeTo(Description description) {
description.appendText("Matcher.first( " + expected.toString() + " )" );
}
};
}
Vous pouvez simplement faire NthMatcher comme:
class NthMatcher internal constructor(private val id: Int, private val n: Int) : TypeSafeMatcher<View>(View::class.Java) {
companion object {
var matchCount: Int = 0
}
init {
var matchCount = 0
}
private var resources: Resources? = null
override fun describeTo(description: Description) {
var idDescription = Integer.toString(id)
if (resources != null) {
try {
idDescription = resources!!.getResourceName(id)
} catch (e: Resources.NotFoundException) {
// No big deal, will just use the int value.
idDescription = String.format("%s (resource name not found)", id)
}
}
description.appendText("with id: $idDescription")
}
public override fun matchesSafely(view: View): Boolean {
resources = view.resources
if (id == view.id) {
matchCount++
if(matchCount == n) {
return true
}
}
return false
}
}
Déclarez comme ceci:
fun withNthId(resId: Int, n: Int) = CustomMatchers.NthMatcher(resId, n)
Et utilisez comme ceci:
onView(withNthId(R.id.textview, 1)).perform(click())
Au plus tard * Exécuter -> Record Espresso Test
En cliquant sur le même ID, la vue avec une position différente génère un code différent pour eux.
C'est en fait résoudre ces problèmes.