根据 Oscar 使用 ListCellRenderer 的想法,我想出了一些几乎可以工作的东西......组件值被正确呈现,但列表的值需要一个丑陋的 hack。
之所以需要 hack,是因为对于列表项,渲染器的大小(来自 getSize())反映了最长项的文本宽度,而不是可用于渲染值的空间的宽度。我尝试使用 JComboBox 本身,但它的宽度包括小下拉按钮,因此如果存在滚动条,则不考虑其宽度。hack 是如果渲染器的宽度小于组合框的宽度,则存储渲染器的宽度,如果渲染器的宽度大于组合框的宽度,则使用存储的宽度。这有一个极端情况,其中渲染器的宽度介于内部 JLabel 的宽度和组合框的宽度之间。
由于渲染空间将是组合框的宽度,减去滚动条和插图的宽度,如果有人对我如何知道列表有滚动条以及如何获取滚动条有任何建议,那么我可以提取宽度,我全是耳朵。也许我可以做 list.getParent() 并期望它是一个 JScrollPane (JComboBox 或 JList doco 确实声明它使用滚动窗格)。
欢迎提出其他更好的建议。
代码如下:
recentDirs.setRenderer(new ComboTextRenderer(recentDirs));
...
static private class ComboTextRenderer
extends DefaultListCellRenderer
implements SwingConstants
{
JComponent parent;
int renderWidth;
ComboTextRenderer(JComponent par) {
super();
parent=par;
renderWidth=-1;
}
public void paint(Graphics gc) {
String txt=getText();
int len=txt.length();
int wid=getSize().width;
Insets ins=getInsets();
FontMetrics met=gc.getFontMetrics();
if(renderWidth==-1 || wid<parent.getSize().width) { renderWidth=wid; }
else { wid=renderWidth; }
wid-=(ins.left+ins.right);
if(met.stringWidth(txt)>wid) {
String rpl=null;
for(int xa=0,pfx=Math.min(15,((len/2)-1)),sfx=(pfx+2); pfx>0 && sfx<len; xa++) {
rpl=(txt.substring(0,pfx)+" ... "+txt.substring(sfx));
if(met.stringWidth(rpl)<=wid) { break; }
rpl=null;
if ((len-sfx)>15) { sfx++; }
else if((xa%2)==0 ) { pfx--; }
else { sfx++; }
}
if(rpl!=null) { setText(rpl); }
}
super.paint(gc);
}
}