[英]Set JTable header horizontal alignment based on value renderer column alignment
我已经看过许多有关设置JTable标头对齐的问题/答案,但是它们似乎并未涵盖这种情况。 我希望JTable标头与其下方的数据具有相同的对齐方式; 使它们全部居中对我来说很奇怪,尤其是对于偶尔有长字符串但有空格但通常只包含短字符串的列来说,标题本身位于空格的中心。
我见过的许多解决方案都依赖DefaultCellTableRenderer -所有表头都使用吗? 我不想设置默认渲染器,因为那样会更改所有列; 我想编写代码以找出列中数据的水平对齐方式,并将标题设置为在水平对齐方式上执行相同的操作。
我是否必须从渲染器获取组件(使用特定的行,列和值)并在组件上设置对齐方式? 为此,我是否需要为标题编写自定义渲染器? 什么是最干净的整体方法?
-编辑
卡米克先生的解决方案很优雅,几乎可以满足我的要求。 它拦截了为列标题创建表单元格渲染器的过程。 我以某种方式消除了标头的常规格式,但是,我的标头上没有阴影且没有边框(尽管它们按我的意愿对齐了!)。
我现在有以下内容:在哪里可以直接获取表头的渲染器,因此可以更改其对齐方式?
import java.awt.Component;
import javax.swing.JTable;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableColumn;
import javax.swing.table.TableColumnModel;
public class TableHeaderRenderer implements TableCellRenderer
{
TableCellRenderer olderRenderer = null;
public TableHeaderRenderer(TableCellRenderer givenRenderer)
{
olderRenderer = givenRenderer;
}
// // get the default renderer for the table header.
// JTableHeader header = table.getTableHeader();
// TableCellRenderer defaultRenderer = header.getDefaultRenderer();
@Override
public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column)
{
try // if the casts fail, we end up in the catch below
{
// get the renderer for the first row
DefaultTableCellRenderer firstRowTableCellRenderer = (DefaultTableCellRenderer)table.getCellRenderer(0, column);
// get the renderer for this column header
JTableHeader tableHeader = table.getTableHeader();
TableColumnModel columnModel = tableHeader.getColumnModel();
TableColumn tableColumn = columnModel.getColumn(column);
DefaultTableCellRenderer headerRenderer = (DefaultTableCellRenderer)tableColumn.getCellRenderer();
// if the renderer for the column is null, create one
if (headerRenderer == null)
{ headerRenderer = new DefaultTableCellRenderer();
}
// finally -- set the header's horizontal alignment to be the same as
// that for the first row, then get the component
headerRenderer.setHorizontalAlignment(firstRowTableCellRenderer.getHorizontalAlignment());
Component result = headerRenderer.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
return result;
}
catch (ClassCastException cce)
{
// either the first row or the header has a renderer that isn't a DefaultTableCellRenderer
// just return the component we already have
return olderRenderer.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
}
}
}
-
好的,编辑2:
import java.awt.BorderLayout;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.JTableHeader;
@SuppressWarnings("serial")
public class TableHeaderRendererDemo extends JFrame
{
Object[][]data = { {new Integer(1), "two", "three"},
{ new Integer(4), "five", "six" }
};
Object[] columnNames = { "first", "second", "third" };
public static void main(String ... arguments)
{ new TableHeaderRendererDemo().go();
}
public void go()
{
JTable table = new JTable(data, columnNames);
JScrollPane scrollPane = new JScrollPane(table);
// set our own default renderer for the headers
JTableHeader header = table.getTableHeader();
try
{ DefaultTableCellRenderer defaultRenderer = (DefaultTableCellRenderer)header.getDefaultRenderer();
header.setDefaultRenderer(new TableHeaderRenderer(defaultRenderer));
}
catch (ClassCastException e)
{
System.err.println("Could not cast default renderer; table headers not aligned");
}
add(scrollPane, BorderLayout.CENTER);
pack();
setVisible(true);
}
}
您可以看到我提到的标题问题; 我希望整数可以默认为右对齐,但是出于某些原因,他们不这样做。 但是您可以从中看到我遇到的问题。
这是一个自定义标题渲染器的示例,该渲染器仅使所选列的文本变为粗体:
class MyTableHeaderRenderer implements TableCellRenderer
{
private TableCellRenderer tableCellRenderer;
public MyTableHeaderRenderer(TableCellRenderer tableCellRenderer)
{
this.tableCellRenderer = tableCellRenderer;
}
public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column)
{
Component c = tableCellRenderer.getTableCellRendererComponent(
table, value, isSelected, hasFocus, row, column);
if (column == table.getSelectedColumn())
c.setFont(getFont().deriveFont(Font.BOLD));
return c;
}
}
您可以通过以下代码使用渲染器:
JTableHeader header = table.getTableHeader();
DefaultTableCellRenderer defaultRenderer = (DefaultTableCellRenderer)header.getDefaultRenderer();
header.setDefaultRenderer( new MyTableHeaderRenderer(defaultRenderer) );
在您的情况下,您需要更改对齐方式,而不是字体。
您可能会使用如下代码:
TableCellRenderer tcr = table.getCellRenderer(0, column);
Component renderer = table.prepareRenderer(tcr, 0, column);
defaultRenderer.setAlignment( renderer.getAlignment() ); // whatever the alignment method is.
编辑:
我以某种方式消除了标头的常规格式
您正在使用:
tableColumn.getCellRenderer();
应该不是:
tableColumn.getHeaderRenderer();
人们很少会按列为表标题指定特殊的渲染器。 因此,如果未为TableColumn指定标题渲染器,则应仅使用JTableHeader的默认渲染器,而不创建DefaultTableCellRenderer,这不是表标题使用的渲染器。
乱七八糟的直到我终于来了。 不知何故,布尔的默认渲染器不是DefaultTableCellRenderer,所以我无法通过这种方式进行对齐。 我尝试将组件放在第一行并使其ITS对齐,但是如果表中没有数据,该操作将失败。 这对于我现在想要的已经足够了。 在我看来,列的对齐必须存在于某处,并且我仍然希望可以以某种方式获得它的值,但是到目前为止,如果列数据使用DefaultTableCellRenderer,我所管理的只是一种获取它的方法。
感谢camickr和MadProgrammer。
import java.awt.Component;
import javax.swing.JTable;
import javax.swing.SwingConstants;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableCellRenderer;
/**
* Renderer that aligns the table's column header with the column's data;
* if the renderer for the first row of the column data cannot be cast to
* DefaultTableCellRenderer, this just sets the alignment to CENTER.
* <P>Use:
* <pre>
* // set our own default renderer for the headers
JTableHeader header = table.getTableHeader();
try
{ DefaultTableCellRenderer defaultRenderer = (DefaultTableCellRenderer)header.getDefaultRenderer();
header.setDefaultRenderer(new TableHeaderRenderer(defaultRenderer));
}
catch (ClassCastException e)
{
System.err.println("Could not cast default renderer; table headers not aligned");
}
*
*/
@SuppressWarnings("serial")
public class TableHeaderRenderer extends DefaultTableCellRenderer implements TableCellRenderer
{
DefaultTableCellRenderer innerRenderer = null;
public TableHeaderRenderer(TableCellRenderer givenRenderer)
{
// save the given renderer, normally the default for the header
innerRenderer = (DefaultTableCellRenderer)givenRenderer;
}
@Override
public Component getTableCellRendererComponent
(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column)
{
try
{
// get the renderer for the first row
DefaultTableCellRenderer firstRowTableCellRenderer = (DefaultTableCellRenderer)table.getCellRenderer(0, column);
int firstRowAlignment = firstRowTableCellRenderer.getHorizontalAlignment();
// set the alignment of this header to that of the first row
innerRenderer.setHorizontalAlignment(firstRowAlignment);
return innerRenderer.getTableCellRendererComponent(table, value, isSelected, hasFocus, firstRowAlignment, column);
}
catch (ClassCastException cce)
{
// the first row has a renderer that isn't a DefaultTableCellRenderer
// just set the alignment to CENTER
innerRenderer.setHorizontalAlignment(SwingConstants.CENTER);
return innerRenderer.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
}
}
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.