我在android studio中使用MaterialDatePicker
,用户可以选择出生日期,选择日期后,返回的值是带有所选日期时间戳的long
(自1970年1月1日起的毫秒(。问题是获得的结果有点错误,例如:如果用户选择日期为1998年5月10日(1998年10月1日(,则返回的日期是在此之前的一天,即日历返回1998年4月10日。
我的代码:
Calendar calendar = Calendar.getInstance();
MaterialDatePicker.Builder<Long> builder =
MaterialDatePicker.Builder.datePicker();
MaterialDatePicker<Long> picker = builder.build();
picker.show(getChildFragmentManager(), picker.toString());
picker.addOnPositiveButtonClickListener(new MaterialPickerOnPositiveButtonClickListener<Long>() {
@Override
public void onPositiveButtonClick(Long selection) {
calendar.setTimeInMillis(selection);
SimpleDateFormat simpleDateFormat = new
SimpleDateFormat("dd/MM//yyyy",Locale.getDefault());
txtBirthDate.setText(simpleDateFormat.format(calendar.getTime()));
}
});
现在具有timeInMillis的calendar
实例具有选定的日期,仅在前一天。我该怎么解决这个问题?
【2022年工作解决方案】
事实证明,MaterialDatePicker
使用UTC时间格式,这意味着它不包含任何时区信息,正如官方在此声明的那样。
这意味着我们必须在UTC日期时间和区域日期时间间进行一些转换工作,其想法是使用LocalDateTime api:
1.为了在选择器上显示正确的日期,我们需要将区域日期时间转换为UTC日期时间:
Zone date-time (Long) -> zone LocalDateTime (object) -> UTC date-time (object)-> UTC date-time (Long).
2.为了从选择器中获取区域日期时间,我们需要将UTC日期时间转换为区域日期时间:
UTC date-time (Long) -> UTC LocalDateTime (object) -> zone date-time (object) -> zone date-time (Long)
创建辅助程序扩展函数:
fun Long.toLocalDateTime() = LocalDateTime.ofInstant(Instant.ofEpochMilli(this), ZoneId.systemDefault()) fun Long.toUTCLocalDateTime() = LocalDateTime.ofInstant(Instant.ofEpochMilli(this), ZoneId.ofOffset("UTC", ZoneOffset.UTC))
在
Fragment
:中显示MaterialDatePicker
private fun showDatePicker() { val datePicker = MaterialDatePicker.Builder.datePicker().run { setTitleText("Select Date") setSelection(viewModel.dateLong.value.toLocalDateTime().atZone(ZoneId.ofOffset("UTC", ZoneOffset.UTC)).toInstant().toEpochMilli()) //setSelection(MaterialDatePicker.todayInUtcMilliseconds()) //show today's date build() } datePicker.addOnPositiveButtonClickListener { val dateLong = datePicker.selection!!.toUTCLocalDateTime().atZone(ZoneId.systemDefault()).toInstant().toEpochMilli() viewModel.updateDateLong(dateLong) } datePicker.show(childFragmentManager, "") }
演示:https://youtu.be/9YWr3fgQ214
这是因为DatePicker从0开始对月份进行索引覆盖了此方法
override fun onDateSet(view: DatePicker?, year: Int, month: Int, dayOfMonth: Int) {
datePickerListener.onDateSet(view,year,month+1,dayOfMonth)
}
唯一对我有效的解决方案是先将所需的时间戳转换为LocalDateTime
,然后截断为DAYS
,最后将其转换为UTC
,如下所示:
val selection = ZonedDateTime.now()
.toLocalDateTime()
.truncatedTo(ChronoUnit.DAYS)
.toInstant(ZoneOffset.UTC)
.toEpochMilli()
val picker = MaterialDatePicker.Builder.datePicker()
.setSelection(selection)
//...
.build()
然后,您可以按如下方式获取所选日期:
picker.addOnPositiveButtonClickListener { selection ->
val selectedDate = Instant.ofEpochMilli(selection)
.atZone(ZoneId.systemDefault())
//...
}