OsmAnd/DataExtractionOSM/src/net/osmand/swing/ProgressDialog.java

233 lines
5.9 KiB
Java

package net.osmand.swing;
import java.awt.BorderLayout;
import java.awt.Component;
import java.lang.reflect.InvocationTargetException;
import javax.swing.BorderFactory;
import javax.swing.JDialog;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JProgressBar;
import javax.swing.SwingUtilities;
import net.osmand.IProgress;
import net.osmand.LogUtil;
import org.apache.commons.logging.Log;
public class ProgressDialog extends JDialog implements IProgress {
private static final long serialVersionUID = -3915486672514402269L;
private final static Log log = LogUtil.getLog(ProgressDialog.class);
private JProgressBar progressBar;
private JLabel label;
private Runnable run;
private InvocationTargetException exception = null;
private Object result;
// Progress variables
private static final float deltaToChange = 0.001f;
private String taskName;
private int deltaWork;
private WorkerThread workerThread;
private String genProgress;
private long previousTaskStarted = 0;
public ProgressDialog(Component parent, String name){
super(JOptionPane.getFrameForComponent(parent), true);
setTitle(name);
initDialog();
}
@Override
public boolean isInterrupted(){
return !isVisible();
}
@SuppressWarnings("deprecation")
public Object run() throws InvocationTargetException, InterruptedException {
result = null;
workerThread = new WorkerThread();
workerThread.start();
setVisible(true);
if(workerThread.checkIsLive()){
// that's really bad solution unless we don't find any problems with that
// means monitor objects & we continue to use because otherwise
// we should protect all places where it is used regular checks that process is interrupted
workerThread.stop();
throw new InterruptedException();
}
if (exception != null) {
throw exception;
}
return result;
}
private class WorkerThread extends Thread {
private boolean isLive = true;
public boolean checkIsLive(){
return isLive;
}
@Override
public void run() {
try {
if (run != null) {
run.run();
}
} catch (RuntimeException e) {
exception = new InvocationTargetException(e);
} finally {
isLive = false;
}
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
setVisible(false);
}
});
}
}
private void initDialog() {
JPanel pane = new JPanel(new BorderLayout());
pane.setBorder(BorderFactory.createEmptyBorder(10,10,10,10));
progressBar = new JProgressBar();
pane.add(progressBar, BorderLayout.SOUTH);
label = new JLabel();
pane.add(label, BorderLayout.CENTER);
add(pane);
label.setText(Messages.getString("OsmExtractionUI.PLEASE.WAIT")); //$NON-NLS-1$
progressBar.setIndeterminate(true);
setSize(550, 100);
double x = getParent().getBounds().getCenterX();
double y = getParent().getBounds().getCenterY();
setLocation((int) x - getWidth() / 2, (int) y - getHeight() / 2);
}
public Object getResult() {
return result;
}
public void setResult(Object result) {
this.result = result;
}
public void setRunnable(Runnable run) {
this.run = run;
}
@Override
public void progress(int deltaWork) {
this.deltaWork += deltaWork;
if(change(progressBar.getValue() + this.deltaWork)){
progressBar.setValue(progressBar.getValue() + this.deltaWork);
this.deltaWork = 0;
updateMessage();
}
}
private void updateMessage() {
if(!progressBar.isIndeterminate()){
String format = String.format("\t %.1f %%", progressBar.getValue() * 100f / ((float) progressBar.getMaximum())); //$NON-NLS-1$
label.setText(taskName + format + (genProgress == null ? "" : (" " + genProgress))); //$NON-NLS-1$ //$NON-NLS-2$
}
}
public boolean change(int newProgress) {
if (newProgress < progressBar.getValue()) {
return false;
}
if ((newProgress - progressBar.getValue()) / ((float) progressBar.getMaximum()) < deltaToChange) {
return false;
}
return true;
}
@Override
public void remaining(int remainingWork) {
if(change(progressBar.getMaximum() - remainingWork)){
progressBar.setValue(progressBar.getMaximum() - remainingWork);
updateMessage();
}
deltaWork = progressBar.getMaximum() - remainingWork - this.progressBar.getValue();
}
@Override
public boolean isIndeterminate(){
return progressBar.isIndeterminate();
}
@Override
public void setGeneralProgress(String genProgress) {
this.genProgress = genProgress;
}
@Override
public void startTask(String taskName, int work) {
if (log.isDebugEnabled()) {
log.debug("Memory before task exec: " + Runtime.getRuntime().totalMemory() + " free : " + Runtime.getRuntime().freeMemory()); //$NON-NLS-1$ //$NON-NLS-2$
if (previousTaskStarted == 0) {
log.debug(taskName + " started"); //$NON-NLS-1$
} else {
log.debug(taskName + " started after " + (System.currentTimeMillis() - previousTaskStarted) + " ms"); //$NON-NLS-1$ //$NON-NLS-2$
}
}
previousTaskStarted = System.currentTimeMillis();
if(taskName == null){
taskName = ""; //$NON-NLS-1$
}
label.setText(taskName + (genProgress == null ? "" : (" "+genProgress))); //$NON-NLS-1$ //$NON-NLS-2$
this.taskName = taskName;
startWork(work);
}
@Override
public void finishTask() {
if (taskName != null) {
label.setText(taskName);
}
progressBar.setIndeterminate(true);
}
@Override
public void startWork(int work) {
if(work == 0){
work = 1;
}
final int w = work;
SwingUtilities.invokeLater(new Runnable(){
@Override
public void run() {
if(w != -1){
progressBar.setMinimum(0);
progressBar.setMaximum(w);
progressBar.setValue(0);
progressBar.setIndeterminate(false);
} else {
progressBar.setIndeterminate(true);
}
}
});
deltaWork = 0;
}
}