TITLE:JTableのセルをエクスプローラ風に表示する

Posted by terai at 2006-07-18

JTableのセルをエクスプローラ風に表示する

セルの中にアイコンと文字列を配置し、エクスプローラ風に見えるよう、文字列だけにフォーカスをかけます。

  • &jnlp;
  • &jar;
  • &zip;

#screenshot

サンプルコード

class TestRenderer extends Box implements TableCellRenderer {
  private final Border emptyBorder = BorderFactory.createEmptyBorder(1,1,1,1);
  private final ImageIcon nicon;
  private final ImageIcon sicon;
  private final JLabel textLabel;
  private final JLabel iconLabel;
  public TestRenderer(JTable table) {
    super(BoxLayout.X_AXIS);
    textLabel = new JLabel("dummy");
    textLabel.setOpaque(true);
    textLabel.setBorder(emptyBorder);
    nicon = new ImageIcon(getClass().getResource("wi0063-16.png"));
    FilteredImageSource fis = new FilteredImageSource(
      nicon.getImage().getSource(), new SelectedImageFilter());
    sicon = new ImageIcon(createImage(fis));
    iconLabel = new JLabel(nicon);
    iconLabel.setBorder(BorderFactory.createEmptyBorder());
    table.setRowHeight(Math.max(textLabel.getPreferredSize().height,
                  iconLabel.getPreferredSize().height));
    add(iconLabel);
    add(textLabel);
    add(Box.createHorizontalGlue());
  }
  public Component getTableCellRendererComponent(JTable table, Object value,
        boolean isSelected, boolean hasFocus, int row, int column) {
    textLabel.setText(value==null?"":value.toString());
    FontMetrics fm = table.getFontMetrics(table.getFont());
    int swidth = fm.stringWidth(textLabel.getText())+textLabel.getInsets().left
                                +textLabel.getInsets().right;
    int cwidth = table.getColumnModel().getColumn(column).getWidth()
                   -iconLabel.getPreferredSize().width;
    textLabel.setPreferredSize(new Dimension(swidth>cwidth?cwidth:swidth,0));
    if(isSelected) {
      textLabel.setForeground(table.getSelectionForeground());
      textLabel.setBackground(table.getSelectionBackground());
    }else{
      textLabel.setForeground(table.getForeground());
      textLabel.setBackground(table.getBackground());
    }
    if(hasFocus) {
      textLabel.setBorder(UIManager.getBorder("Table.focusCellHighlightBorder"));
    }else{
      textLabel.setBorder(emptyBorder);
    }
    textLabel.setFont(table.getFont());
    iconLabel.setIcon(isSelected?sicon:nicon);
    return this;
  }
  private static class SelectedImageFilter extends RGBImageFilter {
    public SelectedImageFilter() {
      canFilterIndexColorModel = true;
    }
    public int filterRGB(int x, int y, int argb) {
      Color color = new Color(argb,true);
      float[] array = new float[4];
      color.getComponents(array);
      return new Color(array[0]*0.5f, array[1]*0.5f, array[2], array[3]).getRGB();
    }
  }
}

解説

Windows Explorer*1の詳細表示風にするため、以下のような設定をしています。

  • セル間の罫線を非表示(JTable#setShowHorizontalLines, JTable#setShowVerticalLines)
  • ひとつのセルの中でアイコンと文字列を表示するセルレンダラーを作成
  • フォーカスがJTableに無い場合、選択されたセルの背景色をパネルの背景色に変更

このレンダラーでは、アイコンと文字列を別々のJLabelで作成して並べることで、フォーカスはセルではなく文字列だけに点線で掛かるようになっています。

選択時にはセル全体の背景色を変更するのではなく、文字列を表示しているJLabelの背景色を変更し、そのPreferredSizeを文字列の長さまで縮小して右側に余白を作成しています。

  • Windows Explorer との相違点
    • アイコンと文字列以外の場所(セル内)をクリックしても、選択できてしまう
      • JTableで文字列をクリックした場合だけセルを選択状態にする
      • Windows L&F での JFileChooser はどうやっているのだろう?
    • 矩形による範囲指定で選択することができない
    • ソートすると選択状態がクリアされてしまう
    • タブキー、矢印キーによる選択状態の移動がおかしい
    • 編集不可

参考リンク

コメント