You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
read/src/net/micode/notes/ui/DateTimePicker.java

520 lines
23 KiB

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

/*
* Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
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 { // 定义日期时间选择器类,继承自 FrameLayout
private static final boolean DEFAULT_ENABLE_STATE = true; // 默认启用状态为 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; // 24小时制下小时选择器的最小值
private static final int HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW = 23; // 24小时制下小时选择器的最大值
private static final int HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW = 1; // 12小时制下小时选择器的最小值
private static final int HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW = 12; // 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; // AM/PM选择器的最小值
private static final int AMPM_SPINNER_MAX_VAL = 1; // AM/PM选择器的最大值
private final NumberPicker mDateSpinner; // 日期选择器
private final NumberPicker mHourSpinner; // 小时选择器
private final NumberPicker mMinuteSpinner; // 分钟选择器
private final NumberPicker mAmPmSpinner; // AM/PM选择器
private Calendar mDate; // 存储日期的日历对象
private String[] mDateDisplayValues = new String[DAYS_IN_ALL_WEEK]; // 用于存储日期显示值的数组
private boolean mIsAm; // 存储当前是否为 AM 的标志
private boolean mIs24HourView; // 存储当前是否为24小时制的标志
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); // 更新日期
updateDateControl(); // 更新日期控制
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) { // 如果是 12 小时制
if (!mIsAm && oldVal == HOURS_IN_HALF_DAY - 1 && newVal == HOURS_IN_HALF_DAY) { // 从 PM 切换到 AM
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, 1); // 日期加 1
isDateChanged = true; // 标记日期变化
} else if (mIsAm && oldVal == HOURS_IN_HALF_DAY && newVal == HOURS_IN_HALF_DAY - 1) { // 从 AM 切换到 PM
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, -1); // 日期减 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; // 切换 AM/PM 状态
updateAmPmControl(); // 更新 AM/PM 控制
}
} else { // 如果是 24 小时制
if (oldVal == HOURS_IN_ALL_DAY - 1 && newVal == 0) { // 从 23 切换到 0
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, 1); // 日期加 1
isDateChanged = true; // 标记日期变化
} else if (oldVal == 0 && newVal == HOURS_IN_ALL_DAY - 1) { // 从 0 切换到 23
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, -1); // 日期减 1
isDateChanged = true; // 标记日期变化
}
}
// 设置小时
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));
setCurrentDay(cal.get(Calendar.DAY_OF_MONTH));
}
}
};
// 分钟选择器值变化的监听器
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; // 减少偏移量
}
if (offset != 0) { // 如果偏移量不为零
mDate.add(Calendar.HOUR_OF_DAY, offset); // 更新小时
mHourSpinner.setValue(getCurrentHour()); // 更新小时选择器的值
updateDateControl(); // 更新日期控制
int newHour = getCurrentHourOfDay(); // 获取当前小时
// 更新 AM/PM 状态
if (newHour >= HOURS_IN_HALF_DAY) {
mIsAm = false; // 设为 PM
updateAmPmControl(); // 更新 AM/PM 控制
} else {
mIsAm = true; // 设为 AM
updateAmPmControl(); // 更新 AM/PM 控制
}
}
mDate.set(Calendar.MINUTE, newVal); // 更新日历中的分钟
onDateTimeChanged(); // 通知日期时间已更改
}
};
// AM/PM 选择器值变化的监听器
private NumberPicker.OnValueChangeListener mOnAmPmChangedListener = new NumberPicker.OnValueChangeListener() {
@Override
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
mIsAm = !mIsAm; // 切换 AM/PM 状态
// 根据 AM/PM 状态更新小时
if (mIsAm) {
mDate.add(Calendar.HOUR_OF_DAY, -HOURS_IN_HALF_DAY); // 如果为 AM减去 12 小时
} else {
mDate.add(Calendar.HOUR_OF_DAY, HOURS_IN_HALF_DAY); // 如果为 PM加上 12 小时
}
updateAmPmControl(); // 更新 AM/PM 控制
onDateTimeChanged(); // 通知日期时间已更改
}
};
// 定义日期时间改变的监听器接口
public interface OnDateTimeChangedListener {
void onDateTimeChanged(DateTimePicker view, int year, int month,
int dayOfMonth, int hourOfDay, int minute); // 方法签名
}
// 构造函数,使用当前时间作为参数
public DateTimePicker(Context context) {
this(context, System.currentTimeMillis()); // 调用重载构造函数
}
public DateTimePicker(Context context, long date) {
this(context, date, DateFormat.is24HourFormat(context)); // 调用重载构造函数,并根据上下文判断是否为 24 小时制
}
public DateTimePicker(Context context, long date, boolean is24HourView) {
super(context); // 调用父类构造函数
mDate = Calendar.getInstance(); // 初始化当前日期
mInitialising = true; // 标记当前处于初始化状态
mIsAm = getCurrentHourOfDay() >= HOURS_IN_HALF_DAY; // 根据当前小时判断是否为 AM
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.setOnValueChangedListener(mOnMinuteChangedListener); // 设置值变化监听器
// 初始化 AM/PM 选择器
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.setOnValueChangedListener(mOnAmPmChangedListener); // 设置值变化监听器
// 更新控件到初始状态
updateDateControl(); // 更新日期选择器状态
updateHourControl(); // 更新小时选择器状态
updateAmPmControl(); // 更新 AM/PM 选择器状态
set24HourView(is24HourView); // 设置是否使用 24 小时制
// 设置为当前时间
setCurrentDate(date); // 设置当前日期为传入的日期
setEnabled(isEnabled()); // 设置选择器的启用状态
// 设置内容描述
mInitialising = false; // 标记初始化完成
}
@Override
public void setEnabled(boolean enabled) { // 重写设置启用状态的方法
if (mIsEnabled == enabled) { // 如果当前状态与目标状态相同,则返回
return;
}
super.setEnabled(enabled); // 调用父类的设置启用方法
// 设置各个选择器的启用状态
mDateSpinner.setEnabled(enabled);
mMinuteSpinner.setEnabled(enabled);
mHourSpinner.setEnabled(enabled);
mAmPmSpinner.setEnabled(enabled);
mIsEnabled = enabled; // 更新当前启用状态
}
@Override
public boolean isEnabled() { // 重写获取启用状态的方法
return mIsEnabled; // 返回当前启用状态
}
/**
* 获取当前日期的毫秒值
*
* @return 当前日期的毫秒值
*/
public long getCurrentDateInTimeMillis() {
return mDate.getTimeInMillis(); // 返回当前日期的毫秒表示
}
/**
* 设置当前日期
*
* @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));
}
/**
* 设置当前日期
*
* @param year 当前年份
* @param month 当前月份
* @param dayOfMonth 当前日
* @param hourOfDay 当前小时
* @param minute 当前分钟
*/
public void setCurrentDate(int year, int month,
int dayOfMonth, int hourOfDay, int minute) {
setCurrentYear(year); // 设置当前年份
setCurrentMonth(month); // 设置当前月份
setCurrentDay(dayOfMonth); // 设置当前日期
setCurrentHour(hourOfDay); // 设置当前小时
setCurrentMinute(minute); // 设置当前分钟
}
/**
* 获取当前年份
*
* @return 当前年份
*/
public int getCurrentYear() {
return mDate.get(Calendar.YEAR); // 返回当前年份
}
/**
* 设置当前年份
*
* @param year 当前年份
*/
public void setCurrentYear(int year) {
if (!mInitialising && year == getCurrentYear()) { // 如果不在初始化并且当前年份未变化,则返回
return;
}
mDate.set(Calendar.YEAR, year); // 更新日历中的年份
updateDateControl(); // 更新日期选择器控制
onDateTimeChanged(); // 通知日期时间已更改
}
/**
* 获取当前月份
*
* @return 当前月份
*/
public int getCurrentMonth() {
return mDate.get(Calendar.MONTH); // 返回当前月份
}
/**
* 设置当前月份
*
* @param month 当前月份
*/
public void setCurrentMonth(int month) {
if (!mInitialising && month == getCurrentMonth()) { // 如果不在初始化且当前月份未变化,则返回
return;
}
mDate.set(Calendar.MONTH, month); // 更新日历中的月份
updateDateControl(); // 更新日期控制
onDateTimeChanged(); // 通知日期时间已更改
}
/**
* 获取当前日期
*
* @return 当前日期
*/
public int getCurrentDay() {
return mDate.get(Calendar.DAY_OF_MONTH); // 返回当前日期
}
/**
* 设置当前日期
*
* @param dayOfMonth 当前日期
*/
public void setCurrentDay(int dayOfMonth) {
if (!mInitialising && dayOfMonth == getCurrentDay()) { // 如果不在初始化且当前日期未变化,则返回
return;
}
mDate.set(Calendar.DAY_OF_MONTH, dayOfMonth); // 更新日历中的日期
updateDateControl(); // 更新日期控制
onDateTimeChanged(); // 通知日期时间已更改
}
/**
* 获取当前小时
*
* @return 以 24 小时制表示的当前小时
*/
public int getCurrentHourOfDay() {
return mDate.get(Calendar.HOUR_OF_DAY); // 返回当前 24 小时制的小时
}
private int getCurrentHour() { // 获取当前小时
if (mIs24HourView){ // 如果是 24 小时制
return getCurrentHourOfDay(); // 返回 24 小时制的当前小时
} else {
int hour = getCurrentHourOfDay(); // 获取当前小时
// 转换为 12 小时制
if (hour > HOURS_IN_HALF_DAY) {
return hour - HOURS_IN_HALF_DAY;
} else {
return hour == 0 ? HOURS_IN_HALF_DAY : hour; // 处理 0 点的情况
}
}
}
/**
* 设置当前小时
*
* @param hourOfDay 当前小时
*/
public void setCurrentHour(int hourOfDay) {
if (!mInitialising && hourOfDay == getCurrentHourOfDay()) { // 如果不在初始化且当前小时未变化,则返回
return;
}
mDate.set(Calendar.HOUR_OF_DAY, hourOfDay); // 更新日历中的小时
if (!mIs24HourView) { // 如果不是 24 小时制
if (hourOfDay >= HOURS_IN_HALF_DAY) { // 如果属于 PM
mIsAm = false; // 设置为 PM
if (hourOfDay > HOURS_IN_HALF_DAY) { // 如果小时大于 12
hourOfDay -= HOURS_IN_HALF_DAY; // 转换为 12 小时制
}
} else { // 如果属于 AM
mIsAm = true; // 设置为 AM
if (hourOfDay == 0) { // 如果是 0 点
hourOfDay = HOURS_IN_HALF_DAY; // 转换为 12 点
}
}
updateAmPmControl(); // 更新 AM/PM 控制
}
mHourSpinner.setValue(hourOfDay); // 更新小时选择器的值
onDateTimeChanged(); // 通知日期时间已更改
}
/**
* 获取当前分钟
*
* @return 当前分钟
*/
public int getCurrentMinute() {
return mDate.get(Calendar.MINUTE); // 返回当前分钟
}
/**
* 设置当前分钟
*
* @param minute 当前分钟
*/
public void setCurrentMinute(int minute) {
if (!mInitialising && minute == getCurrentMinute()) { // 如果不在初始化且当前分钟未变化,则返回
return;
}
mMinuteSpinner.setValue(minute); // 更新分钟选择器的值
mDate.set(Calendar.MINUTE, minute); // 更新日历中的分钟
onDateTimeChanged(); // 通知日期时间已更改
}
/**
* @return 如果在 24 小时制下返回 true否则返回 false。
*/
public boolean is24HourView () {
return mIs24HourView; // 返回当前是否在 24 小时制
}
/**
* 设置是否使用 24 小时制或 AM/PM 模式。
*
* @param is24HourView true 为 24 小时制false 为 AM/PM 模式。
*/
public void set24HourView(boolean is24HourView) {
if (mIs24HourView == is24HourView) { // 如果当前状态与目标状态相同则返回
return;
}
mIs24HourView = is24HourView; // 更新当前模式
mAmPmSpinner.setVisibility(is24HourView ? View.GONE : View.VISIBLE); // 根据模式设置 AM/PM 选择器的可见性
int hour = getCurrentHourOfDay(); // 获取当前小时
updateHourControl(); // 更新小时选择器状态
setCurrentHour(hour); // 设置当前小时
updateAmPmControl(); // 更新 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(); // 刷新日期选择器
}
private void updateAmPmControl() { // 更新 AM/PM 选择器状态
if (mIs24HourView) { // 如果为 24 小时制
mAmPmSpinner.setVisibility(View.GONE); // 隐藏 AM/PM 选择器
} else {
int index = mIsAm ? Calendar.AM : Calendar.PM; // 根据 AM/PM 标志获取索引
mAmPmSpinner.setValue(index); // 设置 AM/PM 选择器的值
mAmPmSpinner.setVisibility(View.VISIBLE); // 显示 AM/PM 选择器
}
}
private void updateHourControl() { // 更新小时选择器状态
if (mIs24HourView) { // 如果为 24 小时制
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW); // 设置 24 小时制的最小值
mHourSpinner.setMaxValue(HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW); // 设置 24 小时制的最大值
} else {
mHourSpinner.setMinValue(HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW); // 设置 12 小时制的最小值
mHourSpinner.setMaxValue(HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW); // 设置 12 小时制的最大值
}
}
/**
* 设置回调函数,用于指示“设置”按钮已被按下。
* @param callback 回调函数,如果为 null 则不执行任何操作
*/
public void setOnDateTimeChangedListener(OnDateTimeChangedListener callback) {
mOnDateTimeChangedListener = callback; // 更新监听器
}
private void onDateTimeChanged() { // 日期时间更改的处理方法
if (mOnDateTimeChangedListener != null) { // 如果有设置的监听器
mOnDateTimeChangedListener.onDateTimeChanged(this, getCurrentYear(), // 调用监听器的方法,传递当前的日期和时间
getCurrentMonth(), getCurrentDay(), getCurrentHourOfDay(), getCurrentMinute());
}
}
}