据说SpringLayout非常强大。我试图使用SpringLayout实现我认为是相当简单的布局,但是失败很严重。

减少到最小,我希望在我的JFrame上并排放置4个JButton:

*=====================*
| +--+ +--+ +--+ +--+ |
| |b1| |b2| |b3| |b4| |
| +--+ +--+ +--+ +--+ |
*=====================*


我希望所有4个文本的大小相同,无论文本如何不同。

我希望最外面的一个(b1和b4)与容器的边界之间保持恒定的水平距离,即5 px,并且与按钮的南北之间的边界都保持5 px,它们的高度均相同。

我希望按钮之间的间隙也为5 px。

到目前为止,非常容易。我还有更多限制:


当框架变宽(用户/鼠标)时,我希望按钮(b1 / b2,b2 / b3,b3 / b4)之间的间隙变宽,而不是按钮。
如果将框架缩小,我想在其他东西不得不让步之前,将按钮之间的间隙缩小到1或0。
我希望框架上的pack()给我一个窗口,尺寸恰好是按钮的默认大小,默认间隔为5px,边框周围为5px。


我编写了以下相当简单的代码,结果令人震惊。间隙不会缩小,并且膨胀只会发生在最右边的按钮(b4)上。

既然我有了自定义按钮,可以自行调整大小,那么使用GridBagLayout可以很轻松地满足这些要求,而使用MigLayout可以更轻松地满足这些要求。那不是我要的答案。具体来说,我的问题是:

使用SpringLayout可以使此布局正常工作吗?怎么样?

布局甚至忽略了我按钮的getMaximumSize()。我是在做错什么还是SpringLayout出错了?

非常感谢。

public class SpringLayoutTest extends JFrame {

public SpringLayoutTest() {
  setLocation(100,100);
  setSize(400, 300);
  Container cp = getContentPane();
  SpringLayout layout = new SpringLayout();
  cp.setLayout(layout);

  SiblingButton b1, b2, b3, b4;
  cp.add(b1 = new SiblingButton("..."));
  cp.add(b2 = new SiblingButton("iii"));
  cp.add(b3 = new SiblingButton("xxx"));
  cp.add(b4 = new SiblingButton("WWW"));

  layout.putConstraint(NORTH, b1, 5, NORTH, cp);
  layout.putConstraint(SOUTH, b1, Spring.minus(Spring.constant(5)), SOUTH, cp);
  layout.putConstraint(NORTH, b2, 5, NORTH, cp);
  layout.putConstraint(SOUTH, b2, Spring.minus(Spring.constant(5)), SOUTH, cp);
  layout.putConstraint(NORTH, b3, 5, NORTH, cp);
  layout.putConstraint(SOUTH, b3, Spring.minus(Spring.constant(5)), SOUTH, cp);
  layout.putConstraint(NORTH, b4, 5, NORTH, cp);
  layout.putConstraint(SOUTH, b4, Spring.minus(Spring.constant(5)), SOUTH, cp);

  layout.putConstraint(WEST, b1, 5, WEST, cp);
  layout.putConstraint(WEST, b2, Spring.constant(1, 5, Integer.MAX_VALUE), EAST, b1);
  layout.putConstraint(WEST, b3, Spring.constant(1, 5, Integer.MAX_VALUE), EAST, b2);
  layout.putConstraint(WEST, b4, Spring.constant(1, 5, Integer.MAX_VALUE), EAST, b3);

  layout.putConstraint(EAST, b4, Spring.minus(Spring.constant(5)), EAST, cp);

  layout.putConstraint(WEST, cp, Spring.minus(Spring.constant(5)), WEST, b1);
}

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


SiblingButton是一个非常巧妙的实现。请忽略其设计错误,这仅是出于演示目的。

class SiblingButton extends JButton {

static ArrayList<SiblingButton> siblings = new ArrayList<SiblingButton>();

public SiblingButton(String text) {
  super(text);
  siblings.add(this);
}

public Dimension getMaximumSize() {
  return getPreferredSize();
}

public Dimension getMinimumSize() {
   return getPreferredSize();
}

public Dimension getPreferredSize() {
  Dimension mx = new Dimension(0, 0);
  for (SiblingButton sb : siblings) {
     mx = new Dimension(Math.max(mx.width, sb.originalPreferredSize().width),
           Math.max(mx.height, sb.originalPreferredSize().height));
  }
  return mx;
}

Dimension originalPreferredSize() {
  return super.getPreferredSize();
}

}




更新/结论

到现在已经24小时了,反响势不可挡。单独的响应(感谢Camickr!)甚至没有尝试触摸SpringLayout。我认为这不是在SO社区而是在SpringLayout的实用程序上反映不佳!

我的印象是SpringLayout是布局管理器的红头发继子,在某些方面很有用,但很少使用,因此没有人有使用经验,也没有人愿意向Sun报告错误。

对我来说,事实证明GroupLayout可以完成我所需的一切,并允许我以合理的编码量准确地完成我想要的事情。对GridBagLayout而言,这是一个可喜的变化,从一开始我就很清楚我需要采取的步骤以实现预期的布局,而我只需要坐下来编写代码。

对于任何关心的人,这些都是GroupLayout的特征,这些特征对我来说是如此有用:


在水平和垂直方向上或多或少地进行布局。大量的灵活性和更简单的代码;
可以嵌套网格而无需引入容器只是为了保持布局。
可以插入具有指定约束的间隙;
可以按共享的尺寸规格形成任意组的组件,即所有组件的尺寸都与该组中最大的组件相同;和
可以更改组件的大小(例如,maximumSize = preferredSize),而无需对组件进行子类化。

最佳答案

您可以使用BoxLayout:

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

public class BoxExample extends JFrame
{
    public BoxExample()
    {
        Box box = Box.createHorizontalBox();
        box.setBorder( new EmptyBorder(5, 5, 5, 5) );
        Dimension size = new Dimension(100, 25);

        box.add( createButton("Button1", size) );
        box.add( createStrut() );
        box.add( createButton("Button2", size) );
        box.add( createStrut() );
        box.add( createButton("Button3", size) );
        box.add( createStrut() );
        box.add( createButton("Button4", size) );

        add( box );
    }

    private JButton createButton(String text, Dimension size)
    {
        JButton button = new JButton(text);
        button.setPreferredSize( size );
        button.setMinimumSize( size );
        button.setMaximumSize( size );
        return button;
    }

    private Component createStrut()
    {
        JComponent component = (JComponent)Box.createHorizontalStrut(5);
        component.setMinimumSize(new Dimension(0, 0));
        component.setMaximumSize(new Dimension(Short.MAX_VALUE, Short.MAX_VALUE));
        return component;
    }

    public static void main(String[] args)
    {
        BoxExample frame = new BoxExample();
        frame.setDefaultCloseOperation( EXIT_ON_CLOSE );
        frame.pack();
        frame.setLocationRelativeTo( null );
        frame.setVisible(true);
    }
}

10-07 18:19