java - JPanel 的首选高度低于表渲染器中其子级的组合高度

标签 java swing jtable jlabel

我有一个JTable渲染器返回 JPanel由多个JLabel组成实例。其中之一JLabel s 可以包含 HTML,用于使用 <br/> 将输出分割为多行。标签。

为了显示表中的多行,渲染器调用getTableCellRendererComponent方法

table.setRowHeight(row, componentToReturn.getPreferredSize().height);

根据内容动态更新行高。这仅在 componentToReturn 时才能正常工作。表示正确的首选尺寸。

但是看起来 getPreferredSize返回虚假值。返回组件的首选高度小于组件内标签的高度之和。

这是一个说明此行为的小程序(不使用 JTable )

import java.awt.*;
import javax.swing.*;

public class SwingLabelTest {
  public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
      @Override
      public void run() {
        LabelPanel renderer = new LabelPanel();
        Component component = renderer.getComponent(false);
        //asking for a bigger component will not
        //update the preferred size of the returned component
        component = renderer.getComponent(true);
      }
    });
  }

  private static class LabelPanel {
    private final JPanel compositePanel;
    private final JLabel titleLabel = new JLabel();
    private final JLabel propertyLabel = new JLabel();

    public LabelPanel() {
      JPanel labelPanel = new JPanel();
      labelPanel.setLayout(new BoxLayout(labelPanel, BoxLayout.PAGE_AXIS));

      labelPanel.add(titleLabel);
      labelPanel.add(propertyLabel);

      compositePanel = new JPanel(new BorderLayout());
      //normally it contains more components,
      //but that is not needed to illustrate the problem
      compositePanel.add(labelPanel, BorderLayout.CENTER);
    }

    public Component getComponent( boolean aMultiLineProperty ) {
      titleLabel.setText("Title");
      if ( aMultiLineProperty ){
        propertyLabel.setText("<html>First line<br/>Property: value</html>");
      } else {
        propertyLabel.setText("Property: value");
      }

      int titleLabelHeight = titleLabel.getPreferredSize().height;
      int propertyLabelHeight = propertyLabel.getPreferredSize().height;
      int compositePanelHeight = compositePanel.getPreferredSize().height;
      if ( compositePanelHeight < titleLabelHeight + propertyLabelHeight){
        throw new RuntimeException("Preferred size of the component returned "
                                   + "by the renderer is incorrect");
      }

      return compositePanel;
    }
  }
}

据我所知,前面的示例有点牵强,这里的示例包含 JTable

import java.awt.*;
import javax.swing.*;
import javax.swing.table.*;

public class SwingTableTest {

  public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
      @Override
      public void run() {
        DefaultTableModel tableModel = new DefaultTableModel(0, 1);
        JTable table = new JTable(tableModel);
        table.setDefaultRenderer(Object.class, new DataResultRenderer());
        tableModel.addRow(new Object[]{new Object()});
        tableModel.addRow(new Object[]{new Object()});
        tableModel.addRow(new Object[]{new Object()});

        JFrame testFrame = new JFrame("TestFrame");
        testFrame.getContentPane().add(new JScrollPane(table));
        testFrame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        testFrame.setSize(new Dimension(300, testFrame.getPreferredSize().height));
        testFrame.setVisible(true);
      }
    });
  }

  private static class DataResultRenderer implements TableCellRenderer {
    private final JPanel compositePanel;
    private final JLabel titleLabel = new JLabel();
    private final JLabel propertyLabel = new JLabel();

    public DataResultRenderer() {

      JPanel labelPanel = new JPanel();
      labelPanel.setOpaque(false);
      labelPanel.setLayout(new BoxLayout(labelPanel, BoxLayout.PAGE_AXIS));

      labelPanel.add(titleLabel);
      labelPanel.add(propertyLabel);

      compositePanel = new JPanel(new BorderLayout());
      //normally it contains more components,
      //but that is not needed to illustrate the problem
      compositePanel.add(labelPanel, BorderLayout.CENTER);
    }

    @Override
    public Component getTableCellRendererComponent(
        JTable table, Object value, boolean isSelected, 
        boolean hasFocus, int row, int column) {
      titleLabel.setText("Title");

      if ( row == 2 ){
        propertyLabel.setText("<html>Single property: value</html>");
      } else {
        String text = "<html>";
        text += "First property<br/>";
        text += "Second property<br/>";
        text += "Third property:value";
        text += "</html>";
        propertyLabel.setText(text);
      }

      int titleLabelHeight = titleLabel.getPreferredSize().height;
      int propertyLabelHeight = propertyLabel.getPreferredSize().height;
      int compositePanelHeight = compositePanel.getPreferredSize().height;
      if ( compositePanelHeight < titleLabelHeight + propertyLabelHeight){
        throw new RuntimeException("Preferred size of the component returned "
                                   + "by the renderer is incorrect");
      }
      table.setRowHeight(row, compositePanel.getPreferredSize().height);
      return compositePanel;
    }

  }
}

我正在寻找一种方法来更新表格的行高,以确保多行内容完全可见,而无需预先知道每行将包含多少行。

因此,要么我需要一种解决方案来检索正确的首选尺寸,要么我的方法完全错误,然后我需要一个更好的解决方案。

请注意,上述示例已被简化。在真实的代码中,“渲染器”(负责创建组件的代码)被修饰了几次。这意味着外部渲染器是唯一可以访问JTable的渲染器。 ,并且它不知道什么样的 Component内部代码返回。

最佳答案

因为setRowHeight() “将所有单元格的高度(以像素为单位)设置为rowHeight,重新验证并重新绘制”,这种方法是不合理的。如果没有抛出异常,分析显示 CPU 使用率为 100%,因为无休止的级联重绘尝试重复更改行高。此外,行选择变得不可靠。

一些替代方案包括:

  • 使用TablePopupEditor根据 TableCellEditor 的请求显示多行内容。

  • TableModelListener 更新相邻的多行面板,如图 here .

关于java - JPanel 的首选高度低于表渲染器中其子级的组合高度,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32827634/

相关文章:

java - 使用 Rhino 调用将回调作为参数的函数

java - Errai 如何序列化/反序列化模型实体

java - 模拟内部最终对象

java - 如何在pdf上打印bufferedImage

java - 更改 JTable 单元格背景颜色

java - JTable 细胞监听器?

java - GWT - 使用布局面板的 Google 可视化问题?

java - 如何在不使用 JFileChooser 的情况下读取 JTextArea 中的文件 (.txt)?

java - 背景图像未显示在 JPanel 中

java - 如何在 Java 中将所有没有任何 ID 或重复 ID 的文件导出到文本文件?