Summary

ひとつのSwingWorkerで、進捗を表示するJProgressBarをふたつ使用します。

Source Code Examples

enum ComponentType { TOTAL, FILE, LOG }
class Progress {
  public final Object value;
  public final ComponentType componentType;
  public Progress(ComponentType componentType, Object value) {
    this.componentType = componentType;
    this.value = value;
  }
}
// ...
worker = new SwingWorker<String, Progress>() {
  private final Random r = new Random();
  @Override public String doInBackground() throws InterruptedException {
    int current = 0;
    int lengthOfTask = 12; // filelist.size();
    publish(new Progress(ComponentType.LOG, "Length Of Task: " + lengthOfTask));
    publish(new Progress(ComponentType.LOG, "\n--------------------------\n"));
    while (current < lengthOfTask && !isCancelled()) {
      if (!bar1.isDisplayable()) {
        return "Disposed";
      }
      convertFileToSomething();
      publish(new Progress(ComponentType.LOG, "*"));
      publish(new Progress(ComponentType.TOTAL, 100 * current / lengthOfTask));
      current++;
    }
    publish(new Progress(ComponentType.LOG, "\n"));
    return "Done";
  }

  private void convertFileToSomething() throws InterruptedException {
    int current = 0;
    int lengthOfTask = 10 + r.nextInt(50); // = file.length();
    while (current <= lengthOfTask && !isCancelled()) {
      int iv = 100 * current / lengthOfTask;
      Thread.sleep(20);
      publish(new Progress(Component.FILE, iv + 1));
      current++;
    }
  }

  @Override protected void process(List<Progress> chunks) {
    for (Progress s: chunks) {
      switch (s.componentType) {
        case TOTAL: bar1.setValue((Integer) s.value); break;
        case FILE: bar2.setValue((Integer) s.value); break;
        case LOG: area.append((String) s.value); break;
      }
    }
  }
  // ...
View in GitHub: Java, Kotlin

Explanation

上記のサンプルでは、デフォルトで用意されているSwingWorker#setProgress(int)は使用せず、以下の3つのコンポーネントの状態を表すProgressクラスを作成し、これをSwingWorker<String, Progress>#publish(Progress)メソッドに与えてEDT上でそれぞれの状態を更新しています。

  • 全体の進捗を表示するJProgressBar
  • 個々のファイル処理(このサンプルではランダムにThread.sleep(...)を実行するだけ)の進捗を表示するJProgressBar
  • ログを表示するJTextArea

  • SwingWorkerに別のPropertyChangeListenerを追加する方法もある
worker.firePropertyChange("file-progress", iv, iv + 1);
// ...
class SubProgressListener implements PropertyChangeListener {
  private final JProgressBar progressBar;
  public SubProgressListener(JProgressBar progressBar) {
    this.progressBar = progressBar;
    this.progressBar.setValue(0);
  }
  @Override public void propertyChange(PropertyChangeEvent e) {
    String strPropertyName = e.getPropertyName();
    if ("file-progress".equals(strPropertyName)) {
      int progress = (Integer) e.getNewValue();
      progressBar.setValue(progress);
    }
  }
}

Reference

Comment