1 /* 2 * Copyright (C) 2013 DroidDriver committers 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 io.appium.droiddriver.helpers; 18 19 import android.annotation.TargetApi; 20 import android.app.Instrumentation; 21 import android.os.Build; 22 23 import io.appium.droiddriver.DroidDriver; 24 import io.appium.droiddriver.exceptions.DroidDriverException; 25 import io.appium.droiddriver.instrumentation.InstrumentationDriver; 26 import io.appium.droiddriver.uiautomation.UiAutomationDriver; 27 import io.appium.droiddriver.util.InstrumentationUtils; 28 29 /** 30 * Static utility methods using a singleton {@link DroidDriver} instance. This class is NOT 31 * required, but it is handy and using a singleton driver can avoid memory leak when you have many 32 * instances around (for example, one in every test - JUnit framework keeps the test instances in 33 * memory after running them). 34 */ 35 public class DroidDrivers { 36 private static DroidDriver driver; 37 38 /** 39 * Gets the singleton driver. Throws if {@link #setSingleton} has not been called. 40 */ get()41 public static DroidDriver get() { 42 if (driver == null) { 43 throw new DroidDriverException("setSingleton() has not been called"); 44 } 45 return driver; 46 } 47 48 /** 49 * Sets the singleton driver. 50 */ setSingleton(DroidDriver driver)51 public static void setSingleton(DroidDriver driver) { 52 if (DroidDrivers.driver != null) { 53 throw new DroidDriverException("setSingleton() can only be called once"); 54 } 55 DroidDrivers.driver = driver; 56 } 57 58 /** 59 * Returns whether the running target (device or emulator) has {@link android.app.UiAutomation} 60 * API, which is introduced in SDK API 18 (JELLY_BEAN_MR2). 61 */ hasUiAutomation()62 public static boolean hasUiAutomation() { 63 return Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2; 64 } 65 66 /** 67 * Returns a new DroidDriver instance. If am instrument options have "driver", treat it as the 68 * fully-qualified-class-name and create a new instance of it with {@code instrumentation} as the 69 * argument; otherwise a new platform-dependent default DroidDriver instance. 70 */ newDriver()71 public static DroidDriver newDriver() { 72 Instrumentation instrumentation = InstrumentationUtils.getInstrumentation(); 73 String driverClass = InstrumentationUtils.getD2Option("driver"); 74 if (driverClass != null) { 75 try { 76 return (DroidDriver) Class.forName(driverClass).getConstructor(Instrumentation.class) 77 .newInstance(instrumentation); 78 } catch (Throwable t) { 79 throw DroidDriverException.propagate(t); 80 } 81 } 82 83 // If "dd.driver" is not specified, return default. 84 if (hasUiAutomation()) { 85 checkUiAutomation(); 86 return new UiAutomationDriver(instrumentation); 87 } 88 return new InstrumentationDriver(instrumentation); 89 } 90 91 /** Checks if UiAutomation API is available */ 92 @TargetApi(18) checkUiAutomation()93 public static void checkUiAutomation() { 94 if (!hasUiAutomation()) { 95 throw new DroidDriverException("UiAutomation is not available below API 18. " 96 + "See http://developer.android.com/reference/android/app/UiAutomation.html"); 97 } 98 if (InstrumentationUtils.getInstrumentation().getUiAutomation() == null) { 99 throw new DroidDriverException( 100 "uiAutomation==null: did you forget to set '-w' flag for 'am instrument'?"); 101 } 102 } 103 } 104