Java 自定义 SpinnerDateModel 以仅编辑小时和分钟
Java Custom SpinnerDateModel to edit only hour and minute
我正在尝试创建一个小时和分钟的 JSpinner 编辑器。 JSpinner 很简单
JSpinner startSpinner = new JSpinner();
SpinnerWindowModel spinnerModel = new SpinnerWindowModel();
startSpinner.setModel(spinnerModel);
JSpinner.DateEditor editor = new JSpinner.DateEditor(startSpinner, "HH:mm");
startSpinner.setEditor(editor);
Spinner WindowModel 是:
public SpinnerWindowModel() {
super();
Calendar vCal = Calendar.getInstance();
vCal.set(Calendar.HOUR_OF_DAY, 8);
vCal.set(Calendar.MINUTE, 0);
vCal.set(Calendar.SECOND, 0);
setValue(vCal.getTime());
setCalendarField(Calendar.MINUTE);
vCal.set(Calendar.HOUR_OF_DAY, 6);
vCal.set(Calendar.MINUTE, 0);
vCal.set(Calendar.SECOND, 0);
setStart(vCal.getTime());
vCal.set(Calendar.HOUR_OF_DAY, 18);
vCal.set(Calendar.MINUTE, 45);
vCal.set(Calendar.SECOND, 0);
setEnd(vCal.getTime());
}
@Override
public Object getPreviousValue() {
Object obj = super.getPreviousValue();
int fld = getCalendarField();
System.out.println("GPV: " + fld);
return obj;
}
@Override
public Object getNextValue() {
Object obj = super.getNextValue();
int fld = getCalendarField();
System.out.println("GNV: " + fld);
return obj;
}
这两个方法getNext/PreviousValue目前只监控是调用了哪些方法,是在什么字段上调用的。使用受限编辑器,这些方法不会被调用
如果我将编辑器更改为一组更完整的字段,它会像我预期的那样工作。小时和分钟字段是可编辑且正确的。我需要做些什么来编辑简化的编辑器吗?
editor = new JSpinner.DateEditor(startSpinner, "dd/MM/yyyy HH:mm:ss.SS");
所以,当我使用你的模型时,它不起作用,但是当我使用诸如...
Calendar cal = Calendar.getInstance();
cal.set(Calendar.HOUR_OF_DAY, 0);
cal.set(Calendar.MINUTE, 0);
Date startTime = cal.getTime();
cal.set(Calendar.HOUR, 23);
cal.set(Calendar.MINUTE, 59);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
// The Calendar field seems to get ignored and overriden
// by the UI delegate based on what part of the field
// the cursor is current on (hour or minute)
SpinnerDateModel model = new SpinnerDateModel(startTime, null, endTime, Calendar.MINUTE);
JSpinner spinner = new JSpinner(model);
spinner.setEditor(new JSpinner.DateEditor(spinner, "HH:mm"));
spinner.setValue(startTime);
效果很好
此时我的“建议”是不使用自定义模型,而是通过所需的工厂方法手动构建模型
已更新...
由于围绕 JSpinner
和 JForamttedTextField
的复杂性,我放弃了让 JSpinner
处理时间的尝试。相反,我通常会退回到使用单独的字段来管理状态的复杂性
我从修改你的模型开始...
public class SpinnerWindowModel extends SpinnerNumberModel {
private Calendar calendar;
private int calendarField;
public SpinnerWindowModel(int calendarField, Calendar calendar, int value, int min, int max, int step) {
super(value, min, max, step);
this.calendar = calendar;
}
@Override
public void setValue(Object value) {
super.setValue(value);
System.out.println(value);
}
public int getCalendarField() {
return calendarField;
}
@Override
public Object getPreviousValue() {
Object obj = super.getPreviousValue();
if (obj instanceof Integer) {
int fld = getCalendarField();
calendar.set(fld, (int)obj);
return obj;
}
return getValue();
}
@Override
public Object getNextValue() {
Object obj = super.getNextValue();
if (obj instanceof Integer) {
int fld = getCalendarField();
calendar.set(fld, (int)obj);
return obj;
}
return getValue();
}
}
然后将其应用于字段...
Calendar cal = Calendar.getInstance();
cal.set(Calendar.HOUR_OF_DAY, 0);
cal.set(Calendar.MINUTE, 0);
Date startTime = cal.getTime();
cal.set(Calendar.HOUR, 23);
cal.set(Calendar.MINUTE, 59);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
SpinnerWindowModel hourModel = new SpinnerWindowModel(Calendar.HOUR, cal, 0, 0, 23, 1);
SpinnerWindowModel minuteModel = new SpinnerWindowModel(Calendar.HOUR, cal, 0, 0, 59, 15);
JSpinner hour = new JSpinner(hourModel);
JSpinner minute = new JSpinner(minuteModel);
add(hour);
add(new JLabel(":"));
add(minute);
现在,这样做的缺点是,分钟字段不会滚动小时字段。我确实考虑过在分钟模型上使用更改侦听器,但问题是,你真的不知道该字段是在上升还是下降,所以没有简单的方法来确定你应该以哪种方式滚动小时。
我会扩展这个概念,使用两个单独的 JTextField
s(应用适当的 DocumentFilter
s)和一个“滚动”控件。想法是,滚动控件(即向上和向下按钮)会影响聚焦区域。然后,您可以在触发滚动控制时提供适当的状态信息,以允许字段确定它们应该做什么以及它们应该如何相互影响。
其中很多可以汇总到一个模型中,该模型控制一个 Calendar
(或者最好是一个 LocalTime
)对象,该对象将更改基础日期值并通知字段它们需要改变。
我相信这种方法会为您提供所需的控制级别,为小时和分钟提供不同的步进速率,同时为用户交互提供灵活性(我可以手动输入数字或通过滚动控件输入数字,如果我想)
更新了……终于……
所以,基于this answer,底层JFormattedTextField
无法验证该值,因为时间格式正在丢弃值的“日期”部分,只保留这个时间,这是套管 min/max 值检查失败。
相反,我们需要将 min/max 值的“日期”部分定义为我们可以控制(并且更容易定义)的内容。这也意味着任何输入都必须限制在这个特定的时间点
Calendar cal = Calendar.getInstance();
cal.clear();
cal.set(1970, Calendar.JANUARY, 0, 0, 0);
cal.set(Calendar.MILLISECOND, 0);
Date startTime = cal.getTime();
cal.set(1970, Calendar.JANUARY, 23, 59, 0);
cal.set(Calendar.MILLISECOND, 0);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
Calendar value = Calendar.getInstance();
value.set(1970, Calendar.JANUARY, 0, 0, 0);
value.set(Calendar.MILLISECOND, 0);
System.out.println(value.getTime());
// The Calendar field seems to get ignored and overriden
// by the UI delegate based on what part of the field
// the cursor is current on (hour or minute)
SpinnerDateModel model = new SpinnerDateModel(value.getTime(), startTime, endTime, Calendar.MINUTE);
JSpinner spinner = new JSpinner(model);
spinner.setEditor(new JSpinner.DateEditor(spinner, "HH:mm"));
spinner.setValue(value.getTime());
add(spinner);
然后让我更新您的模型以支持多步状态...
public class SpinnerWindowModel extends SpinnerDateModel {
public SpinnerWindowModel() {
super();
Calendar cal = Calendar.getInstance();
cal.clear();
cal.set(1970, Calendar.JANUARY, 0, 0, 0);
cal.set(Calendar.MILLISECOND, 0);
Date startTime = cal.getTime();
cal.set(1970, Calendar.JANUARY, 23, 59, 0);
cal.set(Calendar.MILLISECOND, 0);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
Calendar value = Calendar.getInstance();
value.set(1970, Calendar.JANUARY, 0, 0, 0);
value.set(Calendar.MILLISECOND, 0);
setStart(startTime);
setEnd(endTime);
setValue(value.getTime());
setCalendarField(Calendar.MINUTE);
}
@Override
public Object getPreviousValue() {
int fld = getCalendarField();
Object value = super.getPreviousValue();
if (fld == Calendar.MINUTE) {
Calendar cal = Calendar.getInstance();
cal.setTime(getDate());
cal.add(Calendar.MINUTE, -15);
value = cal.getTime();
}
return value;
}
@Override
public Object getNextValue() {
int fld = getCalendarField();
Object value = super.getNextValue();
if (fld == Calendar.MINUTE) {
Calendar cal = Calendar.getInstance();
cal.setTime(getDate());
cal.add(Calendar.MINUTE, 15);
value = cal.getTime();
}
return value;
}
}
我正在尝试创建一个小时和分钟的 JSpinner 编辑器。 JSpinner 很简单
JSpinner startSpinner = new JSpinner();
SpinnerWindowModel spinnerModel = new SpinnerWindowModel();
startSpinner.setModel(spinnerModel);
JSpinner.DateEditor editor = new JSpinner.DateEditor(startSpinner, "HH:mm");
startSpinner.setEditor(editor);
Spinner WindowModel 是:
public SpinnerWindowModel() {
super();
Calendar vCal = Calendar.getInstance();
vCal.set(Calendar.HOUR_OF_DAY, 8);
vCal.set(Calendar.MINUTE, 0);
vCal.set(Calendar.SECOND, 0);
setValue(vCal.getTime());
setCalendarField(Calendar.MINUTE);
vCal.set(Calendar.HOUR_OF_DAY, 6);
vCal.set(Calendar.MINUTE, 0);
vCal.set(Calendar.SECOND, 0);
setStart(vCal.getTime());
vCal.set(Calendar.HOUR_OF_DAY, 18);
vCal.set(Calendar.MINUTE, 45);
vCal.set(Calendar.SECOND, 0);
setEnd(vCal.getTime());
}
@Override
public Object getPreviousValue() {
Object obj = super.getPreviousValue();
int fld = getCalendarField();
System.out.println("GPV: " + fld);
return obj;
}
@Override
public Object getNextValue() {
Object obj = super.getNextValue();
int fld = getCalendarField();
System.out.println("GNV: " + fld);
return obj;
}
这两个方法getNext/PreviousValue目前只监控是调用了哪些方法,是在什么字段上调用的。使用受限编辑器,这些方法不会被调用
如果我将编辑器更改为一组更完整的字段,它会像我预期的那样工作。小时和分钟字段是可编辑且正确的。我需要做些什么来编辑简化的编辑器吗?
editor = new JSpinner.DateEditor(startSpinner, "dd/MM/yyyy HH:mm:ss.SS");
所以,当我使用你的模型时,它不起作用,但是当我使用诸如...
Calendar cal = Calendar.getInstance();
cal.set(Calendar.HOUR_OF_DAY, 0);
cal.set(Calendar.MINUTE, 0);
Date startTime = cal.getTime();
cal.set(Calendar.HOUR, 23);
cal.set(Calendar.MINUTE, 59);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
// The Calendar field seems to get ignored and overriden
// by the UI delegate based on what part of the field
// the cursor is current on (hour or minute)
SpinnerDateModel model = new SpinnerDateModel(startTime, null, endTime, Calendar.MINUTE);
JSpinner spinner = new JSpinner(model);
spinner.setEditor(new JSpinner.DateEditor(spinner, "HH:mm"));
spinner.setValue(startTime);
效果很好
此时我的“建议”是不使用自定义模型,而是通过所需的工厂方法手动构建模型
已更新...
由于围绕 JSpinner
和 JForamttedTextField
的复杂性,我放弃了让 JSpinner
处理时间的尝试。相反,我通常会退回到使用单独的字段来管理状态的复杂性
我从修改你的模型开始...
public class SpinnerWindowModel extends SpinnerNumberModel {
private Calendar calendar;
private int calendarField;
public SpinnerWindowModel(int calendarField, Calendar calendar, int value, int min, int max, int step) {
super(value, min, max, step);
this.calendar = calendar;
}
@Override
public void setValue(Object value) {
super.setValue(value);
System.out.println(value);
}
public int getCalendarField() {
return calendarField;
}
@Override
public Object getPreviousValue() {
Object obj = super.getPreviousValue();
if (obj instanceof Integer) {
int fld = getCalendarField();
calendar.set(fld, (int)obj);
return obj;
}
return getValue();
}
@Override
public Object getNextValue() {
Object obj = super.getNextValue();
if (obj instanceof Integer) {
int fld = getCalendarField();
calendar.set(fld, (int)obj);
return obj;
}
return getValue();
}
}
然后将其应用于字段...
Calendar cal = Calendar.getInstance();
cal.set(Calendar.HOUR_OF_DAY, 0);
cal.set(Calendar.MINUTE, 0);
Date startTime = cal.getTime();
cal.set(Calendar.HOUR, 23);
cal.set(Calendar.MINUTE, 59);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
SpinnerWindowModel hourModel = new SpinnerWindowModel(Calendar.HOUR, cal, 0, 0, 23, 1);
SpinnerWindowModel minuteModel = new SpinnerWindowModel(Calendar.HOUR, cal, 0, 0, 59, 15);
JSpinner hour = new JSpinner(hourModel);
JSpinner minute = new JSpinner(minuteModel);
add(hour);
add(new JLabel(":"));
add(minute);
现在,这样做的缺点是,分钟字段不会滚动小时字段。我确实考虑过在分钟模型上使用更改侦听器,但问题是,你真的不知道该字段是在上升还是下降,所以没有简单的方法来确定你应该以哪种方式滚动小时。
我会扩展这个概念,使用两个单独的 JTextField
s(应用适当的 DocumentFilter
s)和一个“滚动”控件。想法是,滚动控件(即向上和向下按钮)会影响聚焦区域。然后,您可以在触发滚动控制时提供适当的状态信息,以允许字段确定它们应该做什么以及它们应该如何相互影响。
其中很多可以汇总到一个模型中,该模型控制一个 Calendar
(或者最好是一个 LocalTime
)对象,该对象将更改基础日期值并通知字段它们需要改变。
我相信这种方法会为您提供所需的控制级别,为小时和分钟提供不同的步进速率,同时为用户交互提供灵活性(我可以手动输入数字或通过滚动控件输入数字,如果我想)
更新了……终于……
所以,基于this answer,底层JFormattedTextField
无法验证该值,因为时间格式正在丢弃值的“日期”部分,只保留这个时间,这是套管 min/max 值检查失败。
相反,我们需要将 min/max 值的“日期”部分定义为我们可以控制(并且更容易定义)的内容。这也意味着任何输入都必须限制在这个特定的时间点
Calendar cal = Calendar.getInstance();
cal.clear();
cal.set(1970, Calendar.JANUARY, 0, 0, 0);
cal.set(Calendar.MILLISECOND, 0);
Date startTime = cal.getTime();
cal.set(1970, Calendar.JANUARY, 23, 59, 0);
cal.set(Calendar.MILLISECOND, 0);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
Calendar value = Calendar.getInstance();
value.set(1970, Calendar.JANUARY, 0, 0, 0);
value.set(Calendar.MILLISECOND, 0);
System.out.println(value.getTime());
// The Calendar field seems to get ignored and overriden
// by the UI delegate based on what part of the field
// the cursor is current on (hour or minute)
SpinnerDateModel model = new SpinnerDateModel(value.getTime(), startTime, endTime, Calendar.MINUTE);
JSpinner spinner = new JSpinner(model);
spinner.setEditor(new JSpinner.DateEditor(spinner, "HH:mm"));
spinner.setValue(value.getTime());
add(spinner);
然后让我更新您的模型以支持多步状态...
public class SpinnerWindowModel extends SpinnerDateModel {
public SpinnerWindowModel() {
super();
Calendar cal = Calendar.getInstance();
cal.clear();
cal.set(1970, Calendar.JANUARY, 0, 0, 0);
cal.set(Calendar.MILLISECOND, 0);
Date startTime = cal.getTime();
cal.set(1970, Calendar.JANUARY, 23, 59, 0);
cal.set(Calendar.MILLISECOND, 0);
Date endTime = cal.getTime();
System.out.println(startTime);
System.out.println(endTime);
Calendar value = Calendar.getInstance();
value.set(1970, Calendar.JANUARY, 0, 0, 0);
value.set(Calendar.MILLISECOND, 0);
setStart(startTime);
setEnd(endTime);
setValue(value.getTime());
setCalendarField(Calendar.MINUTE);
}
@Override
public Object getPreviousValue() {
int fld = getCalendarField();
Object value = super.getPreviousValue();
if (fld == Calendar.MINUTE) {
Calendar cal = Calendar.getInstance();
cal.setTime(getDate());
cal.add(Calendar.MINUTE, -15);
value = cal.getTime();
}
return value;
}
@Override
public Object getNextValue() {
int fld = getCalendarField();
Object value = super.getNextValue();
if (fld == Calendar.MINUTE) {
Calendar cal = Calendar.getInstance();
cal.setTime(getDate());
cal.add(Calendar.MINUTE, 15);
value = cal.getTime();
}
return value;
}
}