Java Swing:如何动态更改GUI

5

我需要动态添加组件。此外,我还需要动态更改布局。


6
我想我代表所有人祝愿你好运做到这件事!使用Swing应该是可行的,但像任何事情一样,魔鬼在细节中。您是否阅读了有关布局管理器的教程?您对尝试过的代码是否有任何具体的问题或困难?您能告诉我们任何有关当前问题的细节,以帮助我们帮助您吗?不幸的是,您目前提出的问题过于笼统,无法得到具体的答案。 - Hovercraft Full Of Eels
2个回答

14

为了参考,这里有一个SSCCE,它展示了必要的方法 validate()。这个更加详细的例子展示了两个要求:动态更改布局和添加组件。

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

/** @see https://dev59.com/jVbTa4cB1Zd3GeqP-3Z2 */
public class DynamicLayout extends JPanel {

    private static final LayoutManager H = new GridLayout(1, 0);
    private static final LayoutManager V = new GridLayout(0, 1);

    public DynamicLayout() {
        this.setLayout(H);
        this.setPreferredSize(new Dimension(320, 240));
        for (int i = 0; i < 3; i++) {
            this.add(new JLabel("Label " + String.valueOf(i), JLabel.CENTER));
        }
    }

    private void display() {
        JFrame f = new JFrame("DynamicLayout");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(this);
        JPanel p = new JPanel();
        p.add(new JButton(new AbstractAction("Horizontal") {

            @Override
            public void actionPerformed(ActionEvent e) {
                DynamicLayout.this.setLayout(H);
                DynamicLayout.this.validate();
            }
        }));
        p.add(new JButton(new AbstractAction("Vertical") {

            @Override
            public void actionPerformed(ActionEvent e) {
                DynamicLayout.this.setLayout(V);
                DynamicLayout.this.validate();
            }
        }));
        f.add(p, BorderLayout.SOUTH);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new DynamicLayout().display();
            }
        });
    }
}

1

动态更改布局的例子:

package swinglayout;

import java.awt.BorderLayout;
import java.awt.Container;
import java.awt.FlowLayout;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;

public class LayoutChanger implements ActionListener{
  JButton b1;
  JButton b2;
  JButton b3;
  JButton b4;
  JButton b5;
  JButton b6;
  /** This button set the flowlayout on panel2 with left orientation */
  JButton flowLayout;

  /** This button set the Gridlayout of 2,3 grid on panel2 */
  JButton gridLayout;

  /** This button set the Borderlayout on panel2*/
  JButton borderLayout;

  /** 
   * This panel is control panel where we use button to change
   * layout of another panel
   */
  JPanel panel;

  /** This panel contain multiple button from b1 to b6 */
  JPanel panel2;

  JFrame frame;
  public LayoutChanger() {
    //set Default Look and Feel on frame
    JFrame.setDefaultLookAndFeelDecorated(true);

    frame = new JFrame();
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    Container con = frame.getContentPane();
    con.setLayout(new BorderLayout());

    panel = new JPanel();
    panel2 = new JPanel();
    //This button are used to only showing the layout effect
    b1 = new JButton("HelloButton1");
    b2 = new JButton("HelloButton2");
    b3 = new JButton("HelloButton3");
    b4 = new JButton("HelloButton4");
    b5 = new JButton("HelloButton5");
    b6 = new JButton("HelloButton6");
    // By default panel have layout
    panel2.add(b1);
    panel2.add(b2);
    panel2.add(b3);
    panel2.add(b4);
    panel2.add(b5);
    panel2.add(b6);
    // Layout changer button
    flowLayout = new JButton("FlowLayout");
    gridLayout = new JButton("GridLayout");
    borderLayout = new JButton("BorderLayout");

    //call Action listener on  every layout changer button
    flowLayout.addActionListener(this);
    gridLayout.addActionListener(this);
    borderLayout.addActionListener(this);

    panel.add(flowLayout);
    panel.add(gridLayout);
    panel.add(borderLayout);

    // add layout changer button panel at a top 
    //button panel at the center of container
    con.add(panel, BorderLayout.PAGE_START);
    con.add(panel2, BorderLayout.CENTER);

    frame.setVisible(true);
    frame.pack();

  }

  public void actionPerformed(ActionEvent e) {

    //set the flowlayout on panel2
    if(e.getSource() == flowLayout) {
      FlowLayout flow = new FlowLayout(FlowLayout.LEFT);
      panel2.setLayout(flow);
      panel2.validate();
    }

    //set the gridlayout on panel2
    if(e.getSource() == gridLayout) {
      GridLayout grid = new GridLayout(2,3);
      panel2.setLayout(grid);
      panel2.validate();
    }
    //set the gridlayout but the problem if we don't set the constraint
    //all button are set on center. So you remove the all button from panel
    //Then set grid layout on panel and add them with constraints.
    if(e.getSource() == borderLayout) {
      panel2.remove(b1);
      panel2.remove(b2);
      panel2.remove(b3);
      panel2.remove(b4);
      panel2.remove(b5);
      panel2.remove(b6);

      BorderLayout border = new BorderLayout();
      panel2.setLayout(border);

      panel2.add(b1,BorderLayout.NORTH);
      panel2.add(b2,BorderLayout.SOUTH);
      panel2.add(b3,BorderLayout.EAST);
      panel2.add(b4,BorderLayout.WEST);
      panel2.add(b5,BorderLayout.CENTER);
      panel2.add(b6,BorderLayout.BEFORE_FIRST_LINE);

      panel2.validate();
    }
  }
  public static void main(String args[]) {
    new LayoutChanger();
  }
}

请记住,如果您在面板上设置了新的布局,请不要忘记在面板上调用方法validate()。如果您没有调用此方法,则看不到布局更改的效果。如果您想在不调用该方法的情况下查看效果,则必须调整框架大小。 此外,您可以非常轻松地设置相同的布局,例如FlowLayoutGridLayoutBoxLayout,但是当设置BorderLayout时,需要为添加元素指定约束条件,因此我们首先使用remove(Component comp)方法从面板中删除所有组件,然后按照约束条件将组件添加到面板中。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接