web-dev-qa-db-ja.com

JTableの行の背景色を変更する

3列のJTableがあります。このように3つの列すべてにTableCellRendererを設定しました(おそらくあまり効果的ではないでしょうか?)。

_ for (int i = 0; i < 3; i++) {
     myJTable.getColumnModel().getColumn(i).setCellRenderer(renderer);
 }
_

getTableCellRendererComponent()は、各行の背景色がランダムなコンポーネントを返します。
プログラムの実行中に背景を他のランダムな色に変更するにはどうすればよいですか?

20
user

1つの方法は、モデル内の各行の現在の色を保存することです。 3列3行に固定された単純なモデルを次に示します。

static class MyTableModel extends DefaultTableModel {

    List<Color> rowColours = Arrays.asList(
        Color.RED,
        Color.GREEN,
        Color.CYAN
    );

    public void setRowColour(int row, Color c) {
        rowColours.set(row, c);
        fireTableRowsUpdated(row, row);
    }

    public Color getRowColour(int row) {
        return rowColours.get(row);
    }

    @Override
    public int getRowCount() {
        return 3;
    }

    @Override
    public int getColumnCount() {
        return 3;
    }

    @Override
    public Object getValueAt(int row, int column) {
        return String.format("%d %d", row, column);
    }
}

setRowColourfireTableRowsUpdatedを呼び出すことに注意してください。これにより、テーブルのその行だけが更新されます。

レンダラーはテーブルからモデルを取得できます。

static class MyTableCellRenderer extends DefaultTableCellRenderer {

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
        MyTableModel model = (MyTableModel) table.getModel();
        Component c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
        c.setBackground(model.getRowColour(row));
        return c;
    }
}

行の色の変更は次のように簡単です:

model.setRowColour(1, Color.YELLOW);
21
Richard Fearn

リチャードファーンズの回答を再開して、各2行目を灰色にします。

jTable.setDefaultRenderer(Object.class, new DefaultTableCellRenderer()
{
    @Override
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column)
    {
        final Component c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
        c.setBackground(row % 2 == 0 ? Color.LIGHT_GRAY : Color.WHITE);
        return c;
    }
});
24

すべての列で同じレンダラーを使用しているため、ここに示されている他の回答はうまく機能します。

ただし、一般にJTableを使用する場合、各列に異なるタイプのデータが存在するため、各列に同じレンダラーを使用することはないと考えがちです。これらの場合、 Table Row Rendering アプローチが役立つことがあります。

15
camickr

getTableCellRendererComponent(...)の呼び出しには、レンダラーが検索されるセルの値が含まれます。

その値を使用して色を計算できます。 AbstractTableModelも使用している場合は、レンダラーに任意の型の値を提供できます。

色を取得したら、返すコンポーネントでsetBackground()できます。

1
Andy Thomas

これは基本的に、テーブルの再描画と同じくらい簡単です。ただし、行/列/セルを1つだけ選択的に再描画する方法を見つけていません。

この例では、ボタンをクリックすると行の背景色が変更され、再描画が呼び出されます。

public class TableTest {
    public static void main(String[] args) {
        JFrame frame = new JFrame();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        final Color[] rowColors = new Color[] {
                randomColor(), randomColor(), randomColor()
        };
        final JTable table = new JTable(3, 3);
        table.setDefaultRenderer(Object.class, new TableCellRenderer() {
            @Override
            public Component getTableCellRendererComponent(JTable table,
                    Object value, boolean isSelected, boolean hasFocus,
                    int row, int column) {
                JPanel pane = new JPanel();
                pane.setBackground(rowColors[row]);
                return pane;
            }
        });
        frame.setLayout(new BorderLayout());

        JButton btn = new JButton("Change row2's color");
        btn.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent e) {
                rowColors[1] = randomColor();
                table.repaint();
            }
        });

        frame.add(table, BorderLayout.NORTH);
        frame.add(btn, BorderLayout.SOUTH);
        frame.pack();
        frame.setVisible(true);
    }

    private static Color randomColor() {
        Random rnd = new Random();
        return new Color(rnd.nextInt(256),
                rnd.nextInt(256), rnd.nextInt(256));
    }
}
1
Mark Peters