淺談Android開發(fā)Webview的Loading使用效果
前言
在開發(fā)webview的loading效果的時候會有一些問題,這邊記錄一些碰到的常見的問題,并且設計出一套Loading的方案來解決相關的問題。
1. loading的選擇
開發(fā)loading效果的原因在于webview加載頁面的時候,有時候會耗時,導致不顯示內容又沒有任何提示,效果不太好,所以需要在webview使用的地方加上loading的效果,其實更好的體驗是還要加上EmptyView,我這邊主要就以loadingView來舉例。
那開發(fā)這loading基本有兩種方式,一種是使用window,也就是Dialog這些彈窗的方式,在加載時彈出彈窗,在加載結束后關閉彈窗,有些人可能會封裝好一些loading彈窗,然后在這里復用。
這個方法的好處是如果你封裝好了,能直接復用,省去很多代碼。缺點也很明顯,彈窗彈出的時候是否處于一個不允許交互的情況,如果這個流程有問題,那便一直無法和頁面做交互
另一種方法是直接在webview的上層覆蓋一個LoadingView,webview是繼承FrameLayout,就是也可以直接addView。
這個方法的好處就是不會出現(xiàn)上面的問題,因為我webview所在的頁面關閉了,它的loading也會跟著一起消失,而且顯示的效果會好一些。缺點就是可能一些特殊的webview你會單獨做操作,導致會多寫一些代碼
沒有說哪種方法是實現(xiàn)會比較好,主要看使用的場景和具體的需求。
2. loading顯示時機的問題
我們做loading的思路就是加載開始的時候顯示,加載完成之后關閉,那選擇這個開始的時機和結束的時機就比較重要了。
大多數(shù)人都會直接使用WebViewClient的onPageStarted回調作為開始時機,把onPageFinished的回調,覺得直接這樣寫就行了,無所謂,反正webview會出手。
這個思路確實能在正常的情況下顯示正常,但是在弱網情況下呢?復雜的網絡環(huán)境下呢?有些人可能也會碰到一些這樣的情況,loading的show寫在onPageStarted中,加載時會先白屏一下,才開始顯示loading,但是這個白屏的時間很短,所以覺得無所謂。但有沒有想過這在正常網絡環(huán)境下的白屏一下放到復雜的有問題的網絡環(huán)境中會被放大成什么樣。
這個加載過程其實大體分為兩個階段,從loadurl到WebViewClient的onPageStarted和從WebViewClient的從onPageStarted到onPageFinished
所以我的做法是在loadurl的時候去start loading,而不是WebViewClient的onPageStarted回調的時候。
這個是開始的時機,那結束的時機會不會有問題,還真可能有,有時候你會發(fā)現(xiàn)一種現(xiàn)象,加載完之后,你的H5內容和loading會同時顯示一段時間,才關閉loading(幾年前有碰到過,寫這篇文章的時候測試沒有復現(xiàn)過,不知道是不是版本更新修復了這個問題)
那如果碰到這個問題該怎么解決呢?
碰到這個問題,說明onPageFinished的回調時機在頁面加載完之后,所以不可信。我們知道除了這個方法之外,BaseWebChromeClient也有個方法onProgressChanged表示加載的進度,當然這個進度你拿去判斷也會有問題,因為它并不會每次都會回調100給你,可能有時候給你96,就沒了。我以前的做法是雙重判斷,判斷是進度先返回>85還是onPageFinished先調用,只要有一個調用,我都會關閉loading
3. 體驗優(yōu)化
當然處理好顯示的關閉的時機還不行,想想如果在loadurl中show loading會怎樣,沒錯,就算網速快的情況,頁面讓loading一閃而過,那這樣所造成的體驗就很不好,所以我們需要做一個延遲顯示,我個人習慣是延遲0.5秒。當然延遲顯示也會有延遲顯示的問題,比如延遲到0.3秒的時候你關閉頁面怎么辦,再0.2秒之后我總不不能讓它顯示吧。
說了顯示,再說關閉。無論是onPageFinished方法還是onProgressChanged,你能保證它一定會有回調嗎?這些代碼都不是可控的,里面會不會出現(xiàn)既沒拋異常,也沒給回調的情況。也許有人說不會的,我都用了這么多年了,沒出現(xiàn)過這種問題,但是既然不是我們可控的代碼,加一層保險總沒錯吧。
其實這也簡單,定一個timeout的邏輯就行,我個人是定義10秒超時時間,如果10秒后沒有關閉loading,我就手動關閉并顯示emptyview的error頁面。這個超時時間還是比較實用,最上面說了loading的選擇,如果你的loading做成view,那即便沒有這個邏輯也影響不大,最多就會菊花一直轉,但如果你是window做的,沒有超時的處理,又沒有回調,那你的window會一直顯示卡住頁面。
4. loading最終設計效果
基于上面的情況,我寫個Demo,首先loading的選擇,我選擇基于view,所以要寫個自定義View
public class WebLoadingView extends RelativeLayout {
private Context mContext;
// 0:正常狀態(tài);1:loading狀態(tài);2:顯示loadingview狀態(tài)
private AtomicInteger state;
private Handler lazyHandler;
private Handler timeOutHandler;
public BaseWebLoadingView(Context context) {
super(context);
init(context);
}
public BaseWebLoadingView(Context context, AttributeSet attrs) {
super(context, attrs);
init(context);
}
public BaseWebLoadingView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(context);
}
private void init(Context context) {
this.mContext = context;
state = new AtomicInteger(0);
lazyHandler = new Handler(Looper.getMainLooper());
timeOutHandler = new Handler(Looper.getMainLooper());
initView();
}
private void initView() {
LayoutInflater.from(mContext).inflate(R.layout.demo_loading, this, true);
}
public void show() {
if (state.compareAndSet(0, 1)) {
lazyHandler.postDelayed(new Runnable() {
@Override
public void run() {
if (state.compareAndSet(1, 2)) {
setVisibility(View.VISIBLE);
}
}
}, 500);
timeOutHandler.postDelayed(new Runnable() {
@Override
public void run() {
close();
}
}, 10000);
}
}
public void close() {
state.set(0);
setVisibility(View.GONE);
try {
lazyHandler.removeCallbacksAndMessages(null);
timeOutHandler.removeCallbacksAndMessages(null);
} catch (Exception e) {
e.printStackTrace();
}
}
}
代碼應該都比較好理解,就不過多介紹了,然后在自定義webview的loadurl里面展示
@Override
public void loadUrl(String url) {
if (webLoadingView != null && !TextUtils.isEmpty(url) && url.startsWith("http")) {
webLoadingView.show();
}
super.loadUrl(url);
}
寫這里主要是有個地方要注意,就是調方法時也會執(zhí)行這個loadUrl,所以要判斷是加載網頁的時候才顯示loading。
總結
總結幾個重點吧,第一個是對第三方的東西(webview這個也類似第三方吧,坑真的很多),我們沒辦法把控它的流程,或者說沒辦法把控它的生命周期,所以要封裝一套流程邏輯去給調用端方便去使用。
第二個問題是版本的問題,也許會出現(xiàn)不同的版本所體現(xiàn)的效果不同,這個是需要留意的。
如果要完美解決這堆loading相關的問題,最好的方法就是看源碼,你知道它里面是怎么實現(xiàn)的,為什么會出現(xiàn)onPageStarted之前還會有一段間隔時間,那就去看loadUrl和onPageStarted回調之間的源碼,看它做了什么操作嘛。我個人是沒看源碼,所以這里只能說是淺談。
以上就是淺談Android開發(fā)Webview的Loading使用效果的詳細內容,更多關于Android Webview Loading的資料請關注腳本之家其它相關文章!
相關文章
Android性能優(yōu)化之利用Rxlifecycle解決RxJava內存泄漏詳解
RxJava作為一種響應式編程框架,是目前編程界網紅,可謂是家喻戶曉,其簡潔的編碼風格、易用易讀的鏈式方法調用、強大的異步支持等使得RxJava被廣泛使用。2017-01-01
Android開發(fā)技巧之在a標簽或TextView控件中單擊鏈接彈出Activity(自定義動作)
a標簽以及TextView自動識別的特殊文本(網址、電話號、Email等),這些都可以通過單擊來觸發(fā)不同的動作;但如果讀者想在單擊鏈接時執(zhí)行任意自定義的動作,那么將要介紹的一定是你想要的了2013-01-01
Android 個人理財工具二:使用SQLite實現(xiàn)啟動時初始化數(shù)據
本文主要介紹 Android 使用SQLite實現(xiàn)啟動時初始化數(shù)據,這里對SQLite 的數(shù)據庫進行詳解,附有示例代碼,有興趣的小伙伴可以參考下2016-08-08
android使用Ultra-PullToRefresh實現(xiàn)下拉刷新自定義代碼
本篇文章主要介紹了android使用Ultra-PullToRefresh實現(xiàn)下拉刷新新自定義,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-02-02
Android 安全加密:數(shù)字簽名和數(shù)字證書詳解
本文主要介紹Android 安全加密數(shù)字簽名和數(shù)字證書的資料,這里整理詳細的資料及數(shù)字簽名和數(shù)字證書應用詳解,有需要的小伙伴可以參考下2016-09-09

