我想在一个JPanel里面动态放置一些图片:点击一个按钮,JPanel里就增加一张图片。每张图片的尺寸大小都相同。当第一行满时,图片会自动在第二行增加。如果超过了当前JPanle的高度,则会出现上下滚动条。一开始我用JPanel+FlowLayout.LEFT布局,但是当图片占据空间的高度高于JPanel的高度时,没有上下方向的滚动条出现,也就是说,下面的图片都被遮住了。然后我在JPanel外面套了一个JScrollPane,却发现图片会一直往右边加,超出JPanel的宽度则出现横向的滚动条,与预想的效果相差较大。于是我把JScrollPane的horizontalScrollBarPolicy属性设为ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER,这回横向滚动条没了,但是图片还是一直往右边加,超出JPanle宽度部分被遮住了。后来想到用BoxLayout,但是发现图片会自动占满JPanel的所有空间,图片的大小被改变了,我只想让每张图片都有固定的大小阿。实在没有办法了,盼高手们指点一下,应该怎样做啊?

解决方案 »

  1.   

    你的要求说白了,其实就是这一句:当第一行满时,图片会自动在第二行增加。
    但是这里就有一个矛盾:如果你没有布局管理器,那么java会一直横着添加,这是正常的,它不知道换行,呵呵~~;如果你用了布局管理器,比如GridLayout,你定义为2行2列,那么你可以定义一个顺序去添加,第一次添加在第一行第一列,第二张图片添加在第一行第二列,依次因此从你的需求出发,布局管理器是必须的!至于你说的超过第一行,也就是说当剩余的空间不够插入下一张图片的话,它会自动换到下一行,那也必须要你自己去控制,我试着写一个吧。看看效果如何,你==。
      

  2.   

    超过大小的计算可以用图片的宽度的和,当然监听scroll的横行滚动条的动作也是一个不错的选择,那剩下的就是gridbaglayout的事情了 
      

  3.   

    找到解决办法了,主要问题是当嵌在JScrollPane里的JPanle用FlowLayout布局的时候,往JPanel里面动态加图片的时候一直都是往右加,永远都不会自动换行,这是FlowLayout类自身的缺陷。google了一下,有人给出了解决方案:继承FLowLayout类覆盖minimumLayoutSize方法和preferredLayoutSize方法import java.awt.Component;
    import java.awt.Container;
    import java.awt.Dimension;
    import java.awt.FlowLayout;
    import java.awt.Insets;public class ModifiedFlowLayout extends FlowLayout {    public ModifiedFlowLayout() {
            super();
        }    public ModifiedFlowLayout(int align) {
            super(align);
        }    public ModifiedFlowLayout(int align, int hgap, int vgap) {
            super(align, hgap, vgap);
        }    public Dimension minimumLayoutSize(Container target) {
            return computeSize(target, false);
        }    public Dimension preferredLayoutSize(Container target) {
            return computeSize(target, true);
        }    private Dimension computeSize(Container target, boolean minimum) {
            synchronized (target.getTreeLock()) {
                int hgap = getHgap();
                int vgap = getVgap();
                int w = target.getWidth();            if (w == 0) {
                    w = Integer.MAX_VALUE;
                }            Insets insets = target.getInsets();
                if (insets == null) {
                    insets = new Insets(0, 0, 0, 0);
                }
                int reqdWidth = 0;            int maxwidth = w - (insets.left + insets.right + hgap * 2);
                int n = target.getComponentCount();
                int x = 0;
                int y = insets.top;
                int rowHeight = 0;            for (int i = 0; i < n; i++) {
                    Component c = target.getComponent(i);
                    if (c.isVisible()) {
                        Dimension d =
                                minimum ? c.getMinimumSize() : c.getPreferredSize();
                        if ((x == 0) || ((x + d.width) <= maxwidth)) {
                            if (x > 0) {
                                x += hgap;
                            }
                            x += d.width;
                            rowHeight = Math.max(rowHeight, d.height);
                        } else {
                            x = d.width;
                            y += vgap + rowHeight;
                            rowHeight = d.height;
                        }
                        reqdWidth = Math.max(reqdWidth, x);
                    }
                }
                y += rowHeight;
                return new Dimension(reqdWidth + insets.left + insets.right, y);
            }
        }
    }