/****************************************************************************** * Copyright (c) 2006, 2008 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.gmf.runtime.emf.commands.core.command; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import java.util.Map; import org.eclipse.core.commands.ExecutionException; import org.eclipse.core.commands.operations.IOperationApprover; import org.eclipse.core.commands.operations.IUndoableOperation; import org.eclipse.core.commands.operations.OperationHistoryFactory; import org.eclipse.core.resources.IFile; import org.eclipse.core.runtime.IAdaptable; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Status; import org.eclipse.emf.ecore.EObject; import org.eclipse.emf.ecore.resource.Resource; import org.eclipse.emf.transaction.Transaction; import org.eclipse.emf.transaction.TransactionalEditingDomain; import org.eclipse.emf.transaction.util.CompositeChangeDescription; import org.eclipse.emf.workspace.AbstractEMFOperation; import org.eclipse.emf.workspace.util.WorkspaceSynchronizer; import org.eclipse.gmf.runtime.common.core.command.CommandResult; import org.eclipse.gmf.runtime.common.core.command.ICommand; import org.eclipse.gmf.runtime.common.core.internal.command.ICommandWithSettableResult; import org.eclipse.gmf.runtime.common.core.util.StringStatics; /** * An abstract superclass for GMF {@link IUndoableOperation}s that modify EMF * model resources. * <p> * The operation provides a list of {@link IFile}s that are expected to be * modified when the operation is executed, undone or redone. An * {@link IOperationApprover} is registered with the * {@link OperationHistoryFactory#getOperationHistory()} to validate the * modification to these resources. * <p> * Subclasses must return the command execution result in their implementation * of {@link #doExecuteWithResult(IProgressMonitor, IAdaptable)}. * <p> * This class is meant to be extended by clients. * * @author ldamus */ public abstract class AbstractTransactionalCommand extends AbstractEMFOperation implements ICommand, ICommandWithSettableResult { /** * Convenience method to get a list of workspaces files associated with * <code>eObject</code>. * * @param eObject * the model object, may be <code>null</code> * @return the list of {@link IFile}s */ protected static List getWorkspaceFiles(EObject eObject) { List result = new ArrayList(); if (eObject != null) { Resource resource = eObject.eResource(); if (resource != null) { IFile file = WorkspaceSynchronizer.getFile(resource); if (file != null) { result.add(file); } } } return result; } /** * Convenience method to get a list of workspaces files associated with * {@link EObject}s in <code>eObject</code>. * * @param eObjects * the list of model object * @return the list of {@link IFile}s */ protected static List getWorkspaceFiles(List eObjects) { List result = new ArrayList(); for (Iterator i = eObjects.iterator(); i.hasNext();) { Object next = i.next(); if (next instanceof EObject) { Resource resource = ((EObject) next).eResource(); if (resource != null) { IFile file = WorkspaceSynchronizer.getFile(resource); if (file != null) { result.add(file); } } } } return result; } private final List affectedFiles; private CommandResult commandResult; /** * Initializes me with the editing domain in which I am making model * changes, a label, and a list of {@link IFile}s that I anticipate * modifying when I am executed, undone or redone. * * @param domain * my editing domain * @param label * my user-readable label, should never be <code>null</code>. * @param affectedFiles * the list of affected {@link IFile}s; may be <code>null</code> */ public AbstractTransactionalCommand(TransactionalEditingDomain domain, String label, List affectedFiles) { this(domain, label, null, affectedFiles); } /** * Initializes me with the editing domain, a label, transaction options, and * a list of {@link IFile}s that anticipate modifying when I am executed, * undone or redone. * * @param domain * my editing domain * @param label * my user-readable label, should never be <code>null</code>. * @param options * for the transaction in which I execute myself, or * <code>null</code> for the default options * @param affectedFiles * the list of affected {@link IFile}s; may be <code>null</code> */ public AbstractTransactionalCommand(TransactionalEditingDomain domain, String label, Map options, List affectedFiles) { super(domain, (label == null) ? StringStatics.BLANK : label, options); if (affectedFiles == null) { this.affectedFiles = new ArrayList(); } else { this.affectedFiles = affectedFiles; } } /** * Returns the {@link IFile}s for resources that may be modified when the * operation is executed, undone or redone. */ public List getAffectedFiles() { return affectedFiles; } // Documentation copied from the interface public final CommandResult getCommandResult() { return commandResult; } /** * Sets the command result. * * @param result * the new result for this command. */ protected final void setResult(CommandResult result) { this.commandResult = result; } // Documentation copied from the interface public ICommand compose(IUndoableOperation operation) { if (operation != null) { return new CompositeTransactionalCommand(getEditingDomain(), getLabel()) .compose(this).compose(operation); } return this; } // Documentation copied from the interface public ICommand reduce() { return this; } /** * Implemented by subclasses to perform the model changes. These changes * are applied by manipulation of the EMF metamodel's API, <em>not</em> * by executing commands on the editing domain's command stack. * * @param monitor the progress monitor provided by the operation history * @param info the adaptable provided by the operation history * * @return the result of the execution * * @throws ExecutionException if, for some reason, I fail to complete * the operation */ protected abstract CommandResult doExecuteWithResult( IProgressMonitor monitor, IAdaptable info) throws ExecutionException; protected void didUndo(Transaction tx) { // We will amalgamate any change description that were added by the DiagramEditingDomain's // special post-commit listener. See DiagramEditingDomainFactory for more details. if (tx.getChangeDescription() != null && !tx.getChangeDescription().isEmpty()) { ((CompositeChangeDescription)getChange()).add(tx.getChangeDescription()); } } protected void didRedo(Transaction tx) { // We will amalgamate any change description that were added by the DiagramEditingDomain's // special post-commit listener. See DiagramEditingDomainFactory for more details. if (tx.getChangeDescription() != null && !tx.getChangeDescription().isEmpty()) { ((CompositeChangeDescription)getChange()).add(tx.getChangeDescription()); } } /** * Delegates to {@link #doExecuteWithResult(IProgressMonitor, IAdaptable)} * to perform the model changes. Sets the command result and calls * {@link #cleanup()} to give subclasses a chance to dispose of any objects * that were required for the execution but will not be required for undo or * redo. */ protected IStatus doExecute(IProgressMonitor monitor, IAdaptable info) throws ExecutionException { CommandResult result = doExecuteWithResult(monitor, info); setResult(result); cleanup(); return result != null ? result.getStatus() : Status.OK_STATUS; } /** * Overrides superclass to set the command result. */ protected IStatus doUndo(IProgressMonitor monitor, IAdaptable info) throws ExecutionException { IStatus status = super.doUndo(monitor, info); CommandResult result = new CommandResult(status); setResult(result); return status; } /** * Overrides superclass to set the command result. */ protected IStatus doRedo(IProgressMonitor monitor, IAdaptable info) throws ExecutionException { IStatus status = super.doRedo(monitor, info); CommandResult result = new CommandResult(status); setResult(result); return status; } /** * Considers that the aggregate status may be different from the present * status, and updates the command result accordingly. */ protected IStatus aggregateStatuses(List statuses) { IStatus status = super.aggregateStatuses(statuses); CommandResult result = getCommandResult(); if (result == null) { result = new CommandResult(status); setResult(result); } else if (status != result.getStatus()) { result = new CommandResult(status, result.getReturnValue()); setResult(result); } return status; } /** * Subclasses may implement this method to dispose of objects that were * required for execution, but are no longer require to undo or redo this * operation. * <P> * This method is invoked at the end of * {@link #doExecute(IProgressMonitor, IAdaptable)}. */ protected void cleanup() { // subclasses can use this to cleanup } /** * Internal method to set the command result. * * @param result CommandResult to set * @deprecated internal API */ public void internalSetResult(CommandResult result) { this.commandResult = result; } }