blob: d38ce29323c4c557d85f0e6a850a47ddc3954029 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2005 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.ltk.core.refactoring;
import org.eclipse.text.edits.UndoEdit;
import org.eclipse.core.runtime.Assert;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.filebuffers.FileBuffers;
import org.eclipse.core.filebuffers.ITextFileBuffer;
import org.eclipse.core.filebuffers.ITextFileBufferManager;
import org.eclipse.core.resources.IFile;
import org.eclipse.jface.text.IDocument;
import org.eclipse.ltk.internal.core.refactoring.BufferValidationState;
import org.eclipse.ltk.internal.core.refactoring.Changes;
import org.eclipse.ltk.internal.core.refactoring.ContentStamps;
/**
* A special {@link TextChange} that operates on a <code>IFile</code>.
* <p>
* As of 3.1 the content stamp managed by a text file change maps to the modification
* stamp of its underlying <code>IFile</code>. Undoing a text file change will
* roll back the modification stamp of a resource to its original value using
* the new API {@link org.eclipse.core.resources.IResource#revertModificationStamp(long)}
* </p>
* <p>
* The class should be subclassed by clients which need to perform
* special operation when acquiring or releasing a document.
* </p>
* @since 3.0
*/
public class TextFileChange extends TextChange {
/**
* Flag (value 1) indicating that the file's save state has to be kept. This means an
* unsaved file is still unsaved after performing the change and a saved one
* will be saved.
*/
public static final int KEEP_SAVE_STATE= 1 << 0;
/**
* Flag (value 2) indicating that the file is to be saved after the change has been applied.
*/
public static final int FORCE_SAVE= 1 << 1;
/**
* Flag (value 4) indicating that the file will not be saved after the change has been applied.
*/
public static final int LEAVE_DIRTY= 1 << 2;
// the file to change
private IFile fFile;
private int fSaveMode= KEEP_SAVE_STATE;
// the mapped text buffer
private int fAcquireCount;
private ITextFileBuffer fBuffer;
private boolean fDirty;
private BufferValidationState fValidationState;
private ContentStamp fContentStamp;
/**
* Creates a new <code>TextFileChange</code> for the given file.
*
* @param name the change's name mainly used to render the change in the UI
* @param file the file this text change operates on
*/
public TextFileChange(String name, IFile file) {
super(name);
Assert.isNotNull(file);
fFile= file;
}
/**
* Sets the save state. Must be one of <code>KEEP_SAVE_STATE</code>,
* <code>FORCE_SAVE</code> or <code>LEAVE_DIRTY</code>.
*
* @param saveMode indicating how save is handled when the document
* gets committed
*/
public void setSaveMode(int saveMode) {
fSaveMode= saveMode;
}
/**
* Returns the save state set via {@link #setSaveMode(int)}.
*
* @return the save state
*/
public int getSaveMode() {
return fSaveMode;
}
/**
* Returns the <code>IFile</code> this change is working on.
*
* @return the file this change is working on
*/
public IFile getFile() {
return fFile;
}
/**
* Hook to create an undo change for the given undo edit and content stamp.
* This hook gets called while performing the change to construct the
* corresponding undo change object.
*
* @param edit the {@link UndoEdit} to create an undo change for
* @param stampToRestore the content stamp to restore when the undo
* edit is executed.
*
* @return the undo change or <code>null</code> if no undo change can
* be created. Returning <code>null</code> results in the fact that
* the whole change tree can't be undone. So returning <code>null</code>
* is only recommended if an exception occurred during creating the
* undo change.
*/
protected Change createUndoChange(UndoEdit edit, ContentStamp stampToRestore) {
return new UndoTextFileChange(getName(), fFile, edit, stampToRestore, fSaveMode);
}
/**
* {@inheritDoc}
*/
public Object getModifiedElement(){
return fFile;
}
public Object[] getAffectedObjects() {
Object modifiedElement= getModifiedElement();
if (modifiedElement == null)
return null;
return new Object[] { modifiedElement };
}
/**
* {@inheritDoc}
*/
public void initializeValidationData(IProgressMonitor pm) {
pm.beginTask("", 1); //$NON-NLS-1$
fValidationState= BufferValidationState.create(fFile);
pm.worked(1);
}
/**
* {@inheritDoc}
*/
public RefactoringStatus isValid(IProgressMonitor pm) throws CoreException {
pm.beginTask("", 1); //$NON-NLS-1$
ITextFileBuffer buffer= FileBuffers.getTextFileBufferManager().getTextFileBuffer(fFile.getFullPath());
fDirty= buffer != null && buffer.isDirty();
RefactoringStatus result= fValidationState.isValid(needsSaving());
if (needsSaving()) {
result.merge(Changes.validateModifiesFiles(new IFile[] {fFile}));
} else {
// we are reading the file. So it should be at least in sync
result.merge(Changes.checkInSync(new IFile[] {fFile}));
}
pm.worked(1);
return result;
}
/**
* {@inheritDoc}
*/
public void dispose() {
fValidationState.dispose();
}
/**
* {@inheritDoc}
*/
protected IDocument acquireDocument(IProgressMonitor pm) throws CoreException {
if (fAcquireCount > 0)
return fBuffer.getDocument();
ITextFileBufferManager manager= FileBuffers.getTextFileBufferManager();
IPath path= fFile.getFullPath();
manager.connect(path, pm);
fAcquireCount++;
fBuffer= manager.getTextFileBuffer(path);
IDocument result= fBuffer.getDocument();
fContentStamp= ContentStamps.get(fFile, result);
return result;
}
/**
* {@inheritDoc}
*/
protected void commit(IDocument document, IProgressMonitor pm) throws CoreException {
if (needsSaving()) {
fBuffer.commit(pm, false);
}
}
/**
* {@inheritDoc}
*/
protected void releaseDocument(IDocument document, IProgressMonitor pm) throws CoreException {
Assert.isTrue(fAcquireCount > 0);
if (fAcquireCount == 1) {
ITextFileBufferManager manager= FileBuffers.getTextFileBufferManager();
manager.disconnect(fFile.getFullPath(), pm);
}
fAcquireCount--;
}
/**
* {@inheritDoc}
*/
protected final Change createUndoChange(UndoEdit edit) {
return createUndoChange(edit, fContentStamp);
}
/**
* Is the document to be changed by this text file change currently
* acquired?
*
* @return <code>true</code> if the document is currently acquired,
* <code>false</code> otherwise
* @since 3.2
*/
protected final boolean isDocumentAcquired() {
return fAcquireCount > 0;
}
private boolean needsSaving() {
return (fSaveMode & FORCE_SAVE) != 0 || (!fDirty && (fSaveMode & KEEP_SAVE_STATE) != 0);
}
}