blob: 3ad3a65b42dbbf1e4b85a1a2285ab0bf25e3cd8c [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2006, 2007 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.ide.undo;
import java.util.ArrayList;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.resources.mapping.IResourceChangeDescriptionFactory;
import org.eclipse.core.runtime.CoreException;
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.core.runtime.jobs.ISchedulingRule;
/**
* A DeleteResourcesOperation represents an undoable operation for deleting one
* or more resources in the workspace. Clients may call the public API from a
* background thread.
*
* This class is intended to be instantiated and used by clients. It is not
* intended to be subclassed by clients.
*
* @since 3.3
*
*/
public class DeleteResourcesOperation extends AbstractResourcesOperation {
// Whether to delete project content
private boolean deleteContent = false;
/**
* Create a DeleteResourcesOperation
*
* @param resources
* the resources to be deleted
* @param label
* the label of the operation
* @param deleteContent
* whether or not we are deleting content for projects
*/
public DeleteResourcesOperation(IResource[] resources, String label,
boolean deleteContent) {
super(resources, label);
this.deleteContent = deleteContent;
}
/*
* (non-Javadoc)
*
* Map execution to resource deletion.
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#doExecute(org.eclipse.core.runtime.IProgressMonitor,
* org.eclipse.core.runtime.IAdaptable)
*/
protected void doExecute(IProgressMonitor monitor, IAdaptable uiInfo)
throws CoreException {
delete(monitor, uiInfo, deleteContent);
}
/*
* (non-Javadoc)
*
* Map undo to resource recreation.
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#doUndo(org.eclipse.core.runtime.IProgressMonitor,
* org.eclipse.core.runtime.IAdaptable)
*/
protected void doUndo(IProgressMonitor monitor, IAdaptable uiInfo)
throws CoreException {
recreate(monitor, uiInfo);
}
/*
* (non-Javadoc)
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#updateResourceChangeDescriptionFactory(org.eclipse.core.resources.mapping.IResourceChangeDescriptionFactory,
* int)
*/
protected boolean updateResourceChangeDescriptionFactory(
IResourceChangeDescriptionFactory factory, int operation) {
boolean modified = false;
if (operation == UNDO) {
for (int i = 0; i < resourceDescriptions.length; i++) {
IResource resource = resourceDescriptions[i]
.createResourceHandle();
factory.create(resource);
modified = true;
}
} else {
for (int i = 0; i < resources.length; i++) {
IResource resource = resources[i];
factory.delete(resource);
modified = true;
}
}
return modified;
}
/*
* (non-Javadoc)
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#getExecuteSchedulingRule()
*/
protected ISchedulingRule getExecuteSchedulingRule() {
return super.computeDeleteSchedulingRule();
}
/*
* (non-Javadoc)
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#getUndoSchedulingRule()
*/
protected ISchedulingRule getUndoSchedulingRule() {
return super.computeCreateSchedulingRule();
}
/*
* (non-Javadoc)
*
* Map execution status to deletion status. Provide an extra warning if
* project content is to be deleted.
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#computeExecutionStatus(org.eclipse.core.runtime.IProgressMonitor)
*/
public IStatus computeExecutionStatus(IProgressMonitor monitor) {
IStatus status = super.computeExecutionStatus(monitor);
if (status.isOK()) {
status = computeDeleteStatus();
}
return status;
}
/*
* (non-Javadoc)
*
* Map undo status to resource creation status.
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#computeUndoableStatus(org.eclipse.core.runtime.IProgressMonitor)
*/
public IStatus computeUndoableStatus(IProgressMonitor monitor) {
IStatus status = super.computeUndoableStatus(monitor);
if (status.isOK()) {
// Recreating should not allow overwriting anything that is there,
// because we have no way to restore it.
// See https://bugs.eclipse.org/bugs/show_bug.cgi?id=162655
status = computeCreateStatus(false);
}
return status;
}
/*
* (non-Javadoc)
*
* Map redo status to resource deletion status.
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#computeRedoableStatus(org.eclipse.core.runtime.IProgressMonitor)
*/
public IStatus computeRedoableStatus(IProgressMonitor monitor) {
IStatus status = super.computeRedoableStatus(monitor);
if (status.isOK()) {
status = computeDeleteStatus();
}
return status;
}
/*
* (non-Javadoc)
*
* @see org.eclipse.ui.ide.undo.AbstractWorkspaceOperation#appendDescriptiveText(java.lang.StringBuffer)
*/
protected void appendDescriptiveText(StringBuffer text) {
super.appendDescriptiveText(text);
text.append(" deleteContent: "); //$NON-NLS-1$
text.append(deleteContent);
text.append('\'');
}
/*
* Overridden so that projects whose contents are not to be deleted
* will not be checked. A better solution would be to add API to
* ReadOnlyStateChecker to specify whether project children should
* be checked, but it is too late to do that now.
* See https://bugs.eclipse.org/bugs/show_bug.cgi?id=180758
*/
IStatus checkReadOnlyResources(IResource[] resourcesToCheck) {
// If we aren't deleting content of projects, don't bother
// checking the read only status of projects or their children.
// Clients currently do not mix and match projects and non-projects
// in a DeleteResourcesOperation. However, this is not specified
// in the API, so assume that there could be mixes.
if (!deleteContent) {
ArrayList nonProjectResourcesToCheck = new ArrayList();
for (int i=0; i<resourcesToCheck.length; i++) {
if (resourcesToCheck[i].getType() != IResource.PROJECT) {
nonProjectResourcesToCheck.add(resourcesToCheck[i]);
}
}
if (nonProjectResourcesToCheck.isEmpty()) {
return Status.OK_STATUS;
}
return super.checkReadOnlyResources((IResource[])nonProjectResourcesToCheck
.toArray(new IResource [nonProjectResourcesToCheck.size()]));
}
// We are deleting project content, so do it the normal way
return super.checkReadOnlyResources(resourcesToCheck);
}
}