Java JTable изменить цвет ячейки - PullRequest
11 голосов
/ 15 апреля 2011

Я хотел бы создать редактируемую таблицу и затем проверить данные, чтобы убедиться, что они действительны.Я не уверен, как изменить цвет только одной клетки.Я хотел бы получить ячейку, например (0,0) и перекрасить передний план в красный.Я читал другие посты на SO, а также Oracle о пользовательском ColorRenderer, но я просто не понимаю, как бы это использовать.

Спасибо.

Ответы [ 7 ]

18 голосов
/ 15 апреля 2011

Скажите, что ячейка, которую вы хотите отобразить другим цветом, представляет состояние (в качестве примера я возьму Отклонено и Одобрено). Затем я реализовал бы метод в моей табличной модели с именем getStatus (int row), который возвращает статус для любой данной строки.

Тогда, когда это будет на месте, я бы начал создавать средство визуализации ячеек, отвечающее за рендеринг столбца, которому принадлежит ячейка. Рендерер ячеек будет что-то в строках кода ниже.

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());

Чтобы сделать ячейку редактируемой, просто реализуйте метод isCellEditable (int rowIndex, int columnIndex) в вашей табличной модели. Вам также необходимо реализовать метод setValueAt (значение объекта, int rowIndex, int columnIndex), если вы хотите сохранить значение, предоставленное пользователем (что, я полагаю, вы делаете!).

2 голосов
/ 15 апреля 2011

Я хотел бы создать редактируемую таблицу, а затем проверить данные, чтобы убедиться, что они действительны.

Другой подход заключается в редактировании данных перед их сохранением в модель таблицы для предотвращения ввода неверных данных.

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);
    }
}
1 голос
/ 25 мая 2015

Я считаю, что правильный способ сделать раскраску в таблице - это ColorHighlighter. У средств визуализации таблиц есть проблемы с отображением разных цветов в одном столбце.

Вот пример использования маркеров.В этом случае это для выделения ячейки, которая не редактируется.

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);
1 голос
/ 25 ноября 2012

Это самый простой способ закрасить определенный столбец или ячейку в 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;
    }
}

Этот код получает столбец ячейки для визуализации

TableColumn col = tblExamHistoryAll.getColumnModel().getColumn(5);
DefaultTableModel model3 = (DefaultTableModel)tblExamHistoryAll.getModel();
col.setCellRenderer(new CustomRenderer());

Это для удаления всех предыдущих строк из вашей таблицы. Если вы не хотите их просто удалить эти строки

model3.getDataVector().removeAllElements();
model3.fireTableDataChanged();
1 голос
/ 15 апреля 2011

Самый простой способ - написать простой TableCellRenderer, расширив DefaultTableCellRenderer и переписав метод getTableCellRendererComponent до setBackground( Color.RED ). Например:

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;
    }
});
0 голосов
/ 28 июня 2019

Вы можете расширить DefaultTableCellRenderer, переопределить getTableCellRendererComponent и вызвать что-то вроде

if (myConditions) setBackground(myColor);

перед возвратом «this», когда применяются условия, но у него есть очень раздражающий побочный эффект изменения заднего цвета по умолчанию из-за способа кодирования DefaultTableCellRenderer.setBackGround.

Уловка, которую я обнаружил, заключалась в том, чтобы полностью продублировать код DefaultTableCellRenderer в классе с именем HackedDefaultTableCellRenderer, добавить метод, который напрямую вызывает реализацию компонента setBackground:

public void setComponentBackground(Color c) {
    super.setBackground(c);
}

затем выводят мои настроенные отрисовки из этого взломанного класса, а не из DefaultTableCellRenderer, и, наконец, вызывают setComponentBackground вместо setBackground в моем настроенном getTableCellRendererComponent.

Недостатком является то, что этот HackedDefaultTableCellRenderer использует снимок DefaultTableCellRenderer.

0 голосов
/ 14 июня 2018
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus,
        int row, int col) {
    Component c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, col);
    int control = row;
    control = control % 2;
    control = (control == 0) ? 1 : 0;
    if (control == 1) {
        c.setBackground(Color.green);
    } else {
        c.setBackground(Color.cyan);
    }
    return c;
}
...