使 GridBagLayout JPanel 大小适合内容,而不是容器?

Make GridBagLayout JPanel size to contents, not container?

我有一个 GridBagLayout JPanel 包含在 BorderLayout JPanel 中。我希望 GridBagLayout 能够根据其内部网格调整自身大小。相反,它将自身调整为 BorderLayout,在内部网格的两侧添加空白(GridBagLayout 中的所有权重均为 0)。

我的想法是 BorderLayout 提供空白(使用胶水)。相反, GridBagLayout 提供了空白;你可以通过查看我添加的 TitleBorder 看到这一点。

原因是当我将 GridBagLayout 发送到我的打印机时,我不希望包含所有空格。

import java.awt.*;
import javax.swing.*;

public class GBLDemo {
    public static void main (String[] args) {
        JFrame jframe = new JFrame("GridBagLayout Demo");
        jframe.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        Container contentPane = new JPanel(new BorderLayout());
        jframe.setContentPane(contentPane);

        GridBagConstraints gbc = new GridBagConstraints();
        JPanel gb = new JPanel(new GridBagLayout());
        contentPane.add(gb);

        gbc.gridx = gbc.gridy = 0;
        gb.add(new JLabel("Look "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("at"), gbc);         

        gbc.gridx = 0;
        gbc.gridy = 1;
        gb.add(new JLabel("the "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("border"), gbc);

        gb.setBorder(BorderFactory.createTitledBorder("Border"));

        jframe.pack();
        jframe.setVisible(true);
        jframe.setSize(640, 480);
    } // main(args)
} // GBLDemo

这是我想要的 ASCII 模型:

+-------------------------------------------+
|                                           |
|                                           |
|           +Border-----+                   |
|           |Look  at   |                   |
|           | the border|                   |
|           +-----------+                   |
|                                           |
|                                           |
+-------------------------------------------+

上面的代码生成的显示如下:

具有绝对布局(空)的解决方案如下所示:

import java.awt.Container;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;

import javax.swing.BorderFactory;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;

public class GBLDemo {
    public static void main (String[] args) {
        JFrame jframe = new JFrame("GridBagLayout Demo");
        jframe.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        Container contentPane = new JPanel();
        contentPane.setLayout(null);
        jframe.setContentPane(contentPane);
        jframe.setVisible(true);
        jframe.setSize(640, 480);
        GridBagConstraints gbc = new GridBagConstraints();
        JPanel gb = new JPanel(new GridBagLayout());
        contentPane.add(gb);

        gbc.gridx = gbc.gridy = 0;
        gb.add(new JLabel("Look "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("at"), gbc);         

        gbc.gridx = 0;
        gbc.gridy = 1;
        gb.add(new JLabel("the "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("border"), gbc);

        gb.setBorder(BorderFactory.createTitledBorder("Border"));
        gb.setSize(gb.getPreferredSize());
        gb.setLocation(jframe.getWidth()/2-gb.getWidth()/2, jframe.getHeight()/2-gb.getHeight()/2);

        jframe.addComponentListener(new ComponentAdapter() {
            @Override
            public void componentResized(ComponentEvent arg0) {
                gb.setSize(gb.getPreferredSize());
                gb.setLocation(jframe.getWidth()/2-gb.getWidth()/2, jframe.getHeight()/2-gb.getHeight()/2);
            }
        });        

    } // main(args)
} // GBLDemo

制作这个:

gb JPanel 将始终处于中心位置,ComponentListener 会负责这一点。 gp 将更改其大小以自动适应包含的对象 gb.setSize(gb.getPreferredSize()); .

将组件以其首选大小居中的一个好技巧(例如,问题代码中的 contentPane)是将其作为单个组件添加到具有网格包布局的容器中,而不指定约束。

这是上面的代码,经过调整就是为了做到这一点。

import java.awt.*;
import javax.swing.*;
import javax.swing.border.EmptyBorder;

public class GBLDemo {

    private JComponent ui = null;

    GBLDemo() {
        initUI();
    }

    private Container getContentPanel() {
        Container contentPane = new JPanel(new BorderLayout());

        GridBagConstraints gbc = new GridBagConstraints();
        JPanel gb = new JPanel(new GridBagLayout());
        contentPane.add(gb);

        gbc.gridx = gbc.gridy = 0;
        gb.add(new JLabel("Look "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("at"), gbc);         

        gbc.gridx = 0;
        gbc.gridy = 1;
        gb.add(new JLabel("the "), gbc);

        gbc.gridx = 1;
        gb.add(new JLabel("border"), gbc);

        gb.setBorder(BorderFactory.createTitledBorder("Border"));
        return contentPane;
    }

    public void initUI() {
        if (ui!=null) return;

        // A single object added to a grid bag layout with no constraint, 
        // will be centered within the available space.
        ui = new JPanel(new GridBagLayout()); 
        ui.setBorder(new EmptyBorder(4,4,4,4));

        ui.add(getContentPanel());
    }

    public JComponent getUI() {
        return ui;
    }

    public static void main(String[] args) {
        Runnable r = new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (Exception useDefault) {
                }
                GBLDemo o = new GBLDemo();

                JFrame f = new JFrame(o.getClass().getSimpleName());
                f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                f.setLocationByPlatform(true);

                f.setContentPane(o.getUI());
                f.pack();
                f.setMinimumSize(f.getSize());

                f.setVisible(true);
            }
        };
        SwingUtilities.invokeLater(r);
    }
}

您可以将带有 GridBagLayout 的另一个面板添加到 contentPane,然后将面板 gb 添加到该面板(无填充),而不是直接将面板 gb 添加到 contentPanel。

具有 BorderLayout 的 Panel 的 'center' 组件总是在 north/east/west/south 组件指定其首选尺寸后获得所有剩余的 space。所以上面的中间人组件将 'absorb' BorderLayout 给它的所有 space 并且,只要你没有添加带有 "fill" 约束的 gb,它应该显示出来以您想要的方式居中。