我使用以下代码:
public final class TableCellRendererCenter extends DefaultTableCellRenderer {
public static final TableCellRenderer INSTANCE
= new TableCellRendererCenter();
protected TableCellRendererCenter() {
// Calling super
super();
this.setHorizontalAlignment(SwingConstants.CENTER);
}
}
在JTable列上:
TableColumnModel retrMod = ChartItemsTable.getColumnModel();
TableColumn retrCol = retrMod.getColumn(2);
retrCol.setHeaderRenderer(TableCellRendererCenter.INSTANCE);
retrCol.setCellRenderer(TableCellRendererCenter.INSTANCE);
并且外观与其他列标题不再匹配:
为什么呢?我该如何解决这个问题?
修改
NetBeans似乎正在使用来自DefaultTableCellHeaderRenderer
package sun.swing.table;
我到处读到我不应该使用防晒包。 Grrrr ......这没有用!
编辑2
0verbose的建议产生以下内容:
解
根据eugener的建议,我更新了我的代码如下:
public final class TableCellRendererCenter extends DefaultTableCellRenderer {
@Override
public Component getTableCellRendererComponent(
JTable table, Object value, boolean isSelected,
boolean hasFocus, int row, int column) {
// returns component used for default header rendering
// makes it independent on current L&F
Component retr = table.getTableHeader().getDefaultRenderer().
getTableCellRendererComponent(
table, value, isSelected, hasFocus, row, column);
if ( JLabel.class.isAssignableFrom(retr.getClass()) ) {
JLabel jl = (JLabel) retr;
jl.setHorizontalAlignment(SwingConstants.CENTER);
}
return retr;
}
@Override
public void validate() {}
@Override
public void revalidate() {}
@Override
public void firePropertyChange(
String propertyName, boolean oldValue, boolean newValue) {}
@Override
public void firePropertyChange(
String propertyName, Object oldValue, Object newValue) {}
}
我明白了:
保留了外观并且列标题居中! (其他列标题也是居中的,但我可以通过对int row, int column
参数进行进一步测试来控制它。
答案 0 :(得分:5)
问题是您使用的是不同的单元格渲染器,而不是LAF使用的单元格渲染器。 IMO实现正确LAF的唯一方法是使用已由LAF本身提供的渲染器,例如:
public class TableHeaderRenderer extends JComponent implements TableCellRenderer {
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected,
boolean hasFocus, int row, int column) {
// returns component used for default header rendering
// makes it independent on current L&F
return table.getTableHeader().getDefaultRenderer().
getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
}
// following methods are overriden for performance reasons
@Override
public void validate() {}
@Override
public void revalidate() {}
@Override
public void firePropertyChange(String propertyName, boolean oldValue, boolean newValue) {}
@Override
public void firePropertyChange(String propertyName, Object oldValue, Object newValue) {}
}
显然,一旦你拥有了正确的渲染器,就可以将它投射到JLabel(当然是在检查它之后),然后将文本居中或做其他事情。
答案 1 :(得分:2)
也许你可以尝试用这种方式重构你的课程:
public class TableCellRendererCenter extends JLabel implements TableCellRenderer{
public TableCellRendererCenter(JTable associatedTable){
this.table = associatedTable;
JTableHeader header = this.table.getTableHeader();
this.setHorizontalAlignment(SwingConstants.CENTER);
this.setForeground(header.getForeground());
this.setBackground(header.getBackground());
this.setFont(header.getFont());
}
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
return this;
}}
编辑: 我现在检查了。 DefaultTableCellRenderer已经扩展了JComponent。所以可能只需要这些行:
this.setForeground(header.getForeground());
this.setBackground(header.getBackground());
this.setFont(header.getFont());
(我不确定。我做了类似的事情,但没有列标题,也没有扩展DefaultTableCellRenderer)