MigLayout限制在JScrollPane上滚动JList

时间:2013-07-30 12:41:53

标签: java swing jscrollpane jlist miglayout

我有一个GUI,它由面板上的多个JList组件组成,它位于JScrollPane上。滚动窗格可以一次滚动所有列表。但是,当我在包含列表的面板上使用MigLayout时,我无法滚动超过前1819列表元素。限制似乎与列表的高度有关,而不是元素的数量。如果我使用更大的元素大小,我可以滚动更少的元素。我目前正在使用MigLayout 4.2。

以下是展示此行为的简化示例。

import net.miginfocom.swing.MigLayout;

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

public final class MigLayoutListTest {
    public static void main(final String... args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                final String[] listData = new String[5000];
                for (int i = 0; i < 5000; i++) {
                    listData[i] = "Index " + i;
                }

                final JPanel panel = new JPanel(new MigLayout());
                panel.add(new JList(listData));

                final JFrame frame = new JFrame();
                frame.setContentPane(new JScrollPane(panel));

                frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
                frame.setPreferredSize(new Dimension(600, 400));
                frame.pack();
                frame.setVisible(true);
            }
        });
    }
}

如果我删除了JPanel并将JList直接放在JScrollPane上,那么滚动工作正常,但我无法使用此解决方法,因为我需要在同一个JScrollPane上有多个列表。如果我使用除MigLayout之外的任何布局作为JPanel,则滚动工作正常。

这是MigLayout的错误或限制吗?是否有一些配置选项需要应用于MigLayout才能使其正常工作?

1 个答案:

答案 0 :(得分:2)

我认为你的方法很糟糕。相反,在他们自己的JScrollPane中使用JLists并同步滚动:

import java.awt.*;
import java.util.*;

import javax.swing.*;

public class ScrollingDemo implements Runnable
{
  public static void main(String args[])
  {
    SwingUtilities.invokeLater(new ScrollingDemo());
  }

  public void run()
  {
    JScrollPane sp1 = new JScrollPane(getJList(1));
    sp1.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_ALWAYS);
    JScrollPane sp2 = new JScrollPane(getJList(3));
    sp2.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_ALWAYS);

    JScrollBar sBar1 = sp1.getVerticalScrollBar();
    JScrollBar sBar2 = sp2.getVerticalScrollBar();

    // synchronize:
    sBar2.setModel(sBar1.getModel());

    JPanel p = new JPanel(new GridLayout(1,2));
    p.add(sp1);
    p.add(sp2);

    JFrame frame = new JFrame("Synch Scrolling");
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(p, BorderLayout.CENTER);
    frame.pack();
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
  }

  public JList getJList(int factor)   
  {
    Vector<String> items = new Vector<String>();
    for(int i = 0; i < 5000; i++)
    {
      items.add(Integer.toString((i+1)*factor));
    }
    JList list = new JList(items);
    list.setPrototypeCellValue("XXXXXXXXXX");
    list.setVisibleRowCount(10);
    return list;
  }
}

如果只想查看1个垂直滚动条,只需将滚动条策略设置为JScrollPane.VERTICAL_SCROLLBAR_NEVER即可隐藏。

注意:这假设每个JList具有相同数量的元素。如果你有多个不同长度的列表,你可能(而不是)需要在垂直滚动条中添加AdjustmentListener

编辑:示例:

import java.awt.*;
import java.awt.event.*;
import java.util.*;

import javax.swing.*;

public class ScrollingDemo2 implements Runnable
{
  public static void main(String args[])
  {
    SwingUtilities.invokeLater(new ScrollingDemo2());
  }

  public void run()
  {
    JScrollPane sp1 = new JScrollPane(getJList(100));
    sp1.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_NEVER);
    JScrollPane sp2 = new JScrollPane(getJList(200));
    sp2.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_ALWAYS);

    // synchronize:
    sp2.getVerticalScrollBar().addAdjustmentListener(new Synchronizer(sp1, sp2));

    JPanel p = new JPanel(new GridLayout(1,0));
    p.add(sp1);
    p.add(sp2);

    JFrame frame = new JFrame("Synch Scrolling 2");
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(p, BorderLayout.CENTER);
    frame.pack();
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
  }

  public JList getJList(int count)   
  {
    Vector<String> items = new Vector<String>();
    for(int i = 0; i < count; i++)
    {
      items.add(Integer.toString(i+1));
    }

    JList list = new JList(items);
    list.setPrototypeCellValue("XXXXXXXXXX");
    list.setVisibleRowCount(10);
    return list;
  }

  class Synchronizer implements AdjustmentListener
  {
    JScrollPane sp1, sp2;

    public Synchronizer(JScrollPane sp1, JScrollPane sp2)
    {
      this.sp1 = sp1;
      this.sp2 = sp2;
    }

    public void adjustmentValueChanged(AdjustmentEvent e)
    {
      if (! e.getValueIsAdjusting())
      {
        return;
      }

      JScrollBar vert1 = sp1.getVerticalScrollBar();
      JScrollBar vert2 = sp2.getVerticalScrollBar();

      int range1 = vert1.getMaximum() - vert1.getMinimum() -
                   vert1.getModel().getExtent();

      int range2 = vert2.getMaximum() - vert2.getMinimum() -
                   vert2.getModel().getExtent();

      float percent2 = (float) (vert2.getValue()) / range2;
      int newVal1 = (int) (percent2 * range1);

      vert1.setValue(newVal1);
    }
  }
}