| /******************************************************************************* |
| * Copyright (c) 2000, 2016 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 java.util.Arrays; |
| |
| import org.eclipse.core.runtime.CoreException; |
| |
| import org.eclipse.text.edits.DeleteEdit; |
| import org.eclipse.text.edits.InsertEdit; |
| import org.eclipse.text.edits.MultiTextEdit; |
| import org.eclipse.text.edits.ReplaceEdit; |
| import org.eclipse.text.edits.TextEdit; |
| |
| import org.eclipse.jface.text.BadLocationException; |
| import org.eclipse.jface.text.Region; |
| import org.eclipse.jface.text.TextUtilities; |
| |
| import org.eclipse.ltk.core.refactoring.Change; |
| import org.eclipse.ltk.core.refactoring.TextChange; |
| |
| import org.eclipse.jdt.core.IBuffer; |
| import org.eclipse.jdt.core.ICompilationUnit; |
| import org.eclipse.jdt.core.IPackageFragment; |
| import org.eclipse.jdt.core.IType; |
| import org.eclipse.jdt.core.compiler.InvalidInputException; |
| import org.eclipse.jdt.core.dom.rewrite.ImportRewrite; |
| import org.eclipse.jdt.core.refactoring.CompilationUnitChange; |
| |
| import org.eclipse.jdt.internal.corext.codemanipulation.StubUtility; |
| import org.eclipse.jdt.internal.corext.refactoring.changes.TextChangeCompatibility; |
| import org.eclipse.jdt.internal.corext.util.JavaModelUtil; |
| import org.eclipse.jdt.internal.corext.util.Messages; |
| |
| import org.eclipse.jdt.internal.core.manipulation.util.BasicElementLabels; |
| |
| public class NLSSourceModifier { |
| |
| private final String fSubstitutionPattern; |
| private final boolean fIsEclipseNLS; |
| |
| private NLSSourceModifier(String substitutionPattern, boolean isEclipseNLS) { |
| fSubstitutionPattern= substitutionPattern; |
| fIsEclipseNLS= isEclipseNLS; |
| } |
| |
| public static Change create(ICompilationUnit cu, NLSSubstitution[] subs, String substitutionPattern, IPackageFragment accessorPackage, String accessorClassName, boolean isEclipseNLS) throws CoreException { |
| |
| NLSSourceModifier sourceModification= new NLSSourceModifier(substitutionPattern, isEclipseNLS); |
| |
| String message= Messages.format(NLSMessages.NLSSourceModifier_change_description, BasicElementLabels.getFileName(cu)); |
| |
| TextChange change= new CompilationUnitChange(message, cu); |
| MultiTextEdit multiTextEdit= new MultiTextEdit(); |
| change.setEdit(multiTextEdit); |
| |
| boolean createImportForAccessor= true; |
| for (int i= 0; i < subs.length; i++) { |
| NLSSubstitution substitution= subs[i]; |
| int newState= substitution.getState(); |
| if (newState == NLSSubstitution.EXTERNALIZED && createImportForAccessor) { |
| accessorClassName= sourceModification.createImportForAccessor(multiTextEdit, accessorClassName, accessorPackage, cu); |
| createImportForAccessor= false; |
| } |
| if (substitution.hasStateChanged()) { |
| if (newState == NLSSubstitution.EXTERNALIZED) { |
| if (substitution.getInitialState() == NLSSubstitution.INTERNALIZED) { |
| sourceModification.addNLS(substitution, change, accessorClassName); |
| } else if (substitution.getInitialState() == NLSSubstitution.IGNORED) { |
| sourceModification.addAccessor(substitution, change, accessorClassName); |
| } |
| } else if (newState == NLSSubstitution.INTERNALIZED) { |
| if (substitution.getInitialState() == NLSSubstitution.IGNORED) { |
| sourceModification.deleteTag(substitution, change); |
| if (substitution.isValueRename()) { |
| sourceModification.replaceValue(substitution, change); |
| } |
| } else if (substitution.getInitialState() == NLSSubstitution.EXTERNALIZED) { |
| sourceModification.deleteAccessor(substitution, change, cu); |
| if (!isEclipseNLS) |
| sourceModification.deleteTag(substitution, change); |
| } |
| } else if (newState == NLSSubstitution.IGNORED) { |
| if (substitution.getInitialState() == NLSSubstitution.INTERNALIZED) { |
| sourceModification.addNLS(substitution, change, null); |
| if (substitution.isValueRename()) { |
| sourceModification.replaceValue(substitution, change); |
| } |
| } else { |
| if (substitution.getInitialState() == NLSSubstitution.EXTERNALIZED) { |
| sourceModification.deleteAccessor(substitution, change, cu); |
| } |
| } |
| } |
| } else { |
| if (newState == NLSSubstitution.EXTERNALIZED) { |
| if (substitution.isKeyRename()) { |
| sourceModification.replaceKey(substitution, change); |
| } |
| if (substitution.isAccessorRename()) { |
| sourceModification.replaceAccessor(substitution, change); |
| } |
| } else { |
| if (substitution.isValueRename()) { |
| sourceModification.replaceValue(substitution, change); |
| } |
| } |
| } |
| } |
| |
| return change; |
| } |
| |
| private void replaceAccessor(NLSSubstitution substitution, TextChange change) { |
| AccessorClassReference accessorClassRef= substitution.getAccessorClassReference(); |
| if (accessorClassRef != null) { |
| Region region= accessorClassRef.getRegion(); |
| int len= accessorClassRef.getName().length(); |
| String[] args= {BasicElementLabels.getJavaElementName(accessorClassRef.getName()), BasicElementLabels.getJavaElementName(substitution.getUpdatedAccessor())}; |
| TextChangeCompatibility.addTextEdit(change, Messages.format(NLSMessages.NLSSourceModifier_replace_accessor, args), |
| new ReplaceEdit(region.getOffset(), len, substitution.getUpdatedAccessor())); // |
| } |
| |
| } |
| |
| private void replaceKey(NLSSubstitution substitution, TextChange change) { |
| Region region= substitution.getNLSElement().getPosition(); |
| String[] args= {substitution.getInitialKey(), BasicElementLabels.getJavaElementName(substitution.getKey())}; |
| |
| ReplaceEdit replaceEdit; |
| if (fIsEclipseNLS) |
| replaceEdit= new ReplaceEdit(region.getOffset(), region.getLength(), substitution.getKey()); |
| else |
| replaceEdit= new ReplaceEdit(region.getOffset(), region.getLength(), '\"' + unwindEscapeChars(substitution.getKey()) + '\"'); // |
| |
| TextChangeCompatibility.addTextEdit(change, Messages.format(NLSMessages.NLSSourceModifier_replace_key, args), replaceEdit); |
| } |
| |
| private void replaceValue(NLSSubstitution substitution, TextChange change) { |
| Region region= substitution.getNLSElement().getPosition(); |
| String[] args= {substitution.getInitialValue(), substitution.getValueNonEmpty()}; |
| TextChangeCompatibility.addTextEdit(change, Messages.format(NLSMessages.NLSSourceModifier_replace_value, args), |
| new ReplaceEdit(region.getOffset(), region.getLength(), '\"' + unwindEscapeChars(substitution.getValueNonEmpty()) + '\"')); // |
| } |
| |
| private void deleteAccessor(NLSSubstitution substitution, TextChange change, ICompilationUnit cu) throws CoreException { |
| AccessorClassReference accessorClassRef= substitution.getAccessorClassReference(); |
| if (accessorClassRef != null) { |
| Region region= accessorClassRef.getRegion(); |
| String[] args= {substitution.getValueNonEmpty(), BasicElementLabels.getJavaElementName(substitution.getKey())}; |
| String label= Messages.format(NLSMessages.NLSSourceModifier_remove_accessor, args); |
| String replaceString= '\"' + unwindEscapeChars(substitution.getValueNonEmpty()) + '\"'; |
| TextChangeCompatibility.addTextEdit(change, label, new ReplaceEdit(region.getOffset(), region.getLength(), replaceString)); |
| if (fIsEclipseNLS && substitution.getState() != NLSSubstitution.INTERNALIZED) { |
| |
| Region position= substitution.getNLSElement().getPosition(); |
| int lineStart= getLineStart(cu.getBuffer(), position.getOffset()); |
| int lineEnd= getLineEnd(cu.getBuffer(), position.getOffset()); |
| String cuLine= cu.getBuffer().getText(lineStart, lineEnd - lineStart); |
| StringBuilder buf= new StringBuilder(cuLine); |
| buf.replace(region.getOffset() - lineStart, region.getOffset() + region.getLength() - lineStart, replaceString); |
| try { |
| NLSLine[] allLines= NLSScanner.scan(buf.toString()); |
| |
| NLSLine nlsLine= allLines[0]; |
| NLSElement element= findElement(nlsLine, position.getOffset() - lineStart - accessorClassRef.getName().length() - 1); |
| if (element == null || element.hasTag()) |
| return; |
| |
| NLSElement[] elements= nlsLine.getElements(); |
| int indexInElementList= Arrays.asList(elements).indexOf(element); |
| String editText= ' ' + NLSElement.createTagText(indexInElementList + 1); //tags are 1-based |
| TextChangeCompatibility.addTextEdit(change, label, new InsertEdit(lineEnd, editText)); |
| |
| } catch (InvalidInputException e) { |
| } catch (BadLocationException e) { |
| } |
| } |
| } |
| } |
| |
| private int getLineEnd(IBuffer buffer, int offset) { |
| int pos= offset; |
| int length= buffer.getLength(); |
| while (pos < length && !isDelemiter(buffer.getChar(pos))) { |
| pos++; |
| } |
| return pos; |
| } |
| |
| private int getLineStart(IBuffer buffer, int offset) { |
| int pos= offset; |
| while (pos >= 0 && !isDelemiter(buffer.getChar(pos))) { |
| pos--; |
| } |
| return pos + 1; |
| } |
| |
| private boolean isDelemiter(char ch) { |
| String[] delem= TextUtilities.DELIMITERS; |
| for (int i= 0; i < delem.length; i++) { |
| if (delem[i].length() == 1 && ch == delem[i].charAt(0)) |
| return true; |
| } |
| return false; |
| } |
| |
| private static boolean isPositionInElement(NLSElement element, int position) { |
| Region elementPosition= element.getPosition(); |
| return (elementPosition.getOffset() <= position && position <= elementPosition.getOffset() + elementPosition.getLength()); |
| } |
| |
| private static NLSElement findElement(NLSLine line, int position) { |
| NLSElement[] elements= line.getElements(); |
| for (int i= 0; i < elements.length; i++) { |
| NLSElement element= elements[i]; |
| if (isPositionInElement(element, position)) |
| return element; |
| } |
| return null; |
| } |
| |
| // TODO: not dry |
| private String unwindEscapeChars(String s) { |
| StringBuilder sb= new StringBuilder(s.length()); |
| int length= s.length(); |
| for (int i= 0; i < length; i++) { |
| char c= s.charAt(i); |
| sb.append(getUnwoundString(c)); |
| } |
| return sb.toString(); |
| } |
| |
| private String getUnwoundString(char c) { |
| switch (c) { |
| case '\b' : |
| return "\\b";//$NON-NLS-1$ |
| case '\t' : |
| return "\\t";//$NON-NLS-1$ |
| case '\n' : |
| return "\\n";//$NON-NLS-1$ |
| case '\f' : |
| return "\\f";//$NON-NLS-1$ |
| case '\r' : |
| return "\\r";//$NON-NLS-1$ |
| case '\\' : |
| return "\\\\";//$NON-NLS-1$ |
| } |
| return String.valueOf(c); |
| } |
| |
| private void deleteTag(NLSSubstitution substitution, TextChange change) { |
| Region textRegion= substitution.getNLSElement().getTagPosition(); |
| |
| TextChangeCompatibility.addTextEdit(change, NLSMessages.NLSSourceModifier_remove_tag, |
| new DeleteEdit(textRegion.getOffset(), textRegion.getLength())); |
| } |
| |
| private String createImportForAccessor(MultiTextEdit parent, String accessorClassName, IPackageFragment accessorPackage, ICompilationUnit cu) throws CoreException { |
| IType type= accessorPackage.getCompilationUnit(accessorClassName + JavaModelUtil.DEFAULT_CU_SUFFIX).getType(accessorClassName); |
| String fullyQualifiedName= type.getFullyQualifiedName(); |
| |
| ImportRewrite importRewrite= StubUtility.createImportRewrite(cu, true); |
| String nameToUse= importRewrite.addImport(fullyQualifiedName); |
| TextEdit edit= importRewrite.rewriteImports(null); |
| parent.addChild(edit); |
| |
| return nameToUse; |
| } |
| |
| private void addNLS(NLSSubstitution sub, TextChange change, String accessorName) { |
| if (sub.getState() == NLSSubstitution.INTERNALIZED) |
| return; |
| |
| NLSElement element= sub.getNLSElement(); |
| |
| addAccessor(sub, change, accessorName); |
| |
| if (!fIsEclipseNLS || sub.getState() == NLSSubstitution.IGNORED) { |
| // Add $NON-NLS-n tag |
| String arg= sub.getState() == NLSSubstitution.EXTERNALIZED ? sub.getKey() : sub.getValueNonEmpty(); |
| String name= Messages.format(NLSMessages.NLSSourceModifier_add_tag, arg); |
| TextChangeCompatibility.addTextEdit(change, name, createAddTagChange(element)); |
| } |
| } |
| |
| private void addAccessor(NLSSubstitution sub, TextChange change, String accessorName) { |
| if (sub.getState() == NLSSubstitution.EXTERNALIZED) { |
| NLSElement element= sub.getNLSElement(); |
| Region position= element.getPosition(); |
| String[] args= {sub.getValueNonEmpty(), BasicElementLabels.getJavaElementName(sub.getKey())}; |
| String text= Messages.format(NLSMessages.NLSSourceModifier_externalize, args); |
| |
| String resourceGetter= createResourceGetter(sub.getKey(), accessorName); |
| |
| TextEdit edit= new ReplaceEdit(position.getOffset(), position.getLength(), resourceGetter); |
| if (fIsEclipseNLS && element.getTagPosition() != null) { |
| MultiTextEdit multiEdit= new MultiTextEdit(); |
| multiEdit.addChild(edit); |
| Region tagPosition= element.getTagPosition(); |
| multiEdit.addChild(new DeleteEdit(tagPosition.getOffset(), tagPosition.getLength())); |
| edit= multiEdit; |
| } |
| TextChangeCompatibility.addTextEdit(change, text, edit); |
| } |
| } |
| |
| private TextEdit createAddTagChange(NLSElement element) { |
| int offset= element.getTagPosition().getOffset(); //to be changed |
| String text= ' ' + element.getTagText(); |
| return new InsertEdit(offset, text); |
| } |
| |
| private String createResourceGetter(String key, String accessorName) { |
| StringBuilder buf= new StringBuilder(); |
| buf.append(accessorName); |
| buf.append('.'); |
| |
| if (fIsEclipseNLS) |
| buf.append(key); |
| else { |
| //we just replace the first occurrence of KEY in the pattern |
| int i= fSubstitutionPattern.indexOf(NLSRefactoring.KEY); |
| if (i != -1) { |
| buf.append(fSubstitutionPattern.substring(0, i)); |
| buf.append('"').append(key).append('"'); |
| buf.append(fSubstitutionPattern.substring(i + NLSRefactoring.KEY.length())); |
| } |
| } |
| return buf.toString(); |
| } |
| } |