4

我正在尝试为我的 JTable 创建行标题。我已经在这里这里完成了我的研究,但那里几乎什么都没有。我发现的并没有保持系统的外观。

这是我到目前为止发现的:

在此处输入图像描述

第一列(没有名字的那一列)是我能做到的最好的。我采用了表格标题渲染器并将其应用于行单元格。它部分起作用了。如您所见,降序/升序排序的小箭头图标出现在列的每个单元格中,并且该单元格现在是灰色的。除此之外,它完全失败了。此外,我不希望行标题中有任何箭头图标。

Col 1 是当我尝试从 UIManager 应用外观时发生的。显然它没有用。

Col 2 是photoshop,这是我想要做的。

我对其他 2 列没有做任何事情,所以它是默认设置。

问题是: 如何将系统(不仅是 Windows)默认表格标题的外观和感觉赋予行单元格,以便创建行标题,当然也没有小箭头图标。

这是代码

import java.awt.Component;
import java.awt.Dimension;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.UIManager;
import javax.swing.table.DefaultTableColumnModel;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableColumnModel;
import sun.swing.table.DefaultTableCellHeaderRenderer;

public class RowHeaderTest extends JFrame
{

    public RowHeaderTest()
    {
        initComponents();
    }

    private void initComponents()
    {
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);

        JScrollPane scrollPane = new JScrollPane();
        JTable table = new JTable();
        table.setAutoCreateRowSorter(true);
        table.getTableHeader().setReorderingAllowed(false);
        table.setModel(new DefaultTableModel(
                new Object[][]
                {
                    {
                        "Row 1", "Data 2", "Data 3", "Data 4", "Data 5"
                    },
                    {
                        "Row 2", "Data 6", "Data 7", "Data 8", "Data 9"
                    },
                    {
                        "Row 3", "Data 10", "Data 11", "Data 12", "Data 13"
                    }
                },
                new String[]
                {
                    "", "Col 1", "Col 2", "Col 3", "Col 4"
                }));
        table.getColumnModel().getColumn(0).setHeaderRenderer(HeaderRenderer.THIS);
        table.getColumnModel().getColumn(0).setCellRenderer(HeaderRenderer.THIS);
        table.getColumnModel().getColumn(1).setHeaderRenderer(HeaderRendererUI.THIS);
        table.getColumnModel().getColumn(1).setCellRenderer(HeaderRendererUI.THIS);
        table.getColumnModel().getColumn(3).setHeaderRenderer(RowTableHeaderRendere.THIS);
        table.getColumnModel().getColumn(3).setCellRenderer(RowTableHeaderRendere.THIS);
        scrollPane.setViewportView(table);
        setSize(new Dimension(400, 200));
        setLocationRelativeTo(null);
        add(scrollPane);
    }

    public static void main(String args[])
    {
        try
        {
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
        }
        catch (ClassNotFoundException | InstantiationException | IllegalAccessException | javax.swing.UnsupportedLookAndFeelException ex)
        {
            java.util.logging.Logger.getLogger(RowHeaderTest.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        java.awt.EventQueue.invokeLater(new Runnable()
        {
            public void run()
            {
                new RowHeaderTest().setVisible(true);
            }
        });
    }
}

final class HeaderRenderer extends DefaultTableCellHeaderRenderer
{

    public static final HeaderRenderer THIS = new HeaderRenderer();

    @Override
    public Component getTableCellRendererComponent(
            JTable table, Object value, boolean isSelected,
            boolean hasFocus, int row, int column)
    {
        TableCellRenderer renderer = table.getTableHeader().getDefaultRenderer();
        return renderer.getTableCellRendererComponent(table, value, isSelected,
                hasFocus, row, column);
    }
}

final class HeaderRendererUI extends DefaultTableCellHeaderRenderer
{

    public static final HeaderRendererUI THIS = new HeaderRendererUI();

    @Override
    public Component getTableCellRendererComponent(
            JTable table, Object value, boolean isSelected,
            boolean hasFocus, int row, int column)
    {
        setText(value.toString());
        //TableHeader.ancestorInputMap
        setBackground(isSelected ? UIManager.getColor("TableHeader.focusCellBackground") : UIManager.getColor("TableHeader.background"));
        setBorder(UIManager.getBorder("TableHeader.cellBorder"));
        setFont(UIManager.getFont("TableHeader.font"));
        setForeground(UIManager.getColor("TableHeader.foreground"));
        return this;
    }
}

final class RowTableHeaderRendere extends DefaultTableCellHeaderRenderer
{

    public static final RowTableHeaderRendere THIS = new RowTableHeaderRendere();

    @Override
    public Component getTableCellRendererComponent(
            JTable table, Object value, boolean isSelected,
            boolean hasFocus, int row, int column)
    {
        JTable t = new JTable(new DefaultTableModel(
                null,
                new String[]
                {
                    value.toString()
                }));
        return t.getTableHeader();
    }
}

编辑

我只是在代码中添加了部分解决方案。Col 3 现在看起来不错,但仍然缺乏感觉(它没有鼠标悬停),我不太确定创建一个新表只是为了获取标题是最好的方法。

4

1 回答 1

3

要保持系统 LookAndFeel,您需要稍微更改渲染器。

首先用 UIRessource 扩展你的渲染器。您需要此标记界面来接收 UI 更改:

class HeaderRenderer extends DefaultTableCellHeaderRenderer 
                     implements javax.swing.plaf.UIResource {

然后从您的 JTableHeader 获取并保存原始渲染器(例如,作为构造函数传递的字段 defRenderer)。在你的getTableCellRendererComponent 调用defRenderer.getTableCellRendererComponent,修改并返回这个JLabel。

private final TableCellRenderer defRenderer;

HeaderRenderer (TableCellRenderer defRenderer) {
  this.defRenderer = defRenderer;
}

@Override
public Component getTableCellRendererComponent(
        JTable table, Object value, boolean isSelected,
        boolean hasFocus, int row, int column) {
  Component c = defRenderer.getTableCellRendererComponent (...);
  if ( c instanceof JLabel ) {
    JLabel lbl = (JLabel)c;
    // do anything you want...
  }
  return c;
}

至少覆盖 Renderer 中的 updateUI() 方法并将 updateUI 调用委托给 defRenderer:

@Override
public void updateUI() {
  TableCellRenderer locDefRenderer = defRenderer;
  if (locDefRenderer instanceof JComponent) {
    ((JComponent) locDefRenderer).updateUI();
  } else {
    super.updateUI();
  }
}

有了这张收据,就可以改变很多东西。普通paintComponent-rendering有点复杂;-) - 对于自己的油漆使用JLayer(在Java 7中引入)

于 2014-06-16T15:41:24.007 回答