5

Nimbus で行ごとの色分けがどのように機能するかわかりません。クレイジーなようです!!! ここで整理したいと思います。

デモンストレーションのために、赤とピンクの行を交互に表示する JTable が必要だとしましょう(最初の色がどちらであるかは気にしません)。

独自の「モジュロ 2」を実行するカスタム cellRenderer を再定義せず、JTable からのメソッドをオーバーライドせずに、自分のアプリケーションを開始してから、 Nimbus プロパティのみを使用してカスタムの代替行の色を持つ JTable を取得するまでの必須の手順をリストしたいと思います。

これが私が従うことを期待したステップです:

  1. Nimbus PLAF をインストールする
  2. 「Table.background」ニンバス プロパティをカスタマイズする
  3. 「Table.alternateRowColor」ニンバス プロパティをカスタマイズする
  4. 単純なデータ/ヘッダーで JTable を作成する
  5. JScrollPane で jTable をラップし、JFrame に追加します。
  6. JFrame を表示する

ソースコードは次のとおりです。


public class JTableAlternateRowColors implements Runnable {

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new JTableAlternateRowColors());
    }

    @Override
    public void run() {
        try {
            UIManager.setLookAndFeel(new NimbusLookAndFeel());
        } catch (UnsupportedLookAndFeelException e) {
            e.printStackTrace();
        }

        UIManager.getDefaults().put("Table.background", Color.RED);
        UIManager.getDefaults().put("Table.alternateRowColor", Color.PINK);

        final JFrame jFrame = new JFrame("Nimbus alternate row coloring");
        jFrame.getContentPane().add(new JScrollPane(new JTable(new String[][] {
                {"one","two","three"},
                {"one","two","three"},
                {"one","two","three"}
        }, new String[]{"col1", "col2", "col3"}
        )));
        jFrame.setSize(400, 300);
        jFrame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        jFrame.setVisible(true);
    }
}

これはJDK6コードです。誰かがここで間違っていると教えてもらえますか?


@kleopatra のコメントとコミュニティ全体の貢献によると、Nimbus プロパティのみを使用して行を交互に色付けする方法は次のとおりです。

public class JTableAlternateRowColors は Runnable を実装します {

public static void main(String[] args) {
    SwingUtilities.invokeLater(new JTableAlternateRowColors());
}

@Override
public void run() {
    try {
        UIManager.setLookAndFeel(new NimbusLookAndFeel());
    } catch (UnsupportedLookAndFeelException e) {
        e.printStackTrace();
    }

    UIManager.put("Table.background", new ColorUIResource(Color.RED));
    UIManager.put("Table.alternateRowColor", Color.PINK);
    UIManager.getLookAndFeelDefaults().put("Table:\"Table.cellRenderer\".background", new ColorUIResource(Color.RED));

    final JFrame jFrame = new JFrame("Nimbus alternate row coloring");
    final JTable jTable = new JTable(new String[][]{
            {"one", "two", "three"},
            {"one", "two", "three"},
            {"one", "two", "three"}
    }, new String[]{"col1", "col2", "col3"});
    jTable.setFillsViewportHeight(true);
    jFrame.getContentPane().add(new JScrollPane(jTable));
    jFrame.setSize(400, 300);
    jFrame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    jFrame.setVisible(true);
}

}

4

2 に答える 2

11

いくつかのバグの干渉のように見えます...

デフォルトのテーブルの背景とデフォルトのストライピングの両方を変更する場合、期待される (あなただけでなく、私のものも) UIManager の構成 (alternateRow プロパティを尊重するすべての LAF で同じ) は次のようになります。

UIManager.put("Table.background", Color.RED);
UIManager.put("Table.alternateRowColor", Color.PINK);

MetalでもNimbusでも機能しません

  • メタル: ストライピングなし、テーブルはすべて赤
  • Nimbus: 白/ピンクのストライプ、つまりテーブルの背景は無視されます

最初の根本的な理由は、DefaultTableCellRenderer にあります。

Color background = unselectedBackground != null
                        ? unselectedBackground
                        : table.getBackground();
if (background == null || background instanceof javax.swing.plaf.UIResource) {
    Color alternateColor = DefaultLookup.getColor(this, ui, "Table.alternateRowColor");
    if (alternateColor != null && row % 2 != 0) {
        background = alternateColor;
    }
}

ロジックが曲がっています。代替色は、テーブルの背景が colorUIResource である場合にのみ使用されますが、これはかなり弱い区別です。とにかく、次の試みにつながります。

UIManager.put("Table.background", new ColorUIResource(Color.RED));
UIManager.put("Table.alternateRowColor", Color.PINK);

これは問題ないように見えます (チェックボックス レンダラーの典型的な問題を除きますが、それはまた別のバグ ストーリーです ;-) 金属については、Nimbus についてはまだ運がありません。

次のステップは、関連する可能性のあるNimbusのデフォルトを検索し、適用することです(LAFを設定した後):

UIManager.getLookAndFeelDefaults().put("Table:\"Table.cellRenderer\".background", 
    new ColorUIResource(Color.RED));

編集(コメントで尋ねられたとおり)

JXTable は、この問題を完全に回避しようとします。ストライピングの手段は、HighlighterFactory から取得された Highlighter です。alternateRowColor プロパティを lookAndFeelDefaults から削除し、新しいキー「UIColorHighlighter.stripingBackground」を追加して、Nimbus でダーティにする必要があります。

于 2012-10-22T13:29:15.103 に答える
3

Nimbus プロパティを使用して (@Kleopatra に +1 して、私が間違っていることを証明してください:()、行の色を交互に設定することができます

UIManager.put("Table.alternateRowColor", Color.PINK);
UIManager.getLookAndFeelDefaults().put("Table:\"Table.cellRenderer\".background", Color.RED);

別の方法:

セルを必要な色 ( REDおよびPINK )でペイントするために拡張JTableおよびオーバーライドします。prepareRenderer(TableCellRenderer renderer, int row, int column)

これが役立つことを願っていた短い例です。

追加機能:ビューポートの高さと幅全体の行をペイントpaintComponent(..)する呼び出し もオーバーライドしますが、これは が に設定されている場合にのみ適用されます:paintEmptyRows(Graphics g)JScrollPanesetFillsViewPortHeighttrueMyTable

ここに画像の説明を入力

import java.awt.Color;
import java.awt.Component;
import java.awt.Graphics;
import java.awt.Rectangle;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.WindowConstants;
import javax.swing.plaf.nimbus.NimbusLookAndFeel;
import javax.swing.table.TableCellRenderer;

public class JTableAlternateRowColors {

    public JTableAlternateRowColors() {
        initComponents();
    }

    public static void main(String[] args) {

        try {
            UIManager.setLookAndFeel(new NimbusLookAndFeel());
        } catch (UnsupportedLookAndFeelException e) {
            e.printStackTrace();
        }

        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                new JTableAlternateRowColors();
            }
        });
    }

    private void initComponents() {

        final JFrame jFrame = new JFrame("Nimbus alternate row coloring");

        MyTable table = new MyTable(new String[][]{
                    {"one", "two", "three"},
                    {"one", "two", "three"},
                    {"one", "two", "three"}
                }, new String[]{"col1", "col2", "col3"});

        table.setFillsViewportHeight(true);//will fill the empty spaces too if any

        table.setPreferredScrollableViewportSize(table.getPreferredSize());

        JScrollPane jScrollPane = new JScrollPane(table);

        jFrame.getContentPane().add(jScrollPane);
        jFrame.pack();
        jFrame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        jFrame.setVisible(true);
    }
}

class MyTable extends JTable {

    public MyTable(String[][] data, String[] fields) {
        super(data, fields);
    }

    @Override
    public void paintComponent(Graphics g) {
        super.paintComponent(g);

        if (getFillsViewportHeight()) {
            paintEmptyRows(g);
        }
    }

    /**
     * Paints the backgrounds of the implied empty rows when the table model is
     * insufficient to fill all the visible area available to us. We don't
     * involve cell renderers, because we have no data.
     */
    protected void paintEmptyRows(Graphics g) {
        final int rowCount = getRowCount();
        final Rectangle clip = g.getClipBounds();
        if (rowCount * rowHeight < clip.height) {
            for (int i = rowCount; i <= clip.height / rowHeight; ++i) {
                g.setColor(colorForRow(i));
                g.fillRect(clip.x, i * rowHeight, clip.width, rowHeight);
            }
        }
    }

    /**
     * Returns the appropriate background color for the given row.
     */
    protected Color colorForRow(int row) {
        return (row % 2 == 0) ? Color.RED : Color.PINK;
    }

    /**
     * Shades alternate rows in different colors.
     */
    @Override
    public Component prepareRenderer(TableCellRenderer renderer, int row, int column) {
        Component c = super.prepareRenderer(renderer, row, column);
        if (isCellSelected(row, column) == false) {
            c.setBackground(colorForRow(row));
            c.setForeground(UIManager.getColor("Table.foreground"));
        } else {
            c.setBackground(UIManager.getColor("Table.selectionBackground"));
            c.setForeground(UIManager.getColor("Table.selectionForeground"));
        }
        return c;
    }
}

参考文献:

于 2012-10-22T11:21:11.323 に答える