blob: c5994570bf67b6a85c5e4278c459fa7b30dc5872 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2006 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.jdt.internal.corext.refactoring.nls;
import com.ibm.icu.text.Collator;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.eclipse.text.edits.MultiTextEdit;
import org.eclipse.text.edits.TextEdit;
import org.eclipse.text.edits.TextEditGroup;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.core.filebuffers.FileBuffers;
import org.eclipse.core.filebuffers.ITextFileBuffer;
import org.eclipse.core.filebuffers.ITextFileBufferManager;
import org.eclipse.jface.text.Document;
import org.eclipse.jface.text.IDocument;
import org.eclipse.ltk.core.refactoring.Change;
import org.eclipse.ltk.core.refactoring.TextChange;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.dom.AST;
import org.eclipse.jdt.core.dom.ASTNode;
import org.eclipse.jdt.core.dom.ASTVisitor;
import org.eclipse.jdt.core.dom.AbstractTypeDeclaration;
import org.eclipse.jdt.core.dom.CompilationUnit;
import org.eclipse.jdt.core.dom.FieldDeclaration;
import org.eclipse.jdt.core.dom.Modifier;
import org.eclipse.jdt.core.dom.TypeDeclaration;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
import org.eclipse.jdt.core.dom.rewrite.ListRewrite;
import org.eclipse.jdt.internal.corext.dom.GenericVisitor;
import org.eclipse.jdt.internal.corext.refactoring.changes.CompilationUnitChange;
import org.eclipse.jdt.internal.corext.util.Messages;
import org.eclipse.jdt.ui.JavaUI;
import org.eclipse.jdt.internal.ui.IJavaStatusConstants;
import org.eclipse.jdt.internal.ui.JavaPlugin;
import org.eclipse.jdt.internal.ui.javaeditor.ASTProvider;
public class AccessorClassModifier {
private CompilationUnit fRoot;
private AST fAst;
private ASTRewrite fASTRewrite;
private ListRewrite fListRewrite;
private ICompilationUnit fCU;
private List fFields;
private AccessorClassModifier(ICompilationUnit cu) throws CoreException {
fCU= cu;
fRoot= JavaPlugin.getDefault().getASTProvider().getAST(cu, ASTProvider.WAIT_YES, null);
fAst= fRoot.getAST();
fASTRewrite= ASTRewrite.create(fAst);
AbstractTypeDeclaration parent= null;
if (fRoot.types().size() > 0) {
parent= (AbstractTypeDeclaration)fRoot.types().get(0);
fFields= new ArrayList();
parent.accept(new GenericVisitor() {
/**
* {@inheritDoc}
*/
public boolean visit(FieldDeclaration node) {
int modifiers= node.getModifiers();
if (!Modifier.isPublic(modifiers))
return false;
if (!Modifier.isStatic(modifiers))
return false;
List fragments= node.fragments();
if (fragments.size() != 1)
return false;
VariableDeclarationFragment fragment= (VariableDeclarationFragment)fragments.get(0);
if (fragment.getInitializer() != null)
return false;
fFields.add(node);
return false;
}
});
fListRewrite= fASTRewrite.getListRewrite(parent, TypeDeclaration.BODY_DECLARATIONS_PROPERTY);
} else {
IStatus status= new Status(IStatus.ERROR, JavaUI.ID_PLUGIN, IJavaStatusConstants.INTERNAL_ERROR, NLSMessages.AccessorClassModifier_missingType, null);
throw new CoreException(status);
}
}
private TextEdit getTextEdit() throws CoreException {
IDocument document= null;
ITextFileBufferManager manager= FileBuffers.getTextFileBufferManager();
IPath path= fCU.getPath();
if (manager != null && path != null) {
manager.connect(path, null);
try {
ITextFileBuffer buffer= manager.getTextFileBuffer(path);
if (buffer != null)
document= buffer.getDocument();
} finally {
manager.disconnect(path, null);
}
}
if (document == null)
document= new Document(fCU.getSource());
return fASTRewrite.rewriteAST(document, fCU.getJavaProject().getOptions(true));
}
public static Change create(ICompilationUnit cu, NLSSubstitution[] substitutions) throws CoreException {
Map newKeyToSubstMap= NLSPropertyFileModifier.getNewKeyToSubstitutionMap(substitutions);
Map oldKeyToSubstMap= NLSPropertyFileModifier.getOldKeyToSubstitutionMap(substitutions);
AccessorClassModifier sourceModification= new AccessorClassModifier(cu);
String message= Messages.format(NLSMessages.NLSSourceModifier_change_description, cu.getElementName());
TextChange change= new CompilationUnitChange(message, cu);
MultiTextEdit multiTextEdit= new MultiTextEdit();
change.setEdit(multiTextEdit);
for (int i= 0; i < substitutions.length; i++) {
NLSSubstitution substitution= substitutions[i];
if (NLSPropertyFileModifier.doRemove(substitution, newKeyToSubstMap, oldKeyToSubstMap)) {
sourceModification.removeKey(substitution, change);
}
}
for (int i= 0; i < substitutions.length; i++) {
NLSSubstitution substitution= substitutions[i];
if (substitution.isKeyRename() && NLSPropertyFileModifier.doReplace(substitution, newKeyToSubstMap, oldKeyToSubstMap)) {
sourceModification.renameKey(substitution, change);
}
}
for (int i= 0; i < substitutions.length; i++) {
NLSSubstitution substitution= substitutions[i];
if (NLSPropertyFileModifier.doInsert(substitution, newKeyToSubstMap, oldKeyToSubstMap)) {
sourceModification.addKey(substitution, change);
}
}
if (change.getChangeGroups().length == 0)
return null;
change.addEdit(sourceModification.getTextEdit());
return change;
}
private void removeKey(NLSSubstitution sub, TextChange change) throws CoreException {
ASTNode node= findField(fRoot, sub.getKey());
if (node == null)
return;
String name= Messages.format(NLSMessages.AccessorClassModifier_remove_entry, sub.getKey());
TextEditGroup editGroup= new TextEditGroup(name);
fListRewrite.remove(node, editGroup);
change.addTextEditGroup(editGroup);
fFields.remove(node);
}
private void renameKey(NLSSubstitution sub, TextChange change) throws CoreException {
ASTNode node= findField(fRoot, sub.getInitialKey());
if (node == null)
return;
String name= Messages.format(NLSMessages.AccessorClassModifier_replace_entry, sub.getKey());
TextEditGroup editGroup= new TextEditGroup(name);
fListRewrite.remove(node, editGroup);
fFields.remove(node);
addKey(sub, change, editGroup);
change.addTextEditGroup(editGroup);
}
private ASTNode findField(ASTNode astRoot, final String name) {
class STOP_VISITING extends RuntimeException {
private static final long serialVersionUID= 1L;
}
final ASTNode[] result= new ASTNode[1];
try {
astRoot.accept(new ASTVisitor() {
public boolean visit(VariableDeclarationFragment node) {
if (name.equals(node.getName().getFullyQualifiedName())) {
result[0]= node.getParent();
throw new STOP_VISITING();
}
return true;
}
});
} catch (STOP_VISITING ex) {
// stop visiting AST
}
return result[0];
}
private void addKey(NLSSubstitution sub, TextChange change) throws CoreException {
String name= Messages.format(NLSMessages.AccessorClassModifier_add_entry, sub.getKey());
TextEditGroup editGroup= new TextEditGroup(name);
change.addTextEditGroup(editGroup);
addKey(sub, change, editGroup);
}
private void addKey(NLSSubstitution sub, TextChange change, TextEditGroup editGroup) throws CoreException {
if (fListRewrite == null)
return;
String key= sub.getKey();
FieldDeclaration fieldDeclaration= getNewFinalStringFieldDeclaration(key);
Iterator iter= fFields.iterator();
int insertionPosition= 0;
if (iter.hasNext()) {
Collator collator= Collator.getInstance();
FieldDeclaration existingFieldDecl= (FieldDeclaration)iter.next();
VariableDeclarationFragment fragment= (VariableDeclarationFragment)existingFieldDecl.fragments().get(0);
String identifier= fragment.getName().getIdentifier();
if (collator.compare(key, identifier) != 1) {
insertionPosition= 0;
fListRewrite.insertBefore(fieldDeclaration, existingFieldDecl, editGroup);
} else {
insertionPosition++;
while (iter.hasNext()) {
FieldDeclaration next= (FieldDeclaration)iter.next();
fragment= (VariableDeclarationFragment)next.fragments().get(0);
identifier= fragment.getName().getIdentifier();
if (collator.compare(key, identifier) == -1) {
break;
}
insertionPosition++;
existingFieldDecl= next;
}
fListRewrite.insertAfter(fieldDeclaration, existingFieldDecl, editGroup);
}
} else {
insertionPosition= 0;
fListRewrite.insertLast(fieldDeclaration, editGroup);
}
fFields.add(insertionPosition, fieldDeclaration);
}
private FieldDeclaration getNewFinalStringFieldDeclaration(String name) {
VariableDeclarationFragment variableDeclarationFragment= fAst.newVariableDeclarationFragment();
variableDeclarationFragment.setName(fAst.newSimpleName(name));
FieldDeclaration fieldDeclaration= fAst.newFieldDeclaration(variableDeclarationFragment);
fieldDeclaration.setType(fAst.newSimpleType(fAst.newSimpleName("String"))); //$NON-NLS-1$
fieldDeclaration.modifiers().add(fAst.newModifier(Modifier.ModifierKeyword.PUBLIC_KEYWORD));
fieldDeclaration.modifiers().add(fAst.newModifier(Modifier.ModifierKeyword.STATIC_KEYWORD));
return fieldDeclaration;
}
}