使用Java布局管理器

时间:2012-04-30 23:38:54

标签: java layout gridbaglayout flowlayout

我创建了一个JPanel,它包含任意数量的JLabel,使用左对齐的流布局管理器进行布局。让我们称之为FlowPanel。

每个FlowPanel都有一个深度,我希望将这些FlowPanel的任意数量叠加在一起,每个FlowPanel缩进一个固定的量*深度。最终目标是在JScrollPane中使用垂直滚动条(如果需要)显示这些堆叠面板,而不是水平滚动条。

我尝试使用GridBagLayout堆叠FlowPanels,虽然很容易让它们垂直堆叠并按我想要的方式缩进,但我无法弄清楚如何获得正确的宽度。也就是说,FlowPanels只是从主面板的右侧运行,不关注显示的宽度。

我尝试过使用setPreferredSize,setSize,搞乱GridBagConstraints ......没有运气。

请帮忙! :)

一些示例代码来演示问题。此代码有2个问题:   1.即使只有几行,行也应该是恒定的高度。 2.当FlowPanel包含太多元素时,行会在屏幕右侧滚动。

public class FlowExample {

public static void main(String[] args) throws Exception {
    JFrame frame = new JFrame();

    StackPanel stackPanel = new StackPanel();
    frame.getContentPane().setLayout(new BorderLayout());
    frame.getContentPane().add(
            new JScrollPane(stackPanel, JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED, JScrollPane.HORIZONTAL_SCROLLBAR_NEVER),
            BorderLayout.CENTER);

    frame.setLocation(200, 200);
    frame.setSize(300, 300);
    frame.setVisible(true);

    List<FlowPanel> content = new ArrayList<FlowPanel>();
    for (int i = 0; i < 20; i++) {
        FlowPanel flowPanel = new FlowPanel((int) (4 * Math.random()));
        for (int j = 0; j < (int) (20 * Math.random()); j++) {
            flowPanel.add(new JLabel("label " + j));
        }
        content.add(flowPanel);
        stackPanel.layoutGUI(content);
        Thread.sleep(1000);
    }

    System.in.read();
    System.exit(0);
}
}

class StackPanel extends JPanel {

public StackPanel() {
    setLayout(new BorderLayout());
}

public void layoutGUI(final List<FlowPanel> content) {
    if (!SwingUtilities.isEventDispatchThread()) {
        SwingUtilities.invokeLater(new Runnable() {

            @Override
            public void run() {
                layoutGUI(content);
            }
        });
        return;
    }

    JPanel mainPanel = new JPanel();
    mainPanel.setLayout(new GridLayout(content.size(), 1));
    for (FlowPanel flowPanel : content) {
        flowPanel.setBorder(BorderFactory.createEmptyBorder(0, flowPanel.getDepth() * 10, 0, 0));
        mainPanel.add(flowPanel);
    }

    removeAll();
    add(mainPanel, BorderLayout.CENTER);
    revalidate();
}
}

class FlowPanel extends JPanel {

private int depth;

public FlowPanel(int depth) {
    setLayout(new FlowLayout(FlowLayout.LEFT));
    this.depth = depth;
}

public int getDepth() {
    return depth;
}
}

这是使用GridBagLayout布局主面板的一些示例代码。这是我得到的最好的结果,但正如你所看到的,FlowPanels仍然没有包装......

    JPanel mainPanel = new JPanel();
    mainPanel.setLayout(new GridBagLayout());
    GridBagConstraints constraints = new GridBagConstraints();
    constraints.weightx = 1;
    constraints.gridwidth = GridBagConstraints.REMAINDER;
    constraints.anchor = GridBagConstraints.LINE_START;

    for (FlowPanel flowPanel : content) {
        System.out.println("setting border to " + flowPanel.getDepth());
        flowPanel.setBorder(BorderFactory.createEmptyBorder(0, flowPanel.getDepth() * 10, 0, 0));
        mainPanel.add(flowPanel, constraints);
    }

    constraints.weighty = 1;
    mainPanel.add(new JPanel(), constraints);

    removeAll();
    add(mainPanel, BorderLayout.CENTER);
    revalidate();

1 个答案:

答案 0 :(得分:1)

GridLayout正在尝试将所有内容都放在屏幕上,因此通过将其传递给列表,每次都会重新调整高度。通过从一开始就设置一个大的常数,你就可以告诉它留出足够的空间。尝试修改此行:

mainPanel.setLayout(new GridLayout(content.size(), 1));

mainPanel.setLayout(new GridLayout(500, 1));

要停止从一开始就出现的滚动条,另一种方法是使用BoxLayout。

JPanel mainPanel = new JPanel();
Box box = Box.createVerticalBox();
mainPanel.add( box );
for (FlowPanel flowPanel : content) {
    flowPanel.setBorder(BorderFactory.createEmptyBorder(0, flowPanel.getDepth() * 10, 0, 0));
    box.add(flowPanel);
}
removeAll();
add(mainPanel, BorderLayout.CENTER);
revalidate();

包装问题仅限于关注setPreferredSize的FlowLayout。您可以使用现有代码设置一个首选大小,以满足您的动态调整大小要求。或者,我过去使用过camickr的WrapLayout,效果很好。

http://tips4java.wordpress.com/2008/11/06/wrap-layout/

原样,WrapLayout返回不同的首选大小,具体取决于是否需要包装,懒得避免快速入侵WrapLayout.java:

1)。添加最小宽度变量

private int minWidth = -1;

2)。修改最后一行的'layoutSize'方法:

if(minWidth != -1) {
    dim.width = minWidth;
}
return dim;

完成后,对示例代码的唯一更改是在FlowPanel构造函数中:

public FlowPanel(int depth) {
    WrapLayout layout = new WrapLayout(FlowLayout.LEFT);
    layout.setMinWidth(300);
    setLayout(layout);
    setSize(new Dimension(300, 30));
}