From da5ee1af03033d5bbe9da16e0c595e1e30053833 Mon Sep 17 00:00:00 2001 From: Joaquim Ley Date: Wed, 25 Apr 2018 21:59:00 +0200 Subject: [PATCH] Write UI tests related to request enabled state #29 --- .../ui/home/FavoritesFragmentTest.kt | 78 ++++++++++++++----- .../ui/test/util/CustomMatchers.kt | 10 +++ .../ui/home/favorite/FavoritesFragment.kt | 3 +- .../ui/util/extensions/ViewExtensions.kt | 4 + 4 files changed, 73 insertions(+), 22 deletions(-) create mode 100644 transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/test/util/CustomMatchers.kt diff --git a/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/home/FavoritesFragmentTest.kt b/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/home/FavoritesFragmentTest.kt index ea566f0..da1a5d4 100644 --- a/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/home/FavoritesFragmentTest.kt +++ b/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/home/FavoritesFragmentTest.kt @@ -11,6 +11,7 @@ import android.support.test.espresso.matcher.ViewMatchers.* import android.support.test.filters.MediumTest import android.support.test.rule.ActivityTestRule import android.support.test.runner.AndroidJUnit4 +import android.support.v7.widget.RecyclerView import com.joaquimley.transporteta.R import com.joaquimley.transporteta.presentation.data.Resource import com.joaquimley.transporteta.presentation.home.favorite.FavoritesViewModel @@ -21,6 +22,8 @@ import com.joaquimley.transporteta.ui.home.favorite.FavoritesFragment import com.joaquimley.transporteta.ui.test.util.RecyclerViewMatcher import com.joaquimley.transporteta.ui.testing.TestFragmentActivity import com.joaquimley.transporteta.ui.testing.factory.TestFactoryFavoriteView +import com.joaquimley.transporteta.ui.util.extensions.findViewById +import com.nhaarman.mockito_kotlin.verify import org.hamcrest.CoreMatchers.* import org.junit.Before import org.junit.Rule @@ -183,7 +186,7 @@ class FavoritesFragmentTest { results.postValue(Resource.success(resultsList)) // Then check all items for (favoriteView in resultsList.withIndex()) { - // Scroll to item INDEX + // Scroll to item favoriteView.index onView(withId(R.id.recycler_view)).perform(RecyclerViewActions.scrollToPosition(favoriteView.index)) // Check item is displayed correctly onView(RecyclerViewMatcher.withRecyclerView(R.id.recycler_view).atPosition(favoriteView.index)).check(matches(hasDescendant(withText(resultsList[favoriteView.index].code.toString())))) @@ -192,43 +195,78 @@ class FavoritesFragmentTest { } } + /** + * TODO: Not correctly implemented + */ @Test fun whenRequestButtonIsClickedViewModelRequestIsCalled() { // Given val resultsList = TestFactoryFavoriteView.generateFavoriteViewList() results.postValue(Resource.success(resultsList)) -// requestsAvailable.postValue(false) - - `when`(viewModel.onEtaRequested(resultsList[0])).then{requestsAvailable.postValue(false)} // When onView(withId(R.id.recycler_view)) .perform(RecyclerViewActions.scrollToPosition(0)) - onView(withId(R.id.recycler_view)) - .perform(RecyclerViewActions - .actionOnItemAtPosition(0, click())) - onView(RecyclerViewMatcher.withRecyclerView(R.id.recycler_view).atPosition(0)) .check(matches(hasDescendant(withText(R.string.action_send_sms)))).perform(click()) // Check requestEta was called -// verify(viewModel, times(1)).onEtaRequested(resultsList[0]) + verify(viewModel).onEtaRequested(resultsList[0]) } + @Test + fun whenAcceptingRequestStateIsFalseRequestingTextIsShown() { + // Given (make sure requesting is not being shown) + requestsAvailable.postValue(true) + // When + requestsAvailable.postValue(false) + // Then Snackbar with requesting text and cancel action are shown + onView(allOf(withId(android.support.design.R.id.snackbar_text), withText(R.string.info_requesting))).check(matches(isDisplayed())) + onView(allOf(withId(android.support.design.R.id.snackbar_action), withText(R.string.action_cancel))).check(matches(isDisplayed())) + } @Test - fun whenRequestButtonIsClickedRequestingTextIsShown() { - // Given + fun whenAcceptingRequestStateIsTrueSnackbarWithCancelActionIsDismissed() { + // Given (make sure requesting is showing) + requestsAvailable.postValue(false) + // When + requestsAvailable.postValue(true) + // Then Snackbar with requesting text and cancel action are not shown + onView(allOf(withId(android.support.design.R.id.snackbar_text), withText(R.string.info_requesting))).check(doesNotExist()) + onView(allOf(withId(android.support.design.R.id.snackbar_action), withText(R.string.action_cancel))).check(doesNotExist()) + } + + @Test + fun whenAcceptingRequestStateIsFalseRequestSmsButtonsAreDisabled() { + // Given (make sure requesting is not being shown) val resultsList = TestFactoryFavoriteView.generateFavoriteViewList() results.postValue(Resource.success(resultsList)) // When - onView(RecyclerViewMatcher.withRecyclerView(R.id.recycler_view).atPosition(0)) - .check(matches(hasDescendant(withText(R.string.action_send_sms)))).perform(click()) - // Then show requesting text - onView(withText(R.string.info_requesting)).check(matches(isDisplayed())) + requestsAvailable.postValue(false) + // Then ALL request ETA buttons are disabled + favoritesFragment.findViewById(R.id.recycler_view)?.let { + RecyclerViewMatcher.waitForAdapterChange(it) + for (favoriteView in resultsList.withIndex()) { + // Scroll to item at favoriteView.index + onView(withId(R.id.recycler_view)).perform(RecyclerViewActions.scrollToPosition(favoriteView.index)) + // Check item is displayed correctly + onView(RecyclerViewMatcher.withRecyclerView(R.id.recycler_view).atPosition(favoriteView.index)) + .check(matches(hasDescendant(withId(R.id.eta_button)))).check(matches(not(isEnabled()))) + } + } } -} - -// https://spin.atomicobject.com/2016/04/15/espresso-testing-recyclerviews/ -// https://spin.atomicobject.com/2016/04/15/espresso-testing-recyclerviews/ -// https://medium.com/@_rpiel/recyclerview-and-espresso-a-complicated-story-3f6f4179652e \ No newline at end of file + @Test + fun whenAcceptingRequestStateIsTrueRequestSmsButtonsAreEnabled() { + // Given (make sure requesting is not being shown) + val resultsList = TestFactoryFavoriteView.generateFavoriteViewList() + results.postValue(Resource.success(resultsList)) + // When + requestsAvailable.postValue(true) + // Then ALL request ETA buttons are enabled + for (favoriteView in resultsList.withIndex()) { + // Scroll to item favoriteView.index + onView(RecyclerViewMatcher.withRecyclerView(R.id.recycler_view).atPosition(favoriteView.index)) + .check(matches(hasDescendant(withId(R.id.eta_button)))).check(matches(isEnabled())) + } + } +} \ No newline at end of file diff --git a/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/test/util/CustomMatchers.kt b/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/test/util/CustomMatchers.kt new file mode 100644 index 0000000..e78d529 --- /dev/null +++ b/transport-eta-android/mobile-ui/src/androidTest/java/com/joaquimley/transporteta/ui/test/util/CustomMatchers.kt @@ -0,0 +1,10 @@ +package com.joaquimley.transporteta.ui.test.util + +import com.joaquimley.transporteta.ui.home.favorite.FavoritesAdapter +import org.hamcrest.Description +import org.hamcrest.Matcher +import org.hamcrest.TypeSafeMatcher + +/** + * Matches the {@link CustomAdapter.ViewHolder}s in the middle of the list. + */ diff --git a/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/home/favorite/FavoritesFragment.kt b/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/home/favorite/FavoritesFragment.kt index 54f09fa..a5fc91f 100644 --- a/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/home/favorite/FavoritesFragment.kt +++ b/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/home/favorite/FavoritesFragment.kt @@ -65,7 +65,7 @@ class FavoritesFragment : Fragment() { viewModel.getAcceptingRequests().observe(this, Observer { adapter.setActionEnabledStatus(it ?: true) - requestingSnackbar.show() + requestingSnackbar.setVisible(it) }) } @@ -103,7 +103,6 @@ class FavoritesFragment : Fragment() { message_view?.setVisible(false) recycler_view?.setVisible(true) adapter.submitList(favoriteViewList) - requestingSnackbar.dismiss() } private fun setupScreenEmptyState() { diff --git a/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/util/extensions/ViewExtensions.kt b/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/util/extensions/ViewExtensions.kt index 5ce6da3..966dea0 100644 --- a/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/util/extensions/ViewExtensions.kt +++ b/transport-eta-android/mobile-ui/src/main/java/com/joaquimley/transporteta/ui/util/extensions/ViewExtensions.kt @@ -1,5 +1,6 @@ package com.joaquimley.transporteta.ui.util.extensions +import android.support.design.widget.Snackbar import android.support.v7.recyclerview.extensions.ListAdapter import android.support.v7.widget.RecyclerView import android.text.Editable @@ -17,6 +18,9 @@ fun View.setVisible(isVisible: Boolean) { visibility = if (isVisible) View.VISIBLE else View.GONE } +fun Snackbar.setVisible(isVisible: Boolean?) { + if(isVisible != true) this.show() else this.dismiss() +} fun ImageView.load(resourceId: Int) { Glide.with(context) .load(resourceId)