最近在開發中遇到了一個問題。咱們的app須要統計用戶的頁面路徑,也就是用戶使用各個頁面的狀況。這就須要在不一樣的頁面跳入和跳出時記錄下來。可是咱們的app主要是由Fragment構成的。而在不一樣的使用狀況下,判斷Fragment是否可見的方法是不同的。下面對這些不一樣的使用狀況分開分析。android
這種狀況是最簡單的,也就是在Activity使用XML引入,或者使用FragmentManager 的addFragment或者replaceFrament 動態載入。在這種狀況下,只要監聽Fragment的onResume和onPause方法就可以判斷其顯隱。在onResume和onPause中間是對用戶可見的.app
@Override public void onResume() { super.onResume(); //TODO now visible to user } @Override public void onPause() { super.onPause(); //TODO now invisible to user }
FragmentManager除了addFragment和replaceFragment以外還有showFragment和hideFragment來作Fragment的顯隱,這樣能夠提生頁面切換的速度,是一種用空間換時間的方式。可是這樣使用的Fragment在被hide的時候是不會調用onPause方法的。由於它只是在屏幕中不可見了,可是沒有Pause。這時候咱們須要監聽onHiddenChanged方法ide
@Override public void onHiddenChanged(boolean hidden) { super.onHiddenChanged(hidden); if(hidden){ //TODO now visible to user } else { //TODO now invisible to user } }
可是咱們須要注意一點,那就是,若是用戶直接按home鍵退出了,咱們的程序也沒有對按home鍵事件進行監聽,在按home鍵時調用hideFragment,那麼onHiddenChanged事實上沒有被調用,反而是onPause被調用了。因此咱們須要針對這樣的狀況作特殊的處理。好比在home鍵退出時顯示調用一下hideFragment或者在onPause中也發信號說本Fragment不可見了,這都是能夠的。spa
如今安卓市場上基本上的app都會作tab頁,而tab頁基本都是經過android自帶的ViewPager實現的。ViewPager有這樣一個特色,當滑到某一個Tab時,它會同時加載這個tab的左右兩個tab頁,好比我從1頁面切換到了2頁面,那麼3頁面的onResume也被調用了,可是3頁面其實對用戶是不可見的。這時候咱們就須要監聽setUserVisibleHint來判斷到底對用戶是否可見。code
@Override public void setUserVisibleHint(boolean isVisibleToUser) { super.setUserVisibleHint(isVisibleToUser); if(isVisibleToUser){ //TODO now it's visible to user } else { //TODO now it's invisible to user } }
可是與第二條相似,咱們依然不能徹底依賴setUserVisibleHint這個方法來判斷是否ViewPager中的頁面是否對用戶可見。由於該方法只有在Fragment切換的時候被調用。好比我從一個含有ViewPager的Actvity跳轉到另外一個activity,ViewPager中的Fragment的 setUserVisibleHint方法是不會被調用的,只能經過onPause來判斷。進入該Activity原理相似。那麼怎麼辦呢,咱們須要結合setUserVisibleHint和onResume和onPause。事件
@Override public void setUserVisibleHint(boolean isVisibleToUser) { super.setUserVisibleHint(isVisibleToUser); if ((isVisibleToUser && isResumed())) { onResume(); } else if (!isVisibleToUser) { Timber.i("On Pause on %s Fragment Invisble", getClass().getSimpleName()); onPause() } } @Override public void onResume() { super.onResume(); if (getUserVisibleHint()) { Timber.i("On Resume on %s Fragment Visible", getClass().getSimpleName()); //TODO give the signal that the fragment is visible } } @Override public void onPause() { super.onPause(); //TODO give the signal that the fragment is invisible }
須要注意以上方法並不完美,Fragment不可見的信號會被屢次發送。目前還沒找到一個只發送一次的方法。若是讀者有更好的解決方案,歡迎跟我交流。開發