/*******************************************************************************
* Copyright (c) 2006, 2011 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:
* Wind River Systems - initial API and implementation
*******************************************************************************/
package org.eclipse.debug.ui.actions;
import java.util.Iterator;
import java.util.Map;
import java.util.WeakHashMap;
import org.eclipse.core.commands.AbstractHandler;
import org.eclipse.core.commands.ExecutionEvent;
import org.eclipse.core.commands.ExecutionException;
import org.eclipse.core.commands.HandlerEvent;
import org.eclipse.core.expressions.IEvaluationContext;
import org.eclipse.debug.core.IRequest;
import org.eclipse.debug.internal.ui.commands.actions.DebugCommandService;
import org.eclipse.debug.internal.ui.commands.actions.ICommandParticipant;
import org.eclipse.debug.internal.ui.commands.actions.IEnabledTarget;
import org.eclipse.debug.ui.DebugUITools;
import org.eclipse.debug.ui.contexts.DebugContextEvent;
import org.eclipse.debug.ui.contexts.IDebugContextListener;
import org.eclipse.debug.ui.contexts.IDebugContextService;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.ui.ISources;
import org.eclipse.ui.IWindowListener;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.handlers.HandlerUtil;
/**
* Abstract base class for re-targeting command framework handlers, which
* delegate execution to {@link org.eclipse.debug.core.commands.IDebugCommandHandler}
* handlers. The specific type of IDebugCommandHandler
is
* determined by the abstract {@link #getCommandType()} method.
*
*
Note: This class is not an implementation of the IDebugCommandHandler
* interface, which was somewhat unfortunately named. IDebugCommandHandler
* is an interface that used only by the debugger plug-ins. This class implements
* {@link org.eclipse.core.commands.IHandler} interface and is to be used with the
* platform commands framework.
* Clients may subclass this class. *
* @see org.eclipse.debug.core.commands.IDebugCommandHandler * @see org.eclipse.core.commands.IHandler * * @since 3.6 */ public abstract class DebugCommandHandler extends AbstractHandler { /** * The DebugCommandService is able to evaluate the command handler * enablement in each workbench window separately, however the workbench * command framework uses only a single handler instance for all windows. * This IEnabledTarget implementation tracks enablement of the command * for a given window. When the handler enablement is tested, the * currently active window is used to determine which enabled target * to use. */ private class EnabledTarget implements IEnabledTarget, IDebugContextListener { boolean fEnabled = getInitialEnablement(); IWorkbenchWindow fWindow; EnabledTarget(IWorkbenchWindow window) { fWindow = window; DebugCommandService.getService(fWindow).updateCommand(getCommandType(), this); getContextService(fWindow).addDebugContextListener(this); } public void setEnabled(boolean enabled) { boolean oldEnabled = fEnabled; fEnabled = enabled; if (fEnabled != oldEnabled && fCurrentEnabledTarget == this) { fireHandlerChanged(new HandlerEvent(DebugCommandHandler.this, true, false)); } } public void debugContextChanged(DebugContextEvent event) { DebugCommandService.getService(fWindow).postUpdateCommand(getCommandType(), this); } void dispose() { if (isDisposed()) { return; } getContextService(fWindow).removeDebugContextListener(this); fWindow = null; } boolean isDisposed() { return fWindow == null; } } /** * Window listener is used to make sure that the handler enablement * is updated when the active workbench window is changed. */ private IWindowListener fWindowListener = new IWindowListener() { public void windowOpened(IWorkbenchWindow w) { } public void windowDeactivated(IWorkbenchWindow w) { } public void windowClosed(IWorkbenchWindow w) { EnabledTarget enabledTarget = (EnabledTarget)fEnabledTargetsMap.get(w); if (enabledTarget != null) { enabledTarget.dispose(); } } public void windowActivated(IWorkbenchWindow w) { fCurrentEnabledTarget = (EnabledTarget)fEnabledTargetsMap.get(w); fireHandlerChanged(new HandlerEvent(DebugCommandHandler.this, true, false)); } }; /** * Map of enabled targets keyed by workbench window. */ private Map fEnabledTargetsMap = new WeakHashMap(); /** * The current enabled target, based on the active * workbench window. */ private EnabledTarget fCurrentEnabledTarget = null; /** * The constructor adds the handler as the */ public DebugCommandHandler() { super(); PlatformUI.getWorkbench().addWindowListener(fWindowListener); } /* (non-Javadoc) * @see org.eclipse.core.commands.AbstractHandler#setEnabled(java.lang.Object) */ public void setEnabled(Object evaluationContext) { // This method is called with the current evaluation context // just prior to the isEnabled() being called. Check the active // window and update the current enabled target based on it fCurrentEnabledTarget = null; if (!(evaluationContext instanceof IEvaluationContext)) { return; } IEvaluationContext context = (IEvaluationContext) evaluationContext; Object _window = context.getVariable(ISources.ACTIVE_WORKBENCH_WINDOW_NAME); if (_window instanceof IWorkbenchWindow) { IWorkbenchWindow window = (IWorkbenchWindow)_window; fCurrentEnabledTarget = getEnabledTarget(window); } } /* (non-Javadoc) * @see org.eclipse.core.commands.AbstractHandler#isEnabled() */ public boolean isEnabled() { if (fCurrentEnabledTarget == null) { return false; } return fCurrentEnabledTarget.fEnabled; } private EnabledTarget getEnabledTarget(IWorkbenchWindow window) { EnabledTarget target = (EnabledTarget)fEnabledTargetsMap.get(window); if (target == null) { target = new EnabledTarget(window); fEnabledTargetsMap.put(window, target); } return target; } /* (non-Javadoc) * @see org.eclipse.core.commands.AbstractHandler#execute(org.eclipse.core.commands.ExecutionEvent) */ public Object execute(ExecutionEvent event) throws ExecutionException { IWorkbenchWindow window = HandlerUtil.getActiveWorkbenchWindow(event); if (window == null) { throw new ExecutionException("No active workbench window."); //$NON-NLS-1$ } fCurrentEnabledTarget = getEnabledTarget(window); ISelection selection = getContextService(window).getActiveContext(); if (selection instanceof IStructuredSelection && isEnabled()) { IStructuredSelection ss = (IStructuredSelection) selection; boolean enabledAfterExecute = execute(window, ss.toArray()); // enable/disable the action according to the command fCurrentEnabledTarget.setEnabled(enabledAfterExecute); } return null; } private IDebugContextService getContextService(IWorkbenchWindow window) { return DebugUITools.getDebugContextManager().getContextService(window); } /** * Executes this action on the given target objects * @param window the window * @param targets the targets to execute this action on * @return if the command stays enabled while the command executes */ private boolean execute(IWorkbenchWindow window, final Object[] targets) { DebugCommandService service = DebugCommandService.getService(window); return service.executeCommand( getCommandType(), targets, new ICommandParticipant() { public void requestDone(org.eclipse.debug.core.IRequest request) { DebugCommandHandler.this.postExecute(request, targets); } }); } /** * This method is called after the completion of the execution of this * command. Extending classes may override this method to perform additional * operation after command execution. * * @param request The completed request object which was given the the * debug command handler. * @param targets Objects which were the targets of this action */ protected void postExecute(IRequest request, Object[] targets) { // do nothing by default } /** * Returns the {@link org.eclipse.debug.core.commands.IDebugCommandHandler} * command handler that type this action executes. * * @return command class. * * @see org.eclipse.debug.core.commands.IDebugCommandHandler */ abstract protected Class getCommandType(); /** * Returns whether this action should be enabled when initialized * and there is no active debug context. * * @return false, by default */ protected boolean getInitialEnablement() { return false; } /** * Clean up when removing */ public void dispose() { PlatformUI.getWorkbench().removeWindowListener(fWindowListener); for (Iterator itr = fEnabledTargetsMap.values().iterator(); itr.hasNext();) { EnabledTarget target = (EnabledTarget)itr.next(); if (!target.isDisposed()) { target.dispose(); } } fEnabledTargetsMap.clear(); fCurrentEnabledTarget = null; } }