JCombobox组合框效果实现

JComboboxSwing中比较常用的控件,它显示一个项列表,扩展的是ListModel接口的模型,它的显示绘制器通过实现ListCellBenderer接口来绘制列表单元,下面介绍①普通应用例子;②显示图片选项框例子;③修改下拉按钮的例子;④下拉框可选与否的例子.

对于普通情况下使用JCombobox,没有什么注意事项,只需要把JCombobox new出来,设置它的Model的值就可以了.

先看Sun给的官方的例子:

341.jpg
342.jpg

具体的实现很简单:

String[] petStrings = { "Bird""Cat""Dog""Rabbit""Pig" };

//Create the combo box, select the item at index 4.

JComboBox petList = new JComboBox(petStrings);

petList.setSelectedIndex(4);

petList.addActionListener(this);

也可以通过petList.setEditable(true);设置是否可以编辑.对于Action的处理和普通的一样.

JCombobox默认下拉显示和显示项是文本,为了显示其它内容比如图片或者更复杂的东西,则需要设置新的Renderer,JComboboxRenderer需要实现ListCellRenderer接口.

这个也比较简单,Sun官方也给了例子:

343.jpg
344.jpg

具体的实现其实和普通的一样,先把JCombobox new出来,在使用setRenderer方法设置自己定义的Renderer就可以了.

// Create the combo box.

JComboBox petList = new JComboBox(intArray);

ComboBoxRenderer renderer = new ComboBoxRenderer();

renderer.setPreferredSize(new Dimension(200, 130));

petList.setRenderer(renderer);

当然这个Renderer是实现了ListCellRenderer接口的.

privateclass ComboBoxRenderer extends JLabel implements ListCellRenderer {

这样要是实现接口的方法:

/*

* This method finds the p_w_picpath and text corresponding to the selected

* value and returns the label, set up to display the text and p_w_picpath.

*/

@Override

public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) {

然后然后this就是继承的JLabel,对它可以设置属性了:

setIcon(icon);

setText(pet);

最后把设置好的控件返回就可以了,

returnthis;

当然你也可以设置更复杂的控件,比如继承JButton可以设置成按钮的样式.

SwingMVC模式非常的好,当你需要更改控件的显示的时候只需要继承控件的基本UI,重写你需要重写的部位就可以了,这儿想下拉框的按钮不显示向下的箭头,只需要继承BasicComboBoxUI,重写createArrowButton方法就可以了.

345.jpg 

重写UI

privatestaticclass MyComboBoxUI extends BasicComboBoxUI {

publicstatic ComponentUI createUI(JComponent c) {

returnnew MyComboBoxUI();

}

@Override

protected JButton createArrowButton() {

JButton button = new BasicArrowButton(BasicArrowButton.EAST);

return button;

}

}

当你需要修改JComboboxUI的时候,只需要调用setUI方法就可以了.

JComboBox comboBox = new JComboBox(labels);

comboBox.setUI((ComboBoxUI) MyComboBoxUI.createUI(comboBox));

对于修改JCombobox的显示可以从两个方向考虑,一个是修改MetalComboBoxUI,一个是继承ListCellRenderer.在通过设置UIRenderer使界面修改.

先看完成后的界面:

346.jpg
347.jpg

工程的目录结构如下:

348.jpg 

先设置JCombobox下拉框是否有线,为了使以后扩展容易,设置为接口:

/**

*theinterfacethattheJComboBoxcanhavelineenable.

*/

publicinterface LineEnable {

/**

*setbean.

*@paramisLineEnable

*isLineenable

*/

publicvoid setLineEnabled(boolean isLineEnable);

/**

*getbean.

*@returnisLineenable

*/

publicboolean isLineEnabled();

}

同样的对于JCombobox下拉框是否可以选择也设置为接口:

/**

*theinterfacethattheJComboBoxcanselectenable.

*/

publicinterface SelectEnable {

/**

*setbean.

*@paramisSelectEnable

*isSelectenable

*/

publicvoid setSelectEnabled(boolean isSelectEnable);

/**

*getbean.

*@returnisSelectenable

*/

publicboolean isSelectEnabled();

}

当然需要别的属性,比如颜色、形状等也可以再设置相同的接口.

对于JCombobox的没一个Item,都可以通过实现这些接口获得相应的显示:

/**

*theitemsthatyouwanttoshowinJComboBox.

*/

publicclassMyComboBoxItemimplements SelectEnable, LineEnable {

对于特殊的JCombobox设置Item,设置MyComboBoxItem就可以使Item具有选择可否和是否是线的样式.

它具有3个属性:

/**

*JComboBoxitems.

*/

private Object comboxItem = null;

/**

*isselectenabled.

*/

booleanisSelectEnabled = true;

/**

*islineenabled.

*/

booleanisLineEnabled = false;

可以通过设置它们得到JCombobox样式,这个类就是一个简单的Java Bean.

然后就是设置JCombobox的选项的显示,实现ListCellRenderer接口,通过对组件的重写设置描绘它的新属性:

/**

*JComboBoxcellrenderer.

*/

publicclass MyComboBoxRenderer extends JLabel implementsListCellRenderer, ActionListener {

重写它的方法:

/**

*Returnacomponentthathasbeenconfiguredtodisplaythespecified

*value.

*/

@Override

public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) {

先设置它的显示

String item = (value == null) ? "" : value.toString();

再设置是否是线:

if (((LineEnable) value).isLineEnabled()) {

returnnew JSeparator(JSeparator.HORIZONTAL);

}

接着设置tooltip,提示是否可以显示:

if (-1 < index) {

if (((SelectEnable) value).isSelectEnabled()) {

list.setToolTipText("You select is : " + item);

else {

list.setToolTipText("You cn't select : " + item

", It select unEnable.");

}

}

最后设置是否可以显示:

if (!((SelectEnable) value).isSelectEnabled()) {

setlist.getBackground()); setForeground(UIManager.getColor("Label.disabledForeground"));

}

然后还是需要设置某些选择不可选时候的设置不可选:

/**

*Thelistenerinterfaceforreceivingactionevents.

*/

publicvoid actionPerformed(ActionEvent e) {

Object tempItem = combox.getSelectedItem();

当是线的Item不可选,返回之前选项

if (((LineEnable) tempItem).isLineEnabled()) {

combox.setSelectedItem(currentItem);

else {

currentItem = tempItem;

}

当是不可选的Item不可选,返回之前选项

if (!((SelectEnable) tempItem).isSelectEnabled()) {

combox.setSelectedItem(currentItem);

else {

currentItem = tempItem;

}

}

接下来的类就是设置JComboboxUI,

/**

*MetalUIforJComboBox.

*/

publicclass MyComboBoxUI extends MetalComboBoxUI {

这里的UI设置主要是设置选择不可选的项时清除,并对JCombobox的下拉的菜单设置

/**

*showthepopupmenusize.

*/

@Override

publicvoid show() {

Dimension popupSize = ((MyComboBox) comboBox).getPopupSize();

// reset size.

popupSize.setSize(popupSize.width, getPopupHeightForRowCount(comboBox.getMaximumRowCount()));

Rectangle popupBounds = computePopupBounds(0, comboBox

.getBounds().height, popupSize.width, popupSize.height);

// set max and mini size.

scroller.setMaximumSize(popupBounds.getSize());

scroller.setPreferredSize(popupBounds.getSize());

scroller.setMinimumSize(popupBounds.getSize());

// Invalidates the container.

list.invalidate();

// set select.

intselectedIndex = comboBox.getSelectedIndex();

if (selectedIndex == -1) {

list.clearSelection();

else {

list.setSelectedIndex(selectedIndex);

}

list.ensureIndexIsVisible(list.getSelectedIndex());

setLightWeightPopupEnabled(comboBox.isLightWeightPopupEnabled());

// show it.

show(comboBox, popupBounds.x, popupBounds.y);

}

然后在UInew BasicComboPopup(comboBox) {

popup.getAccessibleContext().setAccessibleParent(comboBox);

就可以了.

最后的类就是自己的MyComboBox,这个类其实也可以不要,只需要你在新建自己特殊的JCombobox,不要忘记设置UI和传入的ItemMyComboBoxItem就可以了,这里为了方便自己实现了,以后使用时只需要New MyComboBox就可以了.

/**

*theJComboBoxthatithavesomeownmethod.

*/

publicclassMyComboBoxextends JComboBox {

在构造函数里设置UI

setUI(new MyComboBoxUI());

另外为了显示宽度合适,它提供了设置popupWidth的方法:

public Dimension getPopupSize() {

Dimension size = getSize();

// reset size.

if (popupWidth < 1) {

popupWidth = size.width;

}

returnnew Dimension(popupWidth, size.height);

}

这样一个属于自己的JComboBox就设置完成了,使用方法如下:

MyComboBoxItem [] items = { new MyComboBoxItem("Astart"),

new MyComboBoxItem("BGold"truetrue),

new MyComboBoxItem("ilove"false),

new MyComboBoxItem("fire your game"true),

new MyComboBoxItem(""truetrue),

new MyComboBoxItem("NIHA"false),

new MyComboBoxItem("生活"),

new MyComboBoxItem(""falsetrue) };

JComboBox jComboBox = new MyComboBox(items);

jComboBox.setRenderer(new MyComboBoxRenderer(jComboBox));

以后就可以当做一个普通的Java控件使用了.