使 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,它应该显示出来以您想要的方式居中。
我有一个 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,它应该显示出来以您想要的方式居中。