TITLE:TableRowSorterでJTableのソート
Posted by aterai at 2007-02-12

TableRowSorterでJTableのソート

JDK 6 で導入された、TableRowSorterを利用して、JTableの行を降順、昇順にソートします。
  • category: swing folder: TableRowSorter title: TableRowSorterでJTableのソート tags: [JTable, TableRowSorter, JTableHeader] author: aterai pubdate: 2007-02-12T18:36:03+09:00 description: JDK 6で導入されたTableRowSorterを利用してJTableの行を降順、昇順にソートします。 image: https://lh4.googleusercontent.com/_9Z4BYR88imo/TQTUnbg2jyI/AAAAAAAAAmU/-7yjlGSjBmo/s800/TableRowSorter.png

概要

JDK 6で導入されたTableRowSorterを利用してJTableの行を降順、昇順にソートします。
TableRowSorter.png

サンプルコード

#spanend
#spanadd
* サンプルコード [#sourcecode]
#spanend
#spanadd
#code(link){{
#spanend
TableModel model = makeTestTableModel();
JTable table = new JTable(model);
RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
table.setRowSorter(sorter);

解説

JDK 6 では、JTableにTableRowSorterを設定することで、カラムヘッダのクリックによる行ソートが行えます。

解説

JDK 6ではJTableTableRowSorterを設定することでカラムヘッダのクリックによる行ソートが行えます。 以下のように、JTable#autoCreateRowSorter(true); メソッドを使用した場合も、JTable が自動的に RowSorter を作成して、ソートが出来るようになります。
  • 以下のようにJTable#autoCreateRowSorter(true)メソッドを使用した場合も、JTableが自動的にRowSorterを作成してソートが可能になる
    JTable table = new JTable(model);
    table.setAutoCreateRowSorter(true);
    

TableRowSorterのデフォルトでは、ヘッダクリックで(降順、昇順)とソートが切り替わり、TableSorterのように(降順、昇順、初期状態)ではなくなっています。上記のサンプルでは、下のボタンをクリックすると、DefaultRowSorter#setSortKeys(List)にnullを代入することで初期状態に戻るようにしています。
第二キーを使ったソートはTableRowSorter版では出来ない?ようなので、TableSorter.javaを使う場合もまだあるかもしれません。 複数キーを使ったソートも可能です。デフォルトではソートキーは3つで、ヘッダにマークが表示されるのは最新のソートキーのみですが、クリックした順に保持されるようです。詳しくはドキュメントやTableの内容をソートなどを参照してください。
  • TableModel に要素を追加した後で、table.setRowSorter(sorter);とすると、IndexOutOfBoundsException: Invalid range が、モデルへの追加、削除、編集中に別の行クリックなどで発生する。
    • 以下のサンプルの場合、model.fireTableDataChanged() メソッドをtable.setRowSorter(sorter)の後で呼び出すと回避できる。
  • TableModelに要素を追加した後でtable.setRowSorter(sorter);とすると、IndexOutOfBoundsException: Invalid rangeがモデルへの追加、削除、編集中に別の行クリックなどで発生する
    • 以下のサンプルの場合model.fireTableDataChanged()メソッドをtable.setRowSorter(sorter)の後で呼び出すと回避可能
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.table.*;
#spandel
public class RowSorterTest extends JPanel{
#spanend
#spanadd

#spanend
#spanadd
public class RowSorterTest extends JPanel {
#spanend
  public RowSorterTest() {
    super(new BorderLayout());
    final DefaultTableModel model = new DefaultTableModel(null, new String[] {"A","B","C"});
    final DefaultTableModel model = new DefaultTableModel(null, new String[] {"A", "B", "C"});
    JTable table = new JTable(model);
    TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
    //table.setRowSorter(sorter); // OK
    // table.setRowSorter(sorter); // OK
    model.addRow(new String[] {"aa", "bb", "cc"});
    model.addRow(new String[] {"dd", "ee", "ff"});
    table.setRowSorter(sorter); // IndexOutOfBoundsException: Invalid range
    //model.fireTableDataChanged(); // <----
    // model.fireTableDataChanged(); // <----
    add(new JButton(new AbstractAction("model.addRow(...); -> IndexOutOfBoundsException") {
      @Override public void actionPerformed(ActionEvent e) {
        model.addRow(new String[] {"gg", "hh", "ii"});
      }
    }), BorderLayout.SOUTH);
    add(new JScrollPane(table));
  }
#spanadd

#spanend
  public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
      @Override public void run() { createAndShowGUI(); }
    });
  }
#spanadd

#spanend
  public static void createAndShowGUI() {
    JFrame frame = new JFrame("Test");
    frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    frame.getContentPane().add(new RowSorterTest());
    frame.setSize(320,240);
    frame.setSize(320, 240);
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
  }
}
  • RowSorter (Java Platform SE 6)
    • RowSorter の具象実装は、TableModel、ListModel などのモデルを参照する必要があります。JTable や JList などのビュークラスも、モデルを参照します。順序の依存性を回避するため、RowSorter 実装がモデル上にリスナーをインストールしないようにしてください。モデルが変更されると、ビュークラスが RowSorter を呼び出します。たとえば、TableModel JTable で行が更新された場合、 rowsUpdated が呼び出されます。モデルが変更されると、ビューは、 modelStructureChanged、allRowsChanged、rowsInserted、rowsDeleted、rowsUpdated のいずれかのメソッドを呼び出します。

参考リンク

コメント

  • DefaultRowSorter#setMaxSortKeys(int)で複数のキーを用いたソートも出来ているように思います(b86)。ただ、ヘッダー部分がわかりにくいですが。 -- syo
    • ご指摘ありがとうございます。なるほど、こちら(Tableの内容をソート)を使えばうまくいきそうですね。 -- aterai
  • rendererがないので、基本的に数字は左揃え、文字列は右揃えで、中央揃えするにはどうすれば宜しいでしょうか? -- パンダ

参考リンク

コメント