我在S / O上看过几个类似的帖子,但我找不到问题的答案。我试图从Web视图中获取html,但是我无法触发我的“onPageFinished()回调。我甚至尝试在WebChromeClient上使用进程回调,但没有任何工作。邻近Web客户端或Web chrome客户端接收任何回调。我正在运行此代码作为Android单元测试并使用资产目录中的测试HTML,我已经通过断言和调试确认了。我缺少什么?
** 更新 ** 我在调试中逐步完成了测试方法。它确实加载了html(它是一个单元测试,所以我实际上看不到渲染的内容)。然后它在wait()调用上阻塞,超时然后在底部执行断言而不从任一客户端接收回调。 (Web客户端和chrome客户端都应该/将调用notify())
这里的区别在于我在Android单元测试中运行。 WebView未附加到任何活动或容器。我想知道这是否与丢失的回调有关。我会尝试将它粘贴在一个Activity上,看它是否有任何区别。 ** / 更新 **
public class MyWebViewTest extends InstrumentationTestCase {
private WebView webView;
private String html;
public void setUp() throws Exception {
super.setUp();
Context context = getInstrumentation().getContext();
this.webView = new WebView(context);
assertTrue("requires test.html",
Arrays.asList(context.getResources().getAssets().list("www")).contains("test.html"));
}
public void tearDown() throws Exception {
}
@JavascriptInterface
public void viewContent(String content) {
this.html = content;
}
public void testCanLoadAndReadWebContentFromWebView() throws Exception {
webView.addJavascriptInterface(this, "_webContainer");
webView.getSettings().setJavaScriptEnabled(true);
webView.setWebViewClient(new WebViewClient() {
@Override
public void onPageFinished(WebView webView, String url) {
super.onPageFinished(webView, url);
onPageLoaded(webView);
}
@Override
public void onPageStarted(WebView view, String url, Bitmap favicon) {
super.onPageStarted(view, url, favicon);
}
});
webView.setWebChromeClient(new WebChromeClient() {
@Override
public void onProgressChanged(WebView view, int newProgress) {
super.onProgressChanged(view, newProgress);
if(newProgress==100)
onPageLoaded(webView);
}
});
webView.loadUrl("file:///android_asset/www/test.html");
synchronized (this) {
wait(5000);
}
assertNotNull(html);
}
private void onPageLoaded(WebView webView) {
webView.loadUrl("javascript:window._webContainer.viewContent(document.documentElement.innerHTML);");
synchronized (this) {
notify();
}
}
}
答案 0 :(得分:2)
我的假设是正确的!当WebView未附加到Activity时,它将不会加载,渲染或执行任何操作。我最终扩展了ActivityInstrumentationTestCase2并将Web视图附加到Activity。那时我的回调中的断点开始亮起。这是我的工作测试用例:
public class MyWebViewTest extends ActivityInstrumentationTestCase2<TestMeActivity> {
private WebView webView;
private String html;
public MyWebViewTest() {
super(TestMeActivity.class);
}
public void setUp() throws Exception {
super.setUp();
Context context = getInstrumentation().getContext();
this.webView = getActivity().getView();
assertTrue("requires test.html",
Arrays.asList(context.getResources().getAssets().list("www")).contains("test.html"));
}
public void tearDown() throws Exception {
}
@JavascriptInterface
public void viewContent(String content) {
this.html = content;
}
public void testCanLoadAndReadWebContentFromWebView() throws Exception {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
configureWebView();
}
});
synchronized (MyWebViewTest.this) {
MyWebViewTest.this.wait(25000);
}
assertEquals("<head></head><body>\n" +
"Hello\n" +
"\n" +
"</body>",html);
}
private void configureWebView() {
webView.addJavascriptInterface(this, "_webContainer");
WebSettings webSettings = webView.getSettings();
webSettings.setJavaScriptEnabled(true);
webView.setWebViewClient(new WebViewClient() {
@Override
public void onPageFinished(WebView webView, String url) {
super.onPageFinished(webView, url);
onPageLoaded(webView);
}
});
webView.loadUrl("file:///android_asset/www/test.html");
}
@TargetApi(Build.VERSION_CODES.KITKAT)
private void onPageLoaded(WebView webView) {
webView.evaluateJavascript("javascript:window._webContainer.viewContent(document.documentElement.innerHTML);",
new ValueCallback<String>() {
@Override
public void onReceiveValue(String value) {
synchronized (MyWebViewTest.this) {
MyWebViewTest.this.notify();
}
}
}
);
}
}
这是我一起投入的活动支持它:
public class TestMeActivity extends Activity{
private WebView view;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
view = new WebView(this);
setContentView(view);
}
public WebView getView() {
return view;
}
}
简而言之,从Android TestCase测试WebView的内容涉及到相当多的工作。您必须向Web视图添加Javascript接口,插入一些Javascript以调用内部Javascript接口对象,传递Web视图的内容,通过接口,最后在您的测试用例中收集它。我当然希望它更容易,但这是我能想到的最好的。