Activity能夠很容易的獲得物理返回鍵的監聽事件,而Fragment卻不能。假設FragmentActivity有三個Fragment,通常安卓用戶指望點擊返回鍵會一層層返回到FragmentActivity。固然,咱們能夠將每一個Fragment對應的Transaction放到BackStack中,可是若是每一個Fragment有對返回事件的特殊消費,那麼在FragmentActivity的onBackPressed()中的代碼就會比較混亂,例如:java
[java] view plaincopygit
@Override github
public void onBackPressed() { app
if(selectedFragment.equals(fragmentA) && fragmentA.hasExpandedRow()) { ide
fragmentA.collapseRow(); this
} else if(selectedFragment.equals(fragmentA) && fragmentA.isShowingLoginView()) { spa
fragmentA.hideLoginView(); .net
} else if(selectedFragment.equals(fragmentA)) { orm
popBackStack(); blog
} else if(selectedFragment.equals(fragmentB) && fragmentB.hasCondition1()) {
fragmentB.reverseCondition1();
} else if(selectedFragment.equals(fragmentB) && fragmentB.hasCondition2()) {
fragmentB.reverseCondition2();
} else if(selectedFragment.equals(fragmentB)) {
popBackStack();
} else {
// handle by activity
super.onBackPressed();
}
}
這對於有代碼潔癖的程序猿顯然是不能容忍的,後來發現了一種優雅的解決方案。
首先建立一個抽象類BackHandledFragment,該類有一個抽象方法onBackPressed(),全部BackHandledFragment的子類在onBackPressed方法中處理各自對Back事件的消費邏輯。onBackPressed返回布爾值,宿主FragmentActivity將會根據該方法的返回值判斷子Fragment是否有消費Back事件。此外,宿主FragmentActivity還會保持一份當前Fragment的引用,當用戶按下Back鍵時,宿主Activity會判斷當前Fragment是否須要消費該事件,若是沒有Fragment消費纔會本身消費。
[java] view plaincopy
public abstract class BackHandledFragment extends Fragment {
protected BackHandledInterface mBackHandledInterface;
/**
* 全部繼承BackHandledFragment的子類都將在這個方法中實現物理Back鍵按下後的邏輯
* FragmentActivity捕捉到物理返回鍵點擊事件後會首先詢問Fragment是否消費該事件
* 若是沒有Fragment消息時FragmentActivity本身才會消費該事件
*/
protected abstract boolean onBackPressed();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if(!(getActivity() instanceof BackHandledInterface)){
throw new ClassCastException("Hosting Activity must implement BackHandledInterface");
}else{
this.mBackHandledInterface = (BackHandledInterface)getActivity();
}
}
@Override
public void onStart() {
super.onStart();
//告訴FragmentActivity,當前Fragment在棧頂
mBackHandledInterface.setSelectedFragment(this);
}
}
宿主FragmentActivity須要繼承BackHandledIntegerface,子Fragment會經過該接口告訴宿主FragmentActivity本身是當前屏幕可見的Fragment。
[java] view plaincopy
public interface BackHandledInterface {
public abstract void setSelectedFragment(BackHandledFragment selectedFragment);
}
因此在Fragment的onCreate中會判斷宿主FragmentActivity是否已繼承了該接口。在Fragment的onStart()方法中就會調用該接口告訴宿主FragmentActivity本身是當前屏幕可見的Fragment。
宿主FragmentActivity就能夠在onBackPressed()方法中對Back事件進行判斷處理了。
[java] view plaincopy
public class MainActivity extends FragmentActivity implements BackHandledInterface{
private BackHandledFragment mBackHandedFragment;
private boolean hadIntercept;
@Override
public void setSelectedFragment(BackHandledFragment selectedFragment) {
this.mBackHandedFragment = selectedFragment;
}
@Override
public void onBackPressed() {
if(mBackHandedFragment == null || !mBackHandedFragment.onBackPressed()){
if(getSupportFragmentManager().getBackStackEntryCount() == 0){
super.onBackPressed();
}else{
getSupportFragmentManager().popBackStack();
}
}
}
}
示例程序Github連接。
參考資料: