Appium + Java8 踩坑记(切换至Webview --
在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具有这个特征。
- 初始化 (先切换到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啦。