1 /* 2 * Copyright (C) 2011 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.contacts.common.test; 18 19 import static android.os.PowerManager.ACQUIRE_CAUSES_WAKEUP; 20 import static android.os.PowerManager.FULL_WAKE_LOCK; 21 import static android.os.PowerManager.ON_AFTER_RELEASE; 22 23 import android.app.Activity; 24 import android.app.Instrumentation; 25 import android.content.Context; 26 import android.os.PowerManager; 27 import android.view.View; 28 import android.view.ViewGroup; 29 import android.widget.TextView; 30 31 import com.google.common.base.Preconditions; 32 33 import junit.framework.Assert; 34 35 import java.util.ArrayList; 36 import java.util.List; 37 import java.util.concurrent.Callable; 38 import java.util.concurrent.ExecutionException; 39 import java.util.concurrent.FutureTask; 40 41 import javax.annotation.concurrent.GuardedBy; 42 import javax.annotation.concurrent.ThreadSafe; 43 44 /** Some utility methods for making integration testing smoother. */ 45 @ThreadSafe 46 public class IntegrationTestUtils { 47 private static final String TAG = "IntegrationTestUtils"; 48 49 private final Instrumentation mInstrumentation; 50 private final Object mLock = new Object(); 51 @GuardedBy("mLock") private PowerManager.WakeLock mWakeLock; 52 IntegrationTestUtils(Instrumentation instrumentation)53 public IntegrationTestUtils(Instrumentation instrumentation) { 54 mInstrumentation = instrumentation; 55 } 56 57 /** 58 * Find a view by a given resource id, from the given activity, and click it, iff it is 59 * enabled according to {@link View#isEnabled()}. 60 */ clickButton(final Activity activity, final int buttonResourceId)61 public void clickButton(final Activity activity, final int buttonResourceId) throws Throwable { 62 runOnUiThreadAndGetTheResult(new Callable<Void>() { 63 @Override 64 public Void call() throws Exception { 65 View view = activity.findViewById(buttonResourceId); 66 Assert.assertNotNull(view); 67 if (view.isEnabled()) { 68 view.performClick(); 69 } 70 return null; 71 } 72 }); 73 } 74 75 /** Returns the result of running {@link TextView#getText()} on the ui thread. */ getText(final TextView view)76 public CharSequence getText(final TextView view) throws Throwable { 77 return runOnUiThreadAndGetTheResult(new Callable<CharSequence>() { 78 @Override 79 public CharSequence call() { 80 return view.getText(); 81 } 82 }); 83 } 84 85 // TODO: Move this class and the appropriate documentation into a test library, having checked 86 // first to see if exactly this code already exists or not. 87 /** 88 * Execute a callable on the ui thread, returning its result synchronously. 89 * <p> 90 * Waits for an idle sync on the main thread (see {@link Instrumentation#waitForIdle(Runnable)}) 91 * before executing this callable. 92 */ 93 public <T> T runOnUiThreadAndGetTheResult(Callable<T> callable) throws Throwable { 94 FutureTask<T> future = new FutureTask<T>(callable); 95 mInstrumentation.waitForIdle(future); 96 try { 97 return future.get(); 98 } catch (ExecutionException e) { 99 // Unwrap the cause of the exception and re-throw it. 100 throw e.getCause(); 101 } 102 } 103 104 /** 105 * Wake up the screen, useful in tests that want or need the screen to be on. 106 * <p> 107 * This is usually called from setUp() for tests that require it. After calling this method, 108 * {@link #releaseScreenWakeLock()} must be called, this is usually done from tearDown(). 109 */ 110 public void acquireScreenWakeLock(Context context) { 111 synchronized (mLock) { 112 Preconditions.checkState(mWakeLock == null, "mWakeLock was already held"); 113 mWakeLock = ((PowerManager) context.getSystemService(Context.POWER_SERVICE)) 114 .newWakeLock( 115 PowerManager.ACQUIRE_CAUSES_WAKEUP | PowerManager.ON_AFTER_RELEASE | PowerManager.FULL_WAKE_LOCK, TAG); 116 mWakeLock.acquire(); 117 } 118 } 119 120 /** Release the wake lock previously acquired with {@link #acquireScreenWakeLock(Context)}. */ 121 public void releaseScreenWakeLock() { 122 synchronized (mLock) { 123 // We don't use Preconditions to force you to have acquired before release. 124 // This is because we don't want unnecessary exceptions in tearDown() since they'll 125 // typically mask the actual exception that happened during the test. 126 // The other reason is that this method is most likely to be called from tearDown(), 127 // which is invoked within a finally block, so it's not infrequently the case that 128 // the setUp() method fails before getting the lock, at which point we don't want 129 // to fail in tearDown(). 130 if (mWakeLock != null) { 131 mWakeLock.release(); 132 mWakeLock = null; 133 } 134 } 135 } 136 137 /** 138 * Gets all {@link TextView} objects whose {@link TextView#getText()} contains the given text as 139 * a substring. 140 */ 141 public List<TextView> getTextViewsWithString(final Activity activity, final String text) 142 throws Throwable { 143 return getTextViewsWithString(getRootView(activity), text); 144 } 145 146 /** 147 * Gets all {@link TextView} objects whose {@link TextView#getText()} contains the given text as 148 * a substring for the given root view. 149 */ 150 public List<TextView> getTextViewsWithString(final View rootView, final String text) 151 throws Throwable { 152 return runOnUiThreadAndGetTheResult(new Callable<List<TextView>>() { 153 @Override 154 public List<TextView> call() throws Exception { 155 List<TextView> matchingViews = new ArrayList<TextView>(); 156 for (TextView textView : getAllViews(TextView.class, rootView)) { 157 if (textView.getText().toString().contains(text)) { 158 matchingViews.add(textView); 159 } 160 } 161 return matchingViews; 162 } 163 }); 164 } 165 166 /** Find the root view for a given activity. */ 167 public static View getRootView(Activity activity) { 168 return activity.findViewById(android.R.id.content).getRootView(); 169 } 170 171 /** 172 * Gets a list of all views of a given type, rooted at the given parent. 173 * <p> 174 * This method will recurse down through all {@link ViewGroup} instances looking for 175 * {@link View} instances of the supplied class type. Specifically it will use the 176 * {@link Class#isAssignableFrom(Class)} method as the test for which views to add to the list, 177 * so if you provide {@code View.class} as your type, you will get every view. The parent itself 178 * will be included also, should it be of the right type. 179 * <p> 180 * This call manipulates the ui, and as such should only be called from the application's main 181 * thread. 182 */ 183 private static <T extends View> List<T> getAllViews(final Class<T> clazz, final View parent) { 184 List<T> results = new ArrayList<T>(); 185 if (parent.getClass().equals(clazz)) { 186 results.add(clazz.cast(parent)); 187 } 188 if (parent instanceof ViewGroup) { 189 ViewGroup viewGroup = (ViewGroup) parent; 190 for (int i = 0; i < viewGroup.getChildCount(); ++i) { 191 results.addAll(getAllViews(clazz, viewGroup.getChildAt(i))); 192 } 193 } 194 return results; 195 } 196 } 197