|
|
|
|
@ -14,99 +14,171 @@
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// 包声明:归属小米便签的UI模块,自定义日期时间选择控件核心类
|
|
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
|
|
|
|
// 导入日期格式化工具类:处理星期/日期的文本展示
|
|
|
|
|
import java.text.DateFormatSymbols;
|
|
|
|
|
// 导入日历类:核心的日期时间管理,处理年/月/日/时/分的计算与联动
|
|
|
|
|
import java.util.Calendar;
|
|
|
|
|
|
|
|
|
|
// 导入资源类:引用布局文件(datetime_picker.xml)
|
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 导入安卓上下文类:创建控件、加载资源
|
|
|
|
|
import android.content.Context;
|
|
|
|
|
// 导入日期格式化工具:判断系统24小时制设置
|
|
|
|
|
import android.text.format.DateFormat;
|
|
|
|
|
// 导入视图相关类:布局容器、视图可见性控制
|
|
|
|
|
import android.view.View;
|
|
|
|
|
import android.widget.FrameLayout;
|
|
|
|
|
// 导入数字选择器:核心的日期/时间选择UI组件
|
|
|
|
|
import android.widget.NumberPicker;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 自定义日期时间选择控件
|
|
|
|
|
* 核心特性:
|
|
|
|
|
* 1. 布局组成:集成日期(近7天)、小时、分钟、上午/下午(AM/PM)四个NumberPicker;
|
|
|
|
|
* 2. 时间联动:处理时间选择的边界联动(如分钟59→0时小时+1、小时23→0时日期+1);
|
|
|
|
|
* 3. 制式适配:支持24小时制/12小时制切换,自动适配系统默认设置;
|
|
|
|
|
* 4. 回调通知:时间选择变化时触发回调,传递最新的年/月/日/时/分;
|
|
|
|
|
* 5. 状态控制:支持整体启用/禁用所有选择器,统一管理交互状态;
|
|
|
|
|
* 典型使用场景:便签提醒时间设置的DateTimePickerDialog中作为核心选择UI。
|
|
|
|
|
*/
|
|
|
|
|
public class DateTimePicker extends FrameLayout {
|
|
|
|
|
|
|
|
|
|
// ======================== 基础常量 - 控件默认状态 ========================
|
|
|
|
|
/** 控件默认启用状态:初始为启用 */
|
|
|
|
|
private static final boolean DEFAULT_ENABLE_STATE = true;
|
|
|
|
|
|
|
|
|
|
// ======================== 常量 - 时间数值范围 ========================
|
|
|
|
|
/** 半天的小时数:12小时制的核心数值(AM/PM分界) */
|
|
|
|
|
private static final int HOURS_IN_HALF_DAY = 12;
|
|
|
|
|
/** 全天的小时数:24小时制的核心数值 */
|
|
|
|
|
private static final int HOURS_IN_ALL_DAY = 24;
|
|
|
|
|
/** 一周的天数:日期选择器展示近7天 */
|
|
|
|
|
private static final int DAYS_IN_ALL_WEEK = 7;
|
|
|
|
|
|
|
|
|
|
// ======================== 常量 - NumberPicker取值范围 ========================
|
|
|
|
|
/** 日期选择器最小值:0(对应近7天的起始索引) */
|
|
|
|
|
private static final int DATE_SPINNER_MIN_VAL = 0;
|
|
|
|
|
/** 日期选择器最大值:6(对应近7天的结束索引,DAYS_IN_ALL_WEEK - 1) */
|
|
|
|
|
private static final int DATE_SPINNER_MAX_VAL = DAYS_IN_ALL_WEEK - 1;
|
|
|
|
|
|
|
|
|
|
/** 24小时制-小时选择器最小值:0(凌晨0点) */
|
|
|
|
|
private static final int HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW = 0;
|
|
|
|
|
/** 24小时制-小时选择器最大值:23(深夜23点) */
|
|
|
|
|
private static final int HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW = 23;
|
|
|
|
|
|
|
|
|
|
/** 12小时制-小时选择器最小值:1(上午/下午1点) */
|
|
|
|
|
private static final int HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW = 1;
|
|
|
|
|
/** 12小时制-小时选择器最大值:12(上午/下午12点) */
|
|
|
|
|
private static final int HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW = 12;
|
|
|
|
|
|
|
|
|
|
/** 分钟选择器最小值:0(整点) */
|
|
|
|
|
private static final int MINUT_SPINNER_MIN_VAL = 0;
|
|
|
|
|
/** 分钟选择器最大值:59(整点前1分钟) */
|
|
|
|
|
private static final int MINUT_SPINNER_MAX_VAL = 59;
|
|
|
|
|
|
|
|
|
|
/** 上午/下午选择器最小值:0(AM/上午) */
|
|
|
|
|
private static final int AMPM_SPINNER_MIN_VAL = 0;
|
|
|
|
|
/** 上午/下午选择器最大值:1(PM/下午) */
|
|
|
|
|
private static final int AMPM_SPINNER_MAX_VAL = 1;
|
|
|
|
|
|
|
|
|
|
// ======================== 成员变量 - UI组件 ========================
|
|
|
|
|
/** 日期选择器:展示近7天的日期(格式如“12.23 星期二”) */
|
|
|
|
|
private final NumberPicker mDateSpinner;
|
|
|
|
|
/** 小时选择器:根据24/12小时制展示不同范围的小时数 */
|
|
|
|
|
private final NumberPicker mHourSpinner;
|
|
|
|
|
/** 分钟选择器:0~59的分钟数选择 */
|
|
|
|
|
private final NumberPicker mMinuteSpinner;
|
|
|
|
|
/** 上午/下午选择器:仅12小时制显示,0=AM/上午,1=PM/下午 */
|
|
|
|
|
private final NumberPicker mAmPmSpinner;
|
|
|
|
|
private Calendar mDate;
|
|
|
|
|
|
|
|
|
|
// ======================== 成员变量 - 日期时间状态 ========================
|
|
|
|
|
/** 核心日历对象:存储当前选中的日期时间,处理所有时间计算/联动 */
|
|
|
|
|
private Calendar mDate;
|
|
|
|
|
/** 日期选择器展示文本数组:存储近7天的格式化日期文本(如“12.23 星期二”) */
|
|
|
|
|
private String[] mDateDisplayValues = new String[DAYS_IN_ALL_WEEK];
|
|
|
|
|
|
|
|
|
|
/** 上午/下午标记:true=AM/上午,false=PM/下午(仅12小时制有效) */
|
|
|
|
|
private boolean mIsAm;
|
|
|
|
|
|
|
|
|
|
/** 24小时制标记:true=24小时制,false=12小时制 */
|
|
|
|
|
private boolean mIs24HourView;
|
|
|
|
|
|
|
|
|
|
/** 控件启用状态:true=启用,false=禁用所有选择器 */
|
|
|
|
|
private boolean mIsEnabled = DEFAULT_ENABLE_STATE;
|
|
|
|
|
|
|
|
|
|
/** 初始化标记:true=控件正在初始化,避免初始化时触发不必要的回调 */
|
|
|
|
|
private boolean mInitialising;
|
|
|
|
|
|
|
|
|
|
// ======================== 成员变量 - 回调监听 ========================
|
|
|
|
|
/** 日期时间变化回调监听器:外部实现,接收时间变化通知 */
|
|
|
|
|
private OnDateTimeChangedListener mOnDateTimeChangedListener;
|
|
|
|
|
|
|
|
|
|
// ======================== 成员变量 - NumberPicker值变化监听器 ========================
|
|
|
|
|
/** 日期选择器值变化监听器:处理日期切换,更新日历对象并触发回调 */
|
|
|
|
|
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);
|
|
|
|
|
// 更新日期选择器的展示文本
|
|
|
|
|
updateDateControl();
|
|
|
|
|
// 触发日期时间变化回调
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** 小时选择器值变化监听器:处理小时切换,包含边界联动(小时→日期)和12/24小时制适配 */
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnHourChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
|
|
@Override
|
|
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
|
|
boolean isDateChanged = false;
|
|
|
|
|
boolean isDateChanged = false; // 日期是否变化标记
|
|
|
|
|
Calendar cal = Calendar.getInstance();
|
|
|
|
|
|
|
|
|
|
// 12小时制下的小时边界处理
|
|
|
|
|
if (!mIs24HourView) {
|
|
|
|
|
// 场景1:下午11点→12点 → 日期+1
|
|
|
|
|
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) {
|
|
|
|
|
}
|
|
|
|
|
// 场景2:上午12点→11点 → 日期-1
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 小时11→12 或 12→11 时,切换上午/下午标记
|
|
|
|
|
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();
|
|
|
|
|
updateAmPmControl(); // 更新上午/下午选择器
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
}
|
|
|
|
|
// 24小时制下的小时边界处理
|
|
|
|
|
else {
|
|
|
|
|
// 场景1:23点→0点 → 日期+1
|
|
|
|
|
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) {
|
|
|
|
|
}
|
|
|
|
|
// 场景2:0点→23点 → 日期-1
|
|
|
|
|
else if (oldVal == 0 && newVal == HOURS_IN_ALL_DAY - 1) {
|
|
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
|
|
cal.add(Calendar.DAY_OF_YEAR, -1);
|
|
|
|
|
isDateChanged = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 计算并设置最终的小时数(适配12/24小时制)
|
|
|
|
|
int newHour = mHourSpinner.getValue() % HOURS_IN_HALF_DAY + (mIsAm ? 0 : HOURS_IN_HALF_DAY);
|
|
|
|
|
mDate.set(Calendar.HOUR_OF_DAY, newHour);
|
|
|
|
|
// 触发时间变化回调
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
|
|
|
|
|
// 日期变化时,更新日历的年/月/日
|
|
|
|
|
if (isDateChanged) {
|
|
|
|
|
setCurrentYear(cal.get(Calendar.YEAR));
|
|
|
|
|
setCurrentMonth(cal.get(Calendar.MONTH));
|
|
|
|
|
@ -115,152 +187,218 @@ public class DateTimePicker extends FrameLayout {
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** 分钟选择器值变化监听器:处理分钟切换,包含边界联动(分钟→小时→日期) */
|
|
|
|
|
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;
|
|
|
|
|
int offset = 0; // 小时偏移量
|
|
|
|
|
|
|
|
|
|
// 场景1:59分→0分 → 小时+1
|
|
|
|
|
if (oldVal == maxValue && newVal == minValue) {
|
|
|
|
|
offset += 1;
|
|
|
|
|
} else if (oldVal == minValue && newVal == maxValue) {
|
|
|
|
|
}
|
|
|
|
|
// 场景2:0分→59分 → 小时-1
|
|
|
|
|
else if (oldVal == minValue && newVal == maxValue) {
|
|
|
|
|
offset -= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 小时需要偏移时,调整日历并更新相关选择器
|
|
|
|
|
if (offset != 0) {
|
|
|
|
|
mDate.add(Calendar.HOUR_OF_DAY, offset);
|
|
|
|
|
mHourSpinner.setValue(getCurrentHour());
|
|
|
|
|
updateDateControl();
|
|
|
|
|
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();
|
|
|
|
|
}
|
|
|
|
|
updateAmPmControl(); // 更新上午/下午选择器
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置最终的分钟数
|
|
|
|
|
mDate.set(Calendar.MINUTE, newVal);
|
|
|
|
|
// 触发时间变化回调
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** 上午/下午选择器值变化监听器:切换上午/下午,调整小时数(±12小时) */
|
|
|
|
|
private NumberPicker.OnValueChangeListener mOnAmPmChangedListener = new NumberPicker.OnValueChangeListener() {
|
|
|
|
|
@Override
|
|
|
|
|
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
|
|
|
|
|
// 切换上午/下午标记
|
|
|
|
|
mIsAm = !mIsAm;
|
|
|
|
|
// 调整小时数:上午→下午 +12小时,下午→上午 -12小时
|
|
|
|
|
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();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ======================== 回调接口 - 日期时间变化通知 ========================
|
|
|
|
|
/**
|
|
|
|
|
* 日期时间变化回调接口
|
|
|
|
|
* 由外部(如DateTimePickerDialog)实现,接收控件选中的最新日期时间
|
|
|
|
|
*/
|
|
|
|
|
public interface OnDateTimeChangedListener {
|
|
|
|
|
/**
|
|
|
|
|
* 日期时间变化回调方法
|
|
|
|
|
* @param view 当前的DateTimePicker控件实例
|
|
|
|
|
* @param year 选中的年份
|
|
|
|
|
* @param month 选中的月份(Calendar.MONTH,0=1月,11=12月)
|
|
|
|
|
* @param dayOfMonth 选中的日期(当月的第几天)
|
|
|
|
|
* @param hourOfDay 选中的小时(24小时制,0~23)
|
|
|
|
|
* @param minute 选中的分钟(0~59)
|
|
|
|
|
*/
|
|
|
|
|
void onDateTimeChanged(DateTimePicker view, int year, int month,
|
|
|
|
|
int dayOfMonth, int hourOfDay, int minute);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 构造方法 ========================
|
|
|
|
|
/**
|
|
|
|
|
* 构造方法1:默认初始化(使用当前系统时间,适配系统24小时制)
|
|
|
|
|
* @param context 应用上下文
|
|
|
|
|
*/
|
|
|
|
|
public DateTimePicker(Context context) {
|
|
|
|
|
this(context, System.currentTimeMillis());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 构造方法2:指定初始时间,适配系统24小时制
|
|
|
|
|
* @param context 应用上下文
|
|
|
|
|
* @param date 初始时间戳(毫秒级)
|
|
|
|
|
*/
|
|
|
|
|
public DateTimePicker(Context context, long date) {
|
|
|
|
|
this(context, date, DateFormat.is24HourFormat(context));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 构造方法3:指定初始时间和24小时制状态,核心初始化逻辑
|
|
|
|
|
* @param context 应用上下文
|
|
|
|
|
* @param date 初始时间戳(毫秒级)
|
|
|
|
|
* @param is24HourView 是否使用24小时制
|
|
|
|
|
*/
|
|
|
|
|
public DateTimePicker(Context context, long date, boolean is24HourView) {
|
|
|
|
|
super(context);
|
|
|
|
|
mDate = Calendar.getInstance();
|
|
|
|
|
mInitialising = true;
|
|
|
|
|
mDate = Calendar.getInstance(); // 初始化核心日历对象
|
|
|
|
|
mInitialising = true; // 标记进入初始化阶段(避免触发不必要的回调)
|
|
|
|
|
// 初始化上午/下午标记:根据当前小时数判断(≥12为下午)
|
|
|
|
|
mIsAm = getCurrentHourOfDay() >= HOURS_IN_HALF_DAY;
|
|
|
|
|
|
|
|
|
|
// 加载控件布局(datetime_picker.xml)
|
|
|
|
|
inflate(context, R.layout.datetime_picker, this);
|
|
|
|
|
|
|
|
|
|
// 初始化日期选择器
|
|
|
|
|
mDateSpinner = (NumberPicker) findViewById(R.id.date);
|
|
|
|
|
mDateSpinner.setMinValue(DATE_SPINNER_MIN_VAL);
|
|
|
|
|
mDateSpinner.setMaxValue(DATE_SPINNER_MAX_VAL);
|
|
|
|
|
mDateSpinner.setOnValueChangedListener(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.setLongPressUpdateInterval(100); // 长按快速调整的间隔(100ms)
|
|
|
|
|
mMinuteSpinner.setOnValueChangedListener(mOnMinuteChangedListener);
|
|
|
|
|
|
|
|
|
|
String[] stringsForAmPm = new DateFormatSymbols().getAmPmStrings();
|
|
|
|
|
// 初始化上午/下午选择器
|
|
|
|
|
String[] stringsForAmPm = new DateFormatSymbols().getAmPmStrings(); // 获取系统AM/PM文本(适配多语言)
|
|
|
|
|
mAmPmSpinner = (NumberPicker) findViewById(R.id.amPm);
|
|
|
|
|
mAmPmSpinner.setMinValue(AMPM_SPINNER_MIN_VAL);
|
|
|
|
|
mAmPmSpinner.setMaxValue(AMPM_SPINNER_MAX_VAL);
|
|
|
|
|
mAmPmSpinner.setDisplayedValues(stringsForAmPm);
|
|
|
|
|
mAmPmSpinner.setDisplayedValues(stringsForAmPm); // 设置AM/PM展示文本
|
|
|
|
|
mAmPmSpinner.setOnValueChangedListener(mOnAmPmChangedListener);
|
|
|
|
|
|
|
|
|
|
// update controls to initial state
|
|
|
|
|
// 更新控件到初始状态
|
|
|
|
|
updateDateControl();
|
|
|
|
|
updateHourControl();
|
|
|
|
|
updateAmPmControl();
|
|
|
|
|
|
|
|
|
|
// 设置24小时制状态
|
|
|
|
|
set24HourView(is24HourView);
|
|
|
|
|
|
|
|
|
|
// set to current time
|
|
|
|
|
// 设置初始时间
|
|
|
|
|
setCurrentDate(date);
|
|
|
|
|
|
|
|
|
|
// 设置控件启用状态
|
|
|
|
|
setEnabled(isEnabled());
|
|
|
|
|
|
|
|
|
|
// set the content descriptions
|
|
|
|
|
// 初始化完成,取消标记
|
|
|
|
|
mInitialising = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 重写方法 - 控件启用/禁用 ========================
|
|
|
|
|
/**
|
|
|
|
|
* 设置控件整体启用/禁用状态
|
|
|
|
|
* @param enabled true=启用(所有选择器可交互),false=禁用(所有选择器不可交互)
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public void setEnabled(boolean enabled) {
|
|
|
|
|
if (mIsEnabled == enabled) {
|
|
|
|
|
return;
|
|
|
|
|
return; // 状态未变化,直接返回
|
|
|
|
|
}
|
|
|
|
|
super.setEnabled(enabled);
|
|
|
|
|
// 同步所有选择器的启用状态
|
|
|
|
|
mDateSpinner.setEnabled(enabled);
|
|
|
|
|
mMinuteSpinner.setEnabled(enabled);
|
|
|
|
|
mHourSpinner.setEnabled(enabled);
|
|
|
|
|
mAmPmSpinner.setEnabled(enabled);
|
|
|
|
|
// 更新启用状态标记
|
|
|
|
|
mIsEnabled = enabled;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 获取控件整体启用状态
|
|
|
|
|
* @return true=启用,false=禁用
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public boolean isEnabled() {
|
|
|
|
|
return mIsEnabled;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 公共方法 - 日期时间获取/设置 ========================
|
|
|
|
|
/**
|
|
|
|
|
* Get the current date in millis
|
|
|
|
|
*
|
|
|
|
|
* @return the current date in millis
|
|
|
|
|
* 获取当前选中的日期时间戳(毫秒级)
|
|
|
|
|
* @return 选中时间的毫秒数
|
|
|
|
|
*/
|
|
|
|
|
public long getCurrentDateInTimeMillis() {
|
|
|
|
|
return mDate.getTimeInMillis();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set the current date
|
|
|
|
|
*
|
|
|
|
|
* @param date The current date in millis
|
|
|
|
|
* 设置当前选中的日期时间(通过时间戳)
|
|
|
|
|
* @param date 要设置的时间戳(毫秒级)
|
|
|
|
|
*/
|
|
|
|
|
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
|
|
|
|
|
* 设置当前选中的日期时间(通过年/月/日/时/分)
|
|
|
|
|
* @param year 年份
|
|
|
|
|
* @param month 月份(Calendar.MONTH,0=1月)
|
|
|
|
|
* @param dayOfMonth 日期(当月的第几天)
|
|
|
|
|
* @param hourOfDay 小时(24小时制,0~23)
|
|
|
|
|
* @param minute 分钟(0~59)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentDate(int year, int month,
|
|
|
|
|
int dayOfMonth, int hourOfDay, int minute) {
|
|
|
|
|
@ -272,86 +410,88 @@ public class DateTimePicker extends FrameLayout {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get current year
|
|
|
|
|
*
|
|
|
|
|
* @return The current year
|
|
|
|
|
* 获取当前选中的年份
|
|
|
|
|
* @return 年份(如2025)
|
|
|
|
|
*/
|
|
|
|
|
public int getCurrentYear() {
|
|
|
|
|
return mDate.get(Calendar.YEAR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set current year
|
|
|
|
|
*
|
|
|
|
|
* @param year The current year
|
|
|
|
|
* 设置当前选中的年份
|
|
|
|
|
* @param year 要设置的年份(如2025)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentYear(int year) {
|
|
|
|
|
// 初始化中或年份未变化时,直接返回
|
|
|
|
|
if (!mInitialising && year == getCurrentYear()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mDate.set(Calendar.YEAR, year);
|
|
|
|
|
updateDateControl();
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
updateDateControl(); // 更新日期选择器展示
|
|
|
|
|
onDateTimeChanged(); // 触发时间变化回调
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get current month in the year
|
|
|
|
|
*
|
|
|
|
|
* @return The current month in the year
|
|
|
|
|
* 获取当前选中的月份
|
|
|
|
|
* @return 月份(Calendar.MONTH,0=1月,11=12月)
|
|
|
|
|
*/
|
|
|
|
|
public int getCurrentMonth() {
|
|
|
|
|
return mDate.get(Calendar.MONTH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set current month in the year
|
|
|
|
|
*
|
|
|
|
|
* @param month The month in the year
|
|
|
|
|
* 设置当前选中的月份
|
|
|
|
|
* @param month 要设置的月份(Calendar.MONTH,0=1月)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentMonth(int month) {
|
|
|
|
|
// 初始化中或月份未变化时,直接返回
|
|
|
|
|
if (!mInitialising && month == getCurrentMonth()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mDate.set(Calendar.MONTH, month);
|
|
|
|
|
updateDateControl();
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
updateDateControl(); // 更新日期选择器展示
|
|
|
|
|
onDateTimeChanged(); // 触发时间变化回调
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get current day of the month
|
|
|
|
|
*
|
|
|
|
|
* @return The day of the month
|
|
|
|
|
* 获取当前选中的日期(当月的第几天)
|
|
|
|
|
* @return 日期(1~31)
|
|
|
|
|
*/
|
|
|
|
|
public int getCurrentDay() {
|
|
|
|
|
return mDate.get(Calendar.DAY_OF_MONTH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set current day of the month
|
|
|
|
|
*
|
|
|
|
|
* @param dayOfMonth The day of the month
|
|
|
|
|
* 设置当前选中的日期(当月的第几天)
|
|
|
|
|
* @param dayOfMonth 要设置的日期(1~31)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentDay(int dayOfMonth) {
|
|
|
|
|
// 初始化中或日期未变化时,直接返回
|
|
|
|
|
if (!mInitialising && dayOfMonth == getCurrentDay()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mDate.set(Calendar.DAY_OF_MONTH, dayOfMonth);
|
|
|
|
|
updateDateControl();
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
updateDateControl(); // 更新日期选择器展示
|
|
|
|
|
onDateTimeChanged(); // 触发时间变化回调
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get current hour in 24 hour mode, in the range (0~23)
|
|
|
|
|
* @return The current hour in 24 hour mode
|
|
|
|
|
* 获取当前选中的小时(24小时制,0~23)
|
|
|
|
|
* @return 小时数(0~23)
|
|
|
|
|
*/
|
|
|
|
|
public int getCurrentHourOfDay() {
|
|
|
|
|
return mDate.get(Calendar.HOUR_OF_DAY);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 内部方法:获取适配当前制式的小时数(24/12小时制)
|
|
|
|
|
* @return 24小时制返回0~23,12小时制返回1~12
|
|
|
|
|
*/
|
|
|
|
|
private int getCurrentHour() {
|
|
|
|
|
if (mIs24HourView){
|
|
|
|
|
return getCurrentHourOfDay();
|
|
|
|
|
return getCurrentHourOfDay(); // 24小时制直接返回
|
|
|
|
|
} else {
|
|
|
|
|
// 12小时制转换:0点→12点,13点→1点,依此类推
|
|
|
|
|
int hour = getCurrentHourOfDay();
|
|
|
|
|
if (hour > HOURS_IN_HALF_DAY) {
|
|
|
|
|
return hour - HOURS_IN_HALF_DAY;
|
|
|
|
|
@ -362,124 +502,160 @@ public class DateTimePicker extends FrameLayout {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set current hour in 24 hour mode, in the range (0~23)
|
|
|
|
|
*
|
|
|
|
|
* @param hourOfDay
|
|
|
|
|
* 设置当前选中的小时(24小时制,0~23)
|
|
|
|
|
* @param hourOfDay 要设置的小时数(0~23)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentHour(int hourOfDay) {
|
|
|
|
|
// 初始化中或小时未变化时,直接返回
|
|
|
|
|
if (!mInitialising && hourOfDay == getCurrentHourOfDay()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mDate.set(Calendar.HOUR_OF_DAY, hourOfDay);
|
|
|
|
|
|
|
|
|
|
// 12小时制下,更新上午/下午标记和选择器
|
|
|
|
|
if (!mIs24HourView) {
|
|
|
|
|
if (hourOfDay >= HOURS_IN_HALF_DAY) {
|
|
|
|
|
mIsAm = false;
|
|
|
|
|
mIsAm = false; // ≥12为下午
|
|
|
|
|
if (hourOfDay > HOURS_IN_HALF_DAY) {
|
|
|
|
|
hourOfDay -= HOURS_IN_HALF_DAY;
|
|
|
|
|
hourOfDay -= HOURS_IN_HALF_DAY; // 转换为12小时制(13→1,14→2...)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
mIsAm = true;
|
|
|
|
|
mIsAm = true; // <12为上午
|
|
|
|
|
if (hourOfDay == 0) {
|
|
|
|
|
hourOfDay = HOURS_IN_HALF_DAY;
|
|
|
|
|
hourOfDay = HOURS_IN_HALF_DAY; // 0点→12点
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
updateAmPmControl();
|
|
|
|
|
updateAmPmControl(); // 更新上午/下午选择器
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置小时选择器的值
|
|
|
|
|
mHourSpinner.setValue(hourOfDay);
|
|
|
|
|
// 触发时间变化回调
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get currentMinute
|
|
|
|
|
*
|
|
|
|
|
* @return The Current Minute
|
|
|
|
|
* 获取当前选中的分钟
|
|
|
|
|
* @return 分钟数(0~59)
|
|
|
|
|
*/
|
|
|
|
|
public int getCurrentMinute() {
|
|
|
|
|
return mDate.get(Calendar.MINUTE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set current minute
|
|
|
|
|
* 设置当前选中的分钟
|
|
|
|
|
* @param minute 要设置的分钟数(0~59)
|
|
|
|
|
*/
|
|
|
|
|
public void setCurrentMinute(int minute) {
|
|
|
|
|
// 初始化中或分钟未变化时,直接返回
|
|
|
|
|
if (!mInitialising && minute == getCurrentMinute()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mMinuteSpinner.setValue(minute);
|
|
|
|
|
mDate.set(Calendar.MINUTE, minute);
|
|
|
|
|
onDateTimeChanged();
|
|
|
|
|
mMinuteSpinner.setValue(minute); // 设置分钟选择器的值
|
|
|
|
|
mDate.set(Calendar.MINUTE, minute); // 更新日历对象
|
|
|
|
|
onDateTimeChanged(); // 触发时间变化回调
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 公共方法 - 24小时制适配 ========================
|
|
|
|
|
/**
|
|
|
|
|
* @return true if this is in 24 hour view else false.
|
|
|
|
|
* 判断当前是否为24小时制
|
|
|
|
|
* @return true=24小时制,false=12小时制
|
|
|
|
|
*/
|
|
|
|
|
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.
|
|
|
|
|
* 设置24小时制/12小时制
|
|
|
|
|
* @param is24HourView true=24小时制(隐藏AM/PM选择器),false=12小时制(显示AM/PM选择器)
|
|
|
|
|
*/
|
|
|
|
|
public void set24HourView(boolean is24HourView) {
|
|
|
|
|
if (mIs24HourView == is24HourView) {
|
|
|
|
|
return;
|
|
|
|
|
return; // 状态未变化,直接返回
|
|
|
|
|
}
|
|
|
|
|
mIs24HourView = is24HourView;
|
|
|
|
|
// 显示/隐藏上午/下午选择器
|
|
|
|
|
mAmPmSpinner.setVisibility(is24HourView ? View.GONE : View.VISIBLE);
|
|
|
|
|
// 获取当前小时数(用于重置选择器)
|
|
|
|
|
int hour = getCurrentHourOfDay();
|
|
|
|
|
// 更新小时选择器的取值范围
|
|
|
|
|
updateHourControl();
|
|
|
|
|
// 重新设置小时数(适配新的制式)
|
|
|
|
|
setCurrentHour(hour);
|
|
|
|
|
// 更新上午/下午选择器状态
|
|
|
|
|
updateAmPmControl();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 内部方法 - 控件更新 ========================
|
|
|
|
|
/**
|
|
|
|
|
* 更新日期选择器的展示文本:生成近7天的格式化日期(如“12.23 星期二”)
|
|
|
|
|
*/
|
|
|
|
|
private void updateDateControl() {
|
|
|
|
|
Calendar cal = Calendar.getInstance();
|
|
|
|
|
cal.setTimeInMillis(mDate.getTimeInMillis());
|
|
|
|
|
// 计算近7天的起始日期(当前日期 - 4天,确保中间为当前日期)
|
|
|
|
|
cal.add(Calendar.DAY_OF_YEAR, -DAYS_IN_ALL_WEEK / 2 - 1);
|
|
|
|
|
mDateSpinner.setDisplayedValues(null);
|
|
|
|
|
|
|
|
|
|
mDateSpinner.setDisplayedValues(null); // 清空原有展示文本
|
|
|
|
|
// 生成近7天的格式化日期文本
|
|
|
|
|
for (int i = 0; i < DAYS_IN_ALL_WEEK; ++i) {
|
|
|
|
|
cal.add(Calendar.DAY_OF_YEAR, 1);
|
|
|
|
|
// 格式化:月.日 星期(如“12.23 星期二”)
|
|
|
|
|
mDateDisplayValues[i] = (String) DateFormat.format("MM.dd EEEE", cal);
|
|
|
|
|
}
|
|
|
|
|
// 设置日期选择器的展示文本
|
|
|
|
|
mDateSpinner.setDisplayedValues(mDateDisplayValues);
|
|
|
|
|
// 设置默认选中中间项(当前日期)
|
|
|
|
|
mDateSpinner.setValue(DAYS_IN_ALL_WEEK / 2);
|
|
|
|
|
mDateSpinner.invalidate();
|
|
|
|
|
mDateSpinner.invalidate(); // 刷新选择器UI
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 更新上午/下午选择器的状态:设置选中项、控制可见性
|
|
|
|
|
*/
|
|
|
|
|
private void updateAmPmControl() {
|
|
|
|
|
if (mIs24HourView) {
|
|
|
|
|
mAmPmSpinner.setVisibility(View.GONE);
|
|
|
|
|
mAmPmSpinner.setVisibility(View.GONE); // 24小时制隐藏
|
|
|
|
|
} else {
|
|
|
|
|
// 设置选中项:0=AM/上午,1=PM/下午
|
|
|
|
|
int index = mIsAm ? Calendar.AM : Calendar.PM;
|
|
|
|
|
mAmPmSpinner.setValue(index);
|
|
|
|
|
mAmPmSpinner.setVisibility(View.VISIBLE);
|
|
|
|
|
mAmPmSpinner.setVisibility(View.VISIBLE); // 12小时制显示
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 更新小时选择器的取值范围:适配24/12小时制
|
|
|
|
|
*/
|
|
|
|
|
private void updateHourControl() {
|
|
|
|
|
if (mIs24HourView) {
|
|
|
|
|
// 24小时制:0~23
|
|
|
|
|
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW);
|
|
|
|
|
mHourSpinner.setMaxValue(HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW);
|
|
|
|
|
} else {
|
|
|
|
|
// 12小时制:1~12
|
|
|
|
|
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW);
|
|
|
|
|
mHourSpinner.setMaxValue(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
|
|
|
|
|
* 设置日期时间变化回调监听器
|
|
|
|
|
* @param callback 外部实现的监听器(null则不触发回调)
|
|
|
|
|
*/
|
|
|
|
|
public void setOnDateTimeChangedListener(OnDateTimeChangedListener callback) {
|
|
|
|
|
mOnDateTimeChangedListener = callback;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ======================== 内部方法 - 回调触发 ========================
|
|
|
|
|
/**
|
|
|
|
|
* 触发日期时间变化回调:传递最新的年/月/日/时/分
|
|
|
|
|
*/
|
|
|
|
|
private void onDateTimeChanged() {
|
|
|
|
|
if (mOnDateTimeChangedListener != null) {
|
|
|
|
|
mOnDateTimeChangedListener.onDateTimeChanged(this, getCurrentYear(),
|
|
|
|
|
getCurrentMonth(), getCurrentDay(), getCurrentHourOfDay(), getCurrentMinute());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|