詳解Android應(yīng)用中DialogFragment的基本用法
DialogFragment的基本用法
1. 創(chuàng)建DialogFragment
public class DialogA extends DialogFragment implements DialogInterface.OnClickListener {
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage(R.string.dialoga_title)
.setPositiveButton(R.string.ok, this)
.setNegativeButton(R.string.cancel, this);
return builder.create();
}
@Override
public void onClick(DialogInterface dialog, int id) {
switch(id) {
case AlertDialog.BUTTON_NEGATIVE:
Toast.makeText(getActivity(), "Negative", Toast.LENGTH_SHORT).show();
break;
case AlertDialog.BUTTON_POSITIVE:
Toast.makeText(getActivity(), "Positive", Toast.LENGTH_SHORT).show();
break;
default:
break;
}
}
}
說明:自定義一個DialogFragment,并重寫它的onCreateDialog()方法。
2. 調(diào)用該DialogFragment
下面是在FragmentActivity中調(diào)用該DialogFragment對話框。
public class DialogTest extends FragmentActivity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
showDialog();
}
private void showDialog() {
FragmentManager fm = getSupportFragmentManager();
DialogA dialoga = new DialogA();
dialoga.show(fm, "fragmenta");
}
}
自定義DialogFragment布局
下面介紹自定義DialogFragment的布局的方法
點擊查看:自定義DialogFragment布局的完整代碼
1. 設(shè)置布局文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="@string/dialoga_intro" /> <ImageView android:id="@+id/image" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/ic_action_video" /> </LinearLayout>
2. 使用布局
public class DialogA extends DialogFragment implements DialogInterface.OnClickListener {
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
LayoutInflater inflater = getActivity().getLayoutInflater();
builder.setView(inflater.inflate(R.layout.dialoga, null))
.setMessage(R.string.dialoga_title)
.setPositiveButton(R.string.ok, this)
.setNegativeButton(R.string.cancel, this);
return builder.create();
}
@Override
public void onClick(DialogInterface dialog, int id) {
switch(id) {
case AlertDialog.BUTTON_NEGATIVE:
Toast.makeText(getActivity(), "Negative", Toast.LENGTH_SHORT).show();
break;
case AlertDialog.BUTTON_POSITIVE:
Toast.makeText(getActivity(), "Positive", Toast.LENGTH_SHORT).show();
break;
default:
break;
}
}
}
DialogFragment和Activity的交互
下面介紹自定義DialogFragment和Activity交互的方法
點擊查看:DialogFragment和Activity交互的完整代碼
1. 定義通信接口
在DialogFragment中定義它們之間的通信接口。
public interface NoticeDialogListener {
public void onDialogPositiveClick(DialogFragment dialog);
public void onDialogNegativeClick(DialogFragment dialog);
}
// Use this instance of the interface to deliver action events
NoticeDialogListener mListener;
// Override the Fragment.onAttach() method to instantiate the NoticeDialogListener
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
// Verify that the host activity implements the callback interface
try {
// Instantiate the NoticeDialogListener so we can send events to the host
mListener = (NoticeDialogListener) activity;
} catch (ClassCastException e) {
// The activity doesn't implement the interface, throw exception
throw new ClassCastException(activity.toString()
+ " must implement NoticeDialogListener");
}
}
2. 在DialogFragment中調(diào)用該接口
@Override
public void onClick(DialogInterface dialog, int id) {
switch(id) {
case AlertDialog.BUTTON_POSITIVE:
//Toast.makeText(getActivity(), "Negative", Toast.LENGTH_SHORT).show();
mListener.onDialogPositiveClick(DialogA.this);
break;
case AlertDialog.BUTTON_NEGATIVE:
//Toast.makeText(getActivity(), "Positive", Toast.LENGTH_SHORT).show();
mListener.onDialogNegativeClick(DialogA.this);
break;
default:
break;
}
}
3. 在Activity中實現(xiàn)該接口
public class DialogTest extends FragmentActivity
implements DialogA.NoticeDialogListener {
...
@Override
public void onDialogPositiveClick(DialogFragment dialog) {
Toast.makeText(this, "Positive Callback", Toast.LENGTH_SHORT).show();
}
@Override
public void onDialogNegativeClick(DialogFragment dialog) {
Toast.makeText(this, "Negative Callback", Toast.LENGTH_SHORT).show();
}
}
Dialog與DialogFragment的對比
從代碼的編寫角度看,Dialog使用起來要更為簡單,但是Google則是推薦盡量使用DialogFragment(對于Android 3.0以下的版本,可以結(jié)合使用support包中提供的DialogFragment以及FragmentActivity)。今天試著用這兩種方式來創(chuàng)建對話框,發(fā)現(xiàn)DialogFragment果然有一個非常好的特性(在手機(jī)配置變化,導(dǎo)致Activity需要重新創(chuàng)建時,例如旋屏,基于DialogFragment的對話框?qū)蒄ragmentManager自動重建,然而基于Dialog實現(xiàn)的對話框則沒有這樣的能力)。
下面是兩段實例代碼:
他們使用的界面都一樣:(dialog.xml)
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <ImageView android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/ic_launcher" /> </LinearLayout>
1.基于Dialog實現(xiàn)的對話框
public class MainActivity extends Activity {
private Button clk;
private Dialog dialog;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
clk = (Button) findViewById(R.id.clk);
dialog = new Dialog(this);
dialog.setContentView(R.layout.dialog);
clk.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
dialog.show();
}
});
}
}
當(dāng)我們點擊按鈕時,會彈出對話框(內(nèi)容為android logo),當(dāng)我們旋轉(zhuǎn)屏幕后,Activity重新創(chuàng)建,整個Activity的界面沒有問題,而對話框消失了。
除此之外,其實還有一個問題,就是在logcat中會看到異常信息:Android..leaked .. window,這是因為在Activity結(jié)束之前,Android要求所有的Dialog必須要關(guān)閉。我們旋屏后,Activity會被重建,而上面的代碼邏輯并沒有考慮到對話框的狀態(tài)以及是否已關(guān)閉。
于是將上述代碼修改為:
public class MainActivity extends Activity {
private Button clk;
private Dialog dialog;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
clk = (Button) findViewById(R.id.clk);
dialog = new Dialog(this);
dialog.setContentView(R.layout.dialog);
clk.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
dialog.show();
}
});
//用戶恢復(fù)對話框的狀態(tài)
if(savedInstanceState != null && savedInstanceState.getBoolean("dialog_show"))
clk.performClick();
}
/**
* 用于保存對話框的狀態(tài)以便恢復(fù)
*/
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
if(dialog != null && dialog.isShowing())
outState.putBoolean("dialog_show", true);
else
outState.putBoolean("dialog_show", false);
}
/**
* 在Activity銷毀之前,確保對話框以關(guān)閉
*/
@Override
protected void onDestroy() {
super.onDestroy();
if(dialog != null && dialog.isShowing())
dialog.dismiss();
}
}
2. 基于DialogFragment的對話框
與上面的對話框使用同樣的界面布局,此處僅僅展現(xiàn)一個簡單對話框,因此只重寫了onCreateView方法
public class MyDialogFragment extends DialogFragment {
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View v = inflater.inflate(R.layout.dialog, container, false);
return v;
}
}
public class MainActivity extends FragmentActivity {
private Button clk;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
clk = (Button) findViewById(R.id.clk);
clk.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
MyDialogFragment mdf = new MyDialogFragment();
FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
mdf.show(ft, "df");
}
});
}
}
這兩段代碼可以實現(xiàn)第一種方式的同樣功能,此處我們并沒有去關(guān)心對話框的重建,以及Activity銷毀前對話框是否已關(guān)閉,這一切都是由FragmentManager來管理。
其實DialogFragment還擁有fragment的優(yōu)點,即可以在一個Activity內(nèi)部實現(xiàn)回退(因為FragmentManager會管理一個回退棧)
相關(guān)文章
Android開發(fā)EditText禁止輸入監(jiān)聽及InputFilter字符過濾
這篇文章主要為大家介紹了Android開發(fā)EditText禁止輸入監(jiān)聽及InputFilter字符過濾示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
Android中RecyclerView實現(xiàn)簡單購物車功能
這篇文章主要為大家詳細(xì)介紹了Android中RecyclerView實現(xiàn)簡單購物車功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02
Android 單例模式實現(xiàn)可復(fù)用數(shù)據(jù)存儲的詳細(xì)過程
本文介紹了如何使用單例模式實現(xiàn)一個可復(fù)用的數(shù)據(jù)存儲類,該類可以存儲不同類型的數(shù)據(jù),并提供統(tǒng)一的接口來訪問這些數(shù)據(jù),通過雙重檢查鎖定機(jī)制,該類在多線程環(huán)境下是線程安全的,感興趣的朋友跟隨小編一起看看吧2025-02-02
Android 自定義view實現(xiàn)水波紋動畫效果
這篇文章主要介紹了 Android 自定義view實現(xiàn)水波紋動畫效果的實例代碼,非常不錯,具有參考借鑒價值,需要的朋友參考下吧2017-01-01
Android Studio debug.keystore位置介紹
這篇文章主要介紹了Android Studio debug.keystore位置,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03

