• 追加された行はこの色です。
  • 削除された行はこの色です。
TITLE:JComboBoxのItemを左右にクリップして配置
#navi(../)
*JComboBoxのItemを左右にクリップして配置 [#td747fe0]
>編集者:[[Terai Atsuhiro>terai]]~
作成日:2005-09-12~
更新日:&lastmod;
---
category: swing
folder: ClippedLRComboBox
title: JComboBoxのItemを左右にクリップして配置
tags: [JComboBox, ListCellRenderer, JLabel, JPanel]
author: aterai
pubdate: 2005-09-12T13:00:56+09:00
description: JComboBoxのItem内のレイアウトをメインとサブの二つに分割し、それぞれ適当な長さに省略した文字列を表示します。
image: https://lh5.googleusercontent.com/_9Z4BYR88imo/TQTJSTVvNXI/AAAAAAAAAUI/RNbSh6R4xi8/s800/ClippedLRComboBox.png
hreflang:
    href: https://java-swing-tips.blogspot.com/2008/08/multi-column-jcombobox.html
    lang: en
---
* 概要 [#summary]
`JComboBox`の`Item`内のレイアウトをメインとサブの二つに分割し、それぞれ適当な長さに省略した文字列を表示します。

#contents
#download(https://lh5.googleusercontent.com/_9Z4BYR88imo/TQTJSTVvNXI/AAAAAAAAAUI/RNbSh6R4xi8/s800/ClippedLRComboBox.png)

**概要 [#p068c669]
JComboBoxのItemにテキストをクリップして左右に分けて配置します。
* サンプルコード [#sourcecode]
#code(link){{
class MultiColumnCellRenderer extends JPanel implements ListCellRenderer {
  private final JLabel leftLabel = new JLabel();
  private final JLabel rightLabel;

#screenshot
  public MultiColumnCellRenderer(int rightWidth) {
    super(new BorderLayout());
    this.setBorder(BorderFactory.createEmptyBorder(1, 1, 1, 1));

**サンプルコード [#n26d00a1]
 class MyCellRenderer extends JPanel implements ListCellRenderer {
   private final JLabel lbl1 = new JLabel();
   private final JLabel lbl2 = new JLabel();
   public MyCellRenderer(Dimension dim, int rightw) {
     super(new BorderLayout());
     lbl1.setOpaque(true);
     lbl2.setOpaque(true);
     this.setOpaque(true);
     lbl1.setBorder(BorderFactory.createEmptyBorder(0,2,0,0));
     lbl2.setPreferredSize(new Dimension(rightw, 0));
     lbl2.setBorder(BorderFactory.createEmptyBorder(0,5,0,0));
     this.add(lbl1, BorderLayout.CENTER);
     this.add(lbl2, BorderLayout.EAST);
     this.setPreferredSize(dim);
   }
   public Component getListCellRendererComponent(
                     JList list, Object value, int index,
                     boolean isSelected, boolean cellHasFocus) {
     LRItem item = (LRItem)value;
     lbl1.setText(item.getLeftText());
     lbl2.setText(item.getRightText());
     lbl1.setBackground(isSelected ? SystemColor.textHighlight :
                                     Color.white);
     lbl2.setBackground(isSelected ? SystemColor.textHighlight :
                                     Color.white);
     this.setBackground(isSelected ? SystemColor.textHighlight :
                                     Color.white);
     lbl1.setForeground(isSelected ? Color.white :
                                     Color.black);
     lbl2.setForeground(isSelected ? Color.gray.brighter() :
                                     Color.gray);
     return this;
   }
 }
    leftLabel.setOpaque(false);
    leftLabel.setBorder(BorderFactory.createEmptyBorder(0, 2, 0, 0));

-&jnlp;
-&jar;
-&zip;
    final Dimension dim = new Dimension(rightWidth, 0);
    rightLabel = new JLabel() {
      @Override public Dimension getPreferredSize() {
        return dim;
      }
    };
    rightLabel.setOpaque(false);
    rightLabel.setBorder(BorderFactory.createEmptyBorder(0, 2, 0, 2));
    rightLabel.setForeground(Color.GRAY);
    rightLabel.setHorizontalAlignment(SwingConstants.RIGHT);

**解説 [#y1b479cc]
上記のサンプルでは、JLabelを二つ並べたJPanelをレンダラーにすることで、Itemに設定した文字列を左右に表示しています。このため文字列が長い場合、JLabelがこれを自動的にクリップしてくれます。
    this.add(leftLabel);
    this.add(rightLabel, BorderLayout.EAST);
  }

**参考リンク [#w60ad6dd]
-[[JComboBoxのItemを左右に配置>Swing/LRComboBox]]
--こちらはhtmlのtableタグを使用して同様の表示(クリップはしない)を行っています。
  @Override public Component getListCellRendererComponent(
      JList list, Object value, int index,
      boolean isSelected, boolean cellHasFocus) {
    LRItem item = (LRItem) value;
    leftLabel.setText(item.getLeftText());
    rightLabel.setText(item.getRightText());

**コメント [#n7dbe41e]
    leftLabel.setFont(list.getFont());
    rightLabel.setFont(list.getFont());

    if (index < 0) {
      leftLabel.setForeground(list.getForeground());
      this.setOpaque(false);
    } else {
      leftLabel.setForeground(
          isSelected ? list.getSelectionForeground() : list.getForeground());
      this.setBackground(
          isSelected ? list.getSelectionBackground() : list.getBackground());
      this.setOpaque(true);
    }
    return this;
  }

  @Override public Dimension getPreferredSize() {
    Dimension d = super.getPreferredSize();
    return new Dimension(0, d.height);
  }

  @Override public void updateUI() {
    super.updateUI();
    this.setName("List.cellRenderer");
  }
}
}}

* 解説 [#explanation]
- `JPanel`に`JLabel`を二つ並べて配置したセルレンダラーを作成し、`Item`に設定した文字列を左右の`JLabel`に表示
- このため左右の文字列が各`JLabel`の推奨サイズより長い場合は自動的に省略表示になる

* 参考リンク [#reference]
- [[JComboBoxのItemを左右に配置>Swing/LRComboBox]]
-- こちらは`html`の`table`タグを使用して同様の表示(クリップはしない)を行っている
- [[JComboBoxのドロップダウンリストとしてJTableを使用する>Swing/DropdownTableComboBox]]
-- こちらは`JList`の代わりに`JTable`を使用している

* コメント [#comment]
#comment
- ポップアップリストが更新されなくなって?、うまくクリップできなくなっていたのを修正。 -- &user(aterai); &new{2008-08-13 (水) 15:14:12};
- 選択時の文字色を修正(`Windows 7`などへの対応)。 -- &user(aterai); &new{2012-02-03 (金) 14:28:48};

#comment