• 追加された行はこの色です。
  • 削除された行はこの色です。
TITLE:JComboBoxの色を変更
#navi(../)
*JComboBoxの色を変更 [#jce4ead6]
>編集者:[[Terai Atsuhiro>terai]]~
作成日:2005-01-10~
更新日:&lastmod;
---
category: swing
folder: ColorComboBox
title: JComboBoxの色を変更
tags: [JComboBox, ListCellRenderer, JTextField]
author: aterai
pubdate: 2005-01-10T01:48:08+09:00
description: JComboBoxのEditor部分と、List部分の色を変更します。
image: https://lh4.googleusercontent.com/_9Z4BYR88imo/TQTJhY0CAaI/AAAAAAAAAUg/J70FCr-EUlI/s800/ColorComboBox.png
hreflang:
    href: https://java-swing-tips.blogspot.com/2009/06/color-jcombobox.html
    lang: en
---
* 概要 [#summary]
`JComboBox`の`Editor`部分と、`List`部分の色を変更します。

#contents
#download(https://lh4.googleusercontent.com/_9Z4BYR88imo/TQTJhY0CAaI/AAAAAAAAAUg/J70FCr-EUlI/s800/ColorComboBox.png)

**概要 [#x42ac8da]
JComboBoxのEditor部分と、List部分の色を変更します。
* サンプルコード [#sourcecode]
#code(link){{
class AlternateRowColorComboBox<E> extends JComboBox<E> {
  private static final Color EVEN_BGC = new Color(0xE1_FF_E1);
  private static final Color ODD_BGC = Color.WHITE;
  private transient ItemListener itemColorListener;

#screenshot
  protected AlternateRowColorComboBox(ComboBoxModel<E> model) {
    super(model);
  }

**サンプルコード [#b2f19a3d]
 combo01.setModel(makeModel());
 combo01.setRenderer(new MyListCellRenderer(combo01.getRenderer()));
 combo01.addItemListener(new ItemListener() {
   public void itemStateChanged(ItemEvent e) {
     if(e.getStateChange()!=ItemEvent.SELECTED) return;
     combo01.setBackground(getOEColor(combo01.getSelectedIndex()));
   }
 });
 combo01.setSelectedIndex(0);
 combo01.setBackground(evenBGColor);
 
 final JTextField field = (JTextField) combo02.getEditor().getEditorComponent();
 field.setOpaque(true);
 field.setBackground(evenBGColor);
 combo02.setEditable(true);
 combo02.setModel(makeModel());
 combo02.setRenderer(new MyListCellRenderer(combo02.getRenderer()));
 combo02.addItemListener(new ItemListener() {
   public void itemStateChanged(ItemEvent e) {
     if(e.getStateChange()!=ItemEvent.SELECTED) return;
     field.setBackground(getOEColor(combo02.getSelectedIndex()));
   }
 });
 combo02.setSelectedIndex(0);
  @Override public void setEditable(boolean flag) {
    super.setEditable(flag);
    Component field = getEditor().getEditorComponent();
    if (flag && field instanceof JComponent) {
      ((JComponent) field).setOpaque(true);
      field.setBackground(getAlternateRowColor(getSelectedIndex()));
    }
  }

  @Override public void updateUI() {
    removeItemListener(itemColorListener);
    super.updateUI();
    ListCellRenderer<? super E> r = getRenderer();
    setRenderer((list, value, index, isSelected, cellHasFocus) -> {
      Component c = r.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
      if (index >= 0 && c instanceof JComponent) {
        ((JComponent) c).setOpaque(true);
      }
      if (!isSelected) {
        c.setBackground(getAlternateRowColor(index));
      }
      return c;
    });
    itemColorListener = e -> {
      if (e.getStateChange() != ItemEvent.SELECTED) {
        return;
      }
      JComboBox<?> cb = (JComboBox<?>) e.getItemSelectable();
      Color rc = getAlternateRowColor(cb.getSelectedIndex());
      if (cb.isEditable()) {
        cb.getEditor().getEditorComponent().setBackground(rc);
      } else {
        cb.setBackground(rc);
      }
    };
    addItemListener(itemColorListener);
    EventQueue.invokeLater(() -> {
      Component c = getEditor().getEditorComponent();
      c.setBackground(getAlternateRowColor(getSelectedIndex()));
      if (c instanceof JComponent) {
        ((JComponent) c).setOpaque(true);
      }
    });
  }

-&jnlp;
-&jar;
-&zip;
  protected static Color getAlternateRowColor(int index) {
    return index % 2 == 0 ? EVEN_BGC : ODD_BGC;
  }
}
}}

**解説 [#bbe45ca3]
JComboBoxを編集可にした状態で、以下のようにList部分、Editor部分に背景色を設定します。
* 解説 [#explanation]
上記のサンプルでは、`JComboBox`を編集可にした状態で以下のように`List`部分、`Editor`部分に行の奇数偶数で背景色を変更しています。

-List部分
--ListCellRendererを使用することで背景色を変更しています。
- `List`部分
-- `ListCellRenderer`を使用することで背景色を変更
- `Editor`部分
-- `getEditor().getEditorComponent()`で`JTextField`オブジェクトを取得して背景色を変更

-Editor部分
--getEditor().getEditorComponent()でJTextFieldオブジェクトを取得して背景色を変更しています。
----
- `GTKLookAndFeel`などで`Box`(`Editor`)部分の色を変更できない場合がある

上記のサンプルでは、下のJComboBoxで行の奇数偶数による背景色の変更を行っています。
#ref(https://lh3.googleusercontent.com/_9Z4BYR88imo/TQTJj4vDxSI/AAAAAAAAAUk/ZZtKylfc0k8/s800/ColorComboBox1.png)

//**参考リンク
**コメント [#i79a7ae9]
- JComboBox#setEditable(true)は必須のようです。編集不可にするにはEditor部分のJTextFieldに対してsetEditable(false) -- [[Y]] &new{2006-10-10 (火) 18:51:38};
-- ご指摘ありがとうございます。せっかくJComboBoxを上下に並べているのだから、編集可の場合と不可の場合のサンプルにすればよかったですね。編集不可の場合(JComboBox#setEditable(false))に色を着けるには、上記の方法と、以下のようにJComboBox#setBackground(Color)メソッドを使う方法があるようです。編集不可の場合は、この部分の色もレンダラーが勝手にやってくれてたような気がするのですが、勘違いだったのかも。 -- [[terai]] &new{2006-10-10 (火) 19:58:51};
 combo01.addItemListener(new ItemListener() {
   public void itemStateChanged(ItemEvent e) {
     if(e.getStateChange()!=ItemEvent.SELECTED) return;
     combo01.setBackground((combo01.getSelectedIndex()%2==0)?evenBGColor:oddBGColor);
   }
 });
-- せっかくなので、上のJComboBoxは編集不可、下は編集可の場合で、色を着けるサンプルに変更しました。 -- [[terai]] &new{2006-10-10 (火) 20:31:03};
* 参考リンク [#reference]
- [[JComboBoxの文字色を変更する>Swing/ComboBoxForegroundColor]]

* コメント [#comment]
#comment
- `JComboBox#setEditable(true)`は必須のようです。編集不可にするには`Editor`部分の`JTextField`に対して`setEditable(false)` -- &user(Y); &new{2006-10-10 (火) 18:51:38};
-- ご指摘ありがとうございます。せっかく`JComboBox`を上下に並べているので、編集可の場合と不可の場合のサンプルにすればよかったですね。編集不可の場合(`JComboBox#setEditable(false)`)に色を着けるには、上記の方法と、以下のように`JComboBox#setBackground(Color)`メソッドを使う方法があるようです。 %%編集不可の場合は、この部分の色もレンダラーが勝手にやってくれてたような気がするのですが、勘違いだったのかも。%% バージョンや`LookAndFeel`で異なる?ようです。 -- &user(aterai); &new{2006-10-10 (火) 19:58:51};

#code{{
final JComboBox c = new JComboBox();
c.addItemListener(new ItemListener() {
  @Override public void itemStateChanged(ItemEvent e) {
    if (e.getStateChange() != ItemEvent.SELECTED) {
      return;
    }
    c.setBackground((c.getSelectedIndex() % 2 == 0) ? evenBGColor : oddBGColor);
  }
});
}}

-- せっかくなので、上の`JComboBox`は編集不可、下は編集可の場合で、色を着けるサンプルに変更しました。 -- &user(aterai); &new{2006-10-10 (火) 20:31:03};
- メモ: [https://bugs.openjdk.org/browse/JDK-6367601 Windows/Motif L&F: Changing the JComboBox background does not change the popup of the JCombobox] -- &user(aterai); &new{2007-12-13 (木) 15:43:50};
- サンプルソースの`LookAndFeel`を設定しないようにすると、編集不可コンボはボタン部分も背景色になってしまう・・ -- &user(han); &new{2008-10-21 (火) 10:41:38};
-- `MetalLookAndFeel`などは、コンボボックスの背景色を変更すると矢印ボタンの色まで変更してしまう仕様?みたいですね。回避するなら、以下のように`UI`で使っている`PropertyChangeListener`をオーバーライドしてしまうのはどうでしょう。 -- &user(aterai); &new{2008-10-21 (火) 15:41:21};

#code{{
combo01.setUI(new MetalComboBoxUI() {
  @Override public PropertyChangeListener createPropertyChangeListener() {
    return new MetalPropertyChangeListener() {
      @Override public void propertyChange(PropertyChangeEvent e) {
        String propertyName = e.getPropertyName();
        if (propertyName == "background") {
          Color color = (Color) e.getNewValue();
          //arrowButton.setBackground(color);
          listBox.setBackground(color);
        } else {
          super.propertyChange( e );
        }
      }
    };
  }
});
combo01.setModel(makeModel());
combo01.setRenderer(new MyListCellRenderer(combo01.getRenderer()));
combo01.addItemListener(new ItemListener() {
  @Override public void itemStateChanged(ItemEvent e) {
    if (e.getStateChange() != ItemEvent.SELECTED) {
      return;
    }
    combo01.setBackground(getOEColor(combo01.getSelectedIndex()));
  }
});
combo01.setSelectedIndex(0);
combo01.setBackground(evenBGColor);
}}

- ありがとうございます。動作確認してませんが`UI`をさわればいろんなことができそうですね。で`UIManager`でなんとかできないか気になったので試すと、`UIManager.put("ComboBox.background", new ColorUIResource(Color.white));`で全てのコンボボックスの背景色を設定できました(リストの色分けはできないですが) -- &user(han); &new{2008-10-24 (金) 11:28:10};

#comment