1 /*
2  * Copyright (C) 2011 The Android Open Source Project
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 android.webkit.cts;
18 
19 import android.graphics.Bitmap;
20 import android.graphics.Canvas;
21 import android.graphics.Picture;
22 import android.graphics.Rect;
23 import android.net.Uri;
24 import android.net.http.SslCertificate;
25 import android.os.Message;
26 import android.print.PrintDocumentAdapter;
27 import android.util.DisplayMetrics;
28 import android.view.View;
29 import android.view.ViewGroup;
30 import android.view.ViewParent;
31 import android.webkit.CookieManager;
32 import android.webkit.DownloadListener;
33 import android.webkit.ValueCallback;
34 import android.webkit.WebBackForwardList;
35 import android.webkit.WebChromeClient;
36 import android.webkit.WebMessage;
37 import android.webkit.WebMessagePort;
38 import android.webkit.WebSettings;
39 import android.webkit.WebView;
40 import android.webkit.WebView.HitTestResult;
41 import android.webkit.WebView.PictureListener;
42 import android.webkit.WebView.VisualStateCallback;
43 import android.webkit.WebViewClient;
44 import android.webkit.WebViewRenderProcessClient;
45 
46 import com.android.compatibility.common.util.PollingCheck;
47 
48 import com.google.common.util.concurrent.SettableFuture;
49 
50 import java.util.concurrent.Executor;
51 
52 /**
53  * Many tests need to run WebView code in the UI thread. This class
54  * wraps a WebView so that calls are ensured to arrive on the UI thread.
55  *
56  * All methods may be run on either the UI thread or test thread.
57  *
58  * This should remain functionally equivalent to androidx.webkit.WebViewOnUiThread.
59  * Modifications to this class should be reflected in that class as necessary. See
60  * http://go/modifying-webview-cts.
61  */
62 public class WebViewOnUiThread extends WebViewSyncLoader {
63     /**
64      * The WebView that calls will be made on.
65      */
66     private WebView mWebView;
67 
68     /**
69      * Wraps a WebView to ensure that methods are run on the UI thread.
70      *
71      * A new WebViewOnUiThread should be called during setUp so as to
72      * reinitialize between calls.
73      *
74      * @param webView The webView that the methods should call.
75      */
WebViewOnUiThread(WebView webView)76     public WebViewOnUiThread(WebView webView) {
77         super(webView);
78         mWebView = webView;
79     }
80 
cleanUp()81     public void cleanUp() {
82         super.destroy();
83     }
84 
setWebViewClient(final WebViewClient webViewClient)85     public void setWebViewClient(final WebViewClient webViewClient) {
86         WebkitUtils.onMainThreadSync(() -> {
87             mWebView.setWebViewClient(webViewClient);
88         });
89     }
90 
setWebChromeClient(final WebChromeClient webChromeClient)91     public void setWebChromeClient(final WebChromeClient webChromeClient) {
92         WebkitUtils.onMainThreadSync(() -> {
93             mWebView.setWebChromeClient(webChromeClient);
94         });
95     }
96 
97     /**
98      * Set the webview renderer client for {@code mWebView}, on the UI thread.
99      */
setWebViewRenderProcessClient( final WebViewRenderProcessClient webViewRenderProcessClient)100     public void setWebViewRenderProcessClient(
101             final WebViewRenderProcessClient webViewRenderProcessClient) {
102         setWebViewRenderProcessClient(mWebView, webViewRenderProcessClient);
103     }
104 
105     /**
106      * Set the webview renderer client for {@code webView}, on the UI thread.
107      */
setWebViewRenderProcessClient( final WebView webView, final WebViewRenderProcessClient webViewRenderProcessClient)108     public static void setWebViewRenderProcessClient(
109             final WebView webView,
110             final WebViewRenderProcessClient webViewRenderProcessClient) {
111         WebkitUtils.onMainThreadSync(() ->
112                 webView.setWebViewRenderProcessClient(webViewRenderProcessClient)
113         );
114     }
115 
116     /**
117      * Set the webview renderer client for {@code mWebView}, on the UI thread, with callbacks
118      * executed by {@code executor}
119      */
setWebViewRenderProcessClient( final Executor executor, final WebViewRenderProcessClient webViewRenderProcessClient)120     public void setWebViewRenderProcessClient(
121             final Executor executor, final WebViewRenderProcessClient webViewRenderProcessClient) {
122         setWebViewRenderProcessClient(mWebView, executor, webViewRenderProcessClient);
123     }
124 
125     /**
126      * Set the webview renderer client for {@code webView}, on the UI thread, with callbacks
127      * executed by {@code executor}
128      */
setWebViewRenderProcessClient( final WebView webView, final Executor executor, final WebViewRenderProcessClient webViewRenderProcessClient)129     public static void setWebViewRenderProcessClient(
130             final WebView webView,
131             final Executor executor,
132             final WebViewRenderProcessClient webViewRenderProcessClient) {
133         WebkitUtils.onMainThreadSync(() ->
134                 webView.setWebViewRenderProcessClient(executor, webViewRenderProcessClient)
135         );
136     }
137 
138     /**
139      * Get the webview renderer client currently set on {@code mWebView}, on the UI thread.
140      */
getWebViewRenderProcessClient()141     public WebViewRenderProcessClient getWebViewRenderProcessClient() {
142         return getWebViewRenderProcessClient(mWebView);
143     }
144 
145     /**
146      * Get the webview renderer client currently set on {@code webView}, on the UI thread.
147      */
getWebViewRenderProcessClient( final WebView webView)148     public static WebViewRenderProcessClient getWebViewRenderProcessClient(
149             final WebView webView) {
150         return WebkitUtils.onMainThreadSync(() -> {
151             return webView.getWebViewRenderProcessClient();
152         });
153     }
154 
setPictureListener(final PictureListener pictureListener)155     public void setPictureListener(final PictureListener pictureListener) {
156         WebkitUtils.onMainThreadSync(() -> {
157             mWebView.setPictureListener(pictureListener);
158         });
159     }
160 
setNetworkAvailable(final boolean available)161     public void setNetworkAvailable(final boolean available) {
162         WebkitUtils.onMainThreadSync(() -> {
163             mWebView.setNetworkAvailable(available);
164         });
165     }
166 
setDownloadListener(final DownloadListener listener)167     public void setDownloadListener(final DownloadListener listener) {
168         WebkitUtils.onMainThreadSync(() -> {
169             mWebView.setDownloadListener(listener);
170         });
171     }
172 
setBackgroundColor(final int color)173     public void setBackgroundColor(final int color) {
174         WebkitUtils.onMainThreadSync(() -> {
175             mWebView.setBackgroundColor(color);
176         });
177     }
178 
clearCache(final boolean includeDiskFiles)179     public void clearCache(final boolean includeDiskFiles) {
180         WebkitUtils.onMainThreadSync(() -> {
181             mWebView.clearCache(includeDiskFiles);
182         });
183     }
184 
clearHistory()185     public void clearHistory() {
186         WebkitUtils.onMainThreadSync(() -> {
187             mWebView.clearHistory();
188         });
189     }
190 
requestFocus()191     public void requestFocus() {
192         new PollingCheck(WebkitUtils.TEST_TIMEOUT_MS) {
193             @Override
194             protected boolean check() {
195                 requestFocusOnUiThread();
196                 return hasFocus();
197             }
198         }.run();
199     }
200 
requestFocusOnUiThread()201     private void requestFocusOnUiThread() {
202         WebkitUtils.onMainThreadSync(() -> {
203             mWebView.requestFocus();
204         });
205     }
206 
hasFocus()207     private boolean hasFocus() {
208         return WebkitUtils.onMainThreadSync(() -> {
209             return mWebView.hasFocus();
210         });
211     }
212 
213     public boolean canZoomIn() {
214         return WebkitUtils.onMainThreadSync(() -> {
215             return mWebView.canZoomIn();
216         });
217     }
218 
219     public boolean canZoomOut() {
220         return WebkitUtils.onMainThreadSync(() -> {
221             return mWebView.canZoomOut();
222         });
223     }
224 
225     public boolean zoomIn() {
226         return WebkitUtils.onMainThreadSync(() -> {
227             return mWebView.zoomIn();
228         });
229     }
230 
231     public boolean zoomOut() {
232         return WebkitUtils.onMainThreadSync(() -> {
233             return mWebView.zoomOut();
234         });
235     }
236 
237     public void zoomBy(final float zoomFactor) {
238         WebkitUtils.onMainThreadSync(() -> {
239             mWebView.zoomBy(zoomFactor);
240         });
241     }
242 
243     public void setFindListener(final WebView.FindListener listener) {
244         WebkitUtils.onMainThreadSync(() -> {
245             mWebView.setFindListener(listener);
246         });
247     }
248 
249     public void removeJavascriptInterface(final String interfaceName) {
250         WebkitUtils.onMainThreadSync(() -> {
251             mWebView.removeJavascriptInterface(interfaceName);
252         });
253     }
254 
255     public WebMessagePort[] createWebMessageChannel() {
256         return WebkitUtils.onMainThreadSync(() -> {
257             return mWebView.createWebMessageChannel();
258         });
259     }
260 
261     public void postWebMessage(final WebMessage message, final Uri targetOrigin) {
262         WebkitUtils.onMainThreadSync(() -> {
263             mWebView.postWebMessage(message, targetOrigin);
264         });
265     }
266 
267     public void addJavascriptInterface(final Object object, final String name) {
268         WebkitUtils.onMainThreadSync(() -> {
269             mWebView.addJavascriptInterface(object, name);
270         });
271     }
272 
273     public void flingScroll(final int vx, final int vy) {
274         WebkitUtils.onMainThreadSync(() -> {
275             mWebView.flingScroll(vx, vy);
276         });
277     }
278 
279     public void requestFocusNodeHref(final Message hrefMsg) {
280         WebkitUtils.onMainThreadSync(() -> {
281             mWebView.requestFocusNodeHref(hrefMsg);
282         });
283     }
284 
285     public void requestImageRef(final Message msg) {
286         WebkitUtils.onMainThreadSync(() -> {
287             mWebView.requestImageRef(msg);
288         });
289     }
290 
291     public void setInitialScale(final int scaleInPercent) {
292         WebkitUtils.onMainThreadSync(() -> {
293                 mWebView.setInitialScale(scaleInPercent);
294         });
295     }
296 
297     public void clearSslPreferences() {
298         WebkitUtils.onMainThreadSync(() -> {
299             mWebView.clearSslPreferences();
300         });
301     }
302 
303     public void clearClientCertPreferences(final Runnable onCleared) {
304         WebkitUtils.onMainThreadSync(() -> {
305             WebView.clearClientCertPreferences(onCleared);
306         });
307     }
308 
309     public void resumeTimers() {
310         WebkitUtils.onMainThreadSync(() -> {
311             mWebView.resumeTimers();
312         });
313     }
314 
315     public void findNext(final boolean forward) {
316         WebkitUtils.onMainThreadSync(() -> {
317             mWebView.findNext(forward);
318         });
319     }
320 
321     public void clearMatches() {
322         WebkitUtils.onMainThreadSync(() -> {
323             mWebView.clearMatches();
324         });
325     }
326 
327     public void loadUrl(final String url) {
328         WebkitUtils.onMainThreadSync(() -> {
329             mWebView.loadUrl(url);
330         });
331     }
332 
333     public void stopLoading() {
334         WebkitUtils.onMainThreadSync(() -> {
335             mWebView.stopLoading();
336         });
337     }
338 
339     /**
340      * Reload the previous URL.
341      */
342     public void reload() {
343         WebkitUtils.onMainThreadSync(() -> {
344             mWebView.reload();
345         });
346     }
347 
348     public String getTitle() {
349         return WebkitUtils.onMainThreadSync(() -> {
350             return mWebView.getTitle();
351         });
352     }
353 
354     public WebSettings getSettings() {
355         return WebkitUtils.onMainThreadSync(() -> {
356             return mWebView.getSettings();
357         });
358     }
359 
360     public WebBackForwardList copyBackForwardList() {
361         return WebkitUtils.onMainThreadSync(() -> {
362             return mWebView.copyBackForwardList();
363         });
364     }
365 
366     public Bitmap getFavicon() {
367         return WebkitUtils.onMainThreadSync(() -> {
368             return mWebView.getFavicon();
369         });
370     }
371 
372     public String getUrl() {
373         return WebkitUtils.onMainThreadSync(() -> {
374             return mWebView.getUrl();
375         });
376     }
377 
378     public int getProgress() {
379         return WebkitUtils.onMainThreadSync(() -> {
380             return mWebView.getProgress();
381         });
382     }
383 
384     public int getHeight() {
385         return WebkitUtils.onMainThreadSync(() -> {
386             return mWebView.getHeight();
387         });
388     }
389 
390     public int getContentHeight() {
391         return WebkitUtils.onMainThreadSync(() -> {
392             return mWebView.getContentHeight();
393         });
394     }
395 
396     public boolean pageUp(final boolean top) {
397         return WebkitUtils.onMainThreadSync(() -> {
398             return mWebView.pageUp(top);
399         });
400     }
401 
402     public boolean pageDown(final boolean bottom) {
403         return WebkitUtils.onMainThreadSync(() -> {
404             return mWebView.pageDown(bottom);
405         });
406     }
407 
408     /**
409      * Post a visual state listener callback for mWebView on the UI thread.
410      */
411     public void postVisualStateCallback(final long requestId, final VisualStateCallback callback) {
412         WebkitUtils.onMainThreadSync(() -> {
413             mWebView.postVisualStateCallback(requestId, callback);
414         });
415     }
416 
417     public int[] getLocationOnScreen() {
418         final int[] location = new int[2];
419         return WebkitUtils.onMainThreadSync(() -> {
420             mWebView.getLocationOnScreen(location);
421             return location;
422         });
423     }
424 
425     public float getScale() {
426         return WebkitUtils.onMainThreadSync(() -> {
427             return mWebView.getScale();
428         });
429     }
430 
431     public boolean requestFocus(final int direction,
432             final Rect previouslyFocusedRect) {
433         return WebkitUtils.onMainThreadSync(() -> {
434             return mWebView.requestFocus(direction, previouslyFocusedRect);
435         });
436     }
437 
438     public HitTestResult getHitTestResult() {
439         return WebkitUtils.onMainThreadSync(() -> {
440             return mWebView.getHitTestResult();
441         });
442     }
443 
444     public int getScrollX() {
445         return WebkitUtils.onMainThreadSync(() -> {
446             return mWebView.getScrollX();
447         });
448     }
449 
450     public int getScrollY() {
451         return WebkitUtils.onMainThreadSync(() -> {
452             return mWebView.getScrollY();
453         });
454     }
455 
456     public final DisplayMetrics getDisplayMetrics() {
457         return WebkitUtils.onMainThreadSync(() -> {
458             return mWebView.getContext().getResources().getDisplayMetrics();
459         });
460     }
461 
462     public boolean requestChildRectangleOnScreen(final View child,
463             final Rect rect,
464             final boolean immediate) {
465         return WebkitUtils.onMainThreadSync(() -> {
466             return mWebView.requestChildRectangleOnScreen(child, rect,
467                     immediate);
468         });
469     }
470 
471     public int findAll(final String find) {
472         return WebkitUtils.onMainThreadSync(() -> {
473             return mWebView.findAll(find);
474         });
475     }
476 
477     public Picture capturePicture() {
478         return WebkitUtils.onMainThreadSync(() -> {
479             return mWebView.capturePicture();
480         });
481     }
482 
483     /**
484      * Execute javascript synchronously, returning the result.
485      */
486     public String evaluateJavascriptSync(final String script) {
487         final SettableFuture<String> future = SettableFuture.create();
488         evaluateJavascript(script, result -> future.set(result));
489         return WebkitUtils.waitForFuture(future);
490     }
491 
492     public void evaluateJavascript(final String script, final ValueCallback<String> result) {
493         WebkitUtils.onMainThread(() -> mWebView.evaluateJavascript(script, result));
494     }
495 
496     public void saveWebArchive(final String basename, final boolean autoname,
497                                final ValueCallback<String> callback) {
498         WebkitUtils.onMainThreadSync(() -> {
499             mWebView.saveWebArchive(basename, autoname, callback);
500         });
501     }
502 
503     public SslCertificate getCertificate() {
504         return WebkitUtils.onMainThreadSync(() -> {
505             return mWebView.getCertificate();
506         });
507     }
508 
509     public WebView createWebView() {
510         return WebkitUtils.onMainThreadSync(() -> {
511             return new WebView(mWebView.getContext());
512         });
513     }
514 
515     public PrintDocumentAdapter createPrintDocumentAdapter() {
516         return WebkitUtils.onMainThreadSync(() -> {
517             return mWebView.createPrintDocumentAdapter();
518         });
519     }
520 
521     public void setLayoutHeightToMatchParent() {
522         WebkitUtils.onMainThreadSync(() -> {
523             ViewParent parent = mWebView.getParent();
524             if (parent instanceof ViewGroup) {
525                 ((ViewGroup) parent).getLayoutParams().height =
526                     ViewGroup.LayoutParams.MATCH_PARENT;
527             }
528             mWebView.getLayoutParams().height = ViewGroup.LayoutParams.MATCH_PARENT;
529             mWebView.requestLayout();
530         });
531     }
532 
533     public void setLayoutToMatchParent() {
534         WebkitUtils.onMainThreadSync(() -> {
535             setMatchParent((View) mWebView.getParent());
536             setMatchParent(mWebView);
537             mWebView.requestLayout();
538         });
539     }
540 
541     public void setAcceptThirdPartyCookies(final boolean accept) {
542         WebkitUtils.onMainThreadSync(() -> {
543             CookieManager.getInstance().setAcceptThirdPartyCookies(mWebView, accept);
544         });
545     }
546 
547     public boolean acceptThirdPartyCookies() {
548         return WebkitUtils.onMainThreadSync(() -> {
549             return CookieManager.getInstance().acceptThirdPartyCookies(mWebView);
550         });
551     }
552 
553     /**
554      * Accessor for underlying WebView.
555      * @return The WebView being wrapped by this class.
556      */
557     public WebView getWebView() {
558         return mWebView;
559     }
560 
561     /**
562      * Wait for the current state of the DOM to be ready to render on the next draw.
563      */
564     public void waitForDOMReadyToRender() {
565         final SettableFuture<Void> future = SettableFuture.create();
566         postVisualStateCallback(0, new VisualStateCallback() {
567             @Override
568             public void onComplete(long requestId) {
569                 future.set(null);
570             }
571         });
572         WebkitUtils.waitForFuture(future);
573     }
574 
575     /**
576      * Capture a bitmap representation of the current WebView state.
577      *
578      * This synchronises so that the bitmap contents reflects the current DOM state, rather than
579      * potentially capturing a previously generated frame.
580      */
581     public Bitmap captureBitmap() {
582         getSettings().setOffscreenPreRaster(true);
583         waitForDOMReadyToRender();
584         return WebkitUtils.onMainThreadSync(() -> {
585             Bitmap bitmap = Bitmap.createBitmap(mWebView.getWidth(), mWebView.getHeight(),
586                     Bitmap.Config.ARGB_8888);
587             Canvas canvas = new Canvas(bitmap);
588             mWebView.draw(canvas);
589             return bitmap;
590         });
591     }
592 
593     /**
594      * Set LayoutParams to MATCH_PARENT.
595      *
596      * @param view Target view
597      */
598     private void setMatchParent(View view) {
599         ViewGroup.LayoutParams params = view.getLayoutParams();
600         params.height = ViewGroup.LayoutParams.MATCH_PARENT;
601         params.width = ViewGroup.LayoutParams.MATCH_PARENT;
602         view.setLayoutParams(params);
603     }
604 }
605