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 啦。


↙↙↙阅读原文可查看相关链接,并与作者交流