我在尝试垂直填充 JPanel
中的 JScrollPane
(包含 JTable
)时遇到问题:不应填充面板在水平方向上,JScrollPane
应该只占用“必要的”空间。
我正在使用在 SO 上找到的一些代码根据其内容设置表格的首选宽度(我还覆盖了 JTable
的 getPreferredScrollableViewportSize
方法)。
这是我想要实现的截图:
JPanel
更宽,因为在我的应用程序中它将成为 CardLayout
的一部分,因此其他卡片的宽度和高度将决定该面板的大小(在下面的代码中为了方便起见,我重写了 getPreferredSize
方法,但我不想使用它)。
这个结果可以使用下面的代码实现,你只需要改变这行代码:
Object [][] data = new Object [100][1];
到:
Object [][] data = new Object [10][1];
问题是当 JTable
有更多的行时,在这种情况下会出现垂直滚动条,但滚动 Pane 会“收缩”:
这是我正在使用的代码:
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.table.TableColumn;
public class TableTest
{
public static void main (String [] a) {
SwingUtilities.invokeLater (new Runnable () {
@Override public void run () {
createAndShowGUI ();
}
});
}
private static void createAndShowGUI () {
JFrame frame = new JFrame ("Table Test");
frame.setDefaultCloseOperation (JFrame.EXIT_ON_CLOSE);
frame.setContentPane (new MainPanel ());
frame.pack ();
frame.setLocationRelativeTo (null);
frame.setVisible (true);
}
}
class MainPanel extends JPanel
{
private static int MAX_HEIGHT = 600;
private static int MAX_WIDTH = 600;
private JTable table;
protected MainPanel () {
super (new GridBagLayout ());
Object [][] data = new Object [100][1];
for (int i = 0; i < 10; i ++) data [i] = new String [] {"Some data ..."};
table = new JTable (data, new String [] {""}) {
@Override public Dimension getPreferredScrollableViewportSize () {
int width = 0;
for (int column = 0; column < getColumnCount (); column ++) width += columnModel.getColumn (column).getPreferredWidth ();
return new Dimension (Math.min (MAX_WIDTH, width), Math.min (MAX_HEIGHT, getRowHeight () * getRowCount ()));
}
};
table.setAutoResizeMode (JTable.AUTO_RESIZE_OFF);
table.setShowGrid (false);
table.setTableHeader (null);
resizeColumns ();
JScrollPane scrollPane = new JScrollPane (table);
scrollPane.setBackground (table.getBackground ());
scrollPane.getViewport ().setBackground (table.getBackground ());
// --- Adding components ---
GridBagConstraints c = new GridBagConstraints ();
c.fill = GridBagConstraints.VERTICAL;
c.weightx = 0.0;
c.weighty = 1.0;
add (scrollPane, c);
}
@Override public Dimension getPreferredSize () {
return new Dimension (500, 400);
}
protected void resizeColumns () {
for (int column = 0; column < table.getColumnCount (); column ++) {
TableColumn tableColumn = table.getColumnModel ().getColumn (column);
int width = 0;
for (int row = 0; row < table.getRowCount (); row ++) {
width = Math.max (table.prepareRenderer (table.getCellRenderer (row, column), row, column).getPreferredSize ().width, width);
}
tableColumn.setPreferredWidth (width + table.getIntercellSpacing ().width);
}
}
}
如果我将 c.fill
设置为 GridBagConstraints.BOTH
并将 c.weightx
设置为任何正值,则 JScrollPane
将水平和垂直填充父面板。
如何在不手动设置 JScrollPane
大小的情况下使用表格的首选宽度?
编辑
我的代码非常少,因为我发现我的问题与表格列数无关。但是我的真实程序必须处理更多列和大量行(在许多情况下为一百万或更多)。
A JList
是我的第一次尝试,因为我喜欢我可以使用原型(prototype)值的方式,但是它加载数据太慢(也可以在运行时更新),所有GUI 卡住超过 200 000 行,这也是因为我使用的是自定义渲染器。
此外,我已经尝试使用BorderLayout
,但我真的希望滚动 Pane 的高度能够填满我的面板,我尝试了不同的解决方案,但仍然无法解决。
EDIT2
正如@camickr 所建议的,我可以为我的面板使用 BorderLayout,在 BorderLayout.WEST 添加滚动 Pane 以垂直填充面板而不占用所有水平空间。这是目前最好的解决方案,但这会导致水平居中对齐丢失。有什么想法吗?
最佳答案
(in the code below i'm overriding getPreferredSize method for convenience, but i don't want to use it
我同意。摆脱覆盖。
降低框架高度时的绘制问题是因为 GridBagLayout 会在空间不足时以最小尺寸绘制组件。
所以,虽然一般来说我不喜欢使用任何一套???硬编码尺寸的方法如下:
scrollPane.setMinimumSize(scrollPane.getPreferredSize());
这将防止滚动 Pane 宽度减小。
如果您不喜欢使用硬编码尺寸,您可以使用 Relative Layout .这将使您可以轻松地将组件居中。
那么基本代码是:
RelativeLayout rl = new RelativeLayout(RelativeLayout.X_AXIS);
rl.setFill( true );
setLayout(rl);
...
add(Box.createHorizontalGlue(), new Float(1));
add(scrollPane);
add(Box.createHorizontalGlue(), new Float(1));
关于java - GridBagLayout:如何使用首选宽度水平垂直填充,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46886137/