如何解决 JToggleButtons、JPanels 和 JScrollPanes 的这些大小调整问题?

How can I resolve these sizing issues with JToggleButtons, JPanels, and JScrollPanes?

我正在尝试开发一种手风琴菜单形式。有少量 (2-12) 个选项可以切换 on/off。开启后,将有一个 JPanel 以及可见的附加设置。关闭后,其他设置将不可见。

我创建了一个 SelectableExpandablePanel class,它扩展了 JPanel 并实现了 ActionListenerComponentListener。该面板包含两个东西 - JToggleButton 和 child Component(通常是 JPanel,但我不想限制自己将来重用这个概念) BoxLayout 强制执行一列。选择切换按钮时,child 变为可见。当取消选择切换时,child 被隐藏。

当我使用这个组件时,我打算将它放在 JScrollPane 内部的 JPanel 上,如示例 main 方法中所示。

似乎有两个问题我无法克服:

  1. 如果我没有指定 JFrame 大小,它只够每个 child 的宽度和三个按钮的高度。当我单击按钮时,我希望 JScrollPane 执行它的操作并生成一个垂直滚动条。这不会发生。

  2. 我希望切换按钮是包含它们的 JPanel 的整个宽度。我以为我在构造函数中所做的加上 Component Listener 会处理这个问题,但事实并非如此。

下面提供的编译器有一个main方法。如果编译并执行,它会驱动我正在构建的组件以提供测试框架和重现我正在谈论的问题的能力。

import javax.swing.BoxLayout;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JToggleButton;

import java.awt.Dimension;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.ComponentEvent;
import java.awt.event.ComponentListener;

public class SelectableExpandablePanel extends JPanel implements
        ActionListener, ComponentListener {
    private JToggleButton titleButton;
    private JComponent childComponent;

    public SelectableExpandablePanel(JComponent child) {
        this(child, null, null);
    }

    public SelectableExpandablePanel(JComponent child, String title) {
        this(child, title, null);
    }

    public SelectableExpandablePanel(JComponent child, String title,
            String tooltip) {
        super();

        if (child == null) {
            throw new IllegalArgumentException("Child component cannot be null");
        }

        childComponent = child;

        titleButton = new JToggleButton();
        titleButton.setText(title);
        titleButton.addActionListener(this);
        titleButton.setPreferredSize(new Dimension(getSize().width, titleButton
                .getPreferredSize().height));
        titleButton.setToolTipText(tooltip);

        setLayout(new BoxLayout(this, BoxLayout.Y_AXIS));
        setPreferredSize(new Dimension(childComponent.getPreferredSize().width,
                titleButton.getPreferredSize().height));
        setSize(new Dimension(childComponent.getPreferredSize().width,
                titleButton.getPreferredSize().height));

        add(titleButton);

        this.addComponentListener(this);
    }

    public void actionPerformed(ActionEvent e) {
        if (titleButton.isSelected()) {
            add(childComponent);
            setSize(new Dimension(childComponent.getPreferredSize().width,
                    titleButton.getPreferredSize().height
                            + childComponent.getPreferredSize().height));

        } else {
            remove(childComponent);
            setSize(new Dimension(childComponent.getPreferredSize().width,
                    titleButton.getPreferredSize().height));
        }

        invalidate();
        revalidate();
    }

    public void componentHidden(ComponentEvent arg0) {
        // Do nothing
    }

    public void componentMoved(ComponentEvent arg0) {
        // Do nothing
    }

    public void componentResized(ComponentEvent arg0) {
        titleButton.setSize(this.getWidth(),
                titleButton.getPreferredSize().height);
    }

    public void componentShown(ComponentEvent arg0) {
        // Do nothing
    }

    public static void main(String[] args) {
        JScrollPane scrollPane = new JScrollPane();

        // These panels simulates a complex, multi-line configuration panel.
        JPanel testPanel = new JPanel();
        testPanel.setLayout(new BoxLayout(testPanel, BoxLayout.Y_AXIS));
        testPanel.add(new JLabel("Test JLabel"));
        testPanel.add(new JLabel("Test JLabel 2"));
        testPanel.add(new JLabel("Test JLabel 3"));

        JPanel testPanel2 = new JPanel();
        testPanel2.setLayout(new BoxLayout(testPanel2, BoxLayout.Y_AXIS));
        testPanel2.add(new JLabel("Test JLabel"));
        testPanel2.add(new JLabel("Test JLabel 2"));
        testPanel2.add(new JLabel("Test JLabel 3"));

        JPanel testPanel3 = new JPanel();
        testPanel3.setLayout(new BoxLayout(testPanel3, BoxLayout.Y_AXIS));
        testPanel3.add(new JLabel("Test JLabel"));
        testPanel3.add(new JLabel("Test JLabel 2"));
        testPanel3.add(new JLabel("Test JLabel 3"));

        // This panel simulates the panel that will contain each of the
        // SelectableExpandablePanels.
        JPanel testHolder = new JPanel();
        testHolder.setLayout(new BoxLayout(testHolder, BoxLayout.Y_AXIS));
        testHolder.add(new SelectableExpandablePanel(testPanel, "Test"));
        testHolder.add(new SelectableExpandablePanel(testPanel2, "Test 2"));
        testHolder.add(new SelectableExpandablePanel(testPanel3, "Test 3"));

        // We add the test holder to the scroll pane. The intention is that if
        // the expansion is too big to fit, the holding JFrame won't expand, but
        // the scroll pane will get scroll bars to let the user scroll up and
        // down through the toggle buttons and any enabled items.
        scrollPane.setViewportView(testHolder);

        JFrame testFrame = new JFrame("Expandable Panel Test");
        testFrame.getContentPane().add(scrollPane);
        testFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        testFrame.pack();
        testFrame.setVisible(true);
    }
}

不要尝试自己管理尺寸:

//titleButton.setPreferredSize(new Dimension(getSize().width, titleButton.getPreferredSize().height));
 titleButton.setToolTipText(tooltip);

 setLayout(new BoxLayout(this, BoxLayout.Y_AXIS));
//setPreferredSize(new Dimension(childComponent.getPreferredSize().width, titleButton.getPreferredSize().height));
//setSize(new Dimension(childComponent.getPreferredSize().width, titleButton.getPreferredSize().height));

此外,删除 ActionListener 中的 setSize() 代码。这将被忽略,因为布局管理器将决定大小。

当面板的首选尺寸大于滚动窗格的尺寸时,将出现滚动条。如果您对首选尺寸进行硬编码,那么您默认了布局管理器的用途,并且首选尺寸不会随着您 add/remove 个组件而改变。

注意像这样的东西我通常使用 BorderLayout。将按钮放在 PAGE_START 中,将另一个面板放在 CENTER 中。组件将自动填充 space available.

  1. 删除所有 setSize/setPreferredSize 调用,让 LayoutManager 做它的事情。
  2. 要允许 JButton 填充面板的宽度,您可以使用 BorderLayout(例如,将按钮添加到 CENTER,将子容器添加到 SOUTH,然后删除所有这些 setSize 值以让 LayoutManager 处理它).