在一个项目中,我有一个JFrame A启动一个JDialog B,它本身启动一个JDialog 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);
}
}
感谢您的帮助。
答案 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上设置任何weightx
或weighty
属性,所以没有足够的空间让GridBagLayout以其首选大小显示您的JScrollPane。当GridBagLayout确定没有足够的空间来显示所需大小的所有内容时,布局&#34; punts&#34;并迫使一切恢复到最小尺寸。
第一次显示B对话框时,您看到JScrollPane的最小大小(即,足够大以显示滚动条和视口边框)。取消C对话框后,setSize(150, 150)
生效,但对GridBagLayout的任何后代的任何后续更改都可能导致(150,150)被JScrollPane覆盖。最小尺寸。
更复杂的答案是好的做法:
删除所有类中对setSize的所有调用。通常,您根本不应设置显式大小,但如果必须这样做,请使用setPreferredSize
。
与JScrollPanes一样,增长和缩小的组件应在其对应的GridBagConstraints中分配正weightx
和weighty
值,并将fill
设置为GridBagConstraints.BOTH as pcej建议。