Rob*_*bin 8 java swing jtable jsplitpane
当我打电话时JTable#scrollRectToVisible,我想要显示的行在某些情况下隐藏在标题下方.
这个问题的其余部分仅在使用以下代码时才有意义.这是一个非常简单的程序,我用它来说明问题.它显示了一个UI JSplitPane,其上部包含一些控制按钮,下部包含一个JTable包裹在a中的JScrollPane(参见本文底部的屏幕截图).
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import javax.swing.*;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableModel;
public class DividerTest {
private final JSplitPane fSplitPane = new JSplitPane(JSplitPane.VERTICAL_SPLIT);
private final JTable fTable;
private final JScrollPane fScrollPane;
private boolean fHideTable = false;
public DividerTest() {
fTable = new JTable( createTableModel(50));
fScrollPane = new JScrollPane(fTable);
fSplitPane.setBottomComponent(fScrollPane);
fSplitPane.setTopComponent(createControlsPanel());
fSplitPane.setDividerLocation(0.5);
}
private JPanel createControlsPanel(){
JPanel result = new JPanel();
result.setLayout(new BoxLayout(result, BoxLayout.PAGE_AXIS));
final JCheckBox checkBox = new JCheckBox("Make table invisible before adjusting divider");
checkBox.addItemListener(new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {
fHideTable = checkBox.isSelected();
}
});
result.add(checkBox);
JButton upperRow = new JButton("Select row 10");
upperRow.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
selectRowInTableAndScroll(10);
}
});
result.add(upperRow);
JButton lowerRow = new JButton("Select row 45");
lowerRow.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
selectRowInTableAndScroll(45);
}
});
result.add(lowerRow);
JButton hideBottom = new JButton("Hide bottom");
hideBottom.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
if (fHideTable) {
fScrollPane.setVisible(false);
}
fSplitPane.setDividerLocation(1.0);
}
});
result.add(hideBottom);
JButton showBottom = new JButton("Show bottom");
showBottom.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
fScrollPane.setVisible(true);
fSplitPane.setDividerLocation(0.5);
}
});
result.add(showBottom);
return result;
}
private void selectRowInTableAndScroll( int aRowIndex ){
fTable.clearSelection();
fTable.getSelectionModel().addSelectionInterval(aRowIndex, aRowIndex);
fTable.scrollRectToVisible(fTable.getCellRect(aRowIndex, 0, true));
}
public JComponent getUI(){
return fSplitPane;
}
private TableModel createTableModel(int aNumberOfRows){
Object[][] data = new Object[aNumberOfRows][1];
for( int i = 0; i < aNumberOfRows; i++ ){
data[i] = new String[]{"Row" + i};
}
return new DefaultTableModel(data, new String[]{"Column"});
}
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
JFrame frame = new JFrame("Test frame");
frame.getContentPane().add(new DividerTest().getUI());
frame.pack();
frame.setVisible(true);
frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
}
});
}
}
Run Code Online (Sandbox Code Playgroud)
不必要的行为
JSplitPane以便只能看到上面板想要行为
重复上面的步骤,但确保选中"在调整分隔符之前使表格不可见"复选框.这将调用隐藏底部面板之前setVisible(false)的JScrollPane周围JTable.
通过这样做,在最后一步中,第10行将作为最顶行显示,这就是我想要的.我只是不想让滚动窗格变得不可见:在我的实际应用程序中,分隔符以动画方式调整,因此您希望在动画期间保持表格可见.
截图
不需要的:执行上述步骤后,第10行是不可见的
通缉:执行上述步骤后,第10行可见
环境
我认为这不重要,但以防万一:我在Linux系统上使用JDK7.
这似乎是由处理其尺寸小于所需矩形的情况的调用JViewport方式引起的。scrollRectToVisible它在 JavaDocs 中包含一条(有些模糊,但可能相关)注释:
请注意,此方法不会滚动到有效视口之外;例如,如果 contentRect 大于视口,则滚动将被限制在视口的边界内。
我没有仔细阅读完整的代码并进行所有数学计算并检查所有案例。所以警告:以下解释包含完全相同的挥手动作。但在这种特殊情况下这对我意味着什么的简化描述:
当底部被隐藏时(通过相应地设置分隔符位置),则 的高度JScrollPane为JViewport0。 现在,当请求scrollRectToVisible高度为 20 的矩形时(以一个表格行为例),然后它会注意到这不合适。根据 的当前视图位置JViewport,这可能会导致视口滚动,以便该矩形的底部可见。
(您可以观察到这一点:手动拖动分隔线位置,使一个表格行的大约一半可见。当单击“选择第 45 行”按钮时,该行的上半部分将可见。当单击“选择第 10 行”按钮时,该行的上半部分将可见。 " 按钮,则该行的下半部分将可见)
这里似乎对我有用的一种实用解决方案是确保它始终滚动,以便矩形的顶部可见(即使矩形根本不适合视口!)。像这样:
private void selectRowInTableAndScroll(int aRowIndex)
{
fTable.clearSelection();
fTable.getSelectionModel().addSelectionInterval(aRowIndex, aRowIndex);
Rectangle r = fTable.getCellRect(aRowIndex, 0, true);
r.height = Math.min(fScrollPane.getHeight(), r.height);
fTable.scrollRectToVisible(r);
}
Run Code Online (Sandbox Code Playgroud)
但我不能保证当动画开始播放时这会给您带来所需的效果......