1 /* 2 * Copyright (C) 2008 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.compatibility.common.util; 18 19 import static android.view.ViewTreeObserver.OnDrawListener; 20 import static android.view.ViewTreeObserver.OnGlobalLayoutListener; 21 22 import static org.mockito.hamcrest.MockitoHamcrest.argThat; 23 24 import android.content.Context; 25 import android.content.res.Resources; 26 import android.graphics.Bitmap; 27 import android.graphics.BitmapFactory; 28 import android.text.Editable; 29 import android.text.TextUtils; 30 import android.view.View; 31 import android.view.ViewTreeObserver; 32 33 import androidx.annotation.NonNull; 34 import androidx.annotation.Nullable; 35 import androidx.test.rule.ActivityTestRule; 36 37 import junit.framework.Assert; 38 39 import org.hamcrest.BaseMatcher; 40 import org.hamcrest.Description; 41 import org.xmlpull.v1.XmlPullParser; 42 import org.xmlpull.v1.XmlPullParserException; 43 44 import java.io.IOException; 45 import java.util.concurrent.CountDownLatch; 46 import java.util.concurrent.TimeUnit; 47 48 /** 49 * The useful methods for widget test. 50 */ 51 public class WidgetTestUtils { 52 /** 53 * Assert that two bitmaps have identical content (same dimensions, same configuration, 54 * same pixel content). 55 * 56 * @param b1 the first bitmap which needs to compare. 57 * @param b2 the second bitmap which needs to compare. 58 */ assertEquals(Bitmap b1, Bitmap b2)59 public static void assertEquals(Bitmap b1, Bitmap b2) { 60 if (b1 == b2) { 61 return; 62 } 63 64 if (b1 == null || b2 == null) { 65 Assert.fail("the bitmaps are not equal"); 66 } 67 68 // b1 and b2 are all not null. 69 if (b1.getWidth() != b2.getWidth() || b1.getHeight() != b2.getHeight() 70 || b1.getConfig() != b2.getConfig()) { 71 Assert.fail("the bitmaps are not equal"); 72 } 73 74 int w = b1.getWidth(); 75 int h = b1.getHeight(); 76 int s = w * h; 77 int[] pixels1 = new int[s]; 78 int[] pixels2 = new int[s]; 79 80 b1.getPixels(pixels1, 0, w, 0, 0, w, h); 81 b2.getPixels(pixels2, 0, w, 0, 0, w, h); 82 83 for (int i = 0; i < s; i++) { 84 if (pixels1[i] != pixels2[i]) { 85 Assert.fail("the bitmaps are not equal"); 86 } 87 } 88 } 89 90 /** 91 * Find beginning of the special element. 92 * @param parser XmlPullParser will be parsed. 93 * @param firstElementName the target element name. 94 * 95 * @throws XmlPullParserException if XML Pull Parser related faults occur. 96 * @throws IOException if I/O-related error occur when parsing. 97 */ beginDocument(XmlPullParser parser, String firstElementName)98 public static final void beginDocument(XmlPullParser parser, String firstElementName) 99 throws XmlPullParserException, IOException { 100 Assert.assertNotNull(parser); 101 Assert.assertNotNull(firstElementName); 102 103 int type; 104 while ((type = parser.next()) != XmlPullParser.START_TAG 105 && type != XmlPullParser.END_DOCUMENT) { 106 ; 107 } 108 109 if (!parser.getName().equals(firstElementName)) { 110 throw new XmlPullParserException("Unexpected start tag: found " + parser.getName() 111 + ", expected " + firstElementName); 112 } 113 } 114 115 /** 116 * Compare the expected pixels with actual, scaling for the target context density 117 * 118 * @throws AssertionFailedError 119 */ assertScaledPixels(int expected, int actual, Context context)120 public static void assertScaledPixels(int expected, int actual, Context context) { 121 Assert.assertEquals(expected * context.getResources().getDisplayMetrics().density, 122 actual, 3); 123 } 124 125 /** Converts dips into pixels using the {@link Context}'s density. */ convertDipToPixels(Context context, int dip)126 public static int convertDipToPixels(Context context, int dip) { 127 float density = context.getResources().getDisplayMetrics().density; 128 return Math.round(density * dip); 129 } 130 131 /** 132 * Retrieve a bitmap that can be used for comparison on any density 133 * @param resources 134 * @return the {@link Bitmap} or <code>null</code> 135 */ getUnscaledBitmap(Resources resources, int resId)136 public static Bitmap getUnscaledBitmap(Resources resources, int resId) { 137 BitmapFactory.Options options = new BitmapFactory.Options(); 138 options.inScaled = false; 139 return BitmapFactory.decodeResource(resources, resId, options); 140 } 141 142 /** 143 * Retrieve a dithered bitmap that can be used for comparison on any density 144 * @param resources 145 * @param config the preferred config for the returning bitmap 146 * @return the {@link Bitmap} or <code>null</code> 147 */ getUnscaledAndDitheredBitmap(Resources resources, int resId, Bitmap.Config config)148 public static Bitmap getUnscaledAndDitheredBitmap(Resources resources, 149 int resId, Bitmap.Config config) { 150 BitmapFactory.Options options = new BitmapFactory.Options(); 151 options.inDither = true; 152 options.inScaled = false; 153 options.inPreferredConfig = config; 154 return BitmapFactory.decodeResource(resources, resId, options); 155 } 156 157 /** 158 * Argument matcher for equality check of a CharSequence. 159 * 160 * @param expected expected CharSequence 161 * 162 * @return 163 */ sameCharSequence(final CharSequence expected)164 public static CharSequence sameCharSequence(final CharSequence expected) { 165 return argThat(new BaseMatcher<CharSequence>() { 166 @Override 167 public boolean matches(Object o) { 168 if (o instanceof CharSequence) { 169 return TextUtils.equals(expected, (CharSequence) o); 170 } 171 return false; 172 } 173 174 @Override 175 public void describeTo(Description description) { 176 description.appendText("doesn't match " + expected); 177 } 178 }); 179 } 180 181 /** 182 * Argument matcher for equality check of an Editable. 183 * 184 * @param expected expected Editable 185 * 186 * @return 187 */ 188 public static Editable sameEditable(final Editable expected) { 189 return argThat(new BaseMatcher<Editable>() { 190 @Override 191 public boolean matches(Object o) { 192 if (o instanceof Editable) { 193 return TextUtils.equals(expected, (Editable) o); 194 } 195 return false; 196 } 197 198 @Override 199 public void describeTo(Description description) { 200 description.appendText("doesn't match " + expected); 201 } 202 }); 203 } 204 205 /** 206 * Runs the specified {@link Runnable} on the main thread and ensures that the specified 207 * {@link View}'s tree is drawn before returning. 208 * 209 * @param activityTestRule the activity test rule used to run the test 210 * @param view the view whose tree should be drawn before returning 211 * @param runner the runnable to run on the main thread, or {@code null} to 212 * simply force invalidation and a draw pass 213 */ 214 public static void runOnMainAndDrawSync(@NonNull final ActivityTestRule activityTestRule, 215 @NonNull final View view, @Nullable final Runnable runner) { 216 final CountDownLatch latch = new CountDownLatch(1); 217 218 try { 219 activityTestRule.runOnUiThread(() -> { 220 final OnDrawListener listener = new OnDrawListener() { 221 @Override 222 public void onDraw() { 223 // posting so that the sync happens after the draw that's about to happen 224 view.post(() -> { 225 view.getViewTreeObserver().removeOnDrawListener(this); 226 latch.countDown(); 227 }); 228 } 229 }; 230 231 view.getViewTreeObserver().addOnDrawListener(listener); 232 233 if (runner != null) { 234 runner.run(); 235 } 236 view.invalidate(); 237 }); 238 239 Assert.assertTrue("Expected draw pass occurred within 5 seconds", 240 latch.await(5, TimeUnit.SECONDS)); 241 } catch (Throwable t) { 242 throw new RuntimeException(t); 243 } 244 } 245 246 /** 247 * Runs the specified {@link Runnable} on the main thread and ensures that the specified 248 * {@link View}'s tree is drawn before returning. 249 * 250 * @param view the view whose tree should be drawn before returning, must be attached to window 251 * @param runner the runnable to run on the main thread, or {@code null} to 252 * simply force invalidation and a draw pass 253 */ 254 public static void runOnMainAndDrawSync(@NonNull final View view, 255 @Nullable final Runnable runner) { 256 final CountDownLatch latch = new CountDownLatch(1); 257 Assert.assertTrue(view.isAttachedToWindow()); 258 259 try { 260 view.post(() -> { 261 final OnDrawListener listener = new OnDrawListener() { 262 @Override 263 public void onDraw() { 264 // posting so that the sync happens after the draw that's about to happen 265 view.post(() -> { 266 view.getViewTreeObserver().removeOnDrawListener(this); 267 latch.countDown(); 268 }); 269 } 270 }; 271 272 view.getViewTreeObserver().addOnDrawListener(listener); 273 274 if (runner != null) { 275 runner.run(); 276 } 277 view.invalidate(); 278 }); 279 280 Assert.assertTrue("Expected draw pass occurred within 5 seconds", 281 latch.await(5, TimeUnit.SECONDS)); 282 } catch (Throwable t) { 283 throw new RuntimeException(t); 284 } 285 } 286 287 288 /** 289 * Runs the specified Runnable on the main thread and ensures that the activity's view tree is 290 * laid out before returning. 291 * 292 * @param activityTestRule the activity test rule used to run the test 293 * @param runner the runnable to run on the main thread. {@code null} is 294 * allowed, and simply means that there no runnable is required. 295 * @param forceLayout true if there should be an explicit call to requestLayout(), 296 * false otherwise 297 */ 298 public static void runOnMainAndLayoutSync(@NonNull final ActivityTestRule activityTestRule, 299 @Nullable final Runnable runner, boolean forceLayout) 300 throws Throwable { 301 runOnMainAndLayoutSync(activityTestRule, 302 activityTestRule.getActivity().getWindow().getDecorView(), runner, forceLayout); 303 } 304 305 /** 306 * Runs the specified Runnable on the main thread and ensures that the specified view is 307 * laid out before returning. 308 * 309 * @param activityTestRule the activity test rule used to run the test 310 * @param view The view 311 * @param runner the runnable to run on the main thread. {@code null} is 312 * allowed, and simply means that there no runnable is required. 313 * @param forceLayout true if there should be an explicit call to requestLayout(), 314 * false otherwise 315 */ 316 public static void runOnMainAndLayoutSync(@NonNull final ActivityTestRule activityTestRule, 317 @NonNull final View view, @Nullable final Runnable runner, boolean forceLayout) 318 throws Throwable { 319 final View rootView = view.getRootView(); 320 321 final CountDownLatch latch = new CountDownLatch(1); 322 323 activityTestRule.runOnUiThread(() -> { 324 final OnGlobalLayoutListener listener = new ViewTreeObserver.OnGlobalLayoutListener() { 325 @Override 326 public void onGlobalLayout() { 327 rootView.getViewTreeObserver().removeOnGlobalLayoutListener(this); 328 // countdown immediately since the layout we were waiting on has happened 329 latch.countDown(); 330 } 331 }; 332 333 rootView.getViewTreeObserver().addOnGlobalLayoutListener(listener); 334 335 if (runner != null) { 336 runner.run(); 337 } 338 339 if (forceLayout) { 340 rootView.requestLayout(); 341 } 342 }); 343 344 try { 345 Assert.assertTrue("Expected layout pass within 5 seconds", 346 latch.await(5, TimeUnit.SECONDS)); 347 } catch (InterruptedException e) { 348 throw new RuntimeException(e); 349 } 350 } 351 352 } 353