我可以离开完成按钮,并可以通过单击对话框外部关闭对话框,但是如果按下Done或在对话框外部,我的onTimeSetListener将被调用.
所以我发现this stackoverflow question描述了使用DatePickerDialog时出现的错误.要使用DatePickerDialog解决问题,在初始化时我必须将onDateSetListener设置为null,并实现我自己的按钮来处理BUTTON_POSITIVE(Set)和BUTTON_NEGATIVE(Cancel)onClick方法.哪个没问题,因为当调用按钮Set时,我可以像这样访问DatePicker值
int yearPicked = dateDlg.getDatePicker().getYear(); int monthPicked = dateDlg.getDatePicker().getMonth(); int dayPicked = dateDlg.getDatePicker().getDayOfMonth();
因此不需要使用onDateSetListener,但是如果我愿意,则在按Set或Cancel时再次调用它.
我尝试以同样的方式使用TimePickerDialog,但问题是,在BUTTON_POSITIVE onClick方法中,我无法像以前那样访问小时和分钟值,因为TimePickerDialog不提供TimePicker,因为DatePickerDialog提供了DatePicker.再次,如果我将使用onTimeSetListener,它将通过按任何东西来调用.
Calendar cal = Calendar.getInstance(); int hour = cal.get(Calendar.HOUR_OF_DAY); int min = cal.get(Calendar.MINUTE); final TimePickerDialog timeDlg = new TimePickerDialog(PreferencesActivity.this,null,hour,min,true); // Make the Set button timeDlg.setButton(DialogInterface.BUTTON_POSITIVE,"Set",new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog,int which) { if (which == DialogInterface.BUTTON_POSITIVE) { // CANNOT ACCES THE VALUES Toast.makeText(PreferencesActivity.this,Toast.LENGTH_SHORT).show(); } } }); // Set the Cancel button timeDlg.setButton(DialogInterface.BUTTON_NEGATIVE,"Cancel",int which) { if (which == DialogInterface.BUTTON_NEGATIVE) { Toast.makeText(PreferencesActivity.this,Toast.LENGTH_SHORT).show(); } } }); timeDlg.show();
解决方法
private boolean mIgnoreTimeSet = false; final TimePickerDialog timeDlg = new TimePickerDialog(PreferencesActivity.this,PreferencesActivity.this,int which) { mIgnoreTimeSet = false; // only manually invoke OnTimeSetListener (through the dialog) on pre-ICS devices if (Build.VERSION.SDK_INT < Build.VERSION_CODES.ICE_CREAM_SANDWICH) timeDlg.onClick(dialog,which); Toast.makeText(getApplicationContext(),Toast.LENGTH_SHORT).show(); } }); // Set the Cancel button timeDlg.setButton(DialogInterface.BUTTON_NEGATIVE,int which) { Toast.makeText(getApplicationContext(),Toast.LENGTH_SHORT).show(); mIgnoreTimeSet = true; dialog.cancel(); } }); timeDlg.show(); @Override public void onTimeSet(TimePicker view,int hourOfDay,int minute) { if (mIgnoreTimeSet) return; // get the values here - this will only run if 'Set' was clicked }
为了示例,我假设您的PreferencesActivity实现了OnTimeSetListener.由于onTimeSet(…)在onClick(…)之后被调用,因此您可以简单地切换一个布尔值,该布尔值将决定您是否应该对回调中的值执行某些操作或者忽略它们.
我不得不同意它并不理想,但至少是功能性的.假期结束后,我会尝试再看看“更好”的解决方案.
一种替代方法是使用反射来获取内部使用的TimePicker小部件的句柄,但这在未来的Android版本中更有可能中断.
//编辑:潜在的替代方案? (未测试)
第二种选择,可能有点’更清洁’:扩展TimePickerDialog并覆盖(空)公共方法:
public void onTimeChanged(TimePicker view,int minute)
在那里,跟踪用户设置的白天和小时的小时,并实现两个返回它们的getter.然后,如果点击了肯定按钮,则只从对话框中获取这些值,但只是为了“取消”.基本上你不再需要一个OnTimeSetListener,因此它是否被调用无关紧要.
// Edit2:支持pre-ICS设备
在收到关于未在ICS前设备上工作的建议解决方案的几点评论之后,我对上面的代码做了一些小改动 – 即使很难,也不在原始问题的范围内.
pre-ICS和post-ICS设备之间的主要区别在于,在运行对话框按钮的onClick()方法后,不会自动调用OnTimeSetListener回调.一个简单的解决方法是在对话框上调用onClick()方法,然后调用侦听器.在上面的解决方案中,我添加了一个版本代码检查,因为否则你的监听器将最终在post-ICS设备上被调用两次,这可能会导致不良的副作用.
Android 2.1-update1和Android 4.2.2上经过测试和确认的行为.