I have written the following view matcher for my custom view
public static Matcher<View> withValue(final Matcher<Long> longMatcher){
return new BoundedMatcher<View, IntegerField>(IntegerField.class) {
@Override
public void describeTo(Description description) {
description.appendText("with value : ");
longMatcher.describeTo(description);
}
@Override
public void describeMismatch(Object item, Description description) {
super.describeMismatch(item, description);
description.appendText("value=" + ((IntegerField)item).getValue());
}
@Override
protected boolean matchesSafely(IntegerField field) {
return longMatcher.matches(field.getValue());
}
};
when the match fails, the log doesn't contain the mismatch description I appended in the descibeMismatch()
function. Is there anything that I missed?
I had the same problem. Until the feature request is implemented, you can use a custom ViewAssertion
that includes the mismatch reason:
public class EspressoUtils {
// this class is copied from Espresso's source code
// (we need to copy it so that we can replace the `assertThat` function it depends on
private final static class MatchesViewAssertion implements ViewAssertion {
final Matcher<? super View> viewMatcher;
private MatchesViewAssertion(final Matcher<? super View> viewMatcher) {
this.viewMatcher = viewMatcher;
}
public void check(View view, NoMatchingViewException noViewException) {
StringDescription description = new StringDescription();
description.appendText("'");
viewMatcher.describeTo(description);
if (noViewException != null) {
description.appendText(
String.format(
"' check could not be performed because view '%s' was not found.\n",
noViewException.getViewMatcherDescription()));
throw noViewException;
} else {
description.appendText("' doesn't match the selected view.");
assertThat(description.toString(), view, viewMatcher);
}
}
/**
* A replacement for ViewMatchers.assertThat that includes the mismatch description (adapted from the source of ViewMatchers.assertThat
*/
private static <T> void assertThat(String message, T actual, Matcher<T> matcher) {
if (!matcher.matches(actual)) {
final StringDescription mismatch = new StringDescription();
matcher.describeMismatch(actual, mismatch);
Description description = new StringDescription();
description.appendText(message)
.appendText("\nExpected: ")
.appendDescriptionOf(matcher);
if(!mismatch.toString().trim().isEmpty()) {
description.appendText("\n But: ").appendText(mismatch.toString());
}
description.appendText("\n Got: ");
if (actual instanceof View) {
description.appendValue(HumanReadables.describe((View) actual));
} else {
description.appendValue(actual);
}
description.appendText("\n");
throw new AssertionFailedError(description.toString());
}
}
}
public static ViewAssertion matches(final Matcher<View> matcher) {
return new MatchesViewAssertion(matcher);
}
}
Use it like this:
onView(...).check(EspressoUtils.matches(...))
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With