更新JDialog时出现奇怪/不同的行为

时间:2014-11-27 17:10:41

标签: java swing awt

在一个项目中,我有一个JFrame A启动一个JDialog B,它本身启动一个JDialog C(全部使用按钮)。但是,当遵循以下程序之一时:

  • 我点击A开始B;或
  • 我点击A开始B,然后点击B开始C,然后点击C的取消按钮,

B中显示的内容不一样(第二个程序给出了一个奇怪的丑陋的东西)。

我不明白为什么会这样,因为在这两种方式中都会调用我的updateAll方法。

我尝试用一​​个小程序重新创建它,以便更容易看到发生了什么。解决这个问题可能会或可能不会在我的实际项目中解决,但它肯定会有所帮助。

因为我不知道它可能来自哪里,所以这是我的(测试)程序的完整代码。支撑你自己。

'A'框架

public class MyFrame extends JFrame {
    private static final long serialVersionUID = 7073064926636937881L;
    public MyFrame() {
        this.setSize(200, 300);
        JButton button = new JButton("Click me");
        button.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent arg0) {
                new MyDialog1().setVisible(true);
            }
        });
        this.getContentPane().add(button);
    }

    public static void main(String args[]) {
        new MyFrame().setVisible(true);
    }
}

'B'对话

 public class MyDialog1 extends JDialog {
        private static final long serialVersionUID = 9181006217120036637L;
        private JScrollPane scrollPane;
        public String text = "aaaaaaaaaaa\naaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa\naaaaaaaa\naaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa\naaaaaaaa";

    public MyDialog1() {
        this.setVisible(false);
        this.setSize(800, 600);

        this.initComponent();
        this.updateAll();
    }

    private void initComponent() {
        this.getContentPane().setLayout(new GridBagLayout());
        GridBagConstraints c = new GridBagConstraints();

        this.scrollPane = new JScrollPane();

        c.gridx = 0;
        c.gridy = 0;
        this.getContentPane().add(this.scrollPane, c);

        c.gridx = 0;
        c.gridy = 1;
        JButton b = new JButton("Supposedly edit stuff");
        final MyDialog1 caller = this;
        b.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent arg0) {
                new MyDialog2(caller).setVisible(true);
            }
        });
        this.getContentPane().add(b, c);

        c.gridx = 1;
        c.gridy = 1;
        b = new JButton("Leave");
        b.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent arg0) {
                setVisible(false);
            }
        });
        this.getContentPane().add(b, c);
    }

    public void updateAll() {
        JPanel mainPanel = new JPanel();
        for (int i = 0 ; i < 5 ; i++) {
            JPanel subPanel = new JPanel();
            JTextArea t = new JTextArea(this.text);
            t.setSize(60, 30);
            t.setVisible(true);// Useful ? What about setSize ?
            subPanel.add(t);
            mainPanel.add(subPanel);
        }
        this.scrollPane.setSize(150, 150); // FIXME When in initComponent, doesn't do anything, and when in updateAll, behavior is inconsistent 
        this.scrollPane.setViewportView(mainPanel); // Replacing previous JPanel

    }
}

'C'对话

public class MyDialog2 extends JDialog {
    private static final long serialVersionUID = 5676648412234106581L;
    private MyDialog1 caller;

    public MyDialog2(MyDialog1 c) {
        this.setSize(100, 150);
        this.caller = c;
        JButton cancelButton = new JButton("Cancel");
        cancelButton.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent arg0) {
                setVisible(false);
                caller.text += "\nbbbbbbbbbbbbb\nbbbbbbbbbbbbbbbbbb\nbbbbbbbbbbb\nbbbbbbbbbbbbbb\ncccccccccccccccccccccccccccccccccccccccccccc\ncccccccccc";
                caller.updateAll();
            }
        });
        this.getContentPane().add(cancelButton);
    }
}

感谢您的帮助。

2 个答案:

答案 0 :(得分:1)

我建议使用

c.gridx = 0;
c.gridy = 0;
c.fill = GridBagConstraints.BOTH; // make the component fill its display area entirely
c.ipady = 150; //height
c.anchor = GridBagConstraints.FIRST_LINE_START; // component start from the left top corner
this.getContentPane().add(this.scrollPane, c);

用于定义JScrollPane约束。

另外,在修改元素

之后添加validate()和repaint()
this.scrollPane.setViewportView(mainPanel); // Replacing previous JPanel
this.validate();
this.repaint();

答案 1 :(得分:0)

忽略良好做法的简单回答:

替换它:

this.scrollPane.setSize(150, 150);

用这个:

this.scrollPane.setMinimumSize(new Dimension(150, 150));

在具有布局的Container中的组件上调用setSize通常不执行任何操作;充其量,它将设置大小,直到下次验证容器时,因为布局管理器将覆盖该大小。但是,设置minimumSize(或preferredSize或maximumSize)会设置(大多数)布局管理器所遵循的持久属性。

为什么setMinimumSize会有所作为?因为您没有在任何GridBagConstraints上设置任何weightxweighty属性,所以没有足够的空间让GridBagLayout以其首选大小显示您的JScrollPane。当GridBagLayout确定没有足够的空间来显示所需大小的所有内容时,布局&#34; punts&#34;并迫使一切恢复到最小尺寸。

第一次显示B对话框时,您看到JScrollPane的最小大小(即,足够大以显示滚动条和视口边框)。取消C对话框后,setSize(150, 150)生效,但对GridBagLayout的任何后代的任何后续更改都可能导致(150,150)被JScrollPane覆盖。最小尺寸。

更复杂的答案是好的做法:

删除所有类中对setSize的所有调用。通常,您根本不应设置显式大小,但如果必须这样做,请使用setPreferredSize

与JScrollPanes一样,增长和缩小的组件应在其对应的GridBagConstraints中分配正weightxweighty值,并将fill设置为GridBagConstraints.BOTH as pcej建议。