JSpinner詳解

今天要做一個類似Windows的時間屬性的東西,原本以爲是一個JComboBox中間放了一個JFormattedTextField,但是baidu之後,發現了一個更好用的,那就是JSpinner,哈哈。轉一個帖子,寫的很詳細。原來的地址是:http://www.bitscn.com/java/swing/200605/20654.html

JSpinner工作起來好像是在JList或者JComboBox中間放了一個JFormattedTextField。在JList或者JComboBox中,用戶可以提前設定好要輸入的值。JSpinner也提供這樣的一種機制。這個控件的另一個部分是JFormattedTextField。如何顯示和輸入不由那些小的控制格控制,比如JList。相反,可以通過JFormattedTextField來輸入或通過邊上的兩個小箭頭來瀏覽不同的可用的值。
用戶使用組件或鍵盤上的上移和下移箭頭進行選擇。他們也可輸入自己的選擇。然而,與 JComboBox 不同,JSpinner 不提供下拉列表選擇,所以各個選擇以及它們的順序應有一定的意義。

  圖示1顯示了Spinner伴隨不同的輸入類型是什麼樣子。圖示1的頂端的JSpinner是一個用來顯示法語星期,通過SpinnerListModel。中間的,是一個通過SpinnerDateModel顯示日期的JSpinner。底部的是使用SpinnerNumberModel的JSpinner。每一個都是通過各自神祕的方式,在本文的後面我們將要學習。

[轉載]Java <wbr>Swing之:好用的JSpinner

圖示1.JSpinner實例

要創建和操縱JSpinner,許多類都將被調用,最重要的是JSpinner自己。最重要的兩個準素集包括SpinnerModel接口,包括可選擇的集合中的選項,還有,JSpinner.DefaultEditor的實現,用來捕獲所有選擇。慶幸的是,許多其它調用的類都是在後臺工作的,比如,一旦你給SpinnerNumberModel提供了數字的範圍,並且用這個類來協助Spinner,你的工作實際上是完成了。


創建JSpinner控件

  JSpinner類包括兩個構造函數來初始化控件:

  public JSpinner()
  JSpinner spinner = new JSpinner();
  public JSpinner(SpinnerModel model)
  SpinnerModel model = new SpinnerListModel(args);
  JSpinner spinner = new JSpinner(model);

  開始的時候可以沒有數據模型,後面可以使用它來跟蹤JSpinner的方法。另一個方法,在創建這個控件的時候使用完整的模型,實現SpinnerModel接口,它裏面有三個具體的子類可以使用:SpinnerDateModel,SpinnerListModel和SpinnerNumberModel,伴隨着他們的抽象父類AbstractSpinnerModel。如果不指名模型,那麼SpinnerNumberModel將默認使用。而顯示和編輯的控件是JFormattedTextField,編輯的基本功能是通過一系列JSpinner的內部類實現的:DateEditor,ListEditor和NumberFormat,還有父類中DefaultEditor的支持。

JSpinner屬性

  除了創建JSpinner對象之外,你還可以通過表一中的九個屬性中的一個來進行配置。

[轉載]Java <wbr>Swing之:好用的JSpinner

Table 1. JSpinner 屬性

value屬性中的值允許你更改當前控件的設置,nextValue和perviousValue可以使你以不同的方向察看模型中的入口。

  使用ChangeListener來監聽JSpinner events

  JSpinner直接支持一種事件監聽:changeListener。在別的地方,當commitEdit()方法被調用,這個事件將被觸發,告訴你spinner的值發生改變。爲了證明,列表1聯繫到一個自定義的ChangeListener,與圖示1的程序相關聯。

  列表 1. JSpinner with ChangeListener
  import java.awt.*;
  import javax.swing.*;
  import javax.swing.event.*;
  import java.text.*;
  import java.util.*;
  public class SpinnerSample {
  public static void main (String args[]) {
  Runnable runner = new Runnable() {
  public void run() {
  JFrame frame = new JFrame("JSpinner Sample");
  frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  DateFormatSymbols symbols =
  new DateFormatSymbols(Locale.FRENCH);

  ChangeListener listener = new ChangeListener() {
  public void stateChanged(ChangeEvent e) {
  System.out.println("Source: " + e.getSource());
  }
  };
  String days[] = symbols.getWeekdays();
  SpinnerModel model1 = new SpinnerListModel(days);
  JSpinner spinner1 = new JSpinner(model1);
  spinner1.addChangeListener(listener);
  JLabel label1 = new JLabel("French Days/List");
  JPanel panel1 = new JPanel(new BorderLayout());
  panel1.add(label1, BorderLayout.WEST);
  panel1.add(spinner1, BorderLayout.CENTER);
  frame.add(panel1, BorderLayout.NORTH);
  SpinnerModel model2 = new SpinnerDateModel();
  JSpinner spinner2 = new JSpinner(model2);
  spinner2.addChangeListener(listener);
  JLabel label2 = new JLabel("Dates/Date");
  JPanel panel2 = new JPanel(new BorderLayout());
  panel2.add(label2, BorderLayout.WEST);
  panel2.add(spinner2, BorderLayout.CENTER);
  frame.add(panel2, BorderLayout.CENTER);
  SpinnerModel model3 = new SpinnerNumberModel();
  JSpinner spinner3 = new JSpinner(model3);
  spinner3.addChangeListener(listener);
  JLabel label3 = new JLabel("Numbers");
  JPanel panel3 = new JPanel(new BorderLayout());
  panel3.add(label3, BorderLayout.WEST);
  panel3.add(spinner3, BorderLayout.CENTER);
  frame.add(panel3, BorderLayout.SOUTH);
  frame.setSize(200, 90);
  frame.setVisible (true);
  }
  };
  EventQueue.invokeLater(runner);
  }

  運行這個程序可以示範listener的用法(當然,你也會發現更多關於ChangeListener的有意義的方法)。

定製JSpinner的外觀

  同所有的Swing控件,JSpinner在不同的系統定義look-and-feel類型下,擁有不同的外觀,如示圖2。這個控件期初看起來像一個textfield,不同點是繪製了兩個箭頭。


[轉載]Java <wbr>Swing之:好用的JSpinner

示圖 2. JSpinner under different look-and-feel types

集合中的11個UIResource屬性在表格2中列舉,有限的方法繪製text field和箭頭。

[轉載]Java <wbr>Swing之:好用的JSpinner

Table 2. JSpinner UIResource 元素


SpinnerModel 接口

  到目前,我們已經看到了如何同一個主JSpinner類連接,SpinnerModel接口是控件的數據模型,SpinnerModel的定義如下:

  public interface SpinnerModel {
  // Properties
  public Object getValue();
  public void setValue(Object);
  public Object getNextValue();
  public Object getPreviousValue();
  // Listeners
  public void addChangeListener(ChangeListener);
  public void removeChangeListener(ChangeListener);
  }

  SpinnerModel中的六個方法直接繪製了JSpinner,而JSpinner的方法間接調用模塊中的方法,在監聽的狀況下,事件將聯繫到監聽器。

AbstractSpinnerModel類

  AbstractSpinnerModel類基本要實現的是SpinnerModel接口,它提供了管理和通知的監聽列表,子類必須實現接口中的四個方法,SpinnerModel中的三個具體實現如下:SpinnerDateModel,SpinnerListModel和SpinnerNumberModel。


SpinnerDateModel類

  從名字可以推斷出,SpinnerDateModel提供了數據的選擇。這個類有兩個構造函數:一個默認選擇所有的數據,另一個要求你給出範圍。

SpinnerDateModel:用於接受日期輸入。該類支持通過將 Calendar 類中的常數設置爲不同的值來更改日期;例如,Calendar.WEEK_OF_MONTH 每次將日期更改一週。


public SpinnerDateModel()
SpinnerModel model = new SpinnerDateModel();
JSpinner spinner = new JSpinner(model);


public SpinnerDateModel(Date value, Comparable start, Comparable end,int calendarField)
Calendar cal = Calendar.getInstance();
Date now = cal.getTime();
cal.add(Calendar.YEAR, -50);
Date startDate = cal.getTime();
cal.add(Calendar.YEAR, 100);
Date endDate = cal.getTime();
SpinnerModel model = new SpinnerDateModel(now, startDate, endDate, Calendar.YEAR);
JSpinner spinner = new JSpinner(model);


  如果不指名任何參數,就沒有開始和結束點。下面的例子展示了使用參數來表示100年的範圍。最後一個成員變量應該是Calendar類中的一個定值:
  ·    Calendar.AM_PM
  ·    Calendar.DAY_OF_MONTH
  ·    Calendar.DAY_OF_WEEK
  ·    Calendar.DAY_OF_WEEK_IN_MONTH

  ·    Calendar.DAY_OF_YEAR
  ·    Calendar.ERA
  ·    Calendar.HOUR
  ·    Calendar.HOUR_OF_DAY
  ·    Calendar.MILLISECOND
  ·    Calendar.MINUTE
  ·    Calendar.MONTH
  ·    Calendar.SECOND
  ·    Calendar.WEEK_OF_MONTH
  ·    Calendar.WEEK_OF_YEAR
  ·    Calendar.YEAR

  注意:SpinnerDateModel不包含任何Calendar類中的時間域,所以不能通過SpinnerDateModel在JSpinner中翻轉。

  表格3列出了SpinnerModel中的三個屬性,四個關於SpinnerDateModel。

[轉載]Java <wbr>Swing之:好用的JSpinner

Table 3. SpinnerDateModel 屬性

典型地,唯一的新屬性中你將要用來獲得最終的日期,儘管所有的結果都被包裹在getValue()中,以適當的數據類型。如果在構造函數中提供了數據的表示範圍,那麼previous和next的值將是null,在邊界條件下。

SpinnerListModel類


  SpinnerListModel用於接受來自值列表的輸入。提供了從一個入口列表中選擇或者至少是字符串表述,這個類有三個構造函數:

  public SpinnerListModel()SpinnerModel model = new SpinnerListModel();
  JSpinner spinner = new JSpinner(model);
  public SpinnerListModel(List <?> values)List <String> list = args;
  SpinnerModel model = new SpinnerListModel(list);
  JSpinner spinner = new JSpinner(model);
  public SpinnerListModel(Object[] values)SpinnerModel model = new SpinnerListModel(args);
  JSpinner spinner = new JSpinner(model);

  當沒有參數提供時,這個模型包括一個元素:字符串empty。List版保留一個對list的引用。而不是list的拷貝。如果改變了list,那麼模型中的list也將改變。數組版本的創建了一個私有的內部類,並且實例化一個list。對於list和數組版本,初始選擇的是第一個元素,否則將拋出一個IllegalArgumentException異常。

  如表格4顯示,屬性中增添的是set和get list。

[轉載]Java <wbr>Swing之:好用的JSpinner
Table 4. SpinnerListModel 屬性


SpinnerNumberModel類

  SpinnerNumberModel提供了從一個開區間或閉區間選擇數字的模式,數字可以使Number類的所有子類,包括Integer和Double。他有四個構造函數。

  public SpinnerNumberModel()SpinnerModel model = new SpinnerNumberModel();
  JSpinner spinner = new JSpinner(model);
  public SpinnerNumberModel(double value, double minimum, double maximum,
  double stepSize)SpinnerModel model = new SpinnerNumberModel(50, 0, 100, .25);
  JSpinner spinner = new JSpinner(model);
  public SpinnerNumberModel(int value, int minimum, int maximum, int stepSize)SpinnerModel model = new SpinnerNumberModel(50, 0, 100, 1);
  JSpinner spinner = new JSpinner(model);
  public SpinnerNumberModel(Number value, Comparable minimum, Comparable maximum,
  Number stepSize)Number value = new Integer(50);
  Number min = new Integer(0);Number max = new Integer(100);
  Number step = new Integer(1);SpinnerModel model = new SpinnerNumberModel(value, min, max, step);

  JSpinner spinner = new JSpinner(model);

  如果最大或最小值爲null,則爲開區間。對於沒有參數的,初始值爲1,步進爲1。步進是整形的,如果你設爲.333,那麼將不會完成。

  表格5展示了SpinnerNumberModel的屬性:

[轉載]Java <wbr>Swing之:好用的JSpinner
Table 5. SpinnerNumberModel 屬性


自定義模型

  一般來說,可用的JSpinner模型已經足夠了,所以沒有必要創建他的子類了。但是,並不是所有場合都能滿足。比如,你可能希望使用一個包裝了SpinnerListModel的模型,代替停止在第一個或最後一個元素,他包裝了另一個結束。在列表2中給出了具體實現:

  Listing 2. RolloverSpinnerListModel 類

  import javax.swing.*;
  import java.util.*;
  public class RolloverSpinnerListModel extends SpinnerListModel {
  public RolloverSpinnerListModel(List values) {
  super(values);
  }
  public RolloverSpinnerListModel(Object[] values) {
  super(values);

  }
  public Object getNextValue() {
  Object returnValue = super.getNextValue();
  if (returnValue == null) {
  returnValue = getList().get(0);
  }
  return returnValue;
  }
  public Object getPreviousValue() {
  Object returnValue = super.getPreviousValue();
  if (returnValue == null) {
  List list = getList();
  returnValue = list.get(list.size() - 1);
  }
  return returnValue;
  }}

JSpinner編輯器

  對於JSpinner每個可用的模型,一個次要的支持類,JSpinner的一個內部類。然而這個模塊可以控制控件是否可選,JSpinner編輯器允許你控制如何顯示和編輯每個可選的值。

JSpinner.DefaultEditor類

  JSpinner的setEditor()方法允許你將任何Jcomponent作爲JSpinner的編輯器,當然你可以那樣做,更典型的是,你將用JSpinner.DefaultEditor的子類作運行。以JformattedTextField作爲簡單的編輯器工作,將提供所有你需要的基本功能。它包括一個基本的構造函數:


  public JSpinner.DefaultEditor(JSpinner spinner)JSpinner spinner = new JSpinner();
  JComponent editor = JSpinner.DefaultEditor(spinner);
  spinner.setEditor(editor);

  在表格6中可以看到,有兩個屬性:

[轉載]Java <wbr>Swing之:好用的JSpinner
Table 6. JSpinner.DefaultEditor properties


  在不知道使用的是哪個模型工作的情況下,在這個級別的你可以做的是改變JformattedTextField中的文字顯示。更典型的是,你將改變模型編輯器的某些自定義方面。

JSpinner.DateEditor類

  DateEditor允許你定製不同的日期顯示方式,使用java.text包中SimpleDateFormat類。察看Javadoc瞭解更多的關於SimpleDateFormat的可用格式模式。如果你不喜歡默認的地顯示方式,可以通過給構造函數的第二個參數傳遞一個新參數來改變顯示模式。

  public JSpinner.DateEditor(JSpinner spinner)SpinnerModel model = new SpinnerDateModel();
  JSpinner spinner = new JSpinner(model);JComponent editor = JSpinner.DateEditor(spinner);

  spinner.setEditor(editor);
  public JSpinner.DateEditor(JSpinner spinner, String dateFormatPattern)SpinnerModel model = new SpinnerDateModel();
  JSpinner spinner = new JSpinner(model);
  JComponent editor = JSpinner.DateEditor(spinner, "MMMM yyyy");
  spinner.setEditor(editor);

  默認情況,格式是M/d/yy h:mm a或者12/25/04 12:34 PM 代表2004年的聖誕節的某個時間。後面的例子將要顯示2004 December。

  編輯器的兩個屬性在表格7中。

[轉載]Java <wbr>Swing之:好用的JSpinner
Table 7. JSpinner.DateEditor屬性


JSpinner.ListEditor類

  當使用SpinnerListModel類工作時,ListEditor不支持任何特殊格式。而是提供了前置類型支持。既然模塊的所有入口都知道了,編輯器將嘗試匹配用戶輸入的字符。這裏只有一個構造函數,但是你可能幾乎用不到。

  public JSpinner.ListEditor(JSpinner spinner)

  在表格8中將看到ListEditor只有一個屬性:


[轉載]Java <wbr>Swing之:好用的JSpinner
Table 7. JSpinner.ListEditor屬性


JSpinner.NumberEditor類

  NumberEditor和DateEditor的工作方式很相似,允許你輸入定製的顯示模式。代替SimpleDateFormat工作,NumberEditor可以協助java.text包中的DecimalFormat類。就像DateEditor一樣,他又兩個構造函數:

  public JSpinner.NumberEditor(JSpinner spinner)SpinnerModel model = new SpinnerNumberModel(50, 0, 100, .25);JSpinner spinner = new JSpinner(model);JComponent editor = JSpinner.NumberEditor(spinner);spinner.setEditor(editor);public JSpinner.NumberEditor(JSpinner spinner, String decimalFormatPattern)SpinnerModel model = new SpinnerNumberModel(50, 0, 100, .25);JSpinner spinner = new JSpinner(model);JComponent editor = JSpinner.NumberEditor(spinner, "#,##0.###");spinner.setEditor(editor);

  第二個構造函數使用默認的字符串格式。如果數字太大,將使用逗號。如果結果是一個完整的數,將不會用十進制顯示。

  表格9中,顯示editor的兩個屬性。


[轉載]Java <wbr>Swing之:好用的JSpinner
Table 9. JSpinner.NumberEditor 屬性


總結

  在這篇文章中,你學習到了Swing中的JSpinner控件。當你要控制某些選擇在一定的範圍中時,JSpinner可以讓你通過翻滾來選擇需要的值。你學習到了如何提供這些要選擇的值:通過使用SpinnerDateModel和DateEditor,SpinnerListModel和ListEditor,SpinnerNumberModel和NumberEditor來設置日期。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章