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

问题描述 投票:0回答:2

当我调用

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

这个问题的其余部分仅在使用以下代码时才有意义。这是一个非常简单的程序,我用它来说明问题。它显示了一个 UI,其中包含一个

JSplitPane
,上部有一些控制按钮,下部包含一个
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);
      }
    });
  }
}

不受欢迎的行为

  • 运行上面的代码
  • 按“选择第 10 行”:第 10 行被选中并可见
  • 按“选择第 45 行”:第 45 行被选中并可见
  • 单击“隐藏底部”按钮。这将调整
    JSplitPane
    的分隔线,以便只有上面板可见
  • 单击“选择第 10 行”按钮。你当然什么也看不到,因为桌子还不可见
  • 单击“显示底部”按钮。分隔线已调整,但第 10 行隐藏在标题下方。我希望它无需滚动即可可见。

想要的行为

重复上面的步骤,但确保选中“调整分隔线之前使表格不可见”复选框。在隐藏底部面板之前,这将在

setVisible(false)
周围的
JScrollPane
上调用
JTable

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

截图

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

想要:执行上述步骤后第 10 行可见

环境

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

java swing jtable jsplitpane
2个回答
3
投票

这似乎是由

JViewport
处理
scrollRectToVisible
调用其大小小于所需矩形的情况引起的。它包含 JavaDocs 中的一条(有些模糊,但可能相关)注释:

注意该方法不会滚动到有效视口之外;例如,如果 contentRect 大于视口,滚动将被限制在视口的边界内。

我没有仔细阅读完整的代码并进行所有数学计算并检查所有案例。所以警告:以下解释包含完全相同的挥手动作。但在这种特殊情况下这对我意味着什么的简化描述:

当底部被隐藏时(通过相应地设置分隔线位置),则

JScrollPane
及其
JViewport
的高度为0。现在,当请求高度为20的矩形的
scrollRectToVisible
时(例如,对于一个表行),那么它会注意到这不适合。根据
JViewport
的当前视图位置,这可能会导致视口滚动,以便该矩形的 bottom 可见。

(您可以观察到这一点:手动拖动分隔线位置,使表格行的大约一半可见。单击“选择第45行”按钮时,该行的一半将可见。当点击“选择第 10 行”按钮时,该行的一半将可见)

这里似乎对我有用的一个实用解决方案是确保它始终滚动,以便矩形的

top 可见(即使矩形根本不适合视口!)。像这样:

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); }

但是当

动画发挥作用时,我不能保证这会给您带来预期的效果...


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 );
    
© www.soinside.com 2019 - 2024. All rights reserved.