3

JTableにJProgressBarを実装しました。ProgressBarNOTEDITORにはレンダラーを使用しました。

ここで、ProgressBarの設定値を実装しようとしましたが、EDTが機能しないため、SwingUtiltiesを使用しましたが、機能しませんでした。

期待される動作-JProgressBarは値を80に設定する必要がありますが、現在は0%しか表示されていません

public class SkillSetTableProgressBarRenderer extends JProgressBar implements
        TableCellRenderer {

    public SkillSetTableProgressBarRenderer() {
        super(0, 100);
        super.setPreferredSize(new Dimension(100, 80));
    }

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value,
            boolean isSelected, boolean hasFocus, int row, int column) {

        final JProgressBar bar = (JProgressBar) value;

        if (bar.getString().equals("JAVA") || bar.getString().equals("SWING"))
            super.setString("Mellow");
        else
            super.setString("GOOD");

        setOpaque(true);
        table.setOpaque(true);

        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                System.err.println("MAIN ANDER");
                setValue(80);
                bar.setValue(80);
            }
        });

        super.setStringPainted(true);
        super.setIndeterminate(true);
        super.setPreferredSize(new Dimension(140, 16));
        if (isSelected) {
            super.setBackground(table.getSelectionBackground());
            super.setForeground(table.getSelectionForeground());
            // this.setBackground(table.getSelectionBackground());
        } else {
            super.setForeground(table.getForeground());
            super.setBackground(Color.WHITE);
        }

        return this;
    }

}
4

3 に答える 3

3

ProgressBar を CellRenderer として使用しようとしているようです。知っておくべきことの 1 つは、CellRenders はセルが描画されているときにのみ呼び出され、他の時点で値を設定しても効果がないということです。これは、Swing がレンダラーに flyweight パターンを使用し、レンダラーを再利用するためです。

必要な効果を得るには、

  • セルが更新されることをテーブルに通知します。たとえば、基礎となるモデルを更新することによって (必要なイベントが発生します)、
  • リターンのに必要なすべての値を設定するため、を削除します(getter は EDT で呼び出されるため、そこでのスレッド化について心配する必要はありません)。getTableCellRendererComponentinvokeLater
于 2012-02-15T14:37:24.707 に答える
2

1)可能ですが、JComponents内部TableCellRendererを作成したり、オブジェクトをre_createしたりしないでください。レンダラーは、セルの内容をフォーマットするためだけのものです。

2)SwingWorker進行中の移動に使用しJProgressBarます(本当に重要な理由がある場合は、を使用しますRunnable#Tread

についての例Runnable#Tread

ここに画像の説明を入力してください

import java.awt.Component;
import java.util.Random;
import javax.swing.JFrame;
import javax.swing.JProgressBar;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableCellRenderer;

public class TableWithProgressBars {

    public static class ProgressRenderer extends JProgressBar implements TableCellRenderer {

        private static final long serialVersionUID = 1L;

        public ProgressRenderer(int min, int max) {
            super(min, max);
            this.setStringPainted(true);
        }

        @Override
        public Component getTableCellRendererComponent(JTable table, Object value,
                boolean isSelected, boolean hasFocus, int row, int column) {
            this.setValue((Integer) value);
            return this;
        }
    }
    private static final int maximum = 100;

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {

            @Override
            public void run() {
                new TableWithProgressBars().createGUI();
            }
        });

    }

    public void createGUI() {
        final JFrame frame = new JFrame("Progressing");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        Integer[] oneRow = {0, 0, 0, 0};
        String[] headers = {"One", "Two", "Three", "Four"};
        Integer[][] data = {oneRow, oneRow, oneRow, oneRow, oneRow,};
        final DefaultTableModel model = new DefaultTableModel(data, headers);
        final JTable table = new JTable(model);
        table.setDefaultRenderer(Object.class, new ProgressRenderer(0, maximum));
        table.setPreferredScrollableViewportSize(table.getPreferredSize());
        frame.add(new JScrollPane(table));
        frame.pack();
        frame.setLocationRelativeTo(null);
        frame.setVisible(true);
        new Thread(new Runnable() {

            @Override
            public void run() {
                Object waiter = new Object();
                synchronized (waiter) {
                    int rows = model.getRowCount();
                    int columns = model.getColumnCount();
                    Random random = new Random(System.currentTimeMillis());
                    boolean done = false;
                    while (!done) {
                        int row = random.nextInt(rows);
                        int column = random.nextInt(columns);
                        Integer value = (Integer) model.getValueAt(row, column);
                        value++;
                        if (value <= maximum) {
                            model.setValueAt(value, row, column);
                            try {
                                waiter.wait(15);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                        }
                        done = true;
                        for (row = 0; row < rows; row++) {
                            for (column = 0; column < columns; column++) {
                                if (!model.getValueAt(row, column).equals(maximum)) {
                                    done = false;
                                    break;
                                }
                            }
                            if (!done) {
                                break;
                            }
                        }
                    }
                    frame.setTitle("All work done");
                }
            }
        }).start();
    }
}
于 2012-02-15T21:10:48.340 に答える
1

Swing テーブルのチュートリアル: renderers and editorsを見てください。レンダラーはコンポーネントを使用して一種の「スタンプ」を作成します。コンポーネントは実際には UI に含まれていないため、後でコンポーネントを変更しても効果はありません。これが、同じコンポーネントを何度も返すことができる理由の 1 つです。これは、コンポーネントが実際にテーブルに含まれていた場合には不可能です。これはすべて、リンクされたチュートリアルで説明されています。

問題を解決するには、メソッド呼び出しを削除し、正しい進行状況が設定されているSwingUtilities.invokeLatera を返すだけです。JProgressBar

これが正しく表示されていれば、TableModelを含む がありJProgressBarます。これには正当な理由があることを願っています。これは、通常はモデル側に含めるべきではない UI コンポーネントであるためです。これはあなたに基づいています

final JProgressBar bar = (JProgressBar) value;

レンダラーを呼び出します。TableModelこれを回避できるかどうかを確認するために、あなたを確認することを強くお勧めします

于 2012-02-15T18:05:52.413 に答える