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

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

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

當(dāng)然制肮,就會用到webview這樣的一個控件冒窍,這里递沪,我把自己使用過程中遇到的一些問題整理下來.

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

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

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

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

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

覆寫WebViewClient中的onReceivedError()方法:

[java]view plaincopy

/**

*?顯示自定義錯誤提示頁面,用一個View覆蓋在WebView

*/

protectedvoidshowErrorPage()?{

LinearLayout?webParentView?=?(LinearLayout)mWebView.getParent();

initErrorPage();

while(webParentView.getChildCount()?>1)?{

webParentView.removeViewAt(0);

}

LinearLayout.LayoutParams?lp?=newLinearLayout.LayoutParams(LayoutParams.FILL_PARENT,LayoutParams.FILL_PARENT);

webParentView.addView(mErrorView,0,?lp);

mIsErrorPage?=true;

}

protectedvoidhideErrorPage()?{

LinearLayout?webParentView?=?(LinearLayout)mWebView.getParent();

mIsErrorPage?=false;

while(webParentView.getChildCount()?>1)?{

webParentView.removeViewAt(0);

}

}

protectedvoidinitErrorPage()?{

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(newOnClickListener()?{

publicvoidonClick(View?v)?{

mWebView.reload();

}

});

mErrorView.setOnClickListener(null);

}

}

[java]view plaincopy

[java]view plaincopy

[java]view plaincopy

@Override

publicvoidonReceivedError(WebView?view,interrorCode,?String?description,?String?failingUrl)?{

??????????????????

??????????mErrorView.setVisibility(View.VISIBLE);

??????????super.onReceivedError(view,?errorCode,?description,?failingUrl);

}

2.WebView cookies清理:

[java]view plaincopy

CookieSyncManager.createInstance(this);

CookieSyncManager.getInstance().startSync();

CookieManager.getInstance().removeSessionCookie();

3.清理cache 和歷史記錄:

[java]view plaincopy

webView.clearCache(true);

webView.clearHistory();

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

[java]view plaincopy

getScrollY()方法返回的是當(dāng)前可見區(qū)域的頂端距整個頁面頂端的距離,也就是當(dāng)前內(nèi)容滾動的距離.

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

getContentHeight?返回的是整個html?的高度,但并不等同于當(dāng)前整個頁面的高度,因為WebView?有縮放功能,?所以當(dāng)前整個頁面的高度實際上應(yīng)該是原始html?的高度再乘上縮放比例.?因此,更正后的結(jié)果,準(zhǔn)確的判斷方法應(yīng)該是:

if(WebView.getContentHeight*WebView.getScale()?==?(webview.getHeight()+WebView.getScrollY())){//已經(jīng)處于底端?}

5.URL攔截:

Android WebView是攔截不到頁面內(nèi)的fragment跳轉(zhuǎn)的超燃。但是url跳轉(zhuǎn)的話区拳,又會引起頁面刷新,H5頁面的體驗又下降了意乓。只能給WebView注入JS方法了樱调。

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

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

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

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

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

Ex:

[java]view plaincopy

@SuppressLint("NewApi")

@Override

publicWebResourceResponse?shouldInterceptRequest(WebView?view,String?url)?{

//?非超鏈接(如Ajax)請求無法直接添加請求頭,現(xiàn)拼接到url末尾,這里拼接一個imei作為示例

String?ajaxUrl?=?url;

//?如標(biāo)識:req=ajax

if(url.contains("req=ajax"))?{

ajaxUrl?+="&imei="+?imei;

}

returnsuper.shouldInterceptRequest(view,?ajaxUrl);

}

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

[java]view plaincopy

@Override

publicvoidonLoadResource(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長按時將會調(diào)用系統(tǒng)的復(fù)制控件:

[java]view plaincopy

mWebView.setOnLongClickListener(newOnLongClickListener()?{

@Override

publicbooleanonLongClick(View?v)?{

returntrue;

}

});

9.在WebView加入 flash支持:

[java]view plaincopy

String?temp?="

+"\">?

+"\"?height=\""+"90%"+"\"?scale=\""+"noscale"

+"\"?type=\""+"application/x-shockwave-flash"

+"\">?";

String?mimeType?="text/html";

String?encoding?="utf-8";

web.loadDataWithBaseURL("null",?temp,?mimeType,?encoding,"");

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

[java]view plaincopy

mWebView.getSettings().setSupportZoom(true);

mWebView.getSettings().setBuiltInZoomControls(true);

if(DeviceUtils.hasHoneycomb())

mWebView.getSettings().setDisplayZoomControls(false);

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

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

8月份更新:

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

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

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

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

[java]view plaincopy

WebChromeClient?webChromeClient?=newWebChromeClient()?{

@Override

publicvoidonReceivedTitle(WebView?view,?String?title)?{

super.onReceivedTitle(view,?title);

txtTitle.setText(title);

}

};

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

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

(1) 可以確定webview中子頁面只有二級頁面应狱,沒有更深的層次共郭,這里只需要判斷當(dāng)前頁面是否為初始的主頁面,可以goBack的話疾呻,只要將標(biāo)題設(shè)置回來即可.

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

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

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

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

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

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

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

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

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

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

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

10月份更新:

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

需要在Activity的onDestory()中調(diào)用

[java]view plaincopy

webView.destroy();

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

[java]view plaincopy

10-1015:01:11.402:?E/ViewRootImpl(7502):?sendUserActionEvent()?mView?==null

10-1015:01:26.818:?E/webview(7502):?java.lang.Throwable:?Error:?WebView.destroy()?calledwhilestill?attached!

10-1015:01:26.818:?E/webview(7502):????at?android.webkit.WebViewClassic.destroy(WebViewClassic.java:4142)

10-1015:01:26.818:?E/webview(7502):????at?android.webkit.WebView.destroy(WebView.java:707)

10-1015:01:26.818:?E/webview(7502):????at?com.didi.taxi.ui.webview.OperatingWebViewActivity.onDestroy(OperatingWebViewActivity.java:236)

10-1015:01:26.818:?E/webview(7502):????at?android.app.Activity.performDestroy(Activity.java:5543)

10-1015:01:26.818:?E/webview(7502):????at?android.app.Instrumentation.callActivityOnDestroy(Instrumentation.java:1134)

10-1015:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.performDestroyActivity(ActivityThread.java:3619)

10-1015:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:3654)

10-1015:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.access$1300(ActivityThread.java:159)

10-1015:01:26.818:?E/webview(7502):????at?android.app.ActivityThread$H.handleMessage(ActivityThread.java:1369)

10-1015:01:26.818:?E/webview(7502):????at?android.os.Handler.dispatchMessage(Handler.java:99)

10-1015:01:26.818:?E/webview(7502):????at?android.os.Looper.loop(Looper.java:137)

10-1015:01:26.818:?E/webview(7502):????at?android.app.ActivityThread.main(ActivityThread.java:5419)

10-1015:01:26.818:?E/webview(7502):????at?java.lang.reflect.Method.invokeNative(Native?Method)

10-1015:01:26.818:?E/webview(7502):????at?java.lang.reflect.Method.invoke(Method.java:525)

10-1015:01:26.818:?E/webview(7502):????at?com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1187)

10-1015:01:26.818:?E/webview(7502):????at?com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003)

10-1015:01:26.818:?E/webview(7502):????at?dalvik.system.NativeStart.main(Native?Method)

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

[java]view plaincopy

rootLayout.removeView(webView);

webView.destroy();

15. WebView長按自定義菜單,實現(xiàn)復(fù)制分享相關(guān)功能

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

(1) 在js中完成:

處理android.selection.longTouch

這里推薦一個開源項目進(jìn)行參考,:

https://github.com/btate/BTAndroidWebViewSelection

(2) 安卓層處理:

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


http://blog.csdn.net/t12x3456/article/details/13769731

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市枚冗,隨后出現(xiàn)的幾起案子缓溅,更是在濱河造成了極大的恐慌,老刑警劉巖赁温,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡瞒津,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進(jìn)店門更啄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人居灯,你說我怎么就攤上這事祭务。” “怎么了怪嫌?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵义锥,是天一觀的道長。 經(jīng)常有香客問我岩灭,道長拌倍,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任噪径,我火速辦了婚禮柱恤,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘找爱。我一直安慰自己梗顺,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布缴允。 她就那樣靜靜地躺著荚守,像睡著了一般珍德。 火紅的嫁衣襯著肌膚如雪练般。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天锈候,我揣著相機(jī)與錄音薄料,去河邊找鬼。 笑死泵琳,一個胖子當(dāng)著我的面吹牛摄职,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播获列,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼谷市,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了击孩?” 一聲冷哼從身側(cè)響起迫悠,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎巩梢,沒想到半個月后创泄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體艺玲,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年鞠抑,在試婚紗的時候發(fā)現(xiàn)自己被綠了饭聚。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡搁拙,死狀恐怖秒梳,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情箕速,我是刑警寧澤端幼,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站弧满,受9級特大地震影響婆跑,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜庭呜,卻給世界環(huán)境...
    茶點故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一滑进、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧募谎,春花似錦扶关、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至拐纱,卻和暖如春铜异,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背秸架。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工揍庄, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人东抹。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓蚂子,卻偏偏與公主長得像,于是被迫代替她去往敵國和親缭黔。 傳聞我的和親對象是個殘疾皇子食茎,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容