TITLE:JListで異なる高さのセルを使用

Posted by at 2006-05-15

JListで異なる高さのセルを使用

JListのレンダラーにJTextAreaを使って、異なる高さのセルを作成します。

  • &jnlp;
  • &jar;
  • &zip;
DifferentCellHeight.png

サンプルコード

class TextAreaRenderer extends JTextArea implements ListCellRenderer {
  private final Border border = new DotBorder(2,2,2,2);
  private final Color evenColor = new Color(230,255,230);
  @Override public Component getListCellRendererComponent(
      JList list, Object object, int index,
      boolean isSelected, boolean cellHasFocus) {
    setText((object==null) ? "" : object.toString());
    setBorder(cellHasFocus ? border 
                : BorderFactory.createEmptyBorder(2,2,2,2));
    if(isSelected) {
      setBackground(list.getSelectionBackground());
      setForeground(list.getSelectionForeground());
    }else{
      setBackground(index%2==0 ? evenColor : list.getBackground());
      setForeground(list.getForeground());
    }
    return this;
  }
}

private DefaultListModel makeList() {
  DefaultListModel model = new DefaultListModel();
  model.addElement("一行");
  model.addElement("一行目\n二行目");
  model.addElement("一行目\n二行目\n三行目");
  model.addElement("四行\n以上ある\nテキスト\nの場合");
  return model;
}
View in GitHub: Java, Kotlin

解説

左が複数行に対応したJList、右が通常のJListになります。左のJListでは、JList#getFixedCellHeight()が-1で、ListCellRendererにJTextAreaを使用しているため、テキストに\nを含めることで複数行を作成することができます。

セルの区切りを分かりやすくするために、偶数奇数で行の背景色を変更しています。

JTextAreaにセルフォーカスがある状態を表現するために、EmptyBorder LineBorder を継承して作成したDotBorderを使用しています。

class DotBorder extends LineBorder {
  public boolean isBorderOpaque() {return true;}
  public DotBorder(Color color, int thickness) {
    super(color, thickness);
  }
  @Override
  public void paintBorder(Component c, Graphics g, int x, int y, int w, int h) {
    Graphics2D g2 = (Graphics2D)g;
    g2.translate(x,y);
    g2.setPaint(getLineColor());
    BasicGraphicsUtils.drawDashedRect(g2, 0, 0, w, h);
    g2.translate(-x,-y);
  }
}

参考リンク

コメント