<del id="d4fwx"><form id="d4fwx"></form></del>
      <del id="d4fwx"><form id="d4fwx"></form></del><del id="d4fwx"><form id="d4fwx"></form></del>

            <code id="d4fwx"><abbr id="d4fwx"></abbr></code>
          • Android WebView常見(jiàn)問(wèn)題及解決方案匯總

            2022-06-23    分類(lèi): 解決方案

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

            首先上張圖對(duì)WebView進(jìn)行一個(gè)基本的回顧:



            以上思維導(dǎo)圖原文件下載地址:

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


            然后看一下具體的問(wèn)題及解決方案:

            1.為WebView自定義錯(cuò)誤顯示界面:

            覆寫(xiě)WebViewClient中的onReceivedError()方法:


            [java] view plain copy
            1. /** 
            2.  * 顯示自定義錯(cuò)誤提示頁(yè)面,用一個(gè)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)滾動(dòng)到頁(yè)面底端:

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

            5.URL攔截:

            Android WebView是攔截不到頁(yè)面內(nèi)的fragment跳轉(zhuǎn)的。但是url跳轉(zhuǎn)的話(huà),又會(huì)引起頁(yè)面刷新,H5頁(yè)面的體驗(yàn)又下降了。只能給WebView注入JS方法了。


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

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

              目前用了一個(gè)臨時(shí)的辦法解決:

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

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

            Ex:

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


            7.在頁(yè)面中先顯示圖片:

            [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(); //請(qǐng)求圖片時(shí)即顯示頁(yè)面  
            6.      mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_HIDE_PROGRESS, view.getUrl());  
            7.      }  
            8.     super.onLoadResource(view, url);  
            9. }  


             8.屏蔽掉長(zhǎng)按事件 因?yàn)閣ebview長(zhǎng)按時(shí)將會(huì)調(diào)用系統(tǒng)的復(fù)制控件:
                  


            [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是在A(yíng)ndroid 3.0中新增的API.


            這些是目前我整理出來(lái)的一些注意事項(xiàng)和問(wèn)題解決方案,也歡迎大家多提一些關(guān)于webview的問(wèn)題,如果有合適的解決方案,我會(huì)直接更新到這篇文章.


            8月份更新:

            11.WebView 在A(yíng)ndroid4.4的手機(jī)上onPageFinished()回調(diào)會(huì)多調(diào)用一次(具體原因待追查)

            需要盡量避免在onPageFinished()中做業(yè)務(wù)操作,否則會(huì)導(dǎo)致重復(fù)調(diào)用,還有可能會(huì)引起邏輯上的錯(cuò)誤.


            12.需要通過(guò)獲取Web頁(yè)中的title用來(lái)設(shè)置自己界面中的title及相關(guān)問(wèn)題:

            需要給WebView設(shè)置 WebChromeClient,并在onReceiveTitle()回調(diào)中獲取

            [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的手機(jī)上,當(dāng)通過(guò)webview.goBack()回退的時(shí)候,并沒(méi)有觸發(fā)onReceiveTitle(),這樣會(huì)導(dǎo)致標(biāo)題仍然是之前子頁(yè)面的標(biāo)題,沒(méi)有切換回來(lái).

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

            (1) 可以確定webview中子頁(yè)面只有二級(jí)頁(yè)面,沒(méi)有更深的層次,這里只需要判斷當(dāng)前頁(yè)面是否為初始的主頁(yè)面,可以goBack的話(huà),只要將標(biāo)題設(shè)置回來(lái)即可.

            (2)webview中可能有多級(jí)頁(yè)面或者以后可能增加多級(jí)頁(yè)面,這種情況處理起來(lái)要復(fù)雜一些:

                因?yàn)檎m樞蚣虞d的情況onReceiveTitle是一定會(huì)觸發(fā)的,所以就需要自己來(lái)維護(hù)webview  loading的一個(gè)url棧及url與title的映射關(guān)系

               那么就需要一個(gè)ArrayList來(lái)保持加載過(guò)的url,一個(gè)HashMap保存url及對(duì)應(yīng)的title.

               正常順序加載時(shí),將url和對(duì)應(yīng)的title保存起來(lái),webview回退時(shí),移除當(dāng)前url并取出將要回退到的web 頁(yè)的url,找到對(duì)應(yīng)的title進(jìn)行設(shè)置即可.


            這里還要說(shuō)一點(diǎn),當(dāng)加載出錯(cuò)的時(shí)候,比如無(wú)網(wǎng)絡(luò),這時(shí)onReceiveTitle中獲取的標(biāo)題為 找不到該網(wǎng)頁(yè),因此建議當(dāng)觸發(fā)onReceiveError時(shí),不要使用獲取到的title.


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

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

            關(guān)于詳細(xì)的情況可以參考下面這篇文章:

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

               

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

            很好地解決了webview  js注入的安全問(wèn)題.


            10月份更新:

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

            需要在A(yíng)ctivity的onDestory()中調(diào)用


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

            但是直接調(diào)用可能會(huì)引起如下錯(cuò)誤:

            [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
          • 分享標(biāo)題:Android WebView常見(jiàn)問(wèn)題及解決方案匯總
            新聞來(lái)源:http://www.jbt999.com/news26/170776.html

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

            廣告

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

            成都網(wǎng)站建設(shè)

              <del id="d4fwx"><form id="d4fwx"></form></del>
              <del id="d4fwx"><form id="d4fwx"></form></del><del id="d4fwx"><form id="d4fwx"></form></del>

                    <code id="d4fwx"><abbr id="d4fwx"></abbr></code>
                  • 欧美性爱内射大奶美女 | 黄片大全在线观看 | 蜜桃久久精品成人无码AV | 4438全国最大无码视频 | 黄色操逼爽烦尼玛 |