原文: http://zetcode.com/tutorials/javaswingtutorial/swinglayoutmanagement/

Java Swing 具有两种组件:容器和子组件。 容器将子项分组为合适的布局。 要创建布局,我们使用布局管理器。

Tweet

ZetCode 为 Swing 布局管理过程提供了 196 页专门的电子书: Java Swing 布局管理教程

Swing 布局管理器

Swing 有大量可用的布局管理器-内置和第三方。 但是,大多数管理器都不适合现代 UI 的创建。

有三种可以正确完成工作的布局管理器:

  • MigLayout
  • GroupLayout
  • FormLayout

MigLayoutGroupLayoutFormLayout是强大,灵活的布局管理器,可以满足大多数布局要求。 在本教程中,我们使用GroupLayout管理器来设计用户界面。

以下布局管理器已过时:

  • FlowLayout
  • GridLayout
  • CardLayout
  • BoxLayout
  • GridBagLayout

这些布局管理器无法满足现代 UI 的要求。

过时的管理器遇到的问题

过时的管理器要么太简单(FlowLayoutGridLayout),要么就是不必要的复杂(GridBagLayout)。 所有这些管理器都有一个基本的设计错误:他们使用组件之间的固定间隙。 在组件之间使用刚性空间是不可移植的:一旦在不同的屏幕分辨率下运行程序,用户界面就会损坏。

过时的管理器试图通过一种称为嵌套的技术来修复其弱点。 在嵌套中,开发者在多个面板中使用几个不同的布局管理器。 尽管可以使用嵌套创建 UI,但它给代码带来了额外的不必要的复杂性。

过时的管理器

在本节中,我们将介绍过时的布局管理器。 不建议使用这些管理器。 如果我们需要维护一些遗留代码,只需花一些时间研究它们。 否则,应拒绝使用它们。

FlowLayout管理器

这是 Java Swing 工具箱中最简单的布局管理器。 它是JPanel组件的默认布局管理器。

它是如此简单,以至于不能用于任何实际布局。 许多 Java Swing 教程都介绍了该管理器,因此,初学者尝试在其项目中使用它,而没有意识到它不能用于任何严重的事情。

在计算其子大小时,流程布局使每个组件都假定其自然(首选)大小。 管理器将组件排成一排。 按顺序添加了它们。 如果它们不适合一排,则进入下一排。 可以从右向左添加组件,反之亦然。 管理器允许对齐组件。 隐式地,组件居中,并且组件之间以及组件的边缘与容器的边缘之间有 5px 的空间。

  1. FlowLayout()
  2. FlowLayout(int align)
  3. FlowLayout(int align, int hgap, int vgap)

FlowLayout 管理器有三个可用的构造器。 第一个创建具有隐式值的管理器。 以 5px 水平和垂直空间居中。 其他允许指定这些参数。

FlowLayoutEx.java

  1. package com.zetcode;
  2. import javax.swing.JButton;
  3. import javax.swing.JFrame;
  4. import javax.swing.JPanel;
  5. import javax.swing.JTextArea;
  6. import javax.swing.JTree;
  7. import java.awt.Dimension;
  8. import java.awt.EventQueue;
  9. public class FlowLayoutEx extends JFrame {
  10. public FlowLayoutEx() {
  11. initUI();
  12. }
  13. private void initUI() {
  14. var panel = new JPanel();
  15. var button = new JButton("button");
  16. panel.add(button);
  17. var tree = new JTree();
  18. panel.add(tree);
  19. var area = new JTextArea("text area");
  20. area.setPreferredSize(new Dimension(100, 100));
  21. panel.add(area);
  22. add(panel);
  23. pack();
  24. setTitle("FlowLayout example");
  25. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  26. setLocationRelativeTo(null);
  27. }
  28. public static void main(String[] args) {
  29. EventQueue.invokeLater(() -> {
  30. var ex = new FlowLayoutEx();
  31. ex.setVisible(true);
  32. });
  33. }
  34. }

该示例显示了窗口中的按钮,树组件和文本区域组件。 如果我们创建一个空树组件,则组件内部会有一些默认值。

  1. var panel = new JPanel();

JPanel组件的隐式布局管理器是FlowLayout。 我们不必手动设置。

  1. var area = new JTextArea("text area");
  2. area.setPreferredSize(new Dimension(100, 100));

流布局管理器为其组件设置首选大小。 因此,在我们的案例中,区域组件将为100x100px。 如果我们未设置首选大小,则组件将具有其文本的大小。 没有文本,该组件将根本不可见。 尝试在区域组件中编写或删除一些文本。 组件将相应地增长和收缩。

  1. panel.add(area);

该组件放置在带有add()的容器内。

Swing 布局管理 - 图1

图:FlowLayout

GridLayout

GridLayout布局管理器将组件布置在矩形网格中。 容器分为大小相等的矩形。 每个矩形中放置一个组件。

GridLayout非常简单,不能用于任何实际布局。

GridLayoutEx.java

  1. package com.zetcode;
  2. import javax.swing.BorderFactory;
  3. import javax.swing.JButton;
  4. import javax.swing.JFrame;
  5. import javax.swing.JLabel;
  6. import javax.swing.JPanel;
  7. import java.awt.EventQueue;
  8. import java.awt.GridLayout;
  9. public class GridLayoutEx extends JFrame {
  10. public GridLayoutEx() {
  11. initUI();
  12. }
  13. private void initUI() {
  14. var panel = new JPanel();
  15. panel.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
  16. panel.setLayout(new GridLayout(5, 4, 5, 5));
  17. String[] buttons = {
  18. "Cls", "Bck", "", "Close", "7", "8", "9", "/", "4",
  19. "5", "6", "*", "1", "2", "3", "-", "0", ".", "=", "+"
  20. };
  21. for (int i = 0; i < buttons.length; i++) {
  22. if (i == 2) {
  23. panel.add(new JLabel(buttons[i]));
  24. } else {
  25. panel.add(new JButton(buttons[i]));
  26. }
  27. }
  28. add(panel);
  29. setTitle("GridLayout");
  30. setSize(350, 300);
  31. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  32. setLocationRelativeTo(null);
  33. }
  34. public static void main(String[] args) {
  35. EventQueue.invokeLater(() -> {
  36. var ex = new GridLayoutEx();
  37. ex.setVisible(true);
  38. });
  39. }
  40. }

该示例显示了一个简单计算器工具的框架。 我们在管理器中放入了 19 个按钮和一个标签。 请注意,每个按钮的大小均相同。

  1. panel.setLayout(new GridLayout(5, 4, 5, 5));

在这里,我们为面板组件设置了网格布局管理器。 布局管理器采用四个参数。 行数,列数以及组件之间的水平和垂直间隙。

Swing 布局管理 - 图2

图:GridLayout

BorderLayout

BorderLayout是一个简单的布局管理器,在某些布局中可以派上用场。 它是JFrameJWindowJDialogJInternalFrameJApplet的默认布局管理器。 它有一个严重的局限性-它以像素为单位设置子元素之间的间隙,从而创建了刚性的布局。 这导致了不可移植的 UI,因此不建议使用它。

BorderLayout将空间分为五个区域:北,西,南,东和中心。 每个区域只能有一个组件。 如果需要在一个区域中放置更多组件,则必须在其中放置一个由我们选择的管理器组成的小组。 N,W,S,E 区域中的组件具有首选大小。 中间的组件占据了剩余的整个空间。

如果子组件彼此之间距离太近,则看起来不太好。 我们必须在它们之间留一些空间。 Swing 工具箱中的每个组件的边缘都可以带有边框。 要创建边框,我们可以创建EmptyBorder类的新实例,或者使用BorderFactory

BorderEx.java

  1. package com.zetcode;
  2. import javax.swing.JFrame;
  3. import javax.swing.JPanel;
  4. import javax.swing.border.EmptyBorder;
  5. import java.awt.BorderLayout;
  6. import java.awt.Color;
  7. import java.awt.Dimension;
  8. import java.awt.EventQueue;
  9. import java.awt.Insets;
  10. public class BorderLayoutEx extends JFrame {
  11. public BorderLayoutEx() {
  12. initUI();
  13. }
  14. private void initUI() {
  15. var bottomPanel = new JPanel(new BorderLayout());
  16. var topPanel = new JPanel();
  17. topPanel.setBackground(Color.gray);
  18. topPanel.setPreferredSize(new Dimension(250, 150));
  19. bottomPanel.add(topPanel);
  20. bottomPanel.setBorder(new EmptyBorder(new Insets(20, 20, 20, 20)));
  21. add(bottomPanel);
  22. pack();
  23. setTitle("BorderLayout");
  24. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  25. setLocationRelativeTo(null);
  26. }
  27. public static void main(String[] args) {
  28. EventQueue.invokeLater(() -> {
  29. var ex = new BorderLayoutEx();
  30. ex.setVisible(true);
  31. });
  32. }
  33. }

该示例将显示一个灰色面板及其周围的边框。

  1. var bottomPanel = new JPanel(new BorderLayout());
  2. var topPanel = new JPanel();

我们将面板放入面板中。 底部面板具有BorderLayout管理器。

  1. bottomPanel.add(topPanel);

在这里,我们将顶部面板放入底部面板组件中。 更准确地说,我们将其放置在BorderLayout管理器的中心区域。

  1. bottomPanel.setBorder(new EmptyBorder(new Insets(20, 20, 20, 20)));

在这里,我们在底部面板周围创建了 20px 的边框。 边框值如下:上,左,下和右。 请注意,创建固定的嵌入(空格)不是可移植的。

Swing 布局管理 - 图3

图:BorderLayout

下一个示例显示BorderLayout管理器的典型用法。

BorderLayoutEx2.java

  1. package com.zetcode;
  2. import javax.swing.ImageIcon;
  3. import javax.swing.JButton;
  4. import javax.swing.JFrame;
  5. import javax.swing.JLabel;
  6. import javax.swing.JMenu;
  7. import javax.swing.JMenuBar;
  8. import javax.swing.JTextArea;
  9. import javax.swing.JToolBar;
  10. import javax.swing.border.EmptyBorder;
  11. import java.awt.BorderLayout;
  12. import java.awt.EventQueue;
  13. import java.awt.Insets;
  14. public class BorderLayoutEx2 extends JFrame {
  15. public BorderLayoutEx2() {
  16. initUI();
  17. }
  18. private void initUI() {
  19. var menubar = new JMenuBar();
  20. var fileMenu = new JMenu("File");
  21. menubar.add(fileMenu);
  22. setJMenuBar(menubar);
  23. var toolbar = new JToolBar();
  24. toolbar.setFloatable(false);
  25. var exitIcon = new ImageIcon("src/resources/exit.png");
  26. var exitBtn = new JButton(exitIcon);
  27. exitBtn.setBorder(new EmptyBorder(0, 0, 0, 0));
  28. toolbar.add(exitBtn);
  29. add(toolbar, BorderLayout.NORTH);
  30. var vertical = new JToolBar(JToolBar.VERTICAL);
  31. vertical.setFloatable(false);
  32. vertical.setMargin(new Insets(10, 5, 5, 5));
  33. var driveIcon = new ImageIcon("src/resources/drive.png");
  34. var compIcon = new ImageIcon("src/resources/computer.png");
  35. var printIcon = new ImageIcon("src/resources/printer.png");
  36. var driveBtn = new JButton(driveIcon);
  37. driveBtn.setBorder(new EmptyBorder(3, 0, 3, 0));
  38. var compBtn = new JButton(compIcon);
  39. compBtn.setBorder(new EmptyBorder(3, 0, 3, 0));
  40. var printBtn = new JButton(printIcon);
  41. printBtn.setBorder(new EmptyBorder(3, 0, 3, 0));
  42. vertical.add(driveBtn);
  43. vertical.add(compBtn);
  44. vertical.add(printBtn);
  45. add(vertical, BorderLayout.WEST);
  46. add(new JTextArea(), BorderLayout.CENTER);
  47. var statusbar = new JLabel(" Statusbar");
  48. add(statusbar, BorderLayout.SOUTH);
  49. setSize(400, 350);
  50. setTitle("BorderLayout");
  51. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  52. setLocationRelativeTo(null);
  53. }
  54. public static void main(String[] args) {
  55. EventQueue.invokeLater(() -> {
  56. var ex = new BorderLayoutEx2();
  57. ex.setVisible(true);
  58. });
  59. }
  60. }

该示例显示了典型的应用框架。 我们显示了一个垂直和水平工具栏,一个状态栏和一个中央组件(文本区域)。

BorderLayoutJFrame容器的默认布局管理器。 因此,我们不必显式设置它。

  1. add(toolbar, BorderLayout.NORTH);

我们将工具栏放置在布局的北部。

  1. var driveBtn = new JButton(driveIcon);
  2. driveBtn.setBorder(new EmptyBorder(3, 0, 3, 0));

为了在按钮周围留一些空白,我们必须使用EmptyBorder。 这会在按钮的顶部和底部增加一些固定的空间。 当我们添加固定空间时,UI 不可移植。 3px 的空间在 1280x720 的屏幕上可能看起来不错,但在1920x1200px的屏幕上不合适。

  1. add(vertical, BorderLayout.WEST);

我们将垂直太杆放在西边。

  1. add(new JTextArea(), BorderLayout.CENTER);

我们将文本区域放置在中间。

  1. add(statusbar, BorderLayout.SOUTH);

状态栏转到南部区域。

Swing 布局管理 - 图4

图:BorderLayout 2

CardLayout

CardLayout是一个简单的布局管理器,将每个组件都视为卡。 容器是这些卡的栈。 一次只能看到一个组件。 其余的隐藏。 最初显示容器时,默认情况下将显示添加到容器的第一个组件。 该管理器的实际用途有限。 它可用于创建向导或选项卡式窗格。

以下示例使用CardLayout管理器创建图像库。 我们使用了 Krasna Horka 城堡的四张图片(2012 年大火之前)。

CardLayoutEx.java

  1. package com.zetcode;
  2. import java.awt.BorderLayout;
  3. import java.awt.CardLayout;
  4. import java.awt.Color;
  5. import java.awt.EventQueue;
  6. import javax.swing.BorderFactory;
  7. import javax.swing.ImageIcon;
  8. import javax.swing.JButton;
  9. import javax.swing.JFrame;
  10. import javax.swing.JLabel;
  11. import javax.swing.JPanel;
  12. public class CardLayoutEx extends JFrame {
  13. private ImageIcon horka1;
  14. private ImageIcon horka2;
  15. private ImageIcon horka3;
  16. private ImageIcon horka4;
  17. private ImageIcon previ;
  18. private ImageIcon nexti;
  19. private JPanel mainPanel;
  20. private CardLayout cardLayout;
  21. public CardLayoutEx() {
  22. initUI();
  23. }
  24. private void initUI() {
  25. mainPanel = new JPanel();
  26. mainPanel.setBackground(new Color(50, 50, 50));
  27. mainPanel.setBorder(
  28. BorderFactory.createEmptyBorder(5, 5, 5, 5)
  29. );
  30. cardLayout = new CardLayout();
  31. mainPanel.setLayout(cardLayout);
  32. horka1 = new ImageIcon("src/resources/horka1.jpg");
  33. horka2 = new ImageIcon("src/resources/horka2.jpg");
  34. horka3 = new ImageIcon("src/resources/horka3.jpg");
  35. horka4 = new ImageIcon("src/resources/horka4.jpg");
  36. previ = new ImageIcon("src/resources/previous.png");
  37. nexti = new ImageIcon("src/resources/next.png");
  38. var label1 = new JLabel(horka1);
  39. var label2 = new JLabel(horka2);
  40. var label3 = new JLabel(horka3);
  41. var label4 = new JLabel(horka4);
  42. mainPanel.add(label1);
  43. mainPanel.add(label2);
  44. mainPanel.add(label3);
  45. mainPanel.add(label4);
  46. add(mainPanel);
  47. var prevButton = new JButton(previ);
  48. prevButton.addActionListener((e) -> cardLayout.previous(mainPanel));
  49. var nextButton = new JButton(nexti);
  50. nextButton.addActionListener((e) -> cardLayout.next(mainPanel));
  51. var btnPanel = new JPanel();
  52. btnPanel.setBackground(new Color(50, 50, 50));
  53. btnPanel.add(prevButton);
  54. btnPanel.add(nextButton);
  55. add(btnPanel, BorderLayout.SOUTH);
  56. pack();
  57. setTitle("Gallery");
  58. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  59. setLocationRelativeTo(null);
  60. }
  61. public static void main(String[] args) {
  62. EventQueue.invokeLater(() -> {
  63. var ex = new CardLayoutEx();
  64. ex.setVisible(true);
  65. });
  66. }
  67. }

我们创建两个按钮来浏览图像。

  1. mainPanel = new JPanel();
  2. mainPanel.setBackground(new Color(50, 50, 50));
  3. mainPanel.setBorder(
  4. BorderFactory.createEmptyBorder(5, 5, 5, 5)
  5. );

我们创建主面板组件。 我们将其颜色设置为深灰色。 我们将 5px 放置在面板周围,以使它的子级不太靠近窗口的边界。

  1. cardLayout = new CardLayout();
  2. mainPanel.setLayout(cardLayout);

CardLayout管理器已创建并设置到主面板。

  1. mainPanel.add(label1);
  2. mainPanel.add(label2);
  3. mainPanel.add(label3);
  4. mainPanel.add(label4);

显示图像的标签组件将添加到面板中。

  1. var prevButton = new JButton(previ);
  2. prevButton.addActionListener((e) -> cardLayout.previous(mainPanel));

单击上一个按钮,将调用管理器的previous()方法。 它会翻转到指定容器的上一张卡片。

  1. add(mainPanel);

我们将主面板添加到框架组件边框布局的中心区域。 如果未明确指定放置组件的位置,则会将其添加到中心区域。

  1. var btnPanel = new JPanel();
  2. btnPanel.setBackground(new Color(50, 50, 50));
  3. btnPanel.add(prevButton);
  4. btnPanel.add(nextButton);

这些按钮将添加到按钮面板。

  1. add(btnPanel, BorderLayout.SOUTH);

最后,将带有按钮的面板放入BorderLayout管理器的南部区域。

Swing 布局管理 - 图5

图:CardLayout

BoxLayout

BoxLayout管理器是一个简单的布局管理器,用于组织列或行中的组件。 它可以使用嵌套创建非常复杂的布局。 但是,这增加了布局创建的复杂性,并使用了额外的资源,尤其是许多其他JPanel组件。 BoxLayout仅能创建固定空间; 因此,其布局不可移植。

BoxLayout具有以下构造器:

  1. BoxLayout(Container target, int axis)

构造器创建一个布局管理器,该管理器将沿给定轴布置组件。 与其他布局管理器不同,BoxLayout将容器实例作为构造器中的第一个参数。 第二个参数确定管理器的方向。 要创建一个水平框,我们可以使用LINE_AXIS常量。 要创建一个垂直框,我们可以使用PAGE_AXIS常量。

框布局管理器通常与Box类一起使用。 此类创建一些不可见的组件,这些组件会影响最终布局。

  • 胶水
  • 支撑
  • 刚性区域

假设我们要在窗口的右下角放置两个按钮。

BoxLayoutButtonsEx.java

  1. package com.zetcode;
  2. import javax.swing.Box;
  3. import javax.swing.BoxLayout;
  4. import javax.swing.JButton;
  5. import javax.swing.JFrame;
  6. import javax.swing.JPanel;
  7. import java.awt.Dimension;
  8. import java.awt.EventQueue;
  9. public class BoxLayoutButtonsEx extends JFrame {
  10. public BoxLayoutButtonsEx() {
  11. initUI();
  12. }
  13. private void initUI() {
  14. var basePanel = new JPanel();
  15. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));
  16. add(basePanel);
  17. basePanel.add(Box.createVerticalGlue());
  18. var bottomPanel = new JPanel();
  19. bottomPanel.setAlignmentX(1f);
  20. bottomPanel.setLayout(new BoxLayout(bottomPanel, BoxLayout.X_AXIS));
  21. var okBtn = new JButton("OK");
  22. var closeBtn = new JButton("Close");
  23. bottomPanel.add(okBtn);
  24. bottomPanel.add(Box.createRigidArea(new Dimension(5, 0)));
  25. bottomPanel.add(closeBtn);
  26. bottomPanel.add(Box.createRigidArea(new Dimension(15, 0)));
  27. basePanel.add(bottomPanel);
  28. basePanel.add(Box.createRigidArea(new Dimension(0, 15)));
  29. setTitle("Two Buttons");
  30. setSize(300, 150);
  31. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  32. setLocationRelativeTo(null);
  33. }
  34. public static void main(String[] args) {
  35. EventQueue.invokeLater(() -> {
  36. var ex = new BoxLayoutButtonsEx();
  37. ex.setVisible(true);
  38. });
  39. }
  40. }

下图说明了该示例。

Swing 布局管理 - 图6

图:两个按钮

我们创建两个面板。 基本面板具有垂直框布局。 底部面板有一个水平面板。 我们将底板插入底板。 底部面板右对齐。 窗口顶部和底部面板之间的空间是可扩展的。 这是通过垂直胶水实现的。

  1. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));

在这里,我们使用垂直BoxLayout创建一个基础面板。

  1. var bottomPanel = new JPanel();
  2. bottomPanel.setAlignmentX(1f);
  3. bottomPanel.setLayout(new BoxLayout(bottomPanel, BoxLayout.X_AXIS));

底部面板右对齐。 这是通过setAlignmentX()方法完成的。 面板具有水平布局。

  1. bottomPanel.add(Box.createRigidArea(new Dimension(5, 0)));

我们在按钮之间留出一些刚性空间。

  1. basePanel.add(bottomPanel);

在这里,我们将具有水平框布局的底部面板放置到垂直基础面板上。

  1. basePanel.add(Box.createRigidArea(new Dimension(0, 15)));

我们还在底部面板和窗口边框之间留出一些空间。

Swing 布局管理 - 图7

图:BoxLayout按钮示例

当使用BoxLayout管理器时,可以在组件之间设置一个刚性区域。

BoxLayoutRigidAreaEx.java

  1. package com.zetcode;
  2. import javax.swing.Box;
  3. import javax.swing.BoxLayout;
  4. import javax.swing.JButton;
  5. import javax.swing.JFrame;
  6. import javax.swing.JPanel;
  7. import javax.swing.border.EmptyBorder;
  8. import java.awt.Dimension;
  9. import java.awt.EventQueue;
  10. import java.awt.Insets;
  11. public class BoxLayoutRigidAreaEx extends JFrame {
  12. public BoxLayoutRigidAreaEx() {
  13. initUI();
  14. }
  15. private void initUI() {
  16. var basePanel = new JPanel();
  17. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));
  18. basePanel.setBorder(new EmptyBorder(new Insets(40, 60, 40, 60)));
  19. basePanel.add(new JButton("Button"));
  20. basePanel.add(Box.createRigidArea(new Dimension(0, 5)));
  21. basePanel.add(new JButton("Button"));
  22. basePanel.add(Box.createRigidArea(new Dimension(0, 5)));
  23. basePanel.add(new JButton("Button"));
  24. basePanel.add(Box.createRigidArea(new Dimension(0, 5)));
  25. basePanel.add(new JButton("Button"));
  26. add(basePanel);
  27. pack();
  28. setTitle("RigidArea");
  29. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  30. setLocationRelativeTo(null);
  31. }
  32. public static void main(String[] args) {
  33. EventQueue.invokeLater(() -> {
  34. var ex = new BoxLayoutRigidAreaEx();
  35. ex.setVisible(true);
  36. });
  37. }
  38. }

在此示例中,我们显示四个按钮。 默认情况下,按钮之间没有空格。 为了在其中留出一些空间,我们增加了一些刚性区域。

  1. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));

我们为面板使用垂直BoxLayout管理器。

  1. basePanel.add(new JButton("Button"));
  2. basePanel.add(Box.createRigidArea(new Dimension(0, 5)));
  3. basePanel.add(new JButton("Button"));

我们添加按钮并使用Box.createRigidArea()在它们之间创建一个刚性区域。

Swing 布局管理 - 图8

图:RigidArea

每日提示

下一个示例创建“每日提示”窗口对话框。 我们结合使用各种布局管理器。

TipOfDayEx.java

  1. package com.zetcode;
  2. import javax.swing.BorderFactory;
  3. import javax.swing.BoxLayout;
  4. import javax.swing.ImageIcon;
  5. import javax.swing.JButton;
  6. import javax.swing.JCheckBox;
  7. import javax.swing.JDialog;
  8. import javax.swing.JLabel;
  9. import javax.swing.JPanel;
  10. import javax.swing.JSeparator;
  11. import javax.swing.JTextPane;
  12. import java.awt.BorderLayout;
  13. import java.awt.Color;
  14. import java.awt.Dimension;
  15. import java.awt.EventQueue;
  16. import java.awt.FlowLayout;
  17. import java.awt.event.KeyEvent;
  18. public class TipOfDayEx extends JDialog {
  19. public TipOfDayEx() {
  20. initUI();
  21. }
  22. private void initUI() {
  23. var basePanel = new JPanel();
  24. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));
  25. add(basePanel);
  26. var topPanel = new JPanel(new BorderLayout(0, 0));
  27. topPanel.setMaximumSize(new Dimension(450, 0));
  28. var hint = new JLabel("Productivity Hints");
  29. hint.setBorder(BorderFactory.createEmptyBorder(0, 25, 0, 0));
  30. topPanel.add(hint);
  31. var icon = new ImageIcon("src/resources/coffee2.png");
  32. var label = new JLabel(icon);
  33. label.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
  34. topPanel.add(label, BorderLayout.EAST);
  35. var separator = new JSeparator();
  36. separator.setForeground(Color.gray);
  37. topPanel.add(separator, BorderLayout.SOUTH);
  38. basePanel.add(topPanel);
  39. var textPanel = new JPanel(new BorderLayout());
  40. textPanel.setBorder(BorderFactory.createEmptyBorder(15, 25, 15, 25));
  41. var pane = new JTextPane();
  42. pane.setContentType("text/html");
  43. var text = "<p><b>Closing windows using the mouse wheel</b></p>" +
  44. "<p>Clicking with the mouse wheel on an editor tab closes the window. " +
  45. "This method works also with dockable windows or Log window tabs.</p>";
  46. pane.setText(text);
  47. pane.setEditable(false);
  48. textPanel.add(pane);
  49. basePanel.add(textPanel);
  50. var boxPanel = new JPanel(new FlowLayout(FlowLayout.LEFT, 20, 0));
  51. var box = new JCheckBox("Show Tips at startup");
  52. box.setMnemonic(KeyEvent.VK_S);
  53. boxPanel.add(box);
  54. basePanel.add(boxPanel);
  55. var bottomPanel = new JPanel(new FlowLayout(FlowLayout.RIGHT));
  56. var tipBtn = new JButton("Next Tip");
  57. tipBtn.setMnemonic(KeyEvent.VK_N);
  58. var closeBtn = new JButton("Close");
  59. closeBtn.setMnemonic(KeyEvent.VK_C);
  60. bottomPanel.add(tipBtn);
  61. bottomPanel.add(closeBtn);
  62. basePanel.add(bottomPanel);
  63. bottomPanel.setMaximumSize(new Dimension(450, 0));
  64. setTitle("Tip of the Day");
  65. setSize(new Dimension(450, 350));
  66. setResizable(false);
  67. setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
  68. setLocationRelativeTo(null);
  69. }
  70. public static void main(String[] args) {
  71. EventQueue.invokeLater(() -> {
  72. var ex = new TipOfDayEx();
  73. ex.setVisible(true);
  74. });
  75. }
  76. }

该示例混合使用了布局管理器。 只需将四个面板放入垂直组织的基本面板中即可。

  1. var basePanel = new JPanel();
  2. basePanel.setLayout(new BoxLayout(basePanel, BoxLayout.Y_AXIS));
  3. add(basePanel);

这是最底部的面板。 它具有垂直框布局管理器。 基本面板已添加到默认的JDialog组件。 默认情况下,此组件具有边框布局管理器。

  1. var topPanel = new JPanel(new BorderLayout(0, 0));

topPanel面板具有边框布局管理器。 我们将包含三个组成部分。 两个标签和一个分隔符。

  1. topPanel.setMaximumSize(new Dimension(450, 0));

如果我们想要的面板不超过其组件,则必须设置其最大大小。 零值将被忽略。 管理器计算必要的高度。

  1. var textPanel = new JPanel(new BorderLayout());
  2. ...
  3. textPanel.add(pane);

文本窗格组件将添加到边框布局管理器的中心区域。 它会占用所有剩余空间。 正是我们想要的。

  1. var boxPanel = new JPanel(new FlowLayout(FlowLayout.LEFT, 20, 0));

该复选框显示在boxPanel面板中。 它保持对齐。 流布局管理器的水平间隙为 20px。 其他组件的像素为 25px。 这是为什么? 这是因为流布局管理器也在组件和边缘之间放置了一些空间。

  1. var bottomPanel = new JPanel(new FlowLayout(FlowLayout.RIGHT));
  2. ...
  3. bottomPanel.setMaximumSize(new Dimension(450, 0));

底部面板显示两个按钮。 它具有右对齐的流布局管理器。 为了在对话框的右边缘显示按钮,面板必须从头到尾水平伸展。

Swing 布局管理 - 图9

图:当天的提示

没有管理器

可以不使用布局管理器。 在少数情况下,我们可能不需要布局管理器。 (也许将一些图像放置在一些不规则的位置。)但是在大多数情况下,要创建真正可移植的复杂应用,我们需要布局管理器。

如果没有布局管理器,我们将使用绝对值来定位组件。

AbsoluteLayoutEx.java

  1. package com.zetcode;
  2. import javax.swing.JButton;
  3. import javax.swing.JFrame;
  4. import java.awt.EventQueue;
  5. public class AbsoluteLayoutEx extends JFrame {
  6. public AbsoluteLayoutEx() {
  7. initUI();
  8. }
  9. private void initUI() {
  10. setLayout(null);
  11. var okBtn = new JButton("OK");
  12. okBtn.setBounds(50, 50, 80, 25);
  13. var closeBtn = new JButton("Close");
  14. closeBtn.setBounds(150, 50, 80, 25);
  15. add(okBtn);
  16. add(closeBtn);
  17. setTitle("Absolute positioning");
  18. setSize(300, 250);
  19. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  20. setLocationRelativeTo(null);
  21. }
  22. public static void main(String[] args) {
  23. EventQueue.invokeLater(() -> {
  24. var ex = new AbsoluteLayoutEx();
  25. ex.setVisible(true);
  26. });
  27. }
  28. }

这个简单的示例显示了两个按钮。

  1. setLayout(null);

我们通过向setLayout()方法提供null使用绝对定位。 (JFrame组件具有默认的布局管理器BorderLayout。)

  1. okBtn.setBounds(50, 50, 80, 25);

setBounds()方法放置“确定”按钮。 参数是 x 和 y 坐标以及组件的宽度和高度。

Swing 布局管理 - 图10

图:绝对布局

在本章中,我们提到了 Swing 中的布局管理。