成人午夜视频全免费观看高清-秋霞福利视频一区二区三区-国产精品久久久久电影小说-亚洲不卡区三一区三区一区

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

2022-06-23    分類: 解決方案

就目前而言,如何應對版本的頻繁更新呢,又如何靈活多變地展示我們的界面呢,這又涉及到了web app與native app之間孰優(yōu)孰劣的爭論. 于是乎,一種混合型的app誕生了,靈活多變的部分,如淘寶商城首頁的活動頁面,一集凡客誠品中我們都可以見到web 頁面與native頁面的混合,既利用了web app的靈活易更新,也借助了native app本身的效率.
當然,就會用到webview這樣的一個控件,這里,我把自己使用過程中遇到的一些問題整理下來.

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



以上思維導圖原文件下載地址:

http://download.csdn.net/detail/t12x3456/6509195


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

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

覆寫WebViewClient中的onReceivedError()方法:


[java] view plain copy
  1. /** 
  2.  * 顯示自定義錯誤提示頁面,用一個View覆蓋在WebView 
  3.  */  
  4. protected void showErrorPage() {  
  5.     LinearLayout webParentView = (LinearLayout)mWebView.getParent();  
  6.       
  7.     initErrorPage();  
  8.     while (webParentView.getChildCount() > 1) {  
  9.         webParentView.removeViewAt(0);  
  10.     }  
  11.     LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT);  
  12.     webParentView.addView(mErrorView, 0, lp);  
  13.     mIsErrorPage = true;  
  14. }  
  15. protected void hideErrorPage() {  
  16.     LinearLayout webParentView = (LinearLayout)mWebView.getParent();  
  17.       
  18.     mIsErrorPage = false;  
  19.     while (webParentView.getChildCount() > 1) {  
  20.         webParentView.removeViewAt(0);  
  21.     }  
  22. }  
  23.   
  24.   
  25.    protected void initErrorPage() {  
  26.     if (mErrorView == null) {  
  27.         mErrorView = View.inflate(this, R.layout.online_error, null);  
  28.         Button button = (Button)mErrorView.findViewById(R.id.online_error_btn_retry);  
  29.         button.setOnClickListener(new OnClickListener() {  
  30.             public void onClick(View v) {  
  31.                 mWebView.reload();  
  32.             }  
  33.         });  
  34.         mErrorView.setOnClickListener(null);  
  35.     }  
  36. }  
[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1.   
[java] view plain copy
  1.   
[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. @Override  
  2. public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {  
  3. <span style="white-space:pre">                  </span>  
  4. <span style="white-space:pre">          </span>mErrorView.setVisibility(View.VISIBLE);  
  5. <span style="white-space:pre">          </span>super.onReceivedError(view, errorCode, description, failingUrl);  
  6. }          

2.WebView cookies清理:

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. CookieSyncManager.createInstance(this);   
  2. CookieSyncManager.getInstance().startSync();   
  3. CookieManager.getInstance().removeSessionCookie();   

3.清理cache 和歷史記錄:

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. webView.clearCache(true);   
  2. webView.clearHistory();  

4.判斷WebView是否已經(jīng)滾動到頁面底端:

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. getScrollY()方法返回的是當前可見區(qū)域的頂端距整個頁面頂端的距離,也就是當前內容滾動的距離.   
  2. getHeight()或者getBottom()方法都返回當前WebView 這個容器的高度   
  3. getContentHeight 返回的是整個html 的高度,但并不等同于當前整個頁面的高度,因為WebView 有縮放功能, 所以當前整個頁面的高度實際上應該是原始html 的高度再乘上縮放比例. 因此,更正后的結果,準確的判斷方法應該是:   
  4. if(WebView.getContentHeight*WebView.getScale() == (webview.getHeight()+WebView.getScrollY())){ //已經(jīng)處于底端 }   

5.URL攔截:

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


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

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

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

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

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

Ex:

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. @SuppressLint("NewApi")  
  2. @Override  
  3. public WebResourceResponse shouldInterceptRequest(WebView view,String url) {  
  4.     // 非超鏈接(如Ajax)請求無法直接添加請求頭,現(xiàn)拼接到url末尾,這里拼接一個imei作為示例  
  5.   
  6.     String ajaxUrl = url;  
  7.     // 如標識:req=ajax  
  8.     if (url.contains("req=ajax")) {  
  9.        ajaxUrl += "&imei=" + imei;  
  10.     }  
  11.   
  12.     return super.shouldInterceptRequest(view, ajaxUrl);  
  13.   
  14. }  


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

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. @Override  
  2. public void onLoadResource(WebView view, String url) {  
  3.   mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_LOAD_RESOURCE, url);  
  4.     if (url.indexOf(".jpg") > 0) {  
  5.      hideProgress(); //請求圖片時即顯示頁面  
  6.      mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_HIDE_PROGRESS, view.getUrl());  
  7.      }  
  8.     super.onLoadResource(view, url);  
  9. }  


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


[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. mWebView.setOnLongClickListener(new OnLongClickListener() {  
  2.             
  3.           @Override  
  4.           public boolean onLongClick(View v) {  
  5.               return true;  
  6.           }  
  7.       });  

9.在WebView加入 flash支持:

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. String temp = "<html><body bgcolor=\"" + "black"  
  2.                 + "\"> <br/><embed src=\"" + url + "\" width=\"" + "100%"  
  3.                 + "\" height=\"" + "90%" + "\" scale=\"" + "noscale"  
  4.                 + "\" type=\"" + "application/x-shockwave-flash"  
  5.                 + "\"> </embed></body></html>";  
  6. String mimeType = "text/html";  
  7. String encoding = "utf-8";  
  8. web.loadDataWithBaseURL("null", temp, mimeType, encoding, "");  

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


[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. mWebView.getSettings().setSupportZoom(true);  
  2.         mWebView.getSettings().setBuiltInZoomControls(true);  
  3.         if (DeviceUtils.hasHoneycomb())  
  4.               mWebView.getSettings().setDisplayZoomControls(false);  

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


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


8月份更新:

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

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


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

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

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. WebChromeClient webChromeClient = new WebChromeClient() {    
  2.             @Override    
  3.             public void onReceivedTitle(WebView view, String title) {    
  4.                 super.onReceivedTitle(view, title);    
  5.                  
  6.                 txtTitle.setText(title);    
  7.             }    
  8.     
  9.         };    

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

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

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

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

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

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

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


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


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

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

關于詳細的情況可以參考下面這篇文章:

.http://blog.csdn.net/leehong2005/article/details/11808557

   

還有一個開源項目可以參考: https://github.com/pedant/safe-java-js-webview-bridge, 該項目利用onJsPrompt() 替代了addJavaScriptInterface(),(解決方案類似上述參考的博客)同時增加了異步回調,

很好地解決了webview  js注入的安全問題.


10月份更新:

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

需要在Activity的onDestory()中調用


[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  1. webView.destroy();  

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

[java] view plain copy
 在CODE上查看代碼片派生到我的代碼片
  • 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
  • 分享標題:Android WebView常見問題及解決方案匯總
    新聞來源:http://jinyejixie.com/news26/170776.html

    網(wǎng)站建設、網(wǎng)絡推廣公司-創(chuàng)新互聯(lián),是專注品牌與效果的網(wǎng)站制作,網(wǎng)絡營銷seo公司;服務項目有解決方案

    廣告

    聲明:本網(wǎng)站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經(jīng)允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)

    成都網(wǎng)站建設
    报价| 阿拉善右旗| 深水埗区| 垦利县| 县级市| 南投县| 安乡县| 翁源县| 玛沁县| 卢龙县| 荥阳市| 芦山县| 茂名市| 荥阳市| 临武县| 神农架林区| 交城县| 诏安县| 将乐县| 涿州市| 文山县| 新沂市| 莎车县| 保靖县| 岱山县| 台南市| 信阳市| 元朗区| 禹城市| 谢通门县| 烟台市| 迁西县| 民县| 台湾省| 乌兰察布市| 靖宇县| 永修县| 台江县| 噶尔县| 江安县| 祥云县|