为啥 GridBagLayout 将我的组件居中而不是放在角落里?

Posted

技术标签:

【中文标题】为啥 GridBagLayout 将我的组件居中而不是放在角落里?【英文标题】:Why does GridBagLayout center my components instead of putting it in the corner?为什么 GridBagLayout 将我的组件居中而不是放在角落里? 【发布时间】:2011-11-08 15:57:52 【问题描述】:

到目前为止,我设法尽可能避免使用GridBagLayout(手动代码),但这次我无法避免,我正在阅读 SUN 的教程GridBagLayout 到目前为止,进展并不顺利。我想我误解了一些东西。 例如,我尝试以下代码(类似于 SUN 帖子中的代码):

public class MainFrame extends JFrame  


    public static void main(String args[]) 
        EventQueue.invokeLater(new Runnable() 
            public void run() 
                try 
                    MainFrame frame = new MainFrame();
                    frame.setVisible(true);
                 catch (Exception e) 
                    e.printStackTrace();
                
            
        );
    

    /**
     * Create the frame
     */
    public MainFrame() 
        super();
        setBounds(100, 100, 500, 375);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        Container mainContainer = getContentPane();

        mainContainer.setLayout(new GridBagLayout());       

        //add label
        JLabel someLabel = new JLabel("Label 1:");
        GridBagConstraints constraints = new GridBagConstraints();

        constraints.gridx = 0;
        constraints.gridy = 0;
        //constraints.anchor = GridBagConstraints.FIRST_LINE_START;
        //constraints.weightx = 0.5;
        mainContainer.add(someLabel, constraints);      

        JTextField someText = new JTextField(30);

        constraints = new GridBagConstraints();

        constraints.gridx = 1;
        constraints.gridy = 0;
        constraints.weightx = 0.5;
        mainContainer.add(someText, constraints);

        //
    


标签和文本字段在框架的中心并排放置。 但我原以为它们会出现在左上角,因为标签的 gridx 和 gridy 为 0。 即使我设置constraints.anchor = GridBagConstraints.FIRST_LINE_START; 仍然是相同的结果。 我错了吗? 来自 SUN 的帖子:

指定组件左上角的行和列。这 最左列的地址为 gridx=0,顶行的地址为 网格=0。

【问题讨论】:

【参考方案1】:

这对我有用:

public class NewJFrame extends javax.swing.JFrame 

    public NewJFrame() 
        initComponents();
    

    @SuppressWarnings("unchecked")
    private void initComponents() 
        java.awt.GridBagConstraints gridBagConstraints;

        jPanel2 = new javax.swing.JPanel();
        jComboBox3 = new javax.swing.JComboBox();
        jComboBox4 = new javax.swing.JComboBox();
        jComboBox5 = new javax.swing.JComboBox();

        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        setBackground(new java.awt.Color(255, 204, 51));
        setMinimumSize(new java.awt.Dimension(800, 600));
        getContentPane().setLayout(null);

        jPanel2.setLayout(new java.awt.GridBagLayout());

        jComboBox3.setModel(new javax.swing.DefaultComboBoxModel(new String[]  "Item 1", "Item 2", "Item 3", "Item 4" ));
        gridBagConstraints = new java.awt.GridBagConstraints();
        gridBagConstraints.gridx = 0;
        gridBagConstraints.gridy = 0;
        gridBagConstraints.fill = java.awt.GridBagConstraints.HORIZONTAL;
        gridBagConstraints.anchor = java.awt.GridBagConstraints.NORTHWEST;
        gridBagConstraints.weightx = 1.0;
        jPanel2.add(jComboBox3, gridBagConstraints);

        jComboBox4.setModel(new javax.swing.DefaultComboBoxModel(new String[]  "Item 1", "Item 2", "Item 3", "Item 4" ));
        gridBagConstraints = new java.awt.GridBagConstraints();
        gridBagConstraints.gridx = 0;
        gridBagConstraints.gridy = 1;
        gridBagConstraints.fill = java.awt.GridBagConstraints.HORIZONTAL;
        gridBagConstraints.anchor = java.awt.GridBagConstraints.NORTHWEST;
        gridBagConstraints.weightx = 1.0;
        jPanel2.add(jComboBox4, gridBagConstraints);

        jComboBox5.setModel(new javax.swing.DefaultComboBoxModel(new String[]  "Item 1", "Item 2", "Item 3", "Item 4" ));
        gridBagConstraints = new java.awt.GridBagConstraints();
        gridBagConstraints.gridx = 0;
        gridBagConstraints.gridy = 2;
        gridBagConstraints.fill = java.awt.GridBagConstraints.HORIZONTAL;
        gridBagConstraints.anchor = java.awt.GridBagConstraints.NORTHWEST;
        gridBagConstraints.weightx = 1.0;
        gridBagConstraints.weighty = 1.0;
        jPanel2.add(jComboBox5, gridBagConstraints);

        getContentPane().add(jPanel2);
        jPanel2.setBounds(30, 40, 150, 260);

        pack();
    

    public static void main(String args[]) 
        java.awt.EventQueue.invokeLater(new Runnable() 
            public void run() 
                new NewJFrame().setVisible(true);
            
        );
    

    private javax.swing.JComboBox jComboBox3;
    private javax.swing.JComboBox jComboBox4;
    private javax.swing.JComboBox jComboBox5;
    private javax.swing.JPanel jPanel2;

【讨论】:

【参考方案2】:

constraints.weighty = 1; 添加到JLabel 约束,将constraints.anchor = GridBagConstraints.NORTHWEST; 添加到TextField 约束。

编辑:

来自 Oracle 的GridBagLayout guide:

较大的数字表示组件的行或列应该获得更多空间。对于每一列,权重与为该列中的组件指定的最高权重 x 相关,每个多列组件的权重在组件所在的列之间以某种方式拆分。类似地,每一行的权重与为某个组件指定的最高权重相关该行中的组件。额外的空间倾向于朝向最右边的列和底行。

【讨论】:

如果我也将constraints.weighty=1 设置为标签并锚定到标签FIRST_LINE_START,它会起作用。但是重量中的值1实际上是什么意思? 另外,在他们的代码示例中,这行代码带有注释:c.weighty = 1.0; //request any extra vertical space 对于 JButton5,我看到了。但我不认为这是一个很好的演示文稿。无论如何为什么是 1 而不是 0.8。我不明白这些数字是如何选择的 老实说,我也不知道。我对一切都使用 MiGLayout...miglayout.com 记录一下:“如果生成的布局横向小于它需要填充的区域,则额外的空间将按与其权重的比例分配到每一列。 权重为零的列不会接收额外的空间。” weightxweighty 的实际数值仅在相互关联时才有意义。【参考方案3】:

您需要进一步阅读 Swing 教程中的 weightX/weightY 部分,其中指出:

除非您为 weightx 或 weighty 指定至少一个非零值,否则所有组件都会在其容器的中心聚集在一起。

您指定了 weightX,但没有指定 weightY。

编辑,它比我想象的要复杂。看来您还需要指定:

constraints.anchor = GridBagConstraints.FIRST_LINE_START;

除了权重之外的两个组件。

【讨论】:

SUN 的引述说 weightx 或 weighty (所以我没想到它需要两者)。无论如何我尝试了 weighty=1 的标签和 weighty=0.5 的文本和 weighty=0.5 的两者但相同的结果 weightx 如果用于水平定位。 weighty 用于垂直定位。我使用 1.0 和 FIRST_LINE_START 都对我有用。 是的,现在它可以工作了。但是重量中的值 1 到底是什么意思?我不明白。而且在我看来(由于需要锚点)网格已经存在而不是通过在上面放置组件来构建的。对吗? GridBagLayout 以首选尺寸显示组件。当有额外空间时,weightX/Y 值会提示布局管理器如何分配该空间。 EON 的提示奏效了,因为他添加了第二行组件并要求该行通过使用 weighty = 1.0 来占用所有额外空间。在您的情况下,您只有一行,因此您需要指定其中一个组件占用所有可用的额外空间。 好的,但是我只对标签使用了 weighty。所以标签获得了额外的垂直空间(我猜以一种用户看不到的方式)。但是为什么这也会影响文本字段以及涨了吗?【参考方案4】:

您可以通过使用技巧来实现这一点,在行之后添加一个虚拟组件并将其扩展以填充垂直空间。您也可以重复使用约束,无需创建新对象:

编辑:好的,忘记诀窍 :( 正确的方法是 Deon Botha 和 BenCole 说,我已经使用锚更新了我的代码

不要接受这个答案,接受 Deon 的或 Ben 的

public class MainFrame extends JFrame  
    public static void main(String args[]) 
        EventQueue.invokeLater(new Runnable() 
            public void run() 
                try 
                    MainFrame frame = new MainFrame();
                    frame.setVisible(true);
                 catch (Exception e) 
                    e.printStackTrace();
                
            
        );
    

public MainFrame() 
    super();
    setBounds(100, 100, 500, 375);
    setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

    Container mainContainer = getContentPane();
    mainContainer.setLayout(new GridBagLayout());       

    JLabel someLabel = new JLabel("Label 1:");
    JTextField someText = new JTextField(30);

    GridBagConstraints constraints = new GridBagConstraints();
    constraints.anchor = GridBagConstraints.FIRST_LINE_START;

    constraints.gridx = 0;
    constraints.gridy = 0;    
    constraints.weightx = 1.0;
    mainContainer.add(someLabel, constraints);      

    constraints.gridx = 1;                       
    constraints.weightx = 1.0;
    constraints.weighty = 1.0;        
    mainContainer.add(someText, constraints);                       


【讨论】:

但是为什么我需要这个虚拟组件呢?组件是否应该被布局管理器放在左上角?那么gridx=gridy=0是什么意思呢? 你是对的,它“应该工作”,但 GridBayLayout 是纯粹的邪恶,实际上它非常好......只是棘手,我不是纯粹主义者,如果它与一个虚拟标签一起工作那为什么不使用呢? 或者你可以设置constraints.weighty = 1.0; constraints.anchor = GridBagConstraints.FIRST_LINE_START;【参考方案5】:

我可能不会直接回答您的问题,但相信我,您应该使用 IDE 对布局进行尝试和错误。我个人建议Netbeans。您可以在那里拖放,然后查看属性。起初,您会在属性检查器中给出一些默认值,因此自动生成的代码更少。但是,一旦您开始使用布局,您就可以看到代码并很好地了解您的工作方式。

【讨论】:

以上是关于为啥 GridBagLayout 将我的组件居中而不是放在角落里?的主要内容,如果未能解决你的问题,请参考以下文章

iOS 8:为啥 NSLayoutAttributeCenterX 无法将我的视图居中? (适用于 iOS 7)

在 Java 中使用 GridBagLayout 时防止 Swing 组件偏离中心

JLabel自动将其自身集中在GridBagLayout中

在 java 中使用 GridBagLayout 时设置最大组件大小

如何使用 GridBagLayout 防止组件抖动?

React Native 中如何使用 Flex 居中组件?