<h1>基本用法</h1>
DatePickerDialog是一个封装了DatePicker(日期选择器)的Dialog对话框组件。DatePicker需要占据较大的布局空间,相比之下,DatePickerDialog则以对话框的形式,在我们需要设置日期时再显示改控件。DatePickerDialog的使用也很方便:
一般我们使用构造方法public DatePickerDialog(@NonNull Context context, @Nullable OnDateSetListener listener, int year, int month, int dayOfMonth)可以创建出一个DatePickerDialog对象。其中OnDateSetListener listener是用户日期设置完成后的回调监听器,该接口的void onDateSet(DatePicker view, int year, int month, int dayOfMonth);方法内可以获取到用户设置的日期信息,在该方法内我们可以对用户输入的日期信息做进一步的操作。
参数year、month、dayOfMonth用来初始化DatePickerDialog默认显示的日期。不过要注意参数month与Calendar类的month类似,都是从0开始表示,即1-12月依次对应0-11。通常我们可以用Calendar.getInstance();获取到一个表示当前系统日期的Calendar对象,用该对象来初始化DatePickerDialog,如下:
<pre>
Calendar c = null;
public void showDatePickerDialog(DatePickerDialog.OnDateSetListener listener) {
c = Calendar.getInstance();
int year = c.get(Calendar.YEAR);
int month = c.get(Calendar.MONTH);
int day = c.get(Calendar.DAY_OF_MONTH);
new CommonDatePickerDialog(DoctorCheckActivity.this, listener, year, month, day).show();
}
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
DatePickerDialog.OnDateSetListener listener = new DatePickerDialog.OnDateSetListener() {
@Override
public void onDateSet(DatePicker view, int year, int month, int dayOfMonth) {
Log.i(year + "年" + ++month + "月" + dayOfMonth + "日");
}
};
showDatePickerDialog (listener);
}
});
</pre>
<h1>出现的问题</h1>
这段代码执行后,观察发现,如果点击“完成”,Log会输出两次,而如果通过点击返回键或Dialog以外的部分来关闭改Dialog的话,则仍会输出一次Log信息。在一般情况下,我们希望只有当用户确定其输入的日期信息后,才调用onDateSet()方法,但在这里无论是否确定输入,都会回调onDateSet()方法,这显然不是我们想要的。为什么会出现这样的情况呢?
首先,我们知道当Dialog被关闭时,如按下“取消”、“确定”、返回键或Dialog以外的部分,都会调用dismiss()方法,进一步调用onStop()方法。而从Android 4.1到Android4.4(API 16-19),DatePickerDialog的源码都重写了onStop()方法:
<pre>
@Override
protected void onStop() {
tryNotifyDateSet();
super.onStop();
}
private void tryNotifyDateSet() {
if (mCallBack != null) {
mDatePicker.clearFocus();
mCallBack.onDateSet(mDatePicker, mDatePicker.getYear(),
mDatePicker.getMonth(), mDatePicker.getDayOfMonth());
}
}
</pre>
由源码可知,这些版本的DatePickerDialog在关闭时会再次调用DatePickerDialog. OnDateSetListener监听器的onDateSet()方法。
<h1>解决方法</h1>
由以上分析可知,问题出现的原因在于在某些版本的系统中,DatePickerDialog的生命周期onStop()方法会调用onDateSet()方法。因此,我们可以自定义CommonDatePickerDialog继承自OnDateSetListener,重写onStop()方法:
<pre>
class CommonDatePickerDialog extends DatePickerDialog {
public CommonDatePickerDialog(@NonNull Context context, @Nullable OnDateSetListener listener, int year, int month, int dayOfMonth) {
super(context, listener, year, month, dayOfMonth);
}
@Override
protected void onStop() {
//super.onStop();
}
}
</pre>
使CommonDatePickerDialog的onStop()方法成为空实现,不要执行父类的onStop()方法。
当然,DatePickerDialog的父类是Dialog,我们也可以自定义Dialog的按钮及其监听回调,即不使用OnDateSetListener来监听“确认”按钮:
<pre>
public void pickDate(View view) {
Calendar cal = Calendar.getInstance();
final DatePickerDialog mDialog = new DatePickerDialog(this, null, cal.get(Calendar.YEAR), cal.get(Calendar.MONTH), cal.get(Calendar.DAY_OF_MONTH));
//手动设置按钮
mDialog.setButton(DialogInterface.BUTTON_POSITIVE, "完成", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//通过mDialog.getDatePicker()获得dialog上的DatePicker组件,然后可以获取日期信息
DatePicker datePicker = mDialog.getDatePicker();
int year = datePicker.getYear();
int month = datePicker.getMonth();
int day = datePicker.getDayOfMonth();
System.out.println(year + "," + month + "," + day);
}
});
//取消按钮,如果不需要直接不设置即可
mDialog.setButton(DialogInterface.BUTTON_NEGATIVE, "取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
System.out.println("BUTTON_NEGATIVE~~");
}
});
Dialog.show();
}
</pre>
<h1>参考</h1>
http://bbs.9ria.com/forum.php?mod=viewthread&tid=205312
http://www.devba.com/index.php/archives/6187.html