/******************************************************************************* * Copyright (c) 2000, 2006 IBM Corporation and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation *******************************************************************************/ package org.eclipse.ui.internal.dialogs; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IProgressMonitorWithBlocking; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.ProgressMonitorWrapper; import org.eclipse.jface.dialogs.Dialog; import org.eclipse.swt.widgets.Display; import org.eclipse.ui.internal.ExceptionHandler; /** * Used to run an event loop whenever progress monitor methods * are invoked. <p> * This is needed since editor save operations are done in the UI thread. * Although save operations should be written to do the work in the non-UI thread, * this was not done for 1.0, so this was added to keep the UI live * (including allowing the cancel button to work). */ public class EventLoopProgressMonitor extends ProgressMonitorWrapper implements IProgressMonitorWithBlocking { /** * Threshold for how often the event loop is spun, in ms. */ private static int T_THRESH = 100; /** * Maximum amount of time to spend processing events, in ms. */ private static int T_MAX = 50; /** * Last time the event loop was spun. */ private long lastTime = System.currentTimeMillis(); /** * The task name is the name of the current task * in the event loop. */ private String taskName; /** * Constructs a new instance of the receiver and forwards to monitor. * @param monitor */ public EventLoopProgressMonitor(IProgressMonitor monitor) { super(monitor); } /** * @see IProgressMonitor#beginTask */ public void beginTask(String name, int totalWork) { super.beginTask(name, totalWork); taskName = name; runEventLoop(); } /* (non-Javadoc) * @see org.eclipse.core.runtime.IProgressMonitorWithBlocking#clearBlocked() */ public void clearBlocked() { Dialog.getBlockedHandler().clearBlocked(); } /** * @see IProgressMonitor#done */ public void done() { super.done(); taskName = null; runEventLoop(); } /** * @see IProgressMonitor#internalWorked */ public void internalWorked(double work) { super.internalWorked(work); runEventLoop(); } /** * @see IProgressMonitor#isCanceled */ public boolean isCanceled() { runEventLoop(); return super.isCanceled(); } /** * Runs an event loop. */ private void runEventLoop() { // Only run the event loop so often, as it is expensive on some platforms // (namely Motif). long t = System.currentTimeMillis(); if (t - lastTime < T_THRESH) { return; } lastTime = t; // Run the event loop. Display disp = Display.getDefault(); if (disp == null) { return; } //Initialize an exception handler from the window class. ExceptionHandler handler = ExceptionHandler.getInstance(); for (;;) { try { if (!disp.readAndDispatch()) { break; } } catch (Throwable e) {//Handle the exception the same way as the workbench handler.handleException(e); break; } // Only run the event loop for so long. // Otherwise, this would never return if some other thread was // constantly generating events. if (System.currentTimeMillis() - t > T_MAX) { break; } } } /* (non-Javadoc) * @see org.eclipse.core.runtime.IProgressMonitorWithBlocking#setBlocked(org.eclipse.core.runtime.IStatus) */ public void setBlocked(IStatus reason) { Dialog.getBlockedHandler().showBlocked(this, reason, taskName); } /** * @see IProgressMonitor#setCanceled */ public void setCanceled(boolean b) { super.setCanceled(b); taskName = null; runEventLoop(); } /** * @see IProgressMonitor#setTaskName */ public void setTaskName(String name) { super.setTaskName(name); taskName = name; runEventLoop(); } /** * @see IProgressMonitor#subTask */ public void subTask(String name) { //Be prepared in case the first task was null if (taskName == null) { taskName = name; } super.subTask(name); runEventLoop(); } /** * @see IProgressMonitor#worked */ public void worked(int work) { super.worked(work); runEventLoop(); } /** * Return the name of the current task. * @return Returns the taskName. */ protected String getTaskName() { return taskName; } }