将固定高度的面板动态添加到 JScrollPane

Dynamically add fixed-height panels to a JScrollPane

JPanel panel = new JPanel(new GridLayout(0,1));
JScrollPane contentpane = new JScrollPane(panel);
JButton add  = new JButton("ADD");
add.actionListener(new ActionListener() {
     public void actionPerformed(){
          MyPanel newpanel = new MyPanel("title","Button"); //MyPanel is a class which extends JPanel and contains constructor MyPanel(String TitleToSet ,String ButtonTitleTOAdd)
           panel.add(newpanel);
           panel.repaint(); 
     }) ;

我使用这段代码时认为它会动态地将 MyPanel 添加到网格布局,如果添加更多 "MyPanel","panel" 将是可滚动的。然而,情况并非如此,第一个 "MyPanel" 填满了整个 "panel" 并且在添加第二个 "MyPanel" 时(通过单击按钮 "Add"),第一个 "MyPanel" 被缩小了使 space 用于添加第二个。在添加更多时,所有 "MyPanel" 都适合视口而不是使 "panel" 可滚动。如何添加那些 "MyPanel"s 动态并使 "panel" 在添加更多这些时可滚动?任何帮助将不胜感激。

http://docs.oracle.com/javase/7/docs/api/javax/swing/JScrollPane.html :

By default JScrollPane uses ScrollPaneLayout to handle the layout of its child Components. ScrollPaneLayout determines the size to make the viewport view in one of two ways:

  1. [...]
  2. getPreferredSize is used.

您应该添加行

panel.setPreferredSize(new Dimension(0, panel.getComponents().size() * SUB_PANEL_HEIGHT));

到您的 ActionListener。

完整示例:

package main;

import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.Panel;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.Timer;

class Test {
    public static void main(String[] args) {
        JFrame frame = new JFrame();
        frame.setPreferredSize(new Dimension(400, 400));
        frame.setSize(400, 400);
        
        JPanel panel = new JPanel(new GridLayout(0, 1));
        panel.add(new JLabel("BOO"));
        panel.add(new JButton("BBBB"));
        
        JScrollPane contentpane = new JScrollPane(panel);
        frame.add(contentpane);
        
        new Timer(1000, new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                JPanel newpanel = new JPanel();
                newpanel.add(new JLabel("LOL"));
                panel.add(newpanel);
                System.out.println(100 * panel.getComponents().length);
                panel.setPreferredSize(new Dimension(0, 100 * panel.getComponents().length));
                contentpane.validate();
            }
        }).start();
        
        frame.setVisible(true);
    }
}