我有一个class GridPanel extends JPanel
, 带有一个静态内部类ToolSelectComboBox extends JComboBox
,它又具有两个静态内部类ToolSelectComboBoxModel implements ComboBoxModel
和ToolSelectComboBoxRenderer implements ListCellRenderer
. 面板显示ToolSelectComboBox
(TSCB),其构造函数将其模型和渲染器设置为我创建的自定义渲染器。盒子创建正确,其模型和渲染器工作正常。
但是,渲染器的getListCellRendererComponent(...)
方法在它返回ImageIcon
的那个上使用了一个。JLabel
该图标已正确加载,但是,当我第一次单击组合框(每次运行时)时,图像的加载时间恰好(或至少非常接近)一秒多一点。我会假设这是加载文件的一些滞后,除了
- 这是我本地文件系统上的一个 4kB 文件
- 当我在
System.out.println
命令之前和之后添加命令时result.setIcon(...)
,它们几乎立即相互跟随。
我注意到的奇怪的事情是println
命令被触发了两次,一次是在我单击框时,一次是在图标加载时。
还可能值得注意的是,由于它旨在与覆盖父抽象类的单个方法的多个类一起使用(以生成图标的路径),当我注意到它工作缓慢时,我将代码从只需使用getIcon
命令检索图标以将各种大小的图标(16、32 和 64 px 平方)存储在一个TreeMap<Tool.ImageSize, ImageIcon>
(Tool
我创建的具有ImageIcon getIcon()
方法的界面在哪里。
我所有的进口都井井有条。
任何帮助,将不胜感激!
如果我发布了太多代码,我深表歉意,但我想确保它是可以理解的。另一方面,如果您需要更多代码才能理解,请不要犹豫。
代码(所有以“”开头*
并具有类似注释的文本的行都是折叠的 JavaDoc 标记,而不仅仅是混乱的代码):
public class GridPanel extends JPanel {
public static class ToolSelectComboBox extends JComboBox {
// Combo box model `ToolSelectComboBoxModel` snipped
* A renderer for the {@link ToolSelectComboBoxModel}. This may
public static class ToolSelectComboBoxRenderer implements
ListCellRenderer {
* The default renderer. Only the icon and text are modified.
protected DefaultListCellRenderer d = new DefaultListCellRenderer();
@Override
public Component getListCellRendererComponent(final JList list,
final Object value, final int index,
final boolean isSelected, final boolean cellHasFocus) {
if (!ToolSelectComboBoxModel.class.isInstance(list.getModel())) {
throw new IllegalStateException(
"Cannot use a ToolSelectComboBoxRenderer on any list model type other than ToolSelectComboBoxModel.");
}
final JLabel result = (JLabel) d.getListCellRendererComponent(
list, value, index, isSelected, cellHasFocus);
result.setText(null);
if (value != null) {
result.setIcon(((Tool) value)
.getIcon(Tool.IconSize.SIZE_32PX));
}
return result;
}
}
public ToolSelectComboBox() {
setModel(new ToolSelectComboBoxModel());
((ToolSelectComboBoxModel) getModel()).add(new CircleTool()); // shown below
setRenderer(new ToolSelectComboBoxRenderer());
}
}
* Create the panel.
public GridPanel() {
setLayout(new BorderLayout(0, 0));
final ToolSelectComboBox toolSelectComboBox = new ToolSelectComboBox();
add(toolSelectComboBox, BorderLayout.NORTH);
final SquareGrid squareGrid = new SquareGrid(); // another class; this works fine
add(squareGrid, BorderLayout.CENTER); // irrelevant to problem
}
}
该类CircleTool
只有一个方法(覆盖AbstractTool
的抽象方法以获取图像路径),并且由于该方法有效(它可以很好地获取路径,它只是加载缓慢的图标),因此我没有包含此类。
AbstractTool
班级:
public abstract class AbstractTool implements Tool {
/**
* A {@link TreeMap} to map the icon sizes to their icons.
*/
protected final TreeMap<Tool.IconSize, ImageIcon> map = new TreeMap<Tool.IconSize, ImageIcon>();
/**
* Constructs the tool and sets up the {@linkplain #map}.
*/
public AbstractTool() {
for (final Tool.IconSize size : Tool.IconSize.values()) {
System.out.println("Putting value for " + size);
map.put(size,
new ImageIcon(Tool.class.getResource(getImagePath(size))));
}
}
@Override
public ImageIcon getIcon(final IconSize size) {
return map.get(size);
}
/**
* Gets the image path for the given image size.
*
* @param size
* the size
* @return the image path
*/
protected abstract String getImagePath(Tool.IconSize size);
}