|
|
// Apache许可证协议
|
|
|
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
import java.text.DateFormatSymbols;
|
|
|
import java.util.Calendar;
|
|
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
|
|
|
|
import android.content.Context;
|
|
|
import android.text.format.DateFormat;
|
|
|
import android.view.View;
|
|
|
import android.widget.FrameLayout;
|
|
|
import android.widget.NumberPicker;
|
|
|
|
|
|
public class DateTimePicker extends FrameLayout {
|
|
|
|
|
|
private static final boolean DEFAULT_ENABLE_STATE = true;
|
|
|
|
|
|
private static final int HOURS_IN_HALF_DAY = 12;
|
|
|
private static final int HOURS_IN_ALL_DAY = 24;
|
|
|
private static final int DAYS_IN_ALL_WEEK = 7;
|
|
|
private static final int DATE_SPINNER_MIN_VAL = 0;
|
|
|
private static final int DATE_SPINNER_MAX_VAL = DAYS_IN_ALL_WEEK - 1;
|
|
|
private static final int HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW = 0;
|
|
|
private static final int HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW = 23;
|
|
|
private static final int HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW = 1;
|
|
|
private static final int HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW = 12;
|
|
|
private static final int MINUT_SPINNER_MIN_VAL = 0;
|
|
|
private static final int MINUT_SPINNER_MAX_VAL = 59;
|
|
|
private static final int AMPM_SPINNER_MIN_VAL = 0;
|
|
|
private static final int AMPM_SPINNER_MAX_VAL = 1;
|
|
|
|
|
|
private final NumberPicker mDateSpinner;
|
|
|
private final NumberPicker mHourSpinner;
|
|
|
private final NumberPicker mMinuteSpinner;
|
|
|
private final NumberPicker mAmPmSpinner;
|
|
|
private Calendar mDate;
|
|
|
|
|
|
private String[] mDateDisplayValues = new String[DAYS_IN_ALL_WEEK];
|
|
|
|
|
|
private boolean mIsAm;
|
|
|
|
|
|
private boolean mIs24HourView;
|
|
|
|
|
|
private boolean mIsEnabled = DEFAULT_ENABLE_STATE;
|
|
|
|
|
|
private boolean mInitialising;
|
|
|
|
|
|
private OnDateTimeChangedListener mOnDateTimeChangedListener;
|
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnDateChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
@Override
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
mDate.add(Calendar.DAY_OF_YEAR, newVal - oldVal);// 使用Calendar类的add方法将mDate对象的日期字段(即Calendar.DAY_OF_YEAR)增加newVal - oldVal天,以更新日期。
|
|
|
updateDateControl();//调用updateDateControl方法,该方法用来更新日期控件的显示。
|
|
|
onDateTimeChanged();//调用onDateTimeChanged方法,该方法用来通知其他组件该日期时间已经发生了变化。
|
|
|
}
|
|
|
};
|
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnHourChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
@Override
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
boolean isDateChanged = false;
|
|
|
Calendar cal = Calendar.getInstance();
|
|
|
if (!mIs24HourView) {
|
|
|
if (!mIsAm && oldVal == HOURS_IN_HALF_DAY - 1 && newVal == HOURS_IN_HALF_DAY) {
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
cal.add(Calendar.DAY_OF_YEAR, 1);
|
|
|
isDateChanged = true;
|
|
|
} else if (mIsAm && oldVal == HOURS_IN_HALF_DAY && newVal == HOURS_IN_HALF_DAY - 1) {
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
cal.add(Calendar.DAY_OF_YEAR, -1);
|
|
|
isDateChanged = true;
|
|
|
}
|
|
|
if (oldVal == HOURS_IN_HALF_DAY - 1 && newVal == HOURS_IN_HALF_DAY ||
|
|
|
oldVal == HOURS_IN_HALF_DAY && newVal == HOURS_IN_HALF_DAY - 1) {
|
|
|
mIsAm = !mIsAm;
|
|
|
updateAmPmControl();
|
|
|
}
|
|
|
} else {
|
|
|
if (oldVal == HOURS_IN_ALL_DAY - 1 && newVal == 0) {
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
cal.add(Calendar.DAY_OF_YEAR, 1);
|
|
|
isDateChanged = true;
|
|
|
} else if (oldVal == 0 && newVal == HOURS_IN_ALL_DAY - 1) {
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
cal.add(Calendar.DAY_OF_YEAR, -1);
|
|
|
isDateChanged = true;
|
|
|
}
|
|
|
}
|
|
|
// 根据24小时制或12小时制以及上下午状态等情况,计算新的时间并更新到mDate字段中。
|
|
|
int newHour = mHourSpinner.getValue() % HOURS_IN_HALF_DAY + (mIsAm ? 0 : HOURS_IN_HALF_DAY);
|
|
|
mDate.set(Calendar.HOUR_OF_DAY, newHour);
|
|
|
onDateTimeChanged();
|
|
|
// 调用onDateTimeChanged方法,该方法用来通知其他组件该日期时间已经发生了变化。
|
|
|
if (isDateChanged) {
|
|
|
setCurrentYear(cal.get(Calendar.YEAR));
|
|
|
setCurrentMonth(cal.get(Calendar.MONTH));
|
|
|
setCurrentDay(cal.get(Calendar.DAY_OF_MONTH));
|
|
|
}
|
|
|
// 如果日期发生了变化,则调用setCurrentYear和setCurrentMonth方法更新年份和月份控件的显示。
|
|
|
}
|
|
|
};
|
|
|
// 这段代码定义了一个监听器对象,用于监听小时NumberPicker控件的值变化事件。
|
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnMinuteChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
@Override
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
int minValue = mMinuteSpinner.getMinValue();
|
|
|
int maxValue = mMinuteSpinner.getMaxValue();
|
|
|
int offset = 0;
|
|
|
if (oldVal == maxValue && newVal == minValue) {
|
|
|
offset += 1;
|
|
|
} else if (oldVal == minValue && newVal == maxValue) {
|
|
|
offset -= 1;
|
|
|
}
|
|
|
// 当监听器被触发时,它首先根据 NumberPicker 的旧值和新值计算出一个 offset。如果旧值是最大值且新值是最小值,则将 offset 增加1。如果旧值是最小值且新值是最大值,则将 offset 减少1。
|
|
|
if (offset != 0) {
|
|
|
mDate.add(Calendar.HOUR_OF_DAY, offset);
|
|
|
mHourSpinner.setValue(getCurrentHour());
|
|
|
updateDateControl();
|
|
|
int newHour = getCurrentHourOfDay();
|
|
|
if (newHour >= HOURS_IN_HALF_DAY) {
|
|
|
mIsAm = false;
|
|
|
updateAmPmControl();
|
|
|
} else {
|
|
|
mIsAm = true;
|
|
|
updateAmPmControl();
|
|
|
}
|
|
|
}
|
|
|
// 如果 offset 不为0,则监听器会更新 mDate 变量,将 offset 添加到小时数中,将小时选择器的值设置为当前小时,调用 updateDateControl() 更新日期控件,并根据新的小时数更新 AM/PM 控件。
|
|
|
mDate.set(Calendar.MINUTE, newVal);
|
|
|
onDateTimeChanged();
|
|
|
// 更新 mDate 变量后,监听器将分钟值设置为新值,并调用 onDateTimeChanged() 通知任何监听器日期和时间已更新。
|
|
|
}
|
|
|
};
|
|
|
// 定义了一个私有字段 mOnMinuteChangedListener,它是 NumberPicker.OnValueChangeListener 的一个实例。当表示分钟的 NumberPicker 的值发生变化时,此监听器将被触发。
|
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnAmPmChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
@Override
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
mIsAm = !mIsAm;
|
|
|
if (mIsAm) {
|
|
|
mDate.add(Calendar.HOUR_OF_DAY, -HOURS_IN_HALF_DAY);
|
|
|
} else {
|
|
|
mDate.add(Calendar.HOUR_OF_DAY, HOURS_IN_HALF_DAY);
|
|
|
}
|
|
|
updateAmPmControl();
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
// 当监听器被触发时,它将 mIsAm 变量取反,表示用户选择了 AM 还是 PM。如果 mIsAm 是 true,则说明用户选择了 AM,此时监听器将 mDate 变量减去半天的小时数。如果 mIsAm 是 false,则说明用户选择了 PM,此时监听器将 mDate 变量加上半天的小时数。然后,监听器会调用 updateAmPmControl() 更新 AM/PM 控件,并调用 onDateTimeChanged() 通知任何监听器日期和时间已更新。
|
|
|
};
|
|
|
// 定义了一个私有字段 mOnAmPmChangedListener,它是 NumberPicker.OnValueChangeListener 的一个实例。当表示 AM/PM 的 NumberPicker 的值发生变化时,此监听器将被触发。
|
|
|
|
|
|
public interface OnDateTimeChangedListener {
|
|
|
void onDateTimeChanged(DateTimePicker view, int year, int month,
|
|
|
int dayOfMonth, int hourOfDay, int minute);
|
|
|
// 该接口有一个抽象方法 onDateTimeChanged(),它在日期或时间发生变化时被调用。该方法接收 6 个参数:view 表示当前 DateTimePicker 实例,year 表示年份,month 表示月份(从 0 开始),dayOfMonth 表示月中的某一天,hourOfDay 表示小时数(24 小时制),minute 表示分钟数。
|
|
|
}
|
|
|
//定义了一个接口 OnDateTimeChangedListener,它用于监听日期和时间的变化。当日期或时间发生变化时,可以调用该接口的 onDateTimeChanged() 方法通知任何实现该接口的监听器。通过实现该接口并在需要的地方注册监听器,可以在日期或时间发生变化时执行自定义操作,例如更新 UI 或执行某些计算。
|
|
|
|
|
|
public DateTimePicker(Context context) {
|
|
|
this(context, System.currentTimeMillis());
|
|
|
}
|
|
|
|
|
|
public DateTimePicker(Context context, long date) {
|
|
|
this(context, date, DateFormat.is24HourFormat(context));
|
|
|
}
|
|
|
|
|
|
public DateTimePicker(Context context, long date, boolean is24HourView) {
|
|
|
super(context);
|
|
|
|
|
|
mDate = Calendar.getInstance();
|
|
|
mInitialising = true;
|
|
|
mIsAm = getCurrentHourOfDay() >= HOURS_IN_HALF_DAY;
|
|
|
inflate(context, R.layout.datetime_picker, this);
|
|
|
// 首先调用了父类 ViewGroup 的构造函数 super(context) 来初始化 DateTimePicker 实例。然后,代码使用 Calendar.getInstance() 获取一个 Calendar 对象,该对象表示当前日期和时间。变量 mIsAm 被初始化为当前小时数是否大于等于 12,以确定当前用户选择的是 AM 还是 PM。
|
|
|
|
|
|
mDateSpinner = (NumberPicker) findViewById(R.id.date);
|
|
|
mDateSpinner.setMinValue(DATE_SPINNER_MIN_VAL);
|
|
|
mDateSpinner.setMaxValue(DATE_SPINNER_MAX_VAL);
|
|
|
mDateSpinner.setOnValueChangedListener(mOnDateChangedListener);
|
|
|
// 获取了布局文件中 ID 为 date 的 NumberPicker 控件,并将其赋值给 mDateSpinner 变量。然后,使用 setMinValue() 和 setMaxValue() 方法将 mDateSpinner 的最小值和最大值分别设置为 DATE_SPINNER_MIN_VAL 和 DATE_SPINNER_MAX_VAL。使用 setOnValueChangedListener() 方法将 mDateSpinner 的值更改监听器设置为 mOnDateChangedListener。这意味着当 mDateSpinner 的值更改时,将会调用 mOnDateChangedListener 中的方法来处理这个事件。
|
|
|
|
|
|
mHourSpinner = (NumberPicker) findViewById(R.id.hour);
|
|
|
mHourSpinner.setOnValueChangedListener(mOnHourChangedListener);
|
|
|
mMinuteSpinner = (NumberPicker) findViewById(R.id.minute);
|
|
|
mMinuteSpinner.setMinValue(MINUT_SPINNER_MIN_VAL);
|
|
|
mMinuteSpinner.setMaxValue(MINUT_SPINNER_MAX_VAL);
|
|
|
mMinuteSpinner.setOnLongPressUpdateInterval(100);
|
|
|
mMinuteSpinner.setOnValueChangedListener(mOnMinuteChangedListener);
|
|
|
// 获取了布局文件中 ID 为 hour 和 minute 的 NumberPicker 控件,并将它们分别赋值给 mHourSpinner 和 mMinuteSpinner 变量。然后,使用 setOnValueChangedListener() 方法将 mHourSpinner 和 mMinuteSpinner 控件的值更改监听器分别设置为 mOnHourChangedListener 和 mOnMinuteChangedListener。
|
|
|
// 接下来,使用 setMinValue() 和 setMaxValue() 方法将 mMinuteSpinner 的最小值和最大值分别设置为 MINUT_SPINNER_MIN_VAL 和 MINUT_SPINNER_MAX_VAL,以限制分钟选择器的范围。
|
|
|
// 最后,使用 setOnLongPressUpdateInterval() 方法将 mMinuteSpinner 的长按更新间隔设置为 100 毫秒。这意味着当用户长按 mMinuteSpinner 中的增加或减少按钮时,它将以每 100 毫秒的速度连续增加或减少 mMinuteSpinner 的值。
|
|
|
// 通过设置 NumberPicker 的值更改监听器和其他属性,代码实现了一个可以选择小时和分钟的控件,并将它们分别绑定到 mHourSpinner 和 mMinuteSpinner 变量上,以便在之后的操作中使用。
|
|
|
|
|
|
String[] stringsForAmPm = new DateFormatSymbols().getAmPmStrings();//获取了当前设备的日期格式符号,并使用 getAmPmStrings() 方法从中获取 AM/PM 格式符号的字符串数组,并将其赋值给 stringsForAmPm 变量。
|
|
|
mAmPmSpinner = (NumberPicker) findViewById(R.id.amPm);
|
|
|
mAmPmSpinner.setMinValue(AMPM_SPINNER_MIN_VAL);
|
|
|
mAmPmSpinner.setMaxValue(AMPM_SPINNER_MAX_VAL);
|
|
|
// 获取了布局文件中 ID 为 amPm 的 NumberPicker 控件,并将其赋值给 mAmPmSpinner 变量。使用 setMinValue() 和 setMaxValue() 方法将 mAmPmSpinner 的最小值和最大值分别设置为 AMPM_SPINNER_MIN_VAL 和 AMPM_SPINNER_MAX_VAL,以限制 AM/PM 选择器的范围。
|
|
|
mAmPmSpinner.setDisplayedValues(stringsForAmPm);//使用 setDisplayedValues() 方法将 stringsForAmPm 数组设置为 mAmPmSpinner 的显示值。这意味着 AM/PM 选择器将显示 stringsForAmPm 数组中的字符串,而不是默认的数字值。
|
|
|
mAmPmSpinner.setOnValueChangedListener(mOnAmPmChangedListener);//使用 setOnValueChangedListener() 方法将 mAmPmSpinner 的值更改监听器设置为 mOnAmPmChangedListener。这意味着当用户更改 AM/PM 选择器的值时,将调用 mOnAmPmChangedListener 中的方法来处理这个事件。
|
|
|
|
|
|
// update controls to initial state
|
|
|
updateDateControl();
|
|
|
updateHourControl();
|
|
|
updateAmPmControl();
|
|
|
|
|
|
set24HourView(is24HourView);
|
|
|
// 代码调用了 updateDateControl()、updateHourControl() 和 updateAmPmControl() 方法,以便将控件显示为当前日期和时间的值。然后,代码调用了 set24HourView() 方法将选择器的时间格式设置为 24 小时制或 12 小时制。
|
|
|
|
|
|
// set to current time
|
|
|
setCurrentDate(date);
|
|
|
|
|
|
setEnabled(isEnabled());
|
|
|
|
|
|
// set the content descriptions
|
|
|
mInitialising = false;
|
|
|
// 调用了 setCurrentDate() 方法将选择器的日期和时间设置为指定的日期和时间,调用 setEnabled() 方法将选择器的可用状态设置为指定的值,并将 mInitialising 标志设置为 false,以便通知选择器已完成初始化。
|
|
|
}
|
|
|
|
|
|
@Override
|
|
|
public void setEnabled(boolean enabled) {
|
|
|
if (mIsEnabled == enabled) {
|
|
|
return;
|
|
|
}
|
|
|
// 当调用 setEnabled() 方法时,如果传入的参数 enabled 与当前的 mIsEnabled 变量的值相同,说明选择器的可用状态没有发生变化,直接返回即可。
|
|
|
super.setEnabled(enabled);
|
|
|
mDateSpinner.setEnabled(enabled);
|
|
|
mMinuteSpinner.setEnabled(enabled);
|
|
|
mHourSpinner.setEnabled(enabled);
|
|
|
mAmPmSpinner.setEnabled(enabled);
|
|
|
mIsEnabled = enabled;
|
|
|
// 代码首先调用父类的 setEnabled() 方法,将整个时间选择器的可用状态设置为传入的参数 enabled。然后,分别调用 mDateSpinner、mMinuteSpinner、mHourSpinner 和 mAmPmSpinner 的 setEnabled() 方法,将它们的可用状态也设置为传入的参数 enabled。
|
|
|
//最后,将 mIsEnabled 变量的值设置为传入的参数 enabled,以便在下一次调用 setEnabled() 方法时使用。
|
|
|
}
|
|
|
// 重写 setEnabled() 方法,代码实现了一个可以同时设置整个时间选择器的可用状态的功能。
|
|
|
|
|
|
@Override
|
|
|
public boolean isEnabled() {
|
|
|
return mIsEnabled;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get the current date in millis
|
|
|
*
|
|
|
* @return the current date in millis
|
|
|
*/
|
|
|
public long getCurrentDateInTimeMillis() {
|
|
|
return mDate.getTimeInMillis();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set the current date
|
|
|
*
|
|
|
* @param date The current date in millis
|
|
|
*/
|
|
|
public void setCurrentDate(long date) {
|
|
|
Calendar cal = Calendar.getInstance();
|
|
|
cal.setTimeInMillis(date);
|
|
|
setCurrentDate(cal.get(Calendar.YEAR), cal.get(Calendar.MONTH), cal.get(Calendar.DAY_OF_MONTH),
|
|
|
cal.get(Calendar.HOUR_OF_DAY), cal.get(Calendar.MINUTE));
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set the current date
|
|
|
*
|
|
|
* @param year The current year
|
|
|
* @param month The current month
|
|
|
* @param dayOfMonth The current dayOfMonth
|
|
|
* @param hourOfDay The current hourOfDay
|
|
|
* @param minute The current minute
|
|
|
*/
|
|
|
public void setCurrentDate(int year, int month,
|
|
|
int dayOfMonth, int hourOfDay, int minute) {
|
|
|
setCurrentYear(year);
|
|
|
setCurrentMonth(month);
|
|
|
setCurrentDay(dayOfMonth);
|
|
|
setCurrentHour(hourOfDay);
|
|
|
setCurrentMinute(minute);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get current year
|
|
|
*
|
|
|
* @return The current year
|
|
|
*/
|
|
|
public int getCurrentYear() {
|
|
|
return mDate.get(Calendar.YEAR);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set current year
|
|
|
*
|
|
|
* @param year The current year
|
|
|
*/
|
|
|
public void setCurrentYear(int year) {
|
|
|
if (!mInitialising && year == getCurrentYear()) {
|
|
|
return;
|
|
|
}
|
|
|
mDate.set(Calendar.YEAR, year);
|
|
|
updateDateControl();
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get current month in the year
|
|
|
*
|
|
|
* @return The current month in the year
|
|
|
*/
|
|
|
public int getCurrentMonth() {
|
|
|
return mDate.get(Calendar.MONTH);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set current month in the year
|
|
|
*
|
|
|
* @param month The month in the year
|
|
|
*/
|
|
|
public void setCurrentMonth(int month) {
|
|
|
if (!mInitialising && month == getCurrentMonth()) {
|
|
|
return;
|
|
|
}
|
|
|
mDate.set(Calendar.MONTH, month);
|
|
|
updateDateControl();
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get current day of the month
|
|
|
*
|
|
|
* @return The day of the month
|
|
|
*/
|
|
|
public int getCurrentDay() {
|
|
|
return mDate.get(Calendar.DAY_OF_MONTH);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set current day of the month
|
|
|
*
|
|
|
* @param dayOfMonth The day of the month
|
|
|
*/
|
|
|
public void setCurrentDay(int dayOfMonth) {
|
|
|
if (!mInitialising && dayOfMonth == getCurrentDay()) {
|
|
|
return;
|
|
|
}
|
|
|
mDate.set(Calendar.DAY_OF_MONTH, dayOfMonth);
|
|
|
updateDateControl();
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get current hour in 24 hour mode, in the range (0~23)
|
|
|
* @return The current hour in 24 hour mode
|
|
|
*/
|
|
|
public int getCurrentHourOfDay() {
|
|
|
return mDate.get(Calendar.HOUR_OF_DAY);
|
|
|
}
|
|
|
|
|
|
private int getCurrentHour() {
|
|
|
if (mIs24HourView){
|
|
|
return getCurrentHourOfDay();
|
|
|
} else {
|
|
|
int hour = getCurrentHourOfDay();
|
|
|
if (hour > HOURS_IN_HALF_DAY) {
|
|
|
return hour - HOURS_IN_HALF_DAY;
|
|
|
} else {
|
|
|
return hour == 0 ? HOURS_IN_HALF_DAY : hour;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set current hour in 24 hour mode, in the range (0~23)
|
|
|
*
|
|
|
* @param hourOfDay
|
|
|
*/
|
|
|
public void setCurrentHour(int hourOfDay) {
|
|
|
if (!mInitialising && hourOfDay == getCurrentHourOfDay()) {
|
|
|
return;
|
|
|
}
|
|
|
mDate.set(Calendar.HOUR_OF_DAY, hourOfDay);
|
|
|
if (!mIs24HourView) {
|
|
|
if (hourOfDay >= HOURS_IN_HALF_DAY) {
|
|
|
mIsAm = false;
|
|
|
if (hourOfDay > HOURS_IN_HALF_DAY) {
|
|
|
hourOfDay -= HOURS_IN_HALF_DAY;
|
|
|
}
|
|
|
} else {
|
|
|
mIsAm = true;
|
|
|
if (hourOfDay == 0) {
|
|
|
hourOfDay = HOURS_IN_HALF_DAY;
|
|
|
}
|
|
|
}
|
|
|
updateAmPmControl();
|
|
|
}
|
|
|
mHourSpinner.setValue(hourOfDay);
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Get currentMinute
|
|
|
*
|
|
|
* @return The Current Minute
|
|
|
*/
|
|
|
public int getCurrentMinute() {
|
|
|
return mDate.get(Calendar.MINUTE);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set current minute
|
|
|
*/
|
|
|
public void setCurrentMinute(int minute) {
|
|
|
if (!mInitialising && minute == getCurrentMinute()) {
|
|
|
return;
|
|
|
}
|
|
|
mMinuteSpinner.setValue(minute);
|
|
|
mDate.set(Calendar.MINUTE, minute);
|
|
|
onDateTimeChanged();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* @return true if this is in 24 hour view else false.
|
|
|
*/
|
|
|
public boolean is24HourView () {
|
|
|
return mIs24HourView;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Set whether in 24 hour or AM/PM mode.
|
|
|
*
|
|
|
* @param is24HourView True for 24 hour mode. False for AM/PM mode.
|
|
|
*/
|
|
|
public void set24HourView(boolean is24HourView) {
|
|
|
if (mIs24HourView == is24HourView) {
|
|
|
return;
|
|
|
}
|
|
|
mIs24HourView = is24HourView;
|
|
|
mAmPmSpinner.setVisibility(is24HourView ? View.GONE : View.VISIBLE);
|
|
|
// 当传入的参数 is24HourView 与当前的 mIs24HourView 变量的值相同时,方法直接返回。否则,将 mIs24HourView 变量的值设置为传入的参数 is24HourView,表示时间选择器的显示格式已经更改。
|
|
|
int hour = getCurrentHourOfDay();
|
|
|
updateHourControl();
|
|
|
setCurrentHour(hour);
|
|
|
updateAmPmControl();
|
|
|
// 获取当前的小时数并调用 updateHourControl() 方法更新小时选择器,以保持小时选择器的一致性。接着,使用 setCurrentHour() 方法将当前的小时数设置回时间选择器,并调用 updateAmPmControl() 方法更新 AM/PM 选择器。
|
|
|
}
|
|
|
//通过设置 AM/PM 选择器的可见性和更新时间选择器的小时控件和 AM/PM 控件,代码实现了一个可以切换时间选择器显示格式的功能。
|
|
|
|
|
|
private void updateDateControl() {
|
|
|
Calendar cal = Calendar.getInstance();
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
cal.add(Calendar.DAY_OF_YEAR, -DAYS_IN_ALL_WEEK / 2 - 1);
|
|
|
mDateSpinner.setDisplayedValues(null);
|
|
|
for (int i = 0; i < DAYS_IN_ALL_WEEK; ++i) {
|
|
|
cal.add(Calendar.DAY_OF_YEAR, 1);
|
|
|
mDateDisplayValues[i] = (String) DateFormat.format("MM.dd EEEE", cal);
|
|
|
}
|
|
|
mDateSpinner.setDisplayedValues(mDateDisplayValues);
|
|
|
mDateSpinner.setValue(DAYS_IN_ALL_WEEK / 2);
|
|
|
mDateSpinner.invalidate();
|
|
|
}
|
|
|
// 用于更新日期选择器的显示值。它首先获取当前日期并将其设置为日历实例 cal 的时间。然后,cal 被设置为一周的第一天,以确保日期选择器显示一周的日期。
|
|
|
//接下来,使用 setDisplayedValues() 方法将日期选择器的显示值设置为 null,以便重新生成新的日期显示值。然后,使用循环从 cal 中获取每一天的日期,并将其格式化为 MM.dd EEEE 的格式,并将其存储在一个字符串数组中。最后,使用 setDisplayedValues() 方法将新的日期显示值设置为 mDateSpinner,并将当前选择的日期值设置为一周的中间值,以保持日期选择器的一致性。
|
|
|
|
|
|
private void updateAmPmControl() {
|
|
|
if (mIs24HourView) {
|
|
|
mAmPmSpinner.setVisibility(View.GONE);
|
|
|
} else {
|
|
|
int index = mIsAm ? Calendar.AM : Calendar.PM;
|
|
|
mAmPmSpinner.setValue(index);
|
|
|
mAmPmSpinner.setVisibility(View.VISIBLE);
|
|
|
}
|
|
|
}
|
|
|
// 用于更新 AM/PM 选择器的显示值。如果时间选择器为 24 小时格式,则将 AM/PM 选择器隐藏;否则,根据当前是否为 AM,将 AM/PM 选择器的值设置为 Calendar.AM 或 Calendar.PM,并将其可见性设置为 View.VISIBLE。
|
|
|
|
|
|
private void updateHourControl() {
|
|
|
if (mIs24HourView) {
|
|
|
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW);
|
|
|
mHourSpinner.setMaxValue(HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW);
|
|
|
} else {
|
|
|
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW);
|
|
|
mHourSpinner.setMaxValue(HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW);
|
|
|
}
|
|
|
}
|
|
|
// 用于更新小时选择器的显示值。如果时间选择器为 24 小时格式,则将小时选择器的最小值和最大值分别设置为 HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW 和 HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW;否则,将小时选择器的最小值和最大值分别设置为 HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW 和 HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW。
|
|
|
|
|
|
/**
|
|
|
* Set the callback that indicates the 'Set' button has been pressed.
|
|
|
* @param callback the callback, if null will do nothing
|
|
|
*/
|
|
|
public void setOnDateTimeChangedListener(OnDateTimeChangedListener callback) {
|
|
|
mOnDateTimeChangedListener = callback;
|
|
|
}
|
|
|
|
|
|
private void onDateTimeChanged() {
|
|
|
if (mOnDateTimeChangedListener != null) {
|
|
|
mOnDateTimeChangedListener.onDateTimeChanged(this, getCurrentYear(),
|
|
|
getCurrentMonth(), getCurrentDay(), getCurrentHourOfDay(), getCurrentMinute());
|
|
|
}
|
|
|
}
|
|
|
}
|