概要

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

サンプルコード

enum Component { TOTAL, FILE, LOG }
class Progress {
  public final Object value;
  public final Component component;
  public Progress(Component component, Object value) {
    this.component = component;
    this.value = value;
  }
}
//...
worker = new SwingWorker<String, Progress>() {
  @Override public String doInBackground() {
    int current = 0;
    int lengthOfTask = 12; //filelist.size();
    publish(new Progress(Component.LOG, "Length Of Task: " + lengthOfTask));
    publish(new Progress(Component.LOG, "\n------------------------------\n"));
    while (current < lengthOfTask && !isCancelled()) {
      if (!bar1.isDisplayable()) {
        return "Disposed";
      }
      try {
        convertFileToSomething();
      } catch (InterruptedException ie) {
        return "Interrupted";
      }
      publish(new Progress(Component.LOG, "*"));
      publish(new Progress(Component.TOTAL, 100 * current / lengthOfTask));
      current++;
    }
    publish(new Progress(Component.LOG, "\n"));
    return "Done";
  }
  private final Random r = new Random();
  private void convertFileToSomething() throws InterruptedException {
    int current = 0;
    int lengthOfTask = 10 + r.nextInt(50); //long lengthOfTask = file.length();
    while (current <= lengthOfTask && !isCancelled()) {
      int iv = 100 * current / lengthOfTask;
      Thread.sleep(20); // dummy
      publish(new Progress(Component.FILE, iv + 1));
      current++;
    }
  }
  @Override protected void process(List<Progress> chunks) {
    for (Progress s: chunks) {
      switch (s.component) {
        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 all

解説

上記のサンプルでは、デフォルトで用意されている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);
    }
  }
}

参考リンク

コメント