WebViewChromium.java revision b7a83153a343e3f11289dd3edc175792ab1b39b5
1/*
2 * Copyright (C) 2012 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
17package com.android.webview.chromium;
18
19import android.content.res.Configuration;
20import android.graphics.Bitmap;
21import android.graphics.Canvas;
22import android.graphics.Paint;
23import android.graphics.Picture;
24import android.graphics.Rect;
25import android.graphics.drawable.Drawable;
26import android.net.http.SslCertificate;
27import android.os.Build;
28import android.os.Bundle;
29import android.os.Message;
30import android.util.Base64;
31import android.util.Log;
32import android.view.HardwareCanvas;
33import android.view.KeyEvent;
34import android.view.MotionEvent;
35import android.view.View;
36import android.view.View.MeasureSpec;
37import android.view.ViewGroup;
38import android.view.accessibility.AccessibilityEvent;
39import android.view.accessibility.AccessibilityNodeInfo;
40import android.view.accessibility.AccessibilityNodeProvider;
41import android.view.inputmethod.EditorInfo;
42import android.view.inputmethod.InputConnection;
43import android.webkit.DownloadListener;
44import android.webkit.FindActionModeCallback;
45import android.webkit.JavascriptInterface;
46import android.webkit.ValueCallback;
47import android.webkit.WebBackForwardList;
48import android.webkit.WebChromeClient;
49import android.webkit.WebSettings;
50import android.webkit.WebView;
51import android.webkit.WebViewClient;
52import android.webkit.WebViewProvider;
53import android.widget.TextView;
54
55import org.chromium.android_webview.AwBrowserContext;
56import org.chromium.android_webview.AwContents;
57import org.chromium.base.ThreadUtils;
58import org.chromium.content.browser.LoadUrlParams;
59import org.chromium.net.NetworkChangeNotifier;
60
61import java.io.BufferedWriter;
62import java.io.File;
63import java.io.OutputStream;
64import java.lang.annotation.Annotation;
65import java.util.HashMap;
66import java.util.Map;
67
68/**
69 * This class is the delegate to which WebViewProxy forwards all API calls.
70 *
71 * Most of the actual functionality is implemented by AwContents (or ContentViewCore within
72 * it). This class also contains WebView-specific APIs that require the creation of other
73 * adapters (otherwise org.chromium.content would depend on the webview.chromium package)
74 * and a small set of no-op deprecated APIs.
75 */
76class WebViewChromium implements WebViewProvider,
77          WebViewProvider.ScrollDelegate, WebViewProvider.ViewDelegate {
78
79    private static final String TAG = WebViewChromium.class.getSimpleName();
80
81    // The WebView that this WebViewChromium is the provider for.
82    WebView mWebView;
83    // Lets us access protected View-derived methods on the WebView instance we're backing.
84    WebView.PrivateAccess mWebViewPrivate;
85    // The client adapter class.
86    private WebViewContentsClientAdapter mContentsClientAdapter;
87
88    // Variables for functionality provided by this adapter ---------------------------------------
89    // WebSettings adapter, lazily initialized in the getter
90    private WebSettings mWebSettings;
91    // The WebView wrapper for ContentViewCore and required browser compontents.
92    private AwContents mAwContents;
93    // Non-null if this webview is using the GL accelerated draw path.
94    private DrawGLFunctor mGLfunctor;
95
96    private AwBrowserContext mBrowserContext;
97
98    private final WebView.HitTestResult mHitTestResult;
99
100    private final int mAppTargetSdkVersion;
101
102    public WebViewChromium(WebView webView, WebView.PrivateAccess webViewPrivate,
103            AwBrowserContext browserContext) {
104        checkThread();
105        mWebView = webView;
106        mWebViewPrivate = webViewPrivate;
107        mHitTestResult = new WebView.HitTestResult();
108        mBrowserContext = browserContext;
109        mAppTargetSdkVersion = mWebView.getContext().getApplicationInfo().targetSdkVersion;
110    }
111
112    static void completeWindowCreation(WebView parent, WebView child) {
113        AwContents parentContents = ((WebViewChromium) parent.getWebViewProvider()).mAwContents;
114        AwContents childContents =
115                child == null ? null : ((WebViewChromium) child.getWebViewProvider()).mAwContents;
116        parentContents.supplyContentsForPopup(childContents);
117    }
118
119    // WebViewProvider methods --------------------------------------------------------------------
120
121    @Override
122    public void init(Map<String, Object> javaScriptInterfaces, boolean privateBrowsing) {
123        // BUG=6790250 |javaScriptInterfaces| was only ever used by the obsolete DumpRenderTree
124        // so is ignored. TODO: remove it from WebViewProvider.
125        final boolean isAccessFromFileURLsGrantedByDefault =
126                 mAppTargetSdkVersion < Build.VERSION_CODES.JELLY_BEAN;
127        mContentsClientAdapter = new WebViewContentsClientAdapter(mWebView);
128        mAwContents = new AwContents(mBrowserContext, mWebView, new InternalAccessAdapter(),
129                mContentsClientAdapter, isAccessFromFileURLsGrantedByDefault);
130
131        if (privateBrowsing) {
132            final String msg = "Private browsing is not supported in WebView.";
133            if (mAppTargetSdkVersion >= Build.VERSION_CODES.KEY_LIME_PIE) {
134                throw new IllegalArgumentException(msg);
135            } else {
136                Log.w(TAG, msg);
137                // Intentionally irreversibly disable the webview instance, so that private
138                // user data cannot leak through misuse of a non-privateBrowing WebView instance.
139                // Can't just null out mAwContents as we never null-check it before use.
140                mAwContents.destroy();
141                TextView warningLabel = new TextView(mWebView.getContext());
142                warningLabel.setText(mWebView.getContext().getString(
143                        com.android.internal.R.string.webviewchromium_private_browsing_warning));
144                mWebView.addView(warningLabel);
145            }
146        }
147
148    }
149
150    private RuntimeException createThreadException() {
151        return new IllegalStateException("Calling View methods on another thread than the UI " +
152                "thread. PLEASE FILE A BUG! go/klp-webview-bug");
153    }
154
155    //  Intentionally not static, as no need to check thread on static methods
156    private void checkThread() {
157        if (!ThreadUtils.runningOnUiThread()) {
158            final RuntimeException threadViolation = createThreadException();
159            ThreadUtils.postOnUiThread(new Runnable() {
160                @Override
161                public void run() {
162                    throw threadViolation;
163                }
164            });
165            throw createThreadException();
166        }
167    }
168
169    @Override
170    public void setHorizontalScrollbarOverlay(boolean overlay) {
171        checkThread();
172        mAwContents.setHorizontalScrollbarOverlay(overlay);
173    }
174
175    @Override
176    public void setVerticalScrollbarOverlay(boolean overlay) {
177        checkThread();
178        mAwContents.setVerticalScrollbarOverlay(overlay);
179    }
180
181    @Override
182    public boolean overlayHorizontalScrollbar() {
183        checkThread();
184        return mAwContents.overlayHorizontalScrollbar();
185    }
186
187    @Override
188    public boolean overlayVerticalScrollbar() {
189        checkThread();
190        return mAwContents.overlayVerticalScrollbar();
191    }
192
193    @Override
194    public int getVisibleTitleHeight() {
195        // This is deprecated in WebView and should always return 0.
196        return 0;
197    }
198
199    @Override
200    public SslCertificate getCertificate() {
201        checkThread();
202        return mAwContents.getCertificate();
203    }
204
205    @Override
206    public void setCertificate(SslCertificate certificate) {
207        checkThread();
208        UnimplementedWebViewApi.invoke();
209    }
210
211    @Override
212    public void savePassword(String host, String username, String password) {
213        // This is a deprecated API: intentional no-op.
214    }
215
216    @Override
217    public void setHttpAuthUsernamePassword(String host, String realm, String username,
218                                            String password) {
219        checkThread();
220        mAwContents.setHttpAuthUsernamePassword(host, realm, username, password);
221    }
222
223    @Override
224    public String[] getHttpAuthUsernamePassword(String host, String realm) {
225        checkThread();
226        return mAwContents.getHttpAuthUsernamePassword(host, realm);
227    }
228
229    @Override
230    public void destroy() {
231        checkThread();
232        mAwContents.destroy();
233        if (mGLfunctor != null) {
234            mGLfunctor.destroy();
235            mGLfunctor = null;
236        }
237    }
238
239    @Override
240    public void setNetworkAvailable(boolean networkUp) {
241        checkThread();
242        // Note that this purely toggles the JS navigator.online property.
243        // It does not in affect chromium or network stack state in any way.
244        mAwContents.setNetworkAvailable(networkUp);
245    }
246
247    @Override
248    public WebBackForwardList saveState(Bundle outState) {
249        checkThread();
250        if (outState == null) return null;
251        if (!mAwContents.saveState(outState)) return null;
252        return copyBackForwardList();
253    }
254
255    @Override
256    public boolean savePicture(Bundle b, File dest) {
257        // Intentional no-op: hidden method on WebView.
258        return false;
259    }
260
261    @Override
262    public boolean restorePicture(Bundle b, File src) {
263        // Intentional no-op: hidden method on WebView.
264        return false;
265    }
266
267    @Override
268    public WebBackForwardList restoreState(Bundle inState) {
269        checkThread();
270        if (inState == null) return null;
271        if (!mAwContents.restoreState(inState)) return null;
272        return copyBackForwardList();
273    }
274
275    @Override
276    public void loadUrl(String url, Map<String, String> additionalHttpHeaders) {
277        // TODO: We may actually want to do some sanity checks here (like filter about://chrome).
278
279        // For backwards compatibility, apps targeting less than K will have JS URLs evaluated
280        // directly and any result of the evaluation will not replace the current page content.
281        // Matching Chrome behavior more closely; apps targetting >= K that load a JS URL will
282        // have the result of that URL replace the content of the current page.
283        final String JAVASCRIPT_SCHEME = "javascript:";
284        if (mAppTargetSdkVersion < Build.VERSION_CODES.KEY_LIME_PIE &&
285                url.startsWith(JAVASCRIPT_SCHEME)) {
286            mAwContents.evaluateJavaScriptEvenIfNotYetNavigated(
287                    url.substring(JAVASCRIPT_SCHEME.length()));
288            return;
289        }
290
291        LoadUrlParams params = new LoadUrlParams(url);
292        if (additionalHttpHeaders != null) params.setExtraHeaders(additionalHttpHeaders);
293        loadUrlOnUiThread(params);
294    }
295
296    @Override
297    public void loadUrl(String url) {
298        loadUrl(url, null);
299    }
300
301    @Override
302    public void postUrl(String url, byte[] postData) {
303        LoadUrlParams params = LoadUrlParams.createLoadHttpPostParams(url, postData);
304        Map<String,String> headers = new HashMap<String,String>();
305        headers.put("Content-Type", "application/x-www-form-urlencoded");
306        params.setExtraHeaders(headers);
307        loadUrlOnUiThread(params);
308    }
309
310    private static boolean isBase64Encoded(String encoding) {
311        return "base64".equals(encoding);
312    }
313
314    @Override
315    public void loadData(String data, String mimeType, String encoding) {
316        loadUrlOnUiThread(LoadUrlParams.createLoadDataParams(
317                data, mimeType, isBase64Encoded(encoding)));
318    }
319
320    @Override
321    public void loadDataWithBaseURL(String baseUrl, String data, String mimeType, String encoding,
322            String historyUrl) {
323        LoadUrlParams loadUrlParams;
324
325        if (baseUrl != null && baseUrl.startsWith("data:")) {
326            // For backwards compatibility with WebViewClassic, we use the value of |encoding|
327            // as the charset, as long as it's not "base64".
328            boolean isBase64 = isBase64Encoded(encoding);
329            loadUrlParams = LoadUrlParams.createLoadDataParamsWithBaseUrl(
330                    data, mimeType, isBase64, baseUrl, historyUrl, isBase64 ? null : encoding);
331        } else {
332            if (baseUrl == null || baseUrl.length() == 0) baseUrl = "about:blank";
333            // When loading data with a non-data: base URL, the classic WebView would effectively
334            // "dump" that string of data into the WebView without going through regular URL
335            // loading steps such as decoding URL-encoded entities. We achieve this same behavior by
336            // base64 encoding the data that is passed here and then loading that as a data: URL.
337            try {
338                loadUrlParams = LoadUrlParams.createLoadDataParamsWithBaseUrl(
339                        Base64.encodeToString(data.getBytes("utf-8"), Base64.DEFAULT), mimeType,
340                        true, baseUrl, historyUrl, "utf-8");
341            } catch (java.io.UnsupportedEncodingException e) {
342                Log.wtf(TAG, "Unable to load data string " + data, e);
343                return;
344            }
345        }
346        loadUrlOnUiThread(loadUrlParams);
347    }
348
349    private void loadUrlOnUiThread(final LoadUrlParams loadUrlParams) {
350        if (ThreadUtils.runningOnUiThread()) {
351            mAwContents.loadUrl(loadUrlParams);
352        } else {
353            // Disallowed in WebView API for apps targetting a new SDK
354            assert mAppTargetSdkVersion < Build.VERSION_CODES.JELLY_BEAN_MR2;
355            ThreadUtils.postOnUiThread(new Runnable() {
356                @Override
357                public void run() {
358                    mAwContents.loadUrl(loadUrlParams);
359                }
360            });
361        }
362    }
363
364    public void evaluateJavaScript(String script, ValueCallback<String> resultCallback) {
365        checkThread();
366        mAwContents.evaluateJavaScript(script, resultCallback);
367    }
368
369    @Override
370    public void saveWebArchive(String filename) {
371        checkThread();
372        saveWebArchive(filename, false, null);
373    }
374
375    @Override
376    public void saveWebArchive(String basename, boolean autoname, ValueCallback<String> callback) {
377        checkThread();
378        mAwContents.saveWebArchive(basename, autoname, callback);
379    }
380
381    @Override
382    public void stopLoading() {
383        checkThread();
384        mAwContents.stopLoading();
385    }
386
387    @Override
388    public void reload() {
389        checkThread();
390        mAwContents.reload();
391    }
392
393    @Override
394    public boolean canGoBack() {
395        checkThread();
396        return mAwContents.canGoBack();
397    }
398
399    @Override
400    public void goBack() {
401        checkThread();
402        mAwContents.goBack();
403    }
404
405    @Override
406    public boolean canGoForward() {
407        checkThread();
408        return mAwContents.canGoForward();
409    }
410
411    @Override
412    public void goForward() {
413        checkThread();
414        mAwContents.goForward();
415    }
416
417    @Override
418    public boolean canGoBackOrForward(int steps) {
419        checkThread();
420        return mAwContents.canGoBackOrForward(steps);
421    }
422
423    @Override
424    public void goBackOrForward(int steps) {
425        checkThread();
426        mAwContents.goBackOrForward(steps);
427    }
428
429    @Override
430    public boolean isPrivateBrowsingEnabled() {
431        // Not supported in this WebView implementation.
432        return false;
433    }
434
435    @Override
436    public boolean pageUp(boolean top) {
437        checkThread();
438        return mAwContents.pageUp(top);
439    }
440
441    @Override
442    public boolean pageDown(boolean bottom) {
443        checkThread();
444        return mAwContents.pageDown(bottom);
445    }
446
447    @Override
448    public void clearView() {
449        checkThread();
450        UnimplementedWebViewApi.invoke();
451    }
452
453    @Override
454    public Picture capturePicture() {
455        checkThread();
456        return mAwContents.capturePicture();
457    }
458
459    @Override
460    public void exportToPdf(OutputStream stream, int width, int height,
461            ValueCallback<Boolean> resultCallback) {
462        checkThread();
463        // TODO(sgurun) enable this only after upstream part lands
464        //mAwContents.exportToPdf(stream, width, height, resultCallback);
465    }
466
467    @Override
468    public float getScale() {
469        checkThread();
470        return mAwContents.getScale();
471    }
472
473    @Override
474    public void setInitialScale(int scaleInPercent) {
475        checkThread();
476        mAwContents.getSettings().setInitialPageScale(scaleInPercent);
477    }
478
479    @Override
480    public void invokeZoomPicker() {
481        checkThread();
482        mAwContents.invokeZoomPicker();
483    }
484
485    @Override
486    public WebView.HitTestResult getHitTestResult() {
487        checkThread();
488        AwContents.HitTestData data = mAwContents.getLastHitTestResult();
489        mHitTestResult.setType(data.hitTestResultType);
490        mHitTestResult.setExtra(data.hitTestResultExtraData);
491        return mHitTestResult;
492    }
493
494    @Override
495    public void requestFocusNodeHref(Message hrefMsg) {
496        checkThread();
497        mAwContents.requestFocusNodeHref(hrefMsg);
498    }
499
500    @Override
501    public void requestImageRef(Message msg) {
502        checkThread();
503        mAwContents.requestImageRef(msg);
504    }
505
506    @Override
507    public String getUrl() {
508        checkThread();
509        String url =  mAwContents.getUrl();
510        if (url == null || url.trim().isEmpty()) return null;
511        return url;
512    }
513
514    @Override
515    public String getOriginalUrl() {
516        checkThread();
517        String url =  mAwContents.getOriginalUrl();
518        if (url == null || url.trim().isEmpty()) return null;
519        return url;
520    }
521
522    @Override
523    public String getTitle() {
524        checkThread();
525        return mAwContents.getTitle();
526    }
527
528    @Override
529    public Bitmap getFavicon() {
530        checkThread();
531        return mAwContents.getFavicon();
532    }
533
534    @Override
535    public String getTouchIconUrl() {
536        // Intentional no-op: hidden method on WebView.
537        return null;
538    }
539
540    @Override
541    public int getProgress() {
542        // No checkThread() because the value is cached java side (workaround for b/10533304).
543        return mAwContents.getMostRecentProgress();
544    }
545
546    @Override
547    public int getContentHeight() {
548        checkThread();
549        return mAwContents.getContentHeightCss();
550    }
551
552    @Override
553    public int getContentWidth() {
554        checkThread();
555        return mAwContents.getContentWidthCss();
556    }
557
558    @Override
559    public void pauseTimers() {
560        checkThread();
561        mAwContents.pauseTimers();
562    }
563
564    @Override
565    public void resumeTimers() {
566        checkThread();
567        mAwContents.resumeTimers();
568    }
569
570    @Override
571    public void onPause() {
572        checkThread();
573        mAwContents.onPause();
574    }
575
576    @Override
577    public void onResume() {
578        checkThread();
579        mAwContents.onResume();
580    }
581
582    @Override
583    public boolean isPaused() {
584        checkThread();
585        return mAwContents.isPaused();
586    }
587
588    @Override
589    public void freeMemory() {
590        checkThread();
591        // Intentional no-op. Memory is managed automatically by Chromium.
592    }
593
594    @Override
595    public void clearCache(boolean includeDiskFiles) {
596        checkThread();
597        mAwContents.clearCache(includeDiskFiles);
598    }
599
600    /**
601     * This is a poorly named method, but we keep it for historical reasons.
602     */
603    @Override
604    public void clearFormData() {
605        checkThread();
606        mAwContents.hideAutofillPopup();
607    }
608
609    @Override
610    public void clearHistory() {
611        checkThread();
612        mAwContents.clearHistory();
613    }
614
615    @Override
616    public void clearSslPreferences() {
617        checkThread();
618        mAwContents.clearSslPreferences();
619    }
620
621    @Override
622    public WebBackForwardList copyBackForwardList() {
623        checkThread();
624        return new WebBackForwardListChromium(
625                mAwContents.getNavigationHistory());
626    }
627
628    @Override
629    public void setFindListener(WebView.FindListener listener) {
630        checkThread();
631        mContentsClientAdapter.setFindListener(listener);
632    }
633
634    @Override
635    public void findNext(boolean forwards) {
636        checkThread();
637        mAwContents.findNext(forwards);
638    }
639
640    @Override
641    public int findAll(String searchString) {
642        checkThread();
643        mAwContents.findAllAsync(searchString);
644        return 0;
645    }
646
647    @Override
648    public void findAllAsync(String searchString) {
649        checkThread();
650        mAwContents.findAllAsync(searchString);
651    }
652
653    @Override
654    public boolean showFindDialog(String text, boolean showIme) {
655        checkThread();
656        if (mWebView.getParent() == null) {
657            return false;
658        }
659
660        FindActionModeCallback findAction = new FindActionModeCallback(mWebView.getContext());
661        if (findAction == null) {
662            return false;
663        }
664
665        mWebView.startActionMode(findAction);
666        findAction.setWebView(mWebView);
667        if (showIme) {
668            findAction.showSoftInput();
669        }
670
671        if (text != null) {
672            findAction.setText(text);
673            findAction.findAll();
674        }
675
676        return true;
677    }
678
679    @Override
680    public void notifyFindDialogDismissed() {
681        checkThread();
682        clearMatches();
683    }
684
685    @Override
686    public void clearMatches() {
687        checkThread();
688        mAwContents.clearMatches();
689    }
690
691    @Override
692    public void documentHasImages(Message response) {
693        checkThread();
694        mAwContents.documentHasImages(response);
695    }
696
697    @Override
698    public void setWebViewClient(WebViewClient client) {
699        checkThread();
700        mContentsClientAdapter.setWebViewClient(client);
701    }
702
703    @Override
704    public void setDownloadListener(DownloadListener listener) {
705        checkThread();
706        mContentsClientAdapter.setDownloadListener(listener);
707    }
708
709    @Override
710    public void setWebChromeClient(WebChromeClient client) {
711        checkThread();
712        mContentsClientAdapter.setWebChromeClient(client);
713    }
714
715    @Override
716    public void setPictureListener(WebView.PictureListener listener) {
717        checkThread();
718        mContentsClientAdapter.setPictureListener(listener);
719        mAwContents.enableOnNewPicture(listener != null,
720                mAppTargetSdkVersion >= Build.VERSION_CODES.JELLY_BEAN_MR2);
721    }
722
723    @Override
724    public void addJavascriptInterface(Object obj, String interfaceName) {
725        checkThread();
726        Class<? extends Annotation> requiredAnnotation = null;
727        if (mAppTargetSdkVersion >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
728           requiredAnnotation = JavascriptInterface.class;
729        }
730        mAwContents.addPossiblyUnsafeJavascriptInterface(obj, interfaceName, requiredAnnotation);
731    }
732
733    @Override
734    public void removeJavascriptInterface(String interfaceName) {
735        checkThread();
736        mAwContents.removeJavascriptInterface(interfaceName);
737    }
738
739    @Override
740    public WebSettings getSettings() {
741        checkThread();
742        if (mWebSettings == null) {
743            mWebSettings = new ContentSettingsAdapter(mAwContents.getSettings());
744        }
745        return mWebSettings;
746    }
747
748    @Override
749    public void setMapTrackballToArrowKeys(boolean setMap) {
750        checkThread();
751        // This is a deprecated API: intentional no-op.
752    }
753
754    @Override
755    public void flingScroll(int vx, int vy) {
756        checkThread();
757        mAwContents.flingScroll(vx, vy);
758    }
759
760    @Override
761    public View getZoomControls() {
762        checkThread();
763        // This was deprecated in 2009 and hidden in JB MR1, so just provide the minimum needed
764        // to stop very out-dated applications from crashing.
765        Log.w(TAG, "WebView doesn't support getZoomControls");
766        return mAwContents.getSettings().supportZoom() ? new View(mWebView.getContext()) : null;
767    }
768
769    @Override
770    public boolean canZoomIn() {
771        checkThread();
772        return mAwContents.canZoomIn();
773    }
774
775    @Override
776    public boolean canZoomOut() {
777        checkThread();
778        return mAwContents.canZoomOut();
779    }
780
781    @Override
782    public boolean zoomIn() {
783        checkThread();
784        return mAwContents.zoomIn();
785    }
786
787    @Override
788    public boolean zoomOut() {
789        checkThread();
790        return mAwContents.zoomOut();
791    }
792
793    @Override
794    public void dumpViewHierarchyWithProperties(BufferedWriter out, int level) {
795        UnimplementedWebViewApi.invoke();
796    }
797
798    @Override
799    public View findHierarchyView(String className, int hashCode) {
800        UnimplementedWebViewApi.invoke();
801        return null;
802    }
803
804    // WebViewProvider glue methods ---------------------------------------------------------------
805
806    @Override
807    // This needs to be kept thread safe!
808    public WebViewProvider.ViewDelegate getViewDelegate() {
809        return this;
810    }
811
812    @Override
813    public WebViewProvider.ScrollDelegate getScrollDelegate() {
814        checkThread();
815        return this;
816    }
817
818
819    // WebViewProvider.ViewDelegate implementation ------------------------------------------------
820
821    // TODO: remove from WebViewProvider and use default implementation from
822    // ViewGroup.
823    // @Override
824    public boolean shouldDelayChildPressedState() {
825        checkThread();
826        return true;
827    }
828
829//    @Override
830    public AccessibilityNodeProvider getAccessibilityNodeProvider() {
831        checkThread();
832        return mAwContents.getAccessibilityNodeProvider();
833    }
834
835    @Override
836    public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
837        checkThread();
838        mAwContents.onInitializeAccessibilityNodeInfo(info);
839    }
840
841    @Override
842    public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
843        checkThread();
844        mAwContents.onInitializeAccessibilityEvent(event);
845    }
846
847    @Override
848    public boolean performAccessibilityAction(int action, Bundle arguments) {
849        checkThread();
850        if (mAwContents.supportsAccessibilityAction(action)) {
851            return mAwContents.performAccessibilityAction(action, arguments);
852        }
853        return mWebViewPrivate.super_performAccessibilityAction(action, arguments);
854    }
855
856    @Override
857    public void setOverScrollMode(int mode) {
858        checkThread();
859        // This gets called from the android.view.View c'tor that WebView inherits from. This
860        // causes the method to be called when mAwContents == null.
861        // It's safe to ignore these calls however since AwContents will read the current value of
862        // this setting when it's created.
863        if (mAwContents != null) {
864            mAwContents.setOverScrollMode(mode);
865        }
866    }
867
868    @Override
869    public void setScrollBarStyle(int style) {
870        checkThread();
871        mAwContents.setScrollBarStyle(style);
872    }
873
874    @Override
875    public void onDrawVerticalScrollBar(Canvas canvas, Drawable scrollBar,
876                                        int l, int t, int r, int b) {
877        checkThread();
878        // WebViewClassic was overriding this method to handle rubberband over-scroll. Since
879        // WebViewChromium doesn't support that the vanilla implementation of this method can be
880        // used.
881        mWebViewPrivate.super_onDrawVerticalScrollBar(canvas, scrollBar, l, t, r, b);
882    }
883
884    @Override
885    public void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY) {
886        checkThread();
887        mAwContents.onContainerViewOverScrolled(scrollX, scrollY, clampedX, clampedY);
888    }
889
890    @Override
891    public void onWindowVisibilityChanged(int visibility) {
892        checkThread();
893        mAwContents.onWindowVisibilityChanged(visibility);
894    }
895
896    @Override
897    public void onDraw(Canvas canvas) {
898        checkThread();
899        mAwContents.onDraw(canvas);
900    }
901
902    @Override
903    public void setLayoutParams(ViewGroup.LayoutParams layoutParams) {
904        checkThread();
905        // TODO: This is the minimum implementation for HTMLViewer
906        // bringup. Likely will need to go up to ContentViewCore for
907        // a complete implementation.
908        mWebViewPrivate.super_setLayoutParams(layoutParams);
909    }
910
911    @Override
912    public boolean performLongClick() {
913        checkThread();
914        return mWebViewPrivate.super_performLongClick();
915    }
916
917    @Override
918    public void onConfigurationChanged(Configuration newConfig) {
919        checkThread();
920        mAwContents.onConfigurationChanged(newConfig);
921    }
922
923    @Override
924    public InputConnection onCreateInputConnection(EditorInfo outAttrs) {
925        checkThread();
926        return mAwContents.onCreateInputConnection(outAttrs);
927    }
928
929    @Override
930    public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) {
931        checkThread();
932        UnimplementedWebViewApi.invoke();
933        return false;
934    }
935
936    @Override
937    public boolean onKeyDown(int keyCode, KeyEvent event) {
938        checkThread();
939        UnimplementedWebViewApi.invoke();
940        return false;
941    }
942
943    @Override
944    public boolean onKeyUp(int keyCode, KeyEvent event) {
945        checkThread();
946        return mAwContents.onKeyUp(keyCode, event);
947    }
948
949    @Override
950    public void onAttachedToWindow() {
951        checkThread();
952        mAwContents.onAttachedToWindow();
953    }
954
955    @Override
956    public void onDetachedFromWindow() {
957        checkThread();
958        mAwContents.onDetachedFromWindow();
959        if (mGLfunctor != null) {
960            mGLfunctor.detach();
961        }
962    }
963
964    @Override
965    public void onVisibilityChanged(View changedView, int visibility) {
966        checkThread();
967        // The AwContents will find out the container view visibility before the first draw so we
968        // can safely ignore onVisibilityChanged callbacks that happen before init().
969        if (mAwContents != null) {
970            mAwContents.onVisibilityChanged(changedView, visibility);
971        }
972    }
973
974    @Override
975    public void onWindowFocusChanged(boolean hasWindowFocus) {
976        checkThread();
977        mAwContents.onWindowFocusChanged(hasWindowFocus);
978    }
979
980    @Override
981    public void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
982        checkThread();
983        mAwContents.onFocusChanged(focused, direction, previouslyFocusedRect);
984    }
985
986    @Override
987    public boolean setFrame(int left, int top, int right, int bottom) {
988        // TODO(joth): This is the minimum implementation for initial
989        // bringup. Likely will need to go up to AwContents for a complete
990        // implementation, e.g. setting the compositor visible region (to
991        // avoid painting tiles that are offscreen due to the view's position).
992        checkThread();
993        return mWebViewPrivate.super_setFrame(left, top, right, bottom);
994    }
995
996    @Override
997    public void onSizeChanged(int w, int h, int ow, int oh) {
998        checkThread();
999        mAwContents.onSizeChanged(w, h, ow, oh);
1000    }
1001
1002    @Override
1003    public void onScrollChanged(int l, int t, int oldl, int oldt) {
1004        checkThread();
1005    }
1006
1007    @Override
1008    public boolean dispatchKeyEvent(KeyEvent event) {
1009        checkThread();
1010        return mAwContents.dispatchKeyEvent(event);
1011    }
1012
1013    @Override
1014    public boolean onTouchEvent(MotionEvent ev) {
1015        checkThread();
1016        return mAwContents.onTouchEvent(ev);
1017    }
1018
1019    @Override
1020    public boolean onHoverEvent(MotionEvent event) {
1021        checkThread();
1022        return mAwContents.onHoverEvent(event);
1023    }
1024
1025    @Override
1026    public boolean onGenericMotionEvent(MotionEvent event) {
1027        checkThread();
1028        return mAwContents.onGenericMotionEvent(event);
1029    }
1030
1031    @Override
1032    public boolean onTrackballEvent(MotionEvent ev) {
1033        checkThread();
1034        // Trackball event not handled, which eventually gets converted to DPAD keyevents
1035        return false;
1036    }
1037
1038    @Override
1039    public boolean requestFocus(int direction, Rect previouslyFocusedRect) {
1040        checkThread();
1041        mAwContents.requestFocus();
1042        return mWebViewPrivate.super_requestFocus(direction, previouslyFocusedRect);
1043    }
1044
1045    @Override
1046    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
1047        checkThread();
1048        mAwContents.onMeasure(widthMeasureSpec, heightMeasureSpec);
1049    }
1050
1051    @Override
1052    public boolean requestChildRectangleOnScreen(View child, Rect rect, boolean immediate) {
1053        checkThread();
1054        return mAwContents.requestChildRectangleOnScreen(child, rect, immediate);
1055    }
1056
1057    @Override
1058    public void setBackgroundColor(final int color) {
1059        if (ThreadUtils.runningOnUiThread()) {
1060            mAwContents.setBackgroundColor(color);
1061        } else {
1062            // Disallowed in WebView API for apps targetting a new SDK
1063            assert mAppTargetSdkVersion < Build.VERSION_CODES.JELLY_BEAN_MR2;
1064            ThreadUtils.postOnUiThread(new Runnable() {
1065                @Override
1066                public void run() {
1067                    mAwContents.setBackgroundColor(color);
1068                }
1069            });
1070        }
1071    }
1072
1073    @Override
1074    public void setLayerType(int layerType, Paint paint) {
1075        checkThread();
1076        UnimplementedWebViewApi.invoke();
1077    }
1078
1079    @Override
1080    public void preDispatchDraw(Canvas canvas) {
1081        checkThread();
1082        // TODO(leandrogracia): remove this method from WebViewProvider if we think
1083        // we won't need it again.
1084    }
1085
1086    // WebViewProvider.ScrollDelegate implementation ----------------------------------------------
1087
1088    @Override
1089    public int computeHorizontalScrollRange() {
1090        checkThread();
1091        return mAwContents.computeHorizontalScrollRange();
1092    }
1093
1094    @Override
1095    public int computeHorizontalScrollOffset() {
1096        checkThread();
1097        return mAwContents.computeHorizontalScrollOffset();
1098    }
1099
1100    @Override
1101    public int computeVerticalScrollRange() {
1102        checkThread();
1103        return mAwContents.computeVerticalScrollRange();
1104    }
1105
1106    @Override
1107    public int computeVerticalScrollOffset() {
1108        checkThread();
1109        return mAwContents.computeVerticalScrollOffset();
1110    }
1111
1112    @Override
1113    public int computeVerticalScrollExtent() {
1114        checkThread();
1115        return mAwContents.computeVerticalScrollExtent();
1116    }
1117
1118    @Override
1119    public void computeScroll() {
1120        checkThread();
1121        mAwContents.computeScroll();
1122    }
1123
1124    // AwContents.InternalAccessDelegate implementation --------------------------------------
1125    private class InternalAccessAdapter implements AwContents.InternalAccessDelegate {
1126        @Override
1127        public boolean drawChild(Canvas arg0, View arg1, long arg2) {
1128            UnimplementedWebViewApi.invoke();
1129            return false;
1130        }
1131
1132        @Override
1133        public boolean super_onKeyUp(int arg0, KeyEvent arg1) {
1134            UnimplementedWebViewApi.invoke();
1135            return false;
1136        }
1137
1138        @Override
1139        public boolean super_dispatchKeyEventPreIme(KeyEvent arg0) {
1140            UnimplementedWebViewApi.invoke();
1141            return false;
1142        }
1143
1144        @Override
1145        public boolean super_dispatchKeyEvent(KeyEvent event) {
1146            return mWebViewPrivate.super_dispatchKeyEvent(event);
1147        }
1148
1149        @Override
1150        public boolean super_onGenericMotionEvent(MotionEvent arg0) {
1151            UnimplementedWebViewApi.invoke();
1152            return false;
1153        }
1154
1155        @Override
1156        public void super_onConfigurationChanged(Configuration arg0) {
1157            UnimplementedWebViewApi.invoke();
1158        }
1159
1160        @Override
1161        public int super_getScrollBarStyle() {
1162            return mWebViewPrivate.super_getScrollBarStyle();
1163        }
1164
1165        @Override
1166        public boolean awakenScrollBars() {
1167            mWebViewPrivate.awakenScrollBars(0);
1168            // TODO: modify the WebView.PrivateAccess to provide a return value.
1169            return true;
1170        }
1171
1172        @Override
1173        public boolean super_awakenScrollBars(int arg0, boolean arg1) {
1174            // TODO: need method on WebView.PrivateAccess?
1175            UnimplementedWebViewApi.invoke();
1176            return false;
1177        }
1178
1179        @Override
1180        public void onScrollChanged(int l, int t, int oldl, int oldt) {
1181            mWebViewPrivate.setScrollXRaw(l);
1182            mWebViewPrivate.setScrollYRaw(t);
1183            mWebViewPrivate.onScrollChanged(l, t, oldl, oldt);
1184        }
1185
1186        @Override
1187        public void overScrollBy(int deltaX, int deltaY,
1188            int scrollX, int scrollY,
1189            int scrollRangeX, int scrollRangeY,
1190            int maxOverScrollX, int maxOverScrollY,
1191            boolean isTouchEvent) {
1192            mWebViewPrivate.overScrollBy(deltaX, deltaY, scrollX, scrollY,
1193                    scrollRangeX, scrollRangeY, maxOverScrollX, maxOverScrollY, isTouchEvent);
1194        }
1195
1196        @Override
1197        public void super_scrollTo(int scrollX, int scrollY) {
1198            mWebViewPrivate.super_scrollTo(scrollX, scrollY);
1199        }
1200
1201        @Override
1202        public void setMeasuredDimension(int measuredWidth, int measuredHeight) {
1203            mWebViewPrivate.setMeasuredDimension(measuredWidth, measuredHeight);
1204        }
1205
1206        @Override
1207        public boolean requestDrawGL(Canvas canvas) {
1208            if (mGLfunctor == null) {
1209                mGLfunctor = new DrawGLFunctor(mAwContents.getAwDrawGLViewContext());
1210            }
1211            return mGLfunctor.requestDrawGL((HardwareCanvas)canvas, mWebView.getViewRootImpl());
1212        }
1213    }
1214}
1215