Android中WebView常見(jiàn)問(wèn)題及解決方案匯總
Android WebView常見(jiàn)問(wèn)題解決方案匯總:
就目前而言,如何應(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è)基本的回顧:
然后看一下具體的問(wèn)題及解決方案:
1.為WebView自定義錯(cuò)誤顯示界面:
覆寫(xiě)WebViewClient中的onReceivedError()方法:
/** * 顯示自定義錯(cuò)誤提示頁(yè)面,用一個(gè)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是否已經(jīng)滾動(dòng)到頁(yè)面底端:
getScrollY()方法返回的是當(dāng)前可見(jiàn)區(qū)域的頂端距整個(gè)頁(yè)面頂端的距離,也就是當(dāng)前內(nèi)容滾動(dòng)的距離.
getHeight()或者getBottom()方法都返回當(dāng)前WebView 這個(gè)容器的高度
getContentHeight 返回的是整個(gè)html 的高度,但并不等同于當(dāng)前整個(gè)頁(yè)面的高度,因?yàn)閃ebView 有縮放功能, 所以當(dāng)前整個(gè)頁(yè)面的
高度實(shí)際上應(yīng)該是原始html 的高度再乘上縮放比例. 因此,更正后的結(jié)果,準(zhǔn)確的判斷方法應(yīng)該是:
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ì)引起頁(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:
@SuppressLint("NewApi") @Override public WebResourceResponse shouldInterceptRequest(WebView view,String url) { // 非超鏈接(如Ajax)請(qǐng)求無(wú)法直接添加請(qǐng)求頭,現(xiàn)拼接到url末尾,這里拼接一個(gè)imei作為示例 String ajaxUrl = url; // 如標(biāo)識(shí):req=ajax if (url.contains("req=ajax")) { ajaxUrl += "&imei=" + imei; } return super.shouldInterceptRequest(view, ajaxUrl); }
7.在頁(yè)面中先顯示圖片:
@Override public void onLoadResource(WebView view, String url) { mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_LOAD_RESOURCE, url); if (url.indexOf(".jpg") > 0) { hideProgress(); //請(qǐng)求圖片時(shí)即顯示頁(yè)面 mEventListener.onWebViewEvent(CustomWebView.this, OnWebViewEventListener.EVENT_ON_HIDE_PROGRESS, view.getUrl()); } super.onLoadResource(view, url); }
8.屏蔽掉長(zhǎng)按事件 因?yàn)閣ebview長(zhǎng)按時(shí)將會(huì)調(diào)用系統(tǒng)的復(fù)制控件:
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.
這些是目前我整理出來(lái)的一些注意事項(xiàng)和問(wèn)題解決方案,也歡迎大家多提一些關(guān)于webview的問(wèn)題,如果有合適的解決方案,我會(huì)直接更新到這篇文章.
11.WebView 在Android4.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)中獲取
WebChromeClient webChromeClient = new WebChromeClient() { @Override public void onReceivedTitle(WebView view, String title) { super.onReceivedTitle(view, title); txtTitle.setText(title); } };
但是發(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的話,只要將標(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)用.
14.WebView頁(yè)面中播放了音頻,退出Activity后音頻仍然在播放
需要在Activity的onDestory()中調(diào)用
webView.destroy();
但是直接調(diào)用可能會(huì)引起如下錯(cuò)誤:
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調(diào)用destory時(shí),webview仍綁定在Activity上.這是由于自定義webview構(gòu)建時(shí)傳入了該Activity的context對(duì)象,因此需要先從父容器中移除webview,然后再銷毀webview:
rootLayout.removeView(webView); webView.destroy();
15. WebView長(zhǎng)按自定義菜單,實(shí)現(xiàn)復(fù)制分享相關(guān)功能
這個(gè)功能首先可以從兩方面完成:
(1) 在js中完成:
處理android.selection.longTouch
(2) 安卓層處理:
首先使用OnTouchListener實(shí)現(xiàn)長(zhǎng)按實(shí)現(xiàn)監(jiān)聽(tīng),然后實(shí)現(xiàn)WebView的Context menu,最后調(diào)用webview中的emulateShiftHeld(),為了適配安卓不同版本,最好使用反射方式調(diào)用.
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android自定義View實(shí)現(xiàn)簡(jiǎn)單的圓形Progress效果
這篇文章主要介紹了Android自定義View實(shí)現(xiàn)簡(jiǎn)單的圓形Progress效果的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09ionic App 解決android端在真機(jī)上tab處于頂部的問(wèn)題
這篇文章主要介紹了ionic App 解決android端在真機(jī)上tab處于頂部的問(wèn)題的相關(guān)資料,需要的朋友可以參考下2017-06-06Android自定義控件實(shí)現(xiàn)底部菜單(上)
這篇文章主要為大家詳細(xì)介紹了Android自定義控件實(shí)現(xiàn)底部菜單的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01Android RecycleView滑動(dòng)停止后自動(dòng)吸附效果的實(shí)現(xiàn)代碼(滑動(dòng)定位)
這篇文章主要介紹了Android RecycleView滑動(dòng)停止后自動(dòng)吸附效果的實(shí)現(xiàn)代碼(滑動(dòng)定位),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10深入了解Android Okio的超時(shí)機(jī)制
Okio是一個(gè)IO庫(kù),底層基于Java原生的輸入輸出流實(shí)現(xiàn)。但原生的輸入輸出流并沒(méi)有提供超時(shí)的檢測(cè)機(jī)制。而Okio實(shí)現(xiàn)了這個(gè)功能,本文就來(lái)為大家詳細(xì)講講2023-02-02android RecyclerView的一些優(yōu)化點(diǎn)介紹
大家好,本篇文章主要講的是android RecyclerView的一些優(yōu)化點(diǎn)介紹,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12深入解析Android系統(tǒng)中應(yīng)用程序前后臺(tái)切換的實(shí)現(xiàn)要點(diǎn)
這篇文章主要介紹了Android系統(tǒng)中應(yīng)用程序前后臺(tái)切換的實(shí)現(xiàn)要點(diǎn),除了切換操作的效果之外還重點(diǎn)講解了判斷程序運(yùn)行于前臺(tái)還是后臺(tái)的方法,需要的朋友可以參考下2016-04-04Flutter調(diào)用Android和iOS原生代碼的方法示例
這篇文章主要給大家介紹了關(guān)于Flutter調(diào)用Android和iOS原生代碼的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Flutter具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04