JTable#scrollRectToVisible与JSplitPlane结合显示错误的行

时间:2015-08-05 13:37:46

标签: java swing jtable jsplitpane

当我拨打JTable#scrollRectToVisible时,我想要显示的行在某些情况下隐藏在标题下方。

使用以下代码时,此问题的其余部分才有意义。这是一个非常简单的程序,我用它来说明问题。它显示了一个包含JSplitPane的UI,上半部分包含一些控制按钮,下半部分包含一个JTable包裹在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);
      }
    });
  }
}

不需要的行为

  • 运行以上代码
  • 按&#34;选择第10行&#34;:选择第10行并显示
  • 按&#34;选择行45&#34;:第45行被选中且可见
  • 点击&#34;隐藏底部&#34;按钮。这将调整JSplitPane的分隔符,以便只显示上面板
  • 点击&#34;选择第10行和第34行;按钮。你当然看不到任何东西,因为桌子还不可见
  • 点击&#34;显示底部&#34;按钮。调整分隔符,但第10行隐藏在标题下方。我希望它在不需要滚动的情况下可见。

通缉行为

重复上面的步骤,但在调整分隔符之前确保&#34; Make表不可见&#34;选中复选框。在隐藏底部面板之前,这会在setVisible(false)周围的JScrollPane上调用JTable

通过这样做,在最后一步中,第10行将作为最顶行显示,这就是我想要的。 我只是不想让滚动窗格不可见:在我的实际应用程序中,分隔符以动画方式调整,因此您希望在动画期间保持表格可见。

截图

不需要:执行上述步骤后,第10行不可见

Unwanted behavior screenshot

通缉:执行上述步骤后,第10行可见

Wanted behavior screenshot

环境

我认为这不重要,但以防万一:我在Linux系统上使用JDK7。

2 个答案:

答案 0 :(得分:3)

这似乎是由JViewport处理scrollRectToVisible调用其大小小于所需矩形的情况的方式引起的。它包含JavaDocs中的(有些模糊但可能相关)注释:

  

请注意,此方法不会在有效视口外滚动;例如,如果contentRect大于视口,则滚动将限制在视口的范围内。

我没有通过完整的代码并完成所有数学并检查所有情况。所以警告:以下解释包含完全相同的挥手。但是在这种特殊情况下,这简单描述了这对我意味着什么:

当隐藏底部部分时(通过相应地设置分隔符位置),JScrollPane及其JViewport的此高度为0.现在,当请求scrollRectToVisible时高度为20的矩形(对于一个表行,作为示例),然后它会注意到这不适合。根据{{​​1}}的当前视图位置,这可能会导致视口滚动,以便可以看到此矩形的底部

(您可以观察到:手动拖动分隔符位置 ,以便可以看到一个表行的大约一半。单击&#34;选择行45& #34;按钮,该行的上部一半将是可见的。当点击&#34;选择第10行&#34;按钮,然后 lower 的一半行将可见)

这里似乎对我有用的一个实用解决方案是确保它总是滚动,以便矩形的顶部可见(即使矩形根本不适合视!)。像这样:

JViewport

但是,当动画发挥作用时,我无法承诺这对你有所期望的效果......

答案 1 :(得分:2)

不完全确定scrollRectToVisible()正在做什么。

您可以使用JViewport.setViewPosition(...)方法。

Rectangle r = fTable.getCellRect(aRowIndex, 0, true);
Point p = new Point(r.x, r.y);
fScrollPane.getViewport().setViewPosition( p );

在这种情况下,所选行将始终显示在视口的顶部(如果可能)。因此视口将始终滚动,除非所选行在顶部是当前行。使用此方法,如果第一行位于视口的顶部,并且您选择第10行,则视口将滚动以显示顶部的第10行。

但是,此行为与使用scrollRectToVisible()方法略有不同。当使用scrollRectToVisible()方法时,只有当矩形不在视口的可见部分时才滚动视口。如果第一行位于视口的顶部并且您选择第10行,则使用此方法,视口将不会滚动,因为第10行已在视口中可见。

不知道这种功能变化是否可以接受。

请注意,如果您不想在选择行时自动滚动视口,可以尝试以下操作:

JViewport viewport = fScrollPane.getViewport();
Rectangle viewRect = viewport.getViewRect();
Rectangle r = fTable.getCellRect(aRowIndex, 0, true);
Point p = new Point(r.x, r.y);

if (! viewRect.contains(p))
    viewport.setViewPosition( p );