在Appium + Java8 一天一坑(切换至 Webview -- 1) 中讲到了如何在多个 Webview 的情况下切换到最新的 Webview。
不过实际上,当真的有多个 Webview 时,很可能不能找到当前最新的 Webview,例如 App 有两个 Webview: A 和 B. 我们需要的是 Webview B,根据之前方法的原理,我只要找到最后一个 Webview,那么这个 Webview 就一定是最新的。确实如此,可是你切换的时机却在影响我们判断谁才是最后一个 Webview。
当 Webview A 出现后,我们立即switchToLastWebView()
显然,最后一个 Webview 依然是 Webview A,因为 Webview B 并没有显示出来。
如果我们需要切换到 Webview B 呢?最简单的方法就是等待。
如何等呢?Sleep 固定时间?貌似可以,其实不然,Sleep 受环境影响太大,加上本身就是 webview 加载就不快。。那么就该轮训
如何等呢?都是 Webview,如果有两个 Webview 都有固定的名字 (例如 Webview B 名字是 second.webview),那太棒了,直接轮训等driver.context('WEBVIEW_second.webview')
就好了啊。
可我们能保证每个 Webview 都有固定的名字,例如某些 iOS 的 App,webview 名字时随机数: WEBVIEW_12306.1
,有时候是:WEBVIEW_12580.3
等等。
如何等?最保险的方案是采用特征轮训,就是找出你想切换的 Webview B 中的特征元素,然后等待包含该元素的 webview 出现。
int MAX_TIMEOUT_SECONDS = 30;
public void switchToSpecificWebView(By selector) {
switchToLastWebView();
long begin = System.currentTimeMillis();
do {
try {
List<MobileElement> elements = driver.findElements(selector);
if (null != elements && elements.size() > 0) {
return;
}
} catch (Exception e) {
e.printStackTrace();
}
switchToLastWebView();
try {
Thread.sleep(300);
} catch (Exception e) {
e.printStackTrace();
}
} while ((System.currentTimeMillis() - begin) < MAX_TIMOUT_SECONDS * 1000);
}
这里面有 3 个部分:
int MAX_TIMEOUT_SECONDS = 30;
public void switchToSpecificWebView(By selector) {
...
long begin = System.currentTimeMillis();
do {
...
try {
Thread.sleep(300);
} catch (Exception e) {
e.printStackTrace();
}
} while ((System.currentTimeMillis() - begin) < MAX_TIMOUT_SECONDS * 1000);
}
它干了一件事,就是每 300 毫秒执行一次 do While 循环里的任务(为了简单代码已省略,替换为 “...”),等到 30 秒后自动结束。
那么这个任务就应该是找元素并切换了。
...
try {
List<MobileElement> elements = driver.findElements(selector);
if (null != elements && elements.size() > 0) {
return;
}
} catch (Exception e) {
e.printStackTrace();
}
switchToLastWebView();
...
如果找到了特征元素,证明当前我们就在正确的 webview 上,如果没有找到,那么就再切换一次 Weview,看有没有新的 webview 具有这个特征。
public void switchToSpecificWebView(By selector) {
switchToLastWebView();
...
do {
...
try {
Thread.sleep(300);
} catch (Exception e) {
e.printStackTrace();
}
} while ((System.currentTimeMillis() - begin) < MAX_TIMOUT_SECONDS * 1000);
}
好了,这样以来我们就可以保证我们可以切换到想要的最新的 webview 了。
完整代码:
int MAX_TIMEOUT_SECONDS = 30;
public void switchToLastWebView() {
Set<String> contextNames = driver.getContextHandles();
List<String> webViewContextNames = contextNames
.stream()
.filter(contextName -> contextName.contains("WEBVIEW_"))
.collect(Collectors.toList());
String currentContextView = "";
if (webViewContextNames.size() > 0) {
currentContextView = (String) webViewContextNames.toArray()[webViewContextNames.size() - 1];
driver.context(currentContextView);
}
LOGGER.log(Level.INFO, "All contexts:" + contextNames);
LOGGER.log(Level.INFO, "All webview contexts:" + webViewContextNames);
LOGGER.log(Level.INFO, "current context:" + driver.getContext());
}
public void switchToSpecificWebView(By selector) {
switchToLastWebView();
long begin = System.currentTimeMillis();
do {
try {
List<MobileElement> elements = driver.findElements(selector);
if (null != elements && elements.size() > 0) {
return;
}
} catch (Exception e) {
e.printStackTrace();
}
switchToLastWebView();
try {
Thread.sleep(300);
} catch (Exception e) {
e.printStackTrace();
}
} while ((System.currentTimeMillis() - begin) < MAX_TIMEOUT_SECONDS * 1000);
}
到这里,我们可以放心的让代码自己去找我们最新的 webview 啦。