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