1

我注意到 SwingX 的Highlighter界面允许荧光笔返回与传入的组件不同的组件。我实际上找不到任何正在使用的示例,但我想我会尝试使用它来创建某种假秒柱子。

预期的结果是左列中的文本应该截断右列开始的位置,所以我不能只使用Painter. 右列应该为整个列表呈现相同的宽度,这是我还没有弄清楚但似乎并不难的问题。

至于现在,我发现行高被压缩到很小,你看不到任何文字。

这就是我的意思:

截屏

示例程序:

import java.awt.BorderLayout;
import java.awt.Component;

import javax.swing.DefaultListModel;
import javax.swing.GroupLayout;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.LayoutStyle;
import javax.swing.SwingUtilities;

import org.jdesktop.swingx.JXList;
import org.jdesktop.swingx.decorator.AbstractHighlighter;
import org.jdesktop.swingx.decorator.ComponentAdapter;
import org.jdesktop.swingx.renderer.JRendererLabel;
import org.jdesktop.swingx.renderer.StringValue;

public class RendererTest implements Runnable
{
    public static void main(String[] args)
    {
        SwingUtilities.invokeLater(new RendererTest());
    }

    @Override
    public void run()
    {
        JFrame frame = new JFrame("Highlighter test");
        JXList list = new JXList();
        DefaultListModel<String> listModel = new DefaultListModel<>();
        listModel.addElement("one");
        listModel.addElement("two");
        listModel.addElement("three");
        list.setModel(listModel);
        list.setVisibleRowCount(8);
        list.setPrototypeCellValue("some string");
        list.addHighlighter(new AddSecondColumnHighlighter(v -> ((String) v).toUpperCase()));
        JScrollPane listScroll = new JScrollPane(list);
        frame.setLayout(new BorderLayout());
        frame.add(listScroll, BorderLayout.CENTER);
        frame.pack();
        frame.setVisible(true);
    }

    private static class AddSecondColumnHighlighter extends AbstractHighlighter
    {
        private final StringValue secondColumnStringValue;

        public AddSecondColumnHighlighter(StringValue secondColumnStringValue)
        {
            this.secondColumnStringValue = secondColumnStringValue;
        }

        @Override
        protected Component doHighlight(Component component, ComponentAdapter adapter)
        {
            JRendererLabel rightColumn = new JRendererLabel();
            rightColumn.setText(secondColumnStringValue.getString(adapter.getValue()));

            return new FixedSecondColumnRendererLabel(component, rightColumn);
        }
    }

    private static class FixedSecondColumnRendererLabel extends JRendererLabel
    {
        private FixedSecondColumnRendererLabel(Component leadingComponent, Component trailingComponent)
        {
            GroupLayout layout = new GroupLayout(this);
            setLayout(layout);

            layout.setHorizontalGroup(layout.createSequentialGroup()
                                            .addComponent(leadingComponent, GroupLayout.DEFAULT_SIZE, GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
                                            .addPreferredGap(LayoutStyle.ComponentPlacement.RELATED)
                                            .addComponent(trailingComponent));

            layout.setVerticalGroup(layout.createParallelGroup(GroupLayout.Alignment.BASELINE)
                                          .addComponent(leadingComponent)
                                          .addComponent(trailingComponent));
        }
    }

}

我想知道是否有正确的方法来使用这部分 API。我故意扩展JRendererLabel以防万一那是问题,但它似乎更微妙......

4

1 回答 1

2

如果你看一下JRendererLabel,你会看到invalidate, revalidate, validate(和一堆其他方法)被设置为“无操作”,这意味着它们不再向布局管理器发出组件应该布局的通知。这样做是为了帮助提高将渲染器“标记”到组件(JXList.)上的性能。

相反,请使用FixedSecondColumnRendererLabelJPanel.

与其在每次调用该方法时创建一个新实例,FixedSecondColumnRendererLabel不如JRendererLabel考虑对其进行优化,以便返回相同的实例,但每次调用该方法时都会配置一个实例。

请记住,此方法会为您的 中的每一行调用,JXList您拥有的越多,调用的次数越多,它将创建的短寿命对象越多......

于 2014-12-02T00:31:16.930 回答