1

我有一个JTree具有TreeCellEditor可变大小的 a (当您在编辑器中的一个组件中选择一个值时,会显示其他组件)。最初显示编辑器时,它的大小是正确的。但是,如果显示的附加组件导致其首选尺寸变大,则编辑器将不会被放大并且其内容将被剪裁。

我发现如果我设置编辑器的大小,它将调整大小,但我想要一种方法让它在首选大小发生变化时自动调整大小。

这是一个使用 aJTextField来模拟我的编辑器的示例。单击该按钮会使文本字段的首选大小变大,但在我的编辑器中,这将在编辑器组件中触发,我不想在此处显式设置大小,因为它用于除树单元格编辑器之外的上下文中。

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.event.ActionEvent;

import javax.swing.AbstractAction;
import javax.swing.DefaultCellEditor;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JTextField;
import javax.swing.JTree;

public class Test extends JFrame {
    @Override
    public void setVisible(boolean b) {
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        JTree tree = new JTree();
        tree.setEditable(true);
        final JTextField field = new JTextField();
        tree.setCellEditor(new DefaultCellEditor(field));

        JButton button = new JButton(new AbstractAction("Make CellEditor Wider") {
            @Override
            public void actionPerformed(ActionEvent e) {
                Dimension size = field.getPreferredSize();
                size.width += 50;
                field.setPreferredSize(size);
                field.setSize(size); // This forces the tree to show the correct size
            }
        });

        setLayout(new BorderLayout());
        add(tree, BorderLayout.CENTER);
        add(button, BorderLayout.SOUTH);
        setSize(500, 500);
        super.setVisible(b);
    }

    public static void main(String[] args) {
        javax.swing.SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                new Test().setVisible(true);
            }
        });
    }
}
4

1 回答 1

1

基本上,一棵树(或更准确地说:它的 ui-delegate)可以处理与同一节点上的渲染器组件具有不同大小的编辑器组件,尽管它们假设在单个节点编辑过程的生命周期内大小不会改变. 用户界面

  • 开始编辑时调整编辑器组件的大小/位置
  • 触发编辑器组件下方节点位置的更新(如果需要)

对于编辑器生命周期中的更改,自定义控制器必须监听这些更改(和/或任何触发这些更改),然后监听这两个任务。下面是一个 textField,其 prefSize 取决于其内容和一个监听 documentChanges 并根据需要操作的控制器。

// both height and width depend on content
final JTextField field = new JTextField() {

    @Override
    public Dimension getPreferredSize() {
        Dimension dim = super.getPreferredSize();
        int length = getText().length();
        dim.width += length * 10;
        dim.height += length * 2;
        return dim;
    }

};
// controller listens to changes to dynamically takes over the ui's job 
DocumentListener l = new DocumentListener() {

    protected void validateEditor(final JTextField field) {
        // the selectionModel's rowMapper is-a AbstractLayoutCache
        // BEWARE: implementation detail!
        TreeSelectionModel model = tree.getSelectionModel();
        // invalidate all cached node sizes/locations
        ((AbstractLayoutCache) model.getRowMapper()).invalidateSizes();
        // just a fancy cover method for revalidate/repaint
        tree.treeDidChange();
        // manually set the component's size
        field.setSize(field.getPreferredSize());
    }
    @Override
    public void insertUpdate(DocumentEvent e) {
        validateEditor(field);
    }

    @Override
    public void removeUpdate(DocumentEvent e) {
        validateEditor(field);
    }

    @Override
    public void changedUpdate(DocumentEvent e) {
    }

};
field.getDocument().addDocumentListener(l);
于 2013-10-11T09:32:56.723 回答