JTableのセルにJProgressBarを表示
Total: 16857
, Today: 2
, Yesterday: 1
Posted by aterai at
Last-modified:
概要
JTable
のセルにJProgressBar
を使用して進捗を表示します。
Screenshot
Advertisement
サンプルコード
class ProgressRenderer extends DefaultTableCellRenderer {
private final JProgressBar b = new JProgressBar(0, 100);
public ProgressRenderer() {
super();
setOpaque(true);
b.setBorder(BorderFactory.createEmptyBorder(1, 1, 1, 1));
}
@Override public Component getTableCellRendererComponent(
JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column) {
Integer i = (Integer) value;
String text = "Done";
if (i < 0) {
text = "Canceled";
} else if (i < 100) {
b.setValue(i);
return b;
}
super.getTableCellRendererComponent(
table, text, isSelected, hasFocus, row, column);
return this;
}
}
private final Executor executor = Executors.newCachedThreadPool();
// ...
int rowNumber = model.getRowCount();
SwingWorker<Integer, Integer> worker = new SwingWorker<Integer, Integer>() {
private int rndSleep = new Random().nextInt(100) + 1;
private int lengthOfTask = 120;
@Override protected Integer doInBackground() {
int current = 0;
while (current < lengthOfTask && !isCancelled()) {
current++;
try {
Thread.sleep(rndSleep);
} catch (InterruptedException ie) {
publish(-1);
break;
}
publish(100 * current / lengthOfTask);
}
return rndSleep * lengthOfTask;
}
@Override protected void process(List<Integer> chunks) {
for (Integer value: chunks) {
model.setValueAt(value, rowNumber, 2);
}
// model.fireTableCellUpdated(rowNumber, 2);
}
@Override protected void done() {
String text = null;
int i = -1;
if (isCancelled()) {
text = "Canceled";
} else {
try {
i = get();
text = "Done";
} catch (Exception ignore) {
ignore.printStackTrace();
text = ignore.getMessage();
}
}
System.out.println(rowNumber + ":" + text + "(" + i + "ms)");
}
};
model.addTest(new Test("example", 0), worker);
executor.execute(worker); // 1.6.0_18
// worker.execute(); //1.6.0_21
View in GitHub: Java, Kotlin解説
上記のサンプルでは、add
ボタンをクリックするとSwingWorker
を使用したランダムな仮タスクが起動し、その進捗状況が2
列目セル内のJProgressBar
で表示されます。
ProgressRenderer
はJProgressBar
を一つ持ち、仮タスクが動いている間はそのJProgressBar
に値を設定して描画用のコンポーネントとして返す- タスクが終了(またはキャンセル)されたら
JLabel
(自分自身、DefaultTableCellRenderer
)に文字列を設定して返す - このサンプルでは、行番号をキーにしているため、例えばモデルから行を削除するときに実行中のタスクが手前の行などに存在する場合はエラーが発生する
- これを回避するため、実際には行の削除を実行せず、フィルタを使って行を非表示に設定する
- 参考: RowFilterでJTableの行をフィルタリング
SwingWorker
スレッドの同時実行上限は10
で固定- Swing - Maximum number of SwingWorker objects in a Swing app?
public abstract class SwingWorker<T, V> implements RunnableFuture<T> { /** * number of worker threads. */ private static final int MAX_WORKER_THREADS = 10;
- Swing - Maximum number of SwingWorker objects in a Swing app?
参考リンク
- SwingWorkerを使った処理の中断と進捗状況表示
- TableCellRendererに進捗文字列を設定したJProgressBarを使用する
- SwingWorkerを一スレッドずつ順番に実行する