国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android中WebView常見問題及解決方案匯總

2019-12-12 04:48:09
字體:
來源:轉載
供稿:網友

Android WebView常見問題解決方案匯總:

就目前而言,如何應對版本的頻繁更新呢,又如何靈活多變地展示我們的界面呢,這又涉及到了web app與native app之間孰優孰劣的爭論. 于是乎,一種混合型的app誕生了,靈活多變的部分,如淘寶商城首頁的活動頁面,一集凡客誠品中我們都可以見到web頁面與native頁面的混合,既利用了web app的靈活易更新,也借助了native app本身的效率.

當然,就會用到webview這樣的一個控件,這里,我把自己使用過程中遇到的一些問題整理下來.

首先上張圖對WebView進行一個基本的回顧:

然后看一下具體的問題及解決方案:

1.為WebView自定義錯誤顯示界面:

覆寫WebViewClient中的onReceivedError()方法:

/**  * 顯示自定義錯誤提示頁面,用一個View覆蓋在WebView  */ protected void showErrorPage() {   LinearLayout webParentView = (LinearLayout)mWebView.getParent();      initErrorPage();   while (webParentView.getChildCount() > 1) {     webParentView.removeViewAt(0);   }   LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT);   webParentView.addView(mErrorView, 0, lp);   mIsErrorPage = true; } protected void hideErrorPage() {   LinearLayout webParentView = (LinearLayout)mWebView.getParent();      mIsErrorPage = false;   while (webParentView.getChildCount() > 1) {     webParentView.removeViewAt(0);   } }     protected void initErrorPage() {   if (mErrorView == null) {     mErrorView = View.inflate(this, R.layout.online_error, null);     Button button = (Button)mErrorView.findViewById(R.id.online_error_btn_retry);     button.setOnClickListener(new OnClickListener() {       public void onClick(View v) {         mWebView.reload();       }     });     mErrorView.setOnClickListener(null);   } } 
@Override public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) { <span style="white-space:pre">         </span> <span style="white-space:pre">     </span>mErrorView.setVisibility(View.VISIBLE); <span style="white-space:pre">     </span>super.onReceivedError(view, errorCode, description, failingUrl); }   

2.WebView cookies清理:

CookieSyncManager.createInstance(this);  CookieSyncManager.getInstance().startSync();  CookieManager.getInstance().removeSessionCookie(); 

3.清理cache 和歷史記錄:

webView.clearCache(true);  webView.clearHistory(); 

4.判斷WebView是否已經滾動到頁面底端:

getScrollY()方法返回的是當前可見區域的頂端距整個頁面頂端的距離,也就是當前內容滾動的距離.  

getHeight()或者getBottom()方法都返回當前WebView 這個容器的高度  

getContentHeight 返回的是整個html 的高度,但并不等同于當前整個頁面的高度,因為WebView 有縮放功能, 所以當前整個頁面的

高度實際上應該是原始html 的高度再乘上縮放比例. 因此,更正后的結果,準確的判斷方法應該是:  
if(WebView.getContentHeight*WebView.getScale() == (webview.getHeight()+WebView.getScrollY())){ //已經處于底端 }  

5.URL攔截:

Android WebView是攔截不到頁面內的fragment跳轉的。但是url跳轉的話,又會引起頁面刷新,H5頁面的體驗又下降了。只能給WebView注入JS方法了。

6.處理WebView中的非超鏈接請求(如Ajax請求): 

有時候需要加上請求頭,但是非超鏈接的請求,沒有辦法再shouldOverrinding中攔截并用webView.loadUrl(String url,HashMap headers)方法添加請求頭

目前用了一個臨時的辦法解決:

首先需要在url中加特殊標記/協議, 如在onWebViewResource方法中攔截對應的請求,然后將要添加的請求頭,以get形式拼接到url末尾

在shouldInterceptRequest()方法中,可以攔截到所有的網頁中資源請求,比如加載JS,圖片以及Ajax請求等等

Ex:

@SuppressLint("NewApi") @Override public WebResourceResponse shouldInterceptRequest(WebView view,String url) {   // 非超鏈接(如Ajax)請求無法直接添加請求頭,現拼接到url末尾,這里拼接一個imei作為示例    String ajaxUrl = url;   // 如標識:req=ajax   if (url.contains("req=ajax")) {     ajaxUrl += "&imei=" + imei;   }    return super.shouldInterceptRequest(view, ajaxUrl);  } 

7.在頁面中先顯示圖片:

@Override public void onLoadResource(WebView view, String url) {  mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_LOAD_RESOURCE, url);   if (url.indexOf(".jpg") > 0) {    hideProgress(); //請求圖片時即顯示頁面    mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_HIDE_PROGRESS, view.getUrl());    }   super.onLoadResource(view, url); } 

 8.屏蔽掉長按事件 因為webview長按時將會調用系統的復制控件:

mWebView.setOnLongClickListener(new OnLongClickListener() {            @Override      public boolean onLongClick(View v) {        return true;      }    }); 

9.在WebView加入 flash支持:

String temp = "<html><body bgcolor=/"" + "black"         + "/"> <br/><embed src=/"" + url + "/" width=/"" + "100%"         + "/" height=/"" + "90%" + "/" scale=/"" + "noscale"         + "/" type=/"" + "application/x-shockwave-flash"         + "/"> </embed></body></html>"; String mimeType = "text/html"; String encoding = "utf-8"; web.loadDataWithBaseURL("null", temp, mimeType, encoding, ""); 

10.WebView保留縮放功能但隱藏縮放控件:

mWebView.getSettings().setSupportZoom(true);     mWebView.getSettings().setBuiltInZoomControls(true);     if (DeviceUtils.hasHoneycomb())        mWebView.getSettings().setDisplayZoomControls(false); 

注意:setDisplayZoomControls是在Android 3.0中新增的API.

這些是目前我整理出來的一些注意事項和問題解決方案,也歡迎大家多提一些關于webview的問題,如果有合適的解決方案,我會直接更新到這篇文章.

11.WebView 在Android4.4的手機上onPageFinished()回調會多調用一次(具體原因待追查)

需要盡量避免在onPageFinished()中做業務操作,否則會導致重復調用,還有可能會引起邏輯上的錯誤.

12.需要通過獲取Web頁中的title用來設置自己界面中的title及相關問題:

需要給WebView設置 WebChromeClient,并在onReceiveTitle()回調中獲取

WebChromeClient webChromeClient = new WebChromeClient() {        @Override        public void onReceivedTitle(WebView view, String title) {          super.onReceivedTitle(view, title);                   txtTitle.setText(title);        }        };  

但是發現在小米3的手機上,當通過webview.goBack()回退的時候,并沒有觸發onReceiveTitle(),這樣會導致標題仍然是之前子頁面的標題,沒有切換回來.

這里可以分兩種情況去處理:

(1) 可以確定webview中子頁面只有二級頁面,沒有更深的層次,這里只需要判斷當前頁面是否為初始的主頁面,可以goBack的話,只要將標題設置回來即可.

(2)webview中可能有多級頁面或者以后可能增加多級頁面,這種情況處理起來要復雜一些:

因為正常順序加載的情況onReceiveTitle是一定會觸發的,所以就需要自己來維護webview  loading的一個url棧及url與title的映射關系

那么就需要一個ArrayList來保持加載過的url,一個HashMap保存url及對應的title.

正常順序加載時,將url和對應的title保存起來,webview回退時,移除當前url并取出將要回退到的web 頁的url,找到對應的title進行設置即可.

這里還要說一點,當加載出錯的時候,比如無網絡,這時onReceiveTitle中獲取的標題為 找不到該網頁,因此建議當觸發onReceiveError時,不要使用獲取到的title.

13.WebView因addJavaScriptInterface()引起的安全問題.

這個問題主要是因為會有惡意的js代碼注入,尤其是在已經獲取root權限的手機上,一些惡意程序可能會利用該漏洞安裝或者卸載應用.

14.WebView頁面中播放了音頻,退出Activity后音頻仍然在播放

需要在Activity的onDestory()中調用

webView.destroy(); 

但是直接調用可能會引起如下錯誤:

10-10 15:01:11.402: E/ViewRootImpl(7502): sendUserActionEvent() mView == null 
10-10 15:01:26.818: E/webview(7502): java.lang.Throwable: Error: WebView.destroy() called while still attached! 
10-10 15:01:26.818: E/webview(7502):    at android.webkit.WebViewClassic.destroy(WebViewClassic.java:4142) 
10-10 15:01:26.818: E/webview(7502):    at android.webkit.WebView.destroy(WebView.java:707) 
10-10 15:01:26.818: E/webview(7502):    at com.didi.taxi.ui.webview.OperatingWebViewActivity.onDestroy(OperatingWebViewActivity.java:236) 
10-10 15:01:26.818: E/webview(7502):    at android.app.Activity.performDestroy(Activity.java:5543) 
10-10 15:01:26.818: E/webview(7502):    at android.app.Instrumentation.callActivityOnDestroy(Instrumentation.java:1134) 
10-10 15:01:26.818: E/webview(7502):    at android.app.ActivityThread.performDestroyActivity(ActivityThread.java:3619) 
10-10 15:01:26.818: E/webview(7502):    at android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:3654) 
10-10 15:01:26.818: E/webview(7502):    at android.app.ActivityThread.access$1300(ActivityThread.java:159) 
10-10 15:01:26.818: E/webview(7502):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1369) 
10-10 15:01:26.818: E/webview(7502):    at android.os.Handler.dispatchMessage(Handler.java:99) 
10-10 15:01:26.818: E/webview(7502):    at android.os.Looper.loop(Looper.java:137) 
10-10 15:01:26.818: E/webview(7502):    at android.app.ActivityThread.main(ActivityThread.java:5419) 
10-10 15:01:26.818: E/webview(7502):    at java.lang.reflect.Method.invokeNative(Native Method) 
10-10 15:01:26.818: E/webview(7502):    at java.lang.reflect.Method.invoke(Method.java:525) 
10-10 15:01:26.818: E/webview(7502):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1187) 
10-10 15:01:26.818: E/webview(7502):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003) 
10-10 15:01:26.818: E/webview(7502):    at dalvik.system.NativeStart.main(Native Method) 

如上所示,webview調用destory時,webview仍綁定在Activity上.這是由于自定義webview構建時傳入了該Activity的context對象,因此需要先從父容器中移除webview,然后再銷毀webview:

rootLayout.removeView(webView); webView.destroy(); 

15. WebView長按自定義菜單,實現復制分享相關功能 

這個功能首先可以從兩方面完成:

(1) 在js中完成:

處理android.selection.longTouch

(2) 安卓層處理:

首先使用OnTouchListener實現長按實現監聽,然后實現WebView的Context menu,最后調用webview中的emulateShiftHeld(),為了適配安卓不同版本,最好使用反射方式調用.

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 大连市| 水富县| 昭苏县| 綦江县| 剑阁县| 南召县| 河南省| 清涧县| 安吉县| 定安县| 睢宁县| 墨江| 于田县| 泽州县| 永定县| 舞阳县| 江川县| 大竹县| 论坛| 通海县| 桃园县| 平山县| 天峨县| 祁门县| 双柏县| 方山县| 和政县| 太谷县| 汾西县| 汉川市| 稻城县| 银川市| 拜城县| 西峡县| 山东| 濮阳市| 泽库县| 肥乡县| 封丘县| 金沙县| 西宁市|