Java JTable更改单元格颜色

Java JTable更改单元格颜色,java,swing,colors,jtable,Java,Swing,Colors,Jtable,我想制作一个可编辑的表格,然后检查数据以确保其有效。我不知道如何改变一个单元格的颜色。我想获得一个单元格,例如(0,0),并将前景颜色设置为红色。我已经阅读了SO和Oracle上关于自定义颜色渲染器的其他帖子,但我不知道如何使用它 谢谢。假设您希望以不同颜色渲染的单元格表示一种状态(我将以“已拒绝”和“已批准”为例)。然后在我的表模型中实现一个名为getStatus(introw)的方法,该方法返回任何给定行的状态 然后,在适当的时候,我会创建一个单元渲染器,负责渲染单元所属的列。单元渲染器将是

我想制作一个可编辑的表格,然后检查数据以确保其有效。我不知道如何改变一个单元格的颜色。我想获得一个单元格,例如(0,0),并将前景颜色设置为红色。我已经阅读了SO和Oracle上关于自定义颜色渲染器的其他帖子,但我不知道如何使用它


谢谢。

假设您希望以不同颜色渲染的单元格表示一种状态(我将以“已拒绝”和“已批准”为例)。然后在我的表模型中实现一个名为getStatus(introw)的方法,该方法返回任何给定行的状态

然后,在适当的时候,我会创建一个单元渲染器,负责渲染单元所属的列。单元渲染器将是下面代码行中的内容

public class StatusColumnCellRenderer extends DefaultTableCellRenderer {
  @Override
  public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int col) {

    //Cells are by default rendered as a JLabel.
    JLabel l = (JLabel) super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, col);

    //Get the status for the current row.
    CustomTableModel tableModel = (CustomTableModel) table.getModel();
    if (tableModel.getStatus(row) == CustomTableModel.APPROVED) {
      l.setBackground(Color.GREEN);
    } else {
      l.setBackground(Color.RED);
    }

  //Return the JLabel which renders the cell.
  return l;

}
然后,当渲染器就位时,只需使用以下代码将渲染器“应用”到表中:

Table.getColumnModel().getColumn(columnIndex).setCellRenderer(new StatusColumnCellRenderer());
关于使单元格可编辑,只需在表模型中实现(introwIndex,intcolumnIndex)方法。您还需要实现该方法
(Object value,int rowIndex,int columnIndex)如果您想保留用户提供的值(我假设您会这样做!)。

最简单的方法是通过扩展并覆盖
gettableCellRenderComponent
方法到
setBackground(Color.RED),编写一个简单的
TableCellRenderer
。例如:

final JTable table = new JTable(...);
table.setCellRenderer( new DefaultTableCellRenderer() {
    public Component getTableCellRenderer(JTable table, Object value, ...) {
        super.getTableCellRenderer(...);

        if ( value should be highlighted ) {
            setBackground( Color.RED );
        }
        return this;
    }
});
我想制作一个可编辑的表格,然后检查数据以确保其有效

另一种方法是在将数据保存到表模型之前编辑数据,以防止输入无效数据

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.text.*;
import javax.swing.event.*;
import javax.swing.border.*;
import javax.swing.table.*;

public class TableEdit extends JFrame
{
    TableEdit()
    {
        JTable table = new JTable(5,5);
        table.setPreferredScrollableViewportSize(table.getPreferredSize());

        JScrollPane scrollpane = new JScrollPane(table);
        getContentPane().add(scrollpane);

        //  Use a custom editor

        TableCellEditor fce = new FiveCharacterEditor();
        table.setDefaultEditor(Object.class, fce);
    }

    class FiveCharacterEditor extends DefaultCellEditor
    {
        FiveCharacterEditor()
        {
            super( new JTextField() );
        }

        public boolean stopCellEditing()
        {
            try
            {
                String editingValue = (String)getCellEditorValue();

                if(editingValue.length() != 5)
                {
                    JTextField textField = (JTextField)getComponent();
                    textField.setBorder(new LineBorder(Color.red));
                    textField.selectAll();
                    textField.requestFocusInWindow();

                    JOptionPane.showMessageDialog(
                        null,
                        "Please enter string with 5 letters.",
                        "Alert!",JOptionPane.ERROR_MESSAGE);
                    return false;
                }
            }
            catch(ClassCastException exception)
            {
                return false;
            }

            return super.stopCellEditing();
        }

        public Component getTableCellEditorComponent(
            JTable table, Object value, boolean isSelected, int row, int column)
        {
            Component c = super.getTableCellEditorComponent(
                table, value, isSelected, row, column);
            ((JComponent)c).setBorder(new LineBorder(Color.black));

            return c;
        }

    }

    public static void main(String [] args)
    {
        JFrame frame = new TableEdit();
        frame.setDefaultCloseOperation(EXIT_ON_CLOSE);
        frame.pack();
        frame.setLocationRelativeTo( null );
        frame.setVisible(true);
    }
}

这是为jTable中的特定列或单元格着色的最简单方法。

首先,只需创建一个简单的
CustomRenderer类

class CustomRenderer extends DefaultTableCellRenderer <br />
{
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column)
    {
        Component c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
        setForeground(Color.blue); >
        return c;
    }
}
这是为了清除表中以前的所有行。如果不需要,只需删除这些行即可

model3.getDataVector().removeAllElements();
model3.fireTableDataChanged();

我认为在表格中着色的正确方法是通过
彩色荧光灯
表格渲染器在同一列中渲染不同颜色时出现问题。

下面是一个如何使用荧光灯的例子。在这种情况下,它用于高亮显示不可编辑的单元格

public class IsCellEditablePredicate implements HighlightPredicate {

   private JXTable table;

   public IsCellEditablePredicate (final JXTable table) {
       this.table = table;
   }

   @Override
   public boolean isHighlighted(Component component, ComponentAdapter componentAdapter) {

        return !table.isCellEditable(componentAdapter.row,
          componentAdapter.column);
   }
}
然后在设置表格的代码中添加荧光灯及其颜色参数:

 ColorHighlighter grayHighlighter = new ColorHighlighter(new IsCellEditablePredicate(table));

    grayHighlighter.setBackground(Color.LIGHT_GRAY);
    grayHighlighter.setForeground(table.getForeground());
    grayHighlighter.setSelectedBackground(table.getSelectionBackground().darker());
    grayHighlighter.setSelectedForeground(table.getSelectionForeground().darker());

    table.setHighlighters(grayHighlighter);

您可以扩展DefaultTableCellRenderer,重写getTableCellRendererComponent并调用

if (myConditions) setBackground(myColor);
在返回“this”之前,如果条件适用,但由于DefaultTableCellRenderer.setBackGround的编码方式,它会产生一个非常恼人的副作用,即更改默认背景色

我发现的诀窍是在名为HackedDefaultTableCellRenderer的类中完全复制DefaultTableCellRenderer的代码,添加一个直接调用组件的setBackground实现的方法:

public void setComponentBackground(Color c) {
    super.setBackground(c);
}
然后从这个被黑客攻击的类而不是从DefaultTableCellRenderer派生自定义呈现,最后在自定义GetTableCellRenderer组件中调用setComponentBackground而不是setBackground


缺点是此HackedDefaultTableCellRenderer依赖于DefaultTableCellRenderer的快照。

或者:如果愿意,也可以避免在表模型中实现getStatus()方法。比方说,如果表模型的getValueAt(int-rowIndex,int-columnIndex)以字符串形式返回状态(“已批准”、“已拒绝”),那么您可以简单地让渲染器检查提供的值[…]renderComponent(表、值、布尔值[…]),并基于此更改单元格的颜色。这样,您就不需要绕道表模型来获取当前行/列的状态。这实际上可能是最简单的方法…最后一点注意:如果您想要实现getStatus(int行)的解决方案,那么请确保使用table.convertRowIndexToModel(int行)转换行索引。提供给GetTableCellRenderComponent()的行是视图索引,如果由于排序而更改了行的顺序,则该行可能与模型索引不同。这可能是我正在寻找的,我将给出一个快照并让您知道。我有一个保存按钮,在保存之前检查所有数据是否有效。我希望能够保存有效的数据,并跳过无效的数据。谢谢否-这是不完整的:DefaultTableCellEditor是一个已损坏的实现,它展示了臭名昭著的“颜色记忆”-当触摸颜色一次时,您必须始终这样做,即添加一个else分支到if@kleopatra没错,应该有一个
else
将颜色重置为黑色。此外,在
JTable
上没有
setCellRenderer
方法,因此这是一个不完整的示例。我的意图是,它将充分展示OP,让他们开始走上正确的道路,他们可以自己学习小细节。-1对于fireXX,从外部(想知道是否是你,名字完全相同,新帐户?)你的粗体句子到底是什么意思?你能举个例子吗?欢迎来到Stack Overflow!请不要把你的源代码扔在这里。对你的答案要和蔼可亲,尽量给它一个很好的描述,这样其他人就会喜欢它并投赞成票。见:
if (myConditions) setBackground(myColor);
public void setComponentBackground(Color c) {
    super.setBackground(c);
}