注释代码4

pq5n3hobs 8 months ago
parent 881731d38a
commit 03df13f053

@ -0,0 +1,272 @@
/*
* 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;
// DateTimePicker类继承自FrameLayout意味着它可以作为一个容器来承载其他视图组件并且具备布局相关的特性
// 这个类主要用于实现一个日期和时间选择的自定义视图组件方便在Android应用中让用户选择特定的日期和时间
public class DateTimePicker extends FrameLayout {
// 定义默认的启用状态默认为启用true用于控制整个DateTimePicker组件及其内部子组件是否可用
private static final boolean DEFAULT_ENABLE_STATE = true;
// 半天包含的小时数用于12小时制时间相关的计算和逻辑判断例如区分上午和下午的时间范围
private static final int HOURS_IN_HALF_DAY = 12;
// 一整天包含的小时数用于24小时制时间相关的操作和判断
private static final int HOURS_IN_ALL_DAY = 24;
// 一周包含的天数,用于日期选择器相关的范围设置和数据处理等操作
private static final int DAYS_IN_ALL_WEEK = 7;
// 日期选择器NumberPicker的最小值通常从0开始对应一周七天的索引等情况
private static final int DATE_SPINNER_MIN_VAL = 0;
// 日期选择器NumberPicker的最大值设置为一周天数减1因为索引从0开始
private static final int DATE_SPINNER_MAX_VAL = DAYS_IN_ALL_WEEK - 1;
// 24小时制视图下小时选择器NumberPicker的最小值范围是0到23小时
private static final int HOUR_SPINNER_MIN_VAL_24_HOUR_VIEW = 0;
// 24小时制视图下小时选择器NumberPicker的最大值范围是0到23小时
private static final int HOUR_SPINNER_MAX_VAL_24_HOUR_VIEW = 23;
// 12小时制视图下小时选择器NumberPicker的最小值通常从1开始上午1点
private static final int HOUR_SPINNER_MIN_VAL_12_HOUR_VIEW = 1;
// 12小时制视图下小时选择器NumberPicker的最大值到12点结束中午12点或者晚上12点
private static final int HOUR_SPINNER_MAX_VAL_12_HOUR_VIEW = 12;
// 分钟选择器NumberPicker的最小值分钟范围从0开始
private static final int MINUT_SPINNER_MIN_VAL = 0;
// 分钟选择器NumberPicker的最大值到59分钟结束
private static final int MINUT_SPINNER_MAX_VAL = 59;
// AM/PM选择器NumberPicker的最小值通常0表示上午AM对应相关逻辑和显示设置
private static final int AMPM_SPINNER_MIN_VAL = 0;
// AM/PM选择器NumberPicker的最大值通常1表示下午PM对应相关逻辑和显示设置
private static final int AMPM_SPINNER_MAX_VAL = 1;
// 用于显示日期的NumberPicker组件用户可以通过它选择具体的日期以一周内某天的形式呈现
private final NumberPicker mDateSpinner;
// 用于显示小时的NumberPicker组件根据设置的是24小时制还是12小时制其显示范围和逻辑有所不同
private final NumberPicker mHourSpinner;
// 用于显示分钟的NumberPicker组件分钟范围固定从0到59
private final NumberPicker mMinuteSpinner;
// 用于显示上午AM或下午PM的NumberPicker组件仅在12小时制下可见并起作用
private final NumberPicker mAmPmSpinner;
// 用于存储当前选择的日期和时间信息的Calendar对象方便进行日期和时间的计算、获取以及设置等操作
private Calendar mDate;
// 用于存储要在日期选择器中显示的一周七天的字符串表示形式,例如 "周一"、"周二" 等
private String[] mDateDisplayValues = new String[DAYS_IN_ALL_WEEK];
// 用于标记当前时间是上午true还是下午false在12小时制下使用初始值根据当前时间判断
private boolean mIsAm;
// 用于标记是否处于24小时制视图模式true表示24小时制false表示12小时制决定了小时选择器等相关组件的显示和操作逻辑
private boolean mIs24HourView;
// 用于记录整个DateTimePicker组件当前的启用状态初始化为默认启用状态
private boolean mIsEnabled = DEFAULT_ENABLE_STATE;
// 用于标记是否正在初始化组件的过程中,在一些设置逻辑中避免不必要的重复操作或者冲突
private boolean mInitialising;
// 定义一个接口类型的成员变量,用于设置当日期和时间发生改变时的回调监听器,外部类可以实现该接口来响应时间日期的变化
private OnDateTimeChangedListener mOnDateTimeChangedListener;
// 为日期选择器mDateSpinner设置的监听器当日期选择器的值发生改变时会触发此监听器的onValueChange方法
// 在此方法中,会根据日期选择器值的变化来更新对应的日期信息,同时通知其他相关组件更新显示,并触发整体的日期时间改变回调(如果有设置)
private NumberPicker.OnValueChangeListener mOnDateChangedListener = new NumberPicker.OnValueChangeListener() {
@Override
public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
// 根据日期选择器新值与旧值的差值调整Calendar对象mDate中的日期实现日期的增减操作
mDate.add(Calendar.DAY_OF_YEAR, newVal - oldVal);
// 更新日期相关的显示控制,例如更新日期选择器中显示的具体日期字符串等
updateDateControl();
// 触发日期时间改变的回调方法,通知外部监听者日期时间已发生变化
onDateTimeChanged();
}
};
// 为小时选择器mHourSpinner设置的监听器当小时选择器的值发生改变时会触发此监听器的onValueChange方法
// 此方法内包含了复杂的逻辑来处理不同时间制12小时制和24小时制下小时变化时日期、上午/下午标记以及其他相关组件显示和值的更新情况
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小时制下如果当前不是上午且旧值是11半天的最后一小时新值变为12中午12点进入下午可能涉及日期变化
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;
}
// 在12小时制下如果当前是上午且旧值是12中午12点新值变为11上午11点回到上午可能涉及日期变化
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();
}
} else {
// 在24小时制下如果旧值是23一天的最后一小时新值变为0进入新的一天涉及日期变化
if (oldVal == HOURS_IN_ALL_DAY - 1 && newVal == 0) {
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, 1);
isDateChanged = true;
}
// 在24小时制下如果旧值是0凌晨0点新值变为23回到前一天的最后一小时涉及日期变化
else if (oldVal == 0 && newVal == HOURS_IN_ALL_DAY - 1) {
cal.setTimeInMillis(mDate.getTimeInMillis());
cal.add(Calendar.DAY_OF_YEAR, -1);
isDateChanged = true;
}
}
// 根据当前是否为24小时制以及上午/下午状态计算并设置当前的小时数到Calendar对象mDate
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));
}
}
};
// 为分钟选择器mMinuteSpinner设置的监听器当分钟选择器的值发生改变时会触发此监听器的onValueChange方法
// 此方法中会根据分钟值的变化情况,处理可能涉及的小时、日期以及上午/下午等相关状态的更新,并触发日期时间改变回调
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();
if (newHour >= HOURS_IN_HALF_DAY) {
mIsAm = false;
updateAmPmControl();
} else {
mIsAm = true;
updateAmPmControl();
}
}
mDate.set(Calendar.MINUTE, newVal);
onDateTimeChanged();
}
};
// 为AM/PM选择器mAmPmSpinner设置的监听器当AM/PM选择器的值发生改变时会触发此监听器的onValueChange方法
// 此方法中会根据选择的上午/下午变化,相应地调整日期时间中的小时数,并更新相关组件显示,触发日期时间改变回调
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();
}
};
// 定义一个接口用于外部类实现当DateTimePicker中的日期和时间发生改变时会回调此接口的onDateTimeChanged方法
// 外部类可以通过实现该接口获取到日期时间变化的具体信息,并进行相应的业务处理,比如更新显示、保存数据等操作
public interface OnDateTimeChangedListener {
void onDateTimeChanged(DateTimePicker view, int year, int month,
int dayOfMonth, int hourOfDay, int minute);
}
// 构造方法调用另一个带有默认当前时间System.currentTimeMillis()的构造方法来初始化DateTimePicker组件
public DateTimePicker(Context context) {
this(context, System.currentTimeMillis());
}
// 构造方法调用另一个带有是否为24小时制参数DateFormat.is24HourFormat(context)的构造方法来初始化DateTimePicker组件
// 传入的参数date用于设置初始的日期时间以毫秒数表示通常是从某个时间原点开始的时间戳
public DateTimePicker(Context context, long date) {
this(context, date, DateFormat.is24HourFormat(context));
}
// 主要的构造方法用于初始化DateTimePicker组件的各个属性、视图组件以及设置初始的日期时间、时间制等相关状态
public DateTimePicker(Context context, long date, boolean is24HourView) {
super(context);
// 获取一个表示当前时间的Calendar实例后续用于设置和操作日期时间相关信息
mDate = Calendar.getInstance();
mInitialising = true;
// 根据当前小时数判断是上午还是下午用于初始化12小时制下的初始上午/下午状态)
mIsAm = getCurrentHourOfDay() >= HOURS_IN_HALF_DAY;
// 加载布局文件R.layout.datetime_picker到当前的DateTimePicker组件中该布局文件应该定义了包含日期、时间选择器等相关视图的结构
inflate(context, R.layout.datetime_picker, this);
// 获取布局文件中定义的日期选择器NumberPicker组件实例
mDateSpinner = (NumberPicker) findViewById(R.id.date);
// 设置日期选择器的最小值和最大值,对应一周七天的范围
mDateSpinner.setMinValue(DATE_SPINNER_MIN_VAL);
mDateSpinner.setMaxValue(DATE_SPINNER_MAX_VAL);
// 为日期选择器设置值改变监听器,当用户选择不同日期时触发相应逻辑
mDateSpinner.setOnValueChangedListener(mOnDateChangedListener);
// 获取布局文件中定义的小时选择器NumberPicker组件实例
mHourSpinner = (NumberPicker) findViewById(R.id.hour);
// 为小时选择器设置值改变监听器,用于处理小时值变化时的各种逻辑
mHourSpinner.setOnValueChangedListener(mOnHourChangedListener);
// 获取布局文件中定义的分钟选择器NumberPicker组件实例
mMinuteSpinner = (NumberPicker) findViewById(R.id.minute);
// 设置分钟选择器的最小值和最大值对应分钟的0到59范围
mMinuteSpinner.setMinValue(MINUT_SPINNER_MIN_VAL);
mMinuteSpinner.setMaxValue(MINUT_SPINNER_MAX_VAL);
// 设置长按更新间隔这里设置为100毫秒用于控制长按操作时的数值更新频率具体根据实际需求和用户体验调整
mMinuteSpinner.setOnLongPressUpdateInterval(100);
// 为分钟选择器设置值改变监听器,处理分钟值变化的相关逻辑
mMinuteSpinner.setOnValueChangedListener(mOnMinuteChangedListener);
// 获取用于表示上午AM和下午PM的字符串数组用于在AM/PM选择器中显示
String[] stringsForAmPm = new DateFormatSymbols().getAmPmStrings();
// 获取布局文件中定义的AM/PM选择器NumberPicker组件实例
mAmPmSpinner = (NumberPicker) findViewById(R.id.amPm);
// 设置AM/PM选择器的最小值和最大值对应上午0和下午1的表示
mAmPmSpinner.setMinValue(AMPM_SPINNER_MIN_VAL);
mAmPmSpinner.setMaxValue(AMPM_SPINNER_MAX_VAL);
// 设置AM/PM选择器要显示的字符串值即上午AM和下午PM的文本表示
mAmPmSpinner.setDisplayedValues(stringsForAmPm);
// 为AM/PM选择器设置值改变监听器处理上午/下午选择变化的相关逻辑
mAmPm
Loading…
Cancel
Save