一钦勘、簡介
DialogFragment 是在 Android 3.0 時引入的淀散,是一種特殊的 Fragment 氧卧,用于在 Acticity 上展示一個靜態(tài)的對話框桃笙,如:提示框,輸入框沙绝,等等
二搏明、DialogFragment 的好處
在 DialogFragment 之前鼠锈,我們一般使用 Dialog 或者 AlertDialog 來創(chuàng)建對話框,包括現(xiàn)在還是很多人以這種方式來創(chuàng)建熏瞄,但是 Google 并不推薦直接使用 Dialog 創(chuàng)建對話框脚祟,而是推薦盡量使用 DialogFragment 來創(chuàng)建,這是因為 DialogFragment 有 Dialog 所沒有的一些很好的特性
- DialogFragment 是 Fragment 的直接子類强饮,和 Fragment 有著基本一致的聲明周期,當(dāng)屏幕旋轉(zhuǎn)或者按后退鍵的時候为黎,能更好的管理其生命周期邮丰,例如,旋轉(zhuǎn)屏幕時铭乾,會導(dǎo)致對話框重新創(chuàng)建剪廉,使用 DialogFragment 的話,會由 FragmentManager 自己來重新創(chuàng)建對話框炕檩,而 Dialog 則沒有這樣的功能斗蒋,必須手動創(chuàng)建。
三笛质、使用 DialogFragment
3.1 使用 DialogFragment 至少需要實現(xiàn) onCreateView() 或者 onCreateDialog() 方法泉沾,如下所示
public class MyDialogFrament extends DialogFragment {
/**
* 使用定義的xml布局文件展示Dialog。
*
* @param inflater
* @param container
* @param savedInstanceState
* @return
*/
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
return super.onCreateView(inflater, container, savedInstanceState);
}
/**
* 使用 Dialog 或者 AlertDialog 創(chuàng)建的來展示
*
* @param savedInstanceState
* @return
*/
@NonNull
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
return super.onCreateDialog(savedInstanceState);
}
3.2 具體的使用參考:http://blog.csdn.net/lmj623565791/article/details/37815413
其實妇押,使用起來還是很簡單的跷究,但是僅僅這樣使用,并不能顯示我們程序員的專業(yè)性 0.0 敲霍, 我們要考慮復(fù)用性和封裝俊马,不想偷懶的程序員不是好程序員!肩杈!
四柴我、封裝成通用的 DialogFragment
這有一個哥們寫的很不錯,github 地址:https://github.com/developerHaoz/DialogFragmentDemos
本人也是分析了大神的思路扩然,覺得很不錯艘儒,就總結(jié)了下來,大概說下封裝思路:主要是以 onCreateDialog 的方式來創(chuàng)建對話框与学,通過回調(diào)的形式彤悔,從外部傳來 AlertDialog 來創(chuàng)建不同類型的提示框。
public class MyDialogFrament extends DialogFragment {
/**
* 使用定義的xml布局文件展示Dialog索守。
*
* @param inflater
* @param container
* @param savedInstanceState
* @return
*/
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
return super.onCreateView(inflater, container, savedInstanceState);
}
/**
* 使用 Dialog 或者 AlertDialog 創(chuàng)建的來展示
* 設(shè)計為回調(diào)接口形式晕窑,讓調(diào)用該接口的人決定顯示哪種類型的 Dialog
*
* @param savedInstanceState
* @return
*/
@NonNull
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
if (mOnCallDialog == null) {
return super.onCreateDialog(savedInstanceState);
}
return mOnCallDialog.getDialog(getActivity());
}
private OnDialogCancelListener mOnDialogCancelListener;
/**
* 監(jiān)聽彈窗是否被取消
*/
public interface OnDialogCancelListener {
void onCancel();
}
private OnCallDialog mOnCallDialog;
/**
* 需要顯示哪種類型的 DialogFragment
*/
public interface OnCallDialog {
Dialog getDialog(Context context);
}
/**
* 創(chuàng)建 MyDialogFrament 對象
*
* @param callDialog Dialog 取消的監(jiān)聽對象
* @param cancelable Dialog 是否可以撤銷
* @param cancelListener 需要展示哪種 Dialog 的回調(diào)對象
* @return MyDialogFrament 對象
*/
public static MyDialogFrament newInstance(OnCallDialog callDialog, boolean cancelable, OnDialogCancelListener cancelListener) {
MyDialogFrament instance = new MyDialogFrament();
//控制顯示的 Dialog 是否可取消
instance.setCancelable(cancelable);
instance.mOnDialogCancelListener = cancelListener;
instance.mOnCallDialog = callDialog;
return instance;
}
/**
* 創(chuàng)建 MyDialogFrament 對象
*
* @param callDialog Dialog 取消的監(jiān)聽對象
* @param cancelable Dialog 是否可以撤銷
* @return
*/
public static MyDialogFrament newInstance(OnCallDialog callDialog, boolean cancelable) {
return newInstance(callDialog, cancelable, null);
}
@Override
public void onStart() {
super.onStart();
Dialog dialog = getDialog();
if (dialog != null) {
// 在 5.0 以下的版本會出現(xiàn)白色背景邊框,若在 5.0 以上設(shè)置則會造成文字部分的背景也變成透明
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.KITKAT) {
// 目前只有這兩個 dialog 會出現(xiàn)邊框
if (dialog instanceof ProgressDialog || dialog instanceof DatePickerDialog) {
getDialog().getWindow().setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
}
}
Window window = getDialog().getWindow();
WindowManager.LayoutParams windowParams = window.getAttributes();
windowParams.dimAmount = 0.0f;
window.setAttributes(windowParams);
}
}
@Override
public void onCancel(DialogInterface dialog) {
super.onCancel(dialog);
if (mOnDialogCancelListener != null) {
mOnDialogCancelListener.onCancel();
}
}
}
- 另一個設(shè)計點是監(jiān)聽對話框中的所做的操作卵佛,可能是選了某一個選項杨赤,又或者是點擊了確定或取消等敞斋,也是以接口回調(diào)的形式來做的,可以更加自由的選擇在什么地方來對這些操作處理疾牲,以其中的一個舉例:
接口為:
public interface IDialogResultListener<T> {
void onDataResult(T result);
}
以一個列表選項的對話框為例:
public static DialogFragment showListDialog(FragmentManager fragmentManager, final String title, final String[] items
, final IDialogResultListener<Integer> resultListener, boolean cancelable ){
MyDialogFrament dialogFragment = MyDialogFrament.newInstance(new MyDialogFrament.OnCallDialog() {
@Override
public Dialog getDialog(Context context) {
AlertDialog.Builder builder = new AlertDialog.Builder(context, LIST_THEME);
builder.setTitle(title);
builder.setItems(items, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
if(resultListener != null){
resultListener.onDataResult(which);
}
}
});
return builder.create();
}
}, cancelable, null);
dialogFragment.show(fragmentManager, LIST_TAG);
return null;
}
其中植捎,列表的點擊事件激活此回調(diào)接口,使用者阳柔,可以在需要的時候?qū)崿F(xiàn)該接口焰枢,并且經(jīng)過這一層的封裝,代碼看起來也更簡潔舌剂,可讀性更強济锄!