gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-11-30 03:15:06 24 4
gpt4 key购买 nike

我有一个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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com