/******************************************************************************* * Copyright (c) 2000, 2017 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.dltk.internal.corext.refactoring.rename; import java.util.StringTokenizer; import org.eclipse.core.resources.IFile; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.OperationCanceledException; import org.eclipse.core.runtime.SubProgressMonitor; import org.eclipse.dltk.core.IDLTKLanguageToolkit; import org.eclipse.dltk.core.IField; import org.eclipse.dltk.core.ILocalVariable; import org.eclipse.dltk.core.IMember; import org.eclipse.dltk.core.IModelElement; import org.eclipse.dltk.core.IScriptProject; import org.eclipse.dltk.core.ISourceModule; import org.eclipse.dltk.core.ISourceRange; import org.eclipse.dltk.core.ModelException; import org.eclipse.dltk.core.manipulation.IScriptRefactorings; import org.eclipse.dltk.core.search.IDLTKSearchConstants; import org.eclipse.dltk.core.search.IDLTKSearchScope; import org.eclipse.dltk.core.search.SearchEngine; import org.eclipse.dltk.core.search.SearchMatch; import org.eclipse.dltk.core.search.SearchParticipant; import org.eclipse.dltk.core.search.SearchPattern; import org.eclipse.dltk.core.search.SearchRequestor; import org.eclipse.dltk.internal.core.manipulation.Messages; import org.eclipse.dltk.internal.core.manipulation.ScriptManipulationPlugin; import org.eclipse.dltk.internal.core.refactoring.descriptors.RenameModelElementDescriptor; import org.eclipse.dltk.internal.corext.refactoring.RefactoringCoreMessages; import org.eclipse.dltk.internal.corext.refactoring.ScriptRefactoringArguments; import org.eclipse.dltk.internal.corext.refactoring.ScriptRefactoringDescriptor; import org.eclipse.dltk.internal.corext.refactoring.changes.DynamicValidationRefactoringChange; import org.eclipse.dltk.internal.corext.refactoring.code.ScriptableRefactoring; import org.eclipse.dltk.internal.corext.refactoring.participants.ScriptProcessors; import org.eclipse.dltk.internal.corext.refactoring.tagging.IReferenceUpdating; import org.eclipse.dltk.internal.corext.refactoring.util.ResourceUtil; import org.eclipse.dltk.internal.corext.refactoring.util.TextChangeManager; import org.eclipse.dltk.internal.corext.util.SearchUtils; import org.eclipse.ltk.core.refactoring.Change; import org.eclipse.ltk.core.refactoring.RefactoringDescriptor; import org.eclipse.ltk.core.refactoring.RefactoringStatus; import org.eclipse.ltk.core.refactoring.TextChange; import org.eclipse.ltk.core.refactoring.participants.CheckConditionsContext; import org.eclipse.ltk.core.refactoring.participants.RefactoringArguments; import org.eclipse.ltk.core.refactoring.participants.RenameArguments; import org.eclipse.text.edits.MalformedTreeException; import org.eclipse.text.edits.MultiTextEdit; import org.eclipse.text.edits.ReplaceEdit; import org.eclipse.text.edits.TextEdit; import org.eclipse.text.edits.TextEditGroup; public abstract class RenameModelElementProcessor extends ScriptRenameProcessor implements IReferenceUpdating { protected IModelElement fModelElement; protected ISourceModule fCu; // the following fields are set or modified after the construction protected boolean fUpdateReferences; protected String fCurrentName; // private CompilationUnit fCompilationUnitNode; // private VariableDeclaration fTempDeclarationNode; // protected SourceModuleChange fChange; // private boolean fIsComposite is always false // private GroupCategorySet fCategorySet; private TextChangeManager fChangeManager; // private RenameAnalyzeUtil.LocalAnalyzePackage fLocalAnalyzePackage; private final IDLTKLanguageToolkit fToolkit; public RenameModelElementProcessor(IModelElement localVariable, IDLTKLanguageToolkit toolkit) { fToolkit = toolkit; fModelElement = localVariable; fCu = (ISourceModule) fModelElement.getAncestor(IModelElement.SOURCE_MODULE); fChangeManager = new TextChangeManager(true); } @Override public RefactoringStatus initialize(RefactoringArguments arguments) { if (!(arguments instanceof ScriptRefactoringArguments)) return RefactoringStatus .createFatalErrorStatus(RefactoringCoreMessages.InitializableRefactoring_inacceptable_arguments); final ScriptRefactoringArguments extended = (ScriptRefactoringArguments) arguments; final String handle = extended.getAttribute(ScriptRefactoringDescriptor.ATTRIBUTE_INPUT); if (handle != null) { final IModelElement element = ScriptRefactoringDescriptor.handleToElement(extended.getProject(), handle, false); if (element != null && element.exists()) { if (element.getElementType() == IModelElement.SOURCE_MODULE) { fCu = (ISourceModule) element; } else if (element.getElementType() == IModelElement.LOCAL_VARIABLE) { fModelElement = element; fCu = (ISourceModule) fModelElement.getAncestor(IModelElement.SOURCE_MODULE); if (fCu == null) return ScriptableRefactoring.createInputFatalStatus(element, getProcessorName(), IScriptRefactorings.RENAME_LOCAL_VARIABLE); } else return ScriptableRefactoring.createInputFatalStatus(element, getProcessorName(), IScriptRefactorings.RENAME_LOCAL_VARIABLE); } else return ScriptableRefactoring.createInputFatalStatus(element, getProcessorName(), IScriptRefactorings.RENAME_LOCAL_VARIABLE); } else return RefactoringStatus.createFatalErrorStatus( Messages.format(RefactoringCoreMessages.InitializableRefactoring_argument_not_exist, ScriptRefactoringDescriptor.ATTRIBUTE_INPUT)); final String name = extended.getAttribute(ScriptRefactoringDescriptor.ATTRIBUTE_NAME); if (name != null && !"".equals(name)) //$NON-NLS-1$ setNewElementName(name); else return RefactoringStatus.createFatalErrorStatus( Messages.format(RefactoringCoreMessages.InitializableRefactoring_argument_not_exist, ScriptRefactoringDescriptor.ATTRIBUTE_NAME)); if (fCu != null && fModelElement == null) { final String selection = extended.getAttribute(ScriptRefactoringDescriptor.ATTRIBUTE_SELECTION); if (selection != null) { int offset = -1; int length = -1; final StringTokenizer tokenizer = new StringTokenizer(selection); if (tokenizer.hasMoreTokens()) offset = Integer.valueOf(tokenizer.nextToken()).intValue(); if (tokenizer.hasMoreTokens()) length = Integer.valueOf(tokenizer.nextToken()).intValue(); if (offset >= 0 && length >= 0) { try { final IModelElement[] elements = fCu.codeSelect(offset, length); if (elements != null) { for (int index = 0; index < elements.length; index++) { final IModelElement element = elements[index]; if (element instanceof ILocalVariable) fModelElement = element; } } if (fModelElement == null) return ScriptableRefactoring.createInputFatalStatus(null, getProcessorName(), IScriptRefactorings.RENAME_LOCAL_VARIABLE); } catch (ModelException exception) { ScriptManipulationPlugin.log(exception); } } else return RefactoringStatus.createFatalErrorStatus( Messages.format(RefactoringCoreMessages.InitializableRefactoring_illegal_argument, new Object[] { selection, ScriptRefactoringDescriptor.ATTRIBUTE_SELECTION })); } else return RefactoringStatus.createFatalErrorStatus( Messages.format(RefactoringCoreMessages.InitializableRefactoring_argument_not_exist, ScriptRefactoringDescriptor.ATTRIBUTE_SELECTION)); } final String references = extended.getAttribute(ScriptRefactoringDescriptor.ATTRIBUTE_REFERENCES); if (references != null) { fUpdateReferences = Boolean.valueOf(references).booleanValue(); } else return RefactoringStatus.createFatalErrorStatus( Messages.format(RefactoringCoreMessages.InitializableRefactoring_argument_not_exist, ScriptRefactoringDescriptor.ATTRIBUTE_REFERENCES)); return new RefactoringStatus(); } @Override public String getCurrentElementName() { return fCurrentName; } @Override public boolean canEnableUpdateReferences() { return true; } @Override public void setUpdateReferences(boolean update) { fUpdateReferences = update; } @Override public boolean getUpdateReferences() { return fUpdateReferences; } @Override protected RenameModifications computeRenameModifications() throws CoreException { RenameModifications result = new RenameModifications(); if (fModelElement instanceof ILocalVariable) { result.rename((ILocalVariable) fModelElement, new RenameArguments(getNewElementName(), getUpdateReferences())); } else if (fModelElement instanceof IField) { // TODO: add switching method in RenameModifications result.rename((IField) fModelElement, new RenameArguments(getNewElementName(), getUpdateReferences())); } return result; } @Override protected IFile[] getChangedFiles() throws CoreException { return ResourceUtil.getFiles(fChangeManager.getAllSourceModules()); } @Override protected RefactoringStatus doCheckFinalConditions(IProgressMonitor pm, CheckConditionsContext context) throws CoreException, OperationCanceledException { try { pm.beginTask("", 1); //$NON-NLS-1$ RefactoringStatus result = checkNewElementName(getNewElementName()); if (result.hasFatalError()) return result; createEdits(pm); // LocalAnalyzePackage[] localAnalyzePackages= new // RenameAnalyzeUtil.LocalAnalyzePackage[] { fLocalAnalyzePackage }; // result.merge(RenameAnalyzeUtil.analyzeLocalRenames(localAnalyzePackages, // fChange, fCompilationUnitNode, true)); return result; } finally { pm.done(); } } private void createEdits(IProgressMonitor pm) throws CoreException { fChangeManager.clear(); IDLTKSearchScope scope = SearchEngine.createWorkspaceScope(fToolkit); SearchEngine engine = new SearchEngine(); if (fUpdateReferences) { SearchPattern pattern = SearchPattern.createPattern(fModelElement, IDLTKSearchConstants.REFERENCES, SearchUtils.GENERICS_AGNOSTIC_MATCH_RULE, fToolkit); IProgressMonitor monitor = new SubProgressMonitor(pm, 1000); engine.search(pattern, new SearchParticipant[] { SearchEngine.getDefaultSearchParticipant() }, scope, new SearchRequestor() { @Override public void acceptSearchMatch(SearchMatch match) throws CoreException { if (!(match.getElement() instanceof IModelElement)) return; IModelElement elem = (IModelElement) match.getElement(); ISourceModule cu = (ISourceModule) elem.getAncestor(IModelElement.SOURCE_MODULE); if (cu != null) { ReplaceEdit edit = createReplaceEdit(match); addTextEdit(fChangeManager.get(cu), getProcessorName(), edit); } // if (match.getResource().equals(fCu.getCorrespondingResource())) // edits.add(new ReplaceEdit(match.getOffset(), fCurrentName.length(), // getNewElementName())); } }, monitor); } ISourceRange decl = null; if (fModelElement instanceof ILocalVariable) { decl = ((ILocalVariable) fModelElement).getNameRange(); } else if (fModelElement instanceof IMember) { decl = ((IMember) fModelElement).getNameRange(); } if (decl != null) { ReplaceEdit edit = new ReplaceEdit(decl.getOffset(), fCurrentName.length(), getNewElementName()); addTextEdit(fChangeManager.get(fCu), getProcessorName(), edit); } // fChange= new // SourceModuleChange(RefactoringCoreMessages.RenameTempRefactoring_rename, // fCu); // MultiTextEdit rootEdit= new MultiTextEdit(); // fChange.setEdit(rootEdit); // fChange.setKeepPreviewEdits(true); // for (TextEdit edit : edits) { // rootEdit.addChild(edit); // fChange.addTextEditGroup(new // TextEditGroup(RefactoringCoreMessages.RenameTempRefactoring_changeName, // edit)); // } } /** * Creates {@link ReplaceEdit} for the specified {@link SearchMatch} * * @param match * @return * @since 5.0.0 */ protected ReplaceEdit createReplaceEdit(SearchMatch match) { return new ReplaceEdit(match.getOffset(), fCurrentName.length(), getNewElementName()); } private static void addTextEdit(TextChange change, String name, TextEdit edit) throws MalformedTreeException { TextEdit root = change.getEdit(); if (root == null) { root = new MultiTextEdit(); change.setEdit(root); } root.addChild(edit); change.addTextEditGroup(new TextEditGroup(name, edit)); } @Override protected String[] getAffectedProjectNatures() throws CoreException { return ScriptProcessors.computeAffectedNatures(fCu); } @Override public Object[] getElements() { return new Object[] { fModelElement }; } @Override public String getNewElement() { return getNewElementName(); } @Override public RefactoringStatus checkInitialConditions(IProgressMonitor pm) throws CoreException, OperationCanceledException { // fCompilationUnitNode = RefactoringASTParser.parseWithASTProvider(fCu, true, // null); // ISourceRange sourceRange= fLocalVariable. // fLocalVariable.get // ASTNode name= NodeFinder.perform(fCompilationUnitNode, sourceRange); // if (name == null) // return; // if (name.getParent() instanceof VariableDeclaration) // fTempDeclarationNode= (VariableDeclaration) name.getParent(); // if (fTempDeclarationNode == null || fTempDeclarationNode.resolveBinding() == // null) // return // RefactoringStatus.createFatalErrorStatus(RefactoringCoreMessages.RenameTempRefactoring_must_select_local); // if (! Checks.isDeclaredIn(fTempDeclarationNode, MethodDeclaration.class) // && ! Checks.isDeclaredIn(fTempDeclarationNode, Initializer.class)) // return // RefactoringStatus.createFatalErrorStatus(RefactoringCoreMessages.RenameTempRefactoring_only_in_methods_and_initializers); // fCurrentName= fTempDeclarationNode.getName().getIdentifier(); fCurrentName = fModelElement.getElementName(); return new RefactoringStatus(); } /* * private AccumulatingProblemReporter getAccumulatingProblemReporter() { final * PerWorkingCopyInfo perWorkingCopyInfo = getPerWorkingCopyInfo(); if * (perWorkingCopyInfo != null && perWorkingCopyInfo.isActive()) { final * IScriptProject project = getScriptProject(); * * // https://bugs.eclipse.org/bugs/show_bug.cgi?id=267008 // Script nature * check is not enough. It's possible that the // external project created // * has a name of ExternalScriptProject.EXTERNAL_PROJECT_NAME, but no // script * nature. // If script nature added during // * WorkingCopyOwner.newWorkingCopy(...), this fix is not relevant. // Does * script nature should be added in // WorkingCopyOwner.newWorkingCopy, or just * script name checked? if (project != null && * (ExternalScriptProject.EXTERNAL_PROJECT_NAME * .equals(project.getProject().getName()) || ScriptProject * .hasScriptNature(project.getProject()))) { return new * AccumulatingProblemReporter(perWorkingCopyInfo); } } return null; } */ @Override public Change createChange(IProgressMonitor monitor) throws CoreException { try { monitor.beginTask(RefactoringCoreMessages.RenameFieldRefactoring_checking, 1); TextChange[] changes = fChangeManager.getAllChanges(); RenameModelElementDescriptor descriptor = createRefactoringDescriptor(); return new DynamicValidationRefactoringChange(descriptor, getProcessorName(), changes); } finally { monitor.done(); } } /* * @Override public Change createChange(IProgressMonitor pm) throws * CoreException, OperationCanceledException { try { * pm.beginTask(RefactoringCoreMessages.RenameTypeProcessor_creating_changes, * 1); RenameModelElementDescriptor descriptor= createRefactoringDescriptor(); * fChange.setDescriptor(new RefactoringChangeDescriptor(descriptor)); return * fChange; } finally { pm.done(); } } */ private RenameModelElementDescriptor createRefactoringDescriptor() { String project = null; IScriptProject scriptProject = fCu.getScriptProject(); if (scriptProject != null) project = scriptProject.getElementName(); // final String header= // Messages.format(RefactoringCoreMessages.RenameLocalVariableProcessor_descriptor_description, // new String[] { BasicElementLabels.getJavaElementName(fCurrentName), // JavaElementLabels.getElementLabel(fLocalVariable.getParent(), // JavaElementLabels.ALL_FULLY_QUALIFIED), // BasicElementLabels.getJavaElementName(fNewName)}); // final String description= // Messages.format(RefactoringCoreMessages.RenameLocalVariableProcessor_descriptor_description_short, // BasicElementLabels.getJavaElementName(fCurrentName)); // final String comment= new JDTRefactoringDescriptorComment(project, this, // header).asString(); final RenameModelElementDescriptor descriptor = new RenameModelElementDescriptor( IScriptRefactorings.RENAME_LOCAL_VARIABLE); descriptor.setProject(project); // descriptor.setDescription(description); // descriptor.setComment(comment); descriptor.setFlags(RefactoringDescriptor.NONE); descriptor.setModelElement(fModelElement); descriptor.setNewName(getNewElementName()); descriptor.setUpdateReferences(fUpdateReferences); return descriptor; } @Override public boolean needsSavedEditors() { return false; } }