我有一个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才能使其正常工作?
答案 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);
}
}
}