/******************************************************************************* * Copyright (c) 2000, 2009 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.wst.jsdt.core.dom.rewrite; import java.util.ArrayList; import java.util.List; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.NullProgressMonitor; import org.eclipse.core.runtime.SubProgressMonitor; import org.eclipse.text.edits.MultiTextEdit; import org.eclipse.text.edits.TextEdit; import org.eclipse.wst.jsdt.core.Flags; import org.eclipse.wst.jsdt.core.IImportDeclaration; import org.eclipse.wst.jsdt.core.IJavaScriptUnit; import org.eclipse.wst.jsdt.core.ITypeRoot; import org.eclipse.wst.jsdt.core.JavaScriptCore; import org.eclipse.wst.jsdt.core.JavaScriptModelException; import org.eclipse.wst.jsdt.core.Signature; import org.eclipse.wst.jsdt.core.compiler.CharOperation; import org.eclipse.wst.jsdt.core.dom.AST; import org.eclipse.wst.jsdt.core.dom.ASTParser; import org.eclipse.wst.jsdt.core.dom.IBinding; import org.eclipse.wst.jsdt.core.dom.IFunctionBinding; import org.eclipse.wst.jsdt.core.dom.ITypeBinding; import org.eclipse.wst.jsdt.core.dom.IVariableBinding; import org.eclipse.wst.jsdt.core.dom.ImportDeclaration; import org.eclipse.wst.jsdt.core.dom.JavaScriptUnit; import org.eclipse.wst.jsdt.core.dom.Modifier; import org.eclipse.wst.jsdt.core.dom.PrimitiveType; import org.eclipse.wst.jsdt.core.dom.Type; import org.eclipse.wst.jsdt.core.infer.IInferenceFile; import org.eclipse.wst.jsdt.core.infer.ImportRewriteSupport; import org.eclipse.wst.jsdt.core.infer.InferrenceManager; import org.eclipse.wst.jsdt.core.infer.InferrenceProvider; import org.eclipse.wst.jsdt.core.infer.RefactoringSupport; import org.eclipse.wst.jsdt.internal.core.dom.rewrite.ImportRewriteAnalyzer; import org.eclipse.wst.jsdt.internal.core.util.Messages; /** * The {@link ImportRewrite} helps updating imports following a import order and on-demand imports threshold as configured by a project. * <p> * The import rewrite is created on a javaScript unit and collects references to types that are added or removed. When adding imports, e.g. using * {@link #addImport(String)}, the import rewrite evaluates if the type can be imported and returns the a reference to the type that can be used in code. * This reference is either unqualified if the import could be added, or fully qualified if the import failed due to a conflict with another element of the same name. * </p> * <p> * On {@link #rewriteImports(IProgressMonitor)} the rewrite translates these descriptions into * text edits that can then be applied to the original source. The rewrite infrastructure tries to generate minimal text changes and only * works on the import statements. It is possible to combine the result of an import rewrite with the result of a {@link org.eclipse.wst.jsdt.core.dom.rewrite.ASTRewrite} * as long as no import statements are modified by the AST rewrite. * </p> * <p>The options controlling the import order and on-demand thresholds are: * <ul><li>{@link #setImportOrder(String[])} specifies the import groups and their preferred order</li> * <li>{@link #setOnDemandImportThreshold(int)} specifies the number of imports in a group needed for a on-demand import statement (star import)</li> * <li>{@link #setStaticOnDemandImportThreshold(int)} specifies the number of static imports in a group needed for a on-demand import statement (star import)</li> *</ul> * This class is not intended to be subclassed. * </p> * * Provisional API: This class/interface is part of an interim API that is still under development and expected to * change significantly before reaching stability. It is being made available at this early stage to solicit feedback * from pioneering adopters on the understanding that any code that uses this API will almost certainly be broken * (repeatedly) as the API evolves. */ public final class ImportRewrite { /** * A {@link ImportRewrite.ImportRewriteContext} can optionally be used in e.g. {@link ImportRewrite#addImport(String, ImportRewrite.ImportRewriteContext)} to * give more information about the types visible in the scope. These types can be for example inherited inner types where it is * unnecessary to add import statements for. * * </p> * <p> * This class can be implemented by clients. * </p> */ public static abstract class ImportRewriteContext { /** * Result constant signaling that the given element is know in the context. */ public final static int RES_NAME_FOUND= 1; /** * Result constant signaling that the given element is not know in the context. */ public final static int RES_NAME_UNKNOWN= 2; /** * Result constant signaling that the given element is conflicting with an other element in the context. */ public final static int RES_NAME_CONFLICT= 3; /** * Kind constant specifying that the element is a type import. */ public final static int KIND_TYPE= 1; /** * Kind constant specifying that the element is a static field import. */ public final static int KIND_STATIC_FIELD= 2; /** * Kind constant specifying that the element is a static method import. */ public final static int KIND_STATIC_METHOD= 3; /** * Searches for the given element in the context and reports if the element is known ({@link #RES_NAME_FOUND}), * unknown ({@link #RES_NAME_UNKNOWN}) or if its name conflicts ({@link #RES_NAME_CONFLICT}) with an other element. * @param qualifier The qualifier of the element, can be package or the qualified name of a type * @param name The simple name of the element; either a type, method or field name or * for on-demand imports. * @param kind The kind of the element. Can be either {@link #KIND_TYPE}, {@link #KIND_STATIC_FIELD} or * {@link #KIND_STATIC_METHOD}. Implementors should be prepared for new, currently unspecified kinds and return * {@link #RES_NAME_UNKNOWN} by default. * @return Returns the result of the lookup. Can be either {@link #RES_NAME_FOUND}, {@link #RES_NAME_UNKNOWN} or * {@link #RES_NAME_CONFLICT}. */ public abstract int findInContext(String qualifier, String name, int kind); } private static final char STATIC_PREFIX= 's'; private static final char NORMAL_PREFIX= 'n'; private final ImportRewriteContext defaultContext; private final IJavaScriptUnit compilationUnit; private final JavaScriptUnit astRoot; private final boolean restoreExistingImports; private final List existingImports; private String[] importOrder; private int importOnDemandThreshold; private int staticImportOnDemandThreshold; private List addedImports; private List removedImports; private String[] createdImports; private String[] createdStaticImports; private boolean filterImplicitImports; private boolean writeImports=false; private ImportRewriteSupport importRewriteExtension; private boolean isImportMatchesType=true; /** * Creates a {@link ImportRewrite} from a {@link IJavaScriptUnit}. If <code>restoreExistingImports</code> * is <code>true</code>, all existing imports are kept, and new imports will be inserted at best matching locations. If * <code>restoreExistingImports</code> is <code>false</code>, the existing imports will be removed and only the * newly added imports will be created. * <p> * Note that {@link #create(IJavaScriptUnit, boolean)} is more efficient than this method if an AST for * the javaScript unit is already available. * </p> * @param cu the javaScript unit to create the imports for * @param restoreExistingImports specifies if the existing imports should be kept or removed. * @return the created import rewriter. * @throws JavaScriptModelException thrown when the javaScript unit could not be accessed. */ public static ImportRewrite create(IJavaScriptUnit cu, boolean restoreExistingImports) throws JavaScriptModelException { if (cu == null) { throw new IllegalArgumentException("JavaScript unit must not be null"); //$NON-NLS-1$ } ImportRewriteSupport importRewriteExtension=null; InferrenceProvider[] inferenceProviders = InferrenceManager.getInstance().getInferenceProviders( (IInferenceFile)cu); if (inferenceProviders.length>0 && inferenceProviders[0].getRefactoringSupport()!=null) { RefactoringSupport refactoringSupport = inferenceProviders[0].getRefactoringSupport(); if (refactoringSupport!=null) importRewriteExtension=refactoringSupport.getImportRewriteSupport(); } List existingImport= null; if (restoreExistingImports) { existingImport= new ArrayList(); IImportDeclaration[] imports= cu.getImports(); for (int i= 0; i < imports.length; i++) { IImportDeclaration curr= imports[i]; char prefix= Flags.isStatic(curr.getFlags()) ? STATIC_PREFIX : NORMAL_PREFIX; existingImport.add(prefix + curr.getElementName()); } } return new ImportRewrite(cu, null, existingImport,importRewriteExtension); } /** * Creates a {@link ImportRewrite} from a an AST ({@link JavaScriptUnit}). The AST has to be created from a * {@link IJavaScriptUnit}, that means {@link ASTParser#setSource(IJavaScriptUnit)} has been used when creating the * AST. If <code>restoreExistingImports</code> is <code>true</code>, all existing imports are kept, and new imports * will be inserted at best matching locations. If <code>restoreExistingImports</code> is <code>false</code>, the * existing imports will be removed and only the newly added imports will be created. * <p> * Note that this method is more efficient than using {@link #create(IJavaScriptUnit, boolean)} if an AST is already available. * </p> * @param astRoot the AST root node to create the imports for * @param restoreExistingImports specifies if the existing imports should be kept or removed. * @return the created import rewriter. * @throws IllegalArgumentException thrown when the passed AST is null or was not created from a javaScript unit. */ public static ImportRewrite create(JavaScriptUnit astRoot, boolean restoreExistingImports) { if (astRoot == null) { throw new IllegalArgumentException("AST must not be null"); //$NON-NLS-1$ } ITypeRoot typeRoot = astRoot.getTypeRoot(); if (!(typeRoot instanceof IJavaScriptUnit)) { throw new IllegalArgumentException("AST must have been constructed from a JavaScript element"); //$NON-NLS-1$ } ImportRewriteSupport importRewriteExtension=null; InferrenceProvider[] inferenceProviders = InferrenceManager.getInstance().getInferenceProviders( (IInferenceFile)typeRoot); if (inferenceProviders.length>0 && inferenceProviders[0].getRefactoringSupport()!=null) { RefactoringSupport refactoringSupport = inferenceProviders[0].getRefactoringSupport(); if (refactoringSupport!=null) importRewriteExtension=refactoringSupport.getImportRewriteSupport(); } List existingImport= null; if (restoreExistingImports) { existingImport= new ArrayList(); List imports= astRoot.imports(); for (int i= 0; i < imports.size(); i++) { ImportDeclaration curr= (ImportDeclaration) imports.get(i); StringBuffer buf= new StringBuffer(); buf.append(curr.isStatic() ? STATIC_PREFIX : NORMAL_PREFIX).append(curr.getName().getFullyQualifiedName()); if (curr.isOnDemand()) { if (buf.length() > 1) buf.append('.'); buf.append('*'); } existingImport.add(buf.toString()); } } return new ImportRewrite((IJavaScriptUnit) typeRoot, astRoot, existingImport, importRewriteExtension); } private ImportRewrite(IJavaScriptUnit cu, JavaScriptUnit astRoot, List existingImports, ImportRewriteSupport importRewriteExtension) { this.compilationUnit= cu; this.astRoot= astRoot; // might be null this.importRewriteExtension=importRewriteExtension; if (this.importRewriteExtension!=null) { this.isImportMatchesType=this.importRewriteExtension.isImportMatchesType(); this.writeImports=true; } if (existingImports != null) { this.existingImports= existingImports; this.restoreExistingImports= !existingImports.isEmpty(); } else { this.existingImports= new ArrayList(); this.restoreExistingImports= false; } this.filterImplicitImports= true; this.defaultContext= new ImportRewriteContext() { public int findInContext(String qualifier, String name, int kind) { return findInImports(qualifier, name, kind); } }; this.addedImports= null; // Initialized on use this.removedImports= null; // Initialized on use this.createdImports= null; this.createdStaticImports= null; this.importOrder= CharOperation.NO_STRINGS; this.importOnDemandThreshold= 99; this.staticImportOnDemandThreshold= 99; } /** * Defines the import groups and order to be used by the {@link ImportRewrite}. * Imports are added to the group matching their qualified name most. The empty group name groups all imports not matching * any other group. Static imports are managed in separate groups. Static import group names are prefixed with a '#' character. * @param order A list of strings defining the import groups. A group name must be a valid package name or empty. If can be * prefixed by the '#' character for static import groups */ public void setImportOrder(String[] order) { if (order == null) throw new IllegalArgumentException("Order must not be null"); //$NON-NLS-1$ this.importOrder= order; } /** * Sets the on-demand import threshold for normal (non-static) imports. * This threshold defines the number of imports that need to be in a group to use * a on-demand (star) import declaration instead. * * @param threshold a positive number defining the on-demand import threshold * for normal (non-static) imports. * @throws IllegalArgumentException a {@link IllegalArgumentException} is thrown * if the number is not positive. */ public void setOnDemandImportThreshold(int threshold) { if (threshold <= 0) throw new IllegalArgumentException("Threshold must be positive."); //$NON-NLS-1$ this.importOnDemandThreshold= threshold; } /** * Sets the on-demand import threshold for static imports. * This threshold defines the number of imports that need to be in a group to use * a on-demand (star) import declaration instead. * * @param threshold a positive number defining the on-demand import threshold * for normal (non-static) imports. * @throws IllegalArgumentException a {@link IllegalArgumentException} is thrown * if the number is not positive. */ public void setStaticOnDemandImportThreshold(int threshold) { if (threshold <= 0) throw new IllegalArgumentException("Threshold must be positive."); //$NON-NLS-1$ this.staticImportOnDemandThreshold= threshold; } /** * The javaScript unit for which this import rewrite was created for. * @return the javaScript unit for which this import rewrite was created for. */ public IJavaScriptUnit getCompilationUnit() { return this.compilationUnit; } /** * Returns the default rewrite context that only knows about the imported types. Clients * can write their own context and use the default context for the default behavior. * @return the default import rewrite context. */ public ImportRewriteContext getDefaultImportRewriteContext() { return this.defaultContext; } /** * Specifies that implicit imports (types in default package, package <code>java.lang</code> or * in the same package as the rewrite javaScript unit should not be created except if necessary * to resolve an on-demand import conflict. The filter is enabled by default. * @param filterImplicitImports if set, implicit imports will be filtered. */ public void setFilterImplicitImports(boolean filterImplicitImports) { this.filterImplicitImports= filterImplicitImports; } private static int compareImport(char prefix, String qualifier, String name, String curr) { if (curr.charAt(0) != prefix || !curr.endsWith(name)) { return ImportRewriteContext.RES_NAME_UNKNOWN; } curr= curr.substring(1); // remove the prefix if (curr.length() == name.length()) { if (qualifier.length() == 0) { return ImportRewriteContext.RES_NAME_FOUND; } return ImportRewriteContext.RES_NAME_CONFLICT; } // at this place: curr.length > name.length int dotPos= curr.length() - name.length() - 1; if (curr.charAt(dotPos) != '.') { return ImportRewriteContext.RES_NAME_UNKNOWN; } if (qualifier.length() != dotPos || !curr.startsWith(qualifier)) { return ImportRewriteContext.RES_NAME_CONFLICT; } return ImportRewriteContext.RES_NAME_FOUND; } /** * Not API, package visibility as accessed from an anonymous type */ /* package */ final int findInImports(String qualifier, String name, int kind) { boolean allowAmbiguity= (kind == ImportRewriteContext.KIND_STATIC_METHOD) || (name.length() == 1 && name.charAt(0) == '*'); List imports= this.existingImports; char prefix= (kind == ImportRewriteContext.KIND_TYPE) ? NORMAL_PREFIX : STATIC_PREFIX; for (int i= imports.size() - 1; i >= 0 ; i--) { String curr= (String) imports.get(i); int res= compareImport(prefix, qualifier, name, curr); if (res != ImportRewriteContext.RES_NAME_UNKNOWN) { if (!allowAmbiguity || res == ImportRewriteContext.RES_NAME_FOUND) { return res; } } } return ImportRewriteContext.RES_NAME_UNKNOWN; } /** * Adds a new import to the rewriter's record and returns a {@link Type} node that can be used * in the code as a reference to the type. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param typeSig the signature of the type to be added. * @param ast the AST to create the returned type for. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public Type addImportFromSignature(String typeSig, AST ast) { return addImportFromSignature(typeSig, ast, this.defaultContext); } /** * Adds a new import to the rewriter's record and returns a {@link Type} node that can be used * in the code as a reference to the type. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param typeSig the signature of the type to be added. * @param ast the AST to create the returned type for. * @param context an optional context that knows about types visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public Type addImportFromSignature(String typeSig, AST ast, ImportRewriteContext context) { if (typeSig == null || typeSig.length() == 0) { throw new IllegalArgumentException("Invalid type signature: empty or null"); //$NON-NLS-1$ } int sigKind= Signature.getTypeSignatureKind(typeSig); switch (sigKind) { case Signature.BASE_TYPE_SIGNATURE: return ast.newPrimitiveType(PrimitiveType.toCode(Signature.toString(typeSig))); case Signature.ARRAY_TYPE_SIGNATURE: Type elementType= addImportFromSignature(Signature.getElementType(typeSig), ast, context); return ast.newArrayType(elementType, Signature.getArrayCount(typeSig)); case Signature.CLASS_TYPE_SIGNATURE: String erasureName= Signature.toString(typeSig); if (typeSig.charAt(0) == Signature.C_RESOLVED) { erasureName= internalAddImport(erasureName, erasureName, context); } Type baseType= ast.newSimpleType(ast.newName(erasureName)); return baseType; default: throw new IllegalArgumentException("Unknown type signature kind: " + typeSig); //$NON-NLS-1$ } } /** * Adds a new import to the rewriter's record and returns a type reference that can be used * in the code. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding the signature of the type to be added. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public String addImport(ITypeBinding binding) { return addImport(binding, this.defaultContext); } /** * Adds a new import to the rewriter's record and returns a type reference that can be used * in the code. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding the signature of the type to be added. * @param context an optional context that knows about types visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public String addImport(ITypeBinding binding, ImportRewriteContext context) { if (binding.isPrimitive()) { return binding.getName(); } ITypeBinding normalizedBinding= normalizeTypeBinding(binding); if (normalizedBinding == null) { return "invalid"; //$NON-NLS-1$ } if (normalizedBinding.isArray()) { StringBuffer res= new StringBuffer(addImport(normalizedBinding.getElementType(), context)); for (int i= normalizedBinding.getDimensions(); i > 0; i--) { res.append("[]"); //$NON-NLS-1$ } return res.toString(); } String qualifiedName= getRawQualifiedName(normalizedBinding); if (qualifiedName.length() > 0) { String str= internalAddImport(qualifiedName, qualifiedName, context); return str; } return getRawName(normalizedBinding); } private static ITypeBinding normalizeTypeBinding(ITypeBinding binding) { if (binding != null && !binding.isNullType() && !"void".equals(binding.getName())) { //$NON-NLS-1$ if (binding.isAnonymous()) { return binding.getSuperclass(); } return binding; } return null; } /** * Adds a new import to the rewriter's record and returns a {@link Type} that can be used * in the code. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding the signature of the type to be added. * @param ast the AST to create the returned type for. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public Type addImport(ITypeBinding binding, AST ast) { return addImport(binding, ast, this.defaultContext); } /** * Adds a new import to the rewriter's record and returns a {@link Type} that can be used * in the code. The type binding can be an array binding or type variable. * If the binding is a generic type, the type parameters are ignored. For parameterized types, also the type * arguments are processed and imports added if necessary. Anonymous types inside type arguments are normalized to their base type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding the signature of the type to be added. * @param ast the AST to create the returned type for. * @param context an optional context that knows about types visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public Type addImport(ITypeBinding binding, AST ast, ImportRewriteContext context) { if (binding.isPrimitive() || binding.isRecovered()) { return ast.newPrimitiveType(PrimitiveType.toCode(binding.getName())); } ITypeBinding normalizedBinding= normalizeTypeBinding(binding); if (normalizedBinding == null) { return ast.newSimpleType(ast.newSimpleName("invalid")); //$NON-NLS-1$ } if (normalizedBinding.isArray()) { Type elementType= addImport(normalizedBinding.getElementType(), ast, context); return ast.newArrayType(elementType, normalizedBinding.getDimensions()); } String qualifiedName= getRawQualifiedName(normalizedBinding); if (qualifiedName.length() > 0) { String res= internalAddImport(qualifiedName, qualifiedName, context); return ast.newSimpleType(ast.newName(res)); } return ast.newSimpleType(ast.newName(getRawName(normalizedBinding))); } /** * Adds a new import to the rewriter's record and returns a type reference that can be used * in the code. The type binding can only be an array or non-generic type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param qualifiedTypeName the qualified type name of the type to be added * @param context an optional context that knows about types visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public String addImport(String qualifiedTypeName, String packageName, ImportRewriteContext context) { if (packageName==null) packageName=qualifiedTypeName; if (JavaScriptCore.IS_ECMASCRIPT4) { int angleBracketOffset = qualifiedTypeName.indexOf('<'); if (angleBracketOffset != -1) { return internalAddImport(qualifiedTypeName.substring(0, angleBracketOffset), packageName, context) + qualifiedTypeName.substring(angleBracketOffset); } int bracketOffset = qualifiedTypeName.indexOf('['); if (bracketOffset != -1) { return internalAddImport(qualifiedTypeName.substring(0, bracketOffset), packageName, context) + qualifiedTypeName.substring(bracketOffset); } } return internalAddImport(qualifiedTypeName, packageName, context); } /** * Adds a new import to the rewriter's record and returns a type reference that can be used * in the code. The type binding can only be an array or non-generic type. * <p> * No imports are added for types that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param qualifiedTypeName the qualified type name of the type to be added * @return returns a type to which the type binding can be assigned to. The returned type contains is unqualified * when an import could be added or was already known. It is fully qualified, if an import conflict prevented the import. */ public String addImport(String qualifiedTypeName) { return addImport(qualifiedTypeName, qualifiedTypeName, this.defaultContext); } /** * Adds a new static import to the rewriter's record and returns a reference that can be used in the code. The reference will * be fully qualified if an import conflict prevented the import or unqualified if the import succeeded or was already * existing. * <p> * No imports are added for members that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding The binding of the static field or method to be added. * @return returns either the simple member name if the import was successful or else the qualified name if * an import conflict prevented the import. * @throws IllegalArgumentException an {@link IllegalArgumentException} is thrown if the binding is not a static field * or method. */ public String addStaticImport(IBinding binding) { return addStaticImport(binding, this.defaultContext); } /** * Adds a new static import to the rewriter's record and returns a reference that can be used in the code. The reference will * be fully qualified if an import conflict prevented the import or unqualified if the import succeeded or was already * existing. * <p> * No imports are added for members that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param binding The binding of the static field or method to be added. * @param context an optional context that knows about members visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns either the simple member name if the import was successful or else the qualified name if * an import conflict prevented the import. * @throws IllegalArgumentException an {@link IllegalArgumentException} is thrown if the binding is not a static field * or method. */ public String addStaticImport(IBinding binding, ImportRewriteContext context) { if (Modifier.isStatic(binding.getModifiers())) { if (binding instanceof IVariableBinding) { IVariableBinding variableBinding= (IVariableBinding) binding; if (variableBinding.isField()) { ITypeBinding declaringType= variableBinding.getDeclaringClass(); return addStaticImport(getRawQualifiedName(declaringType), binding.getName(), true, context); } } else if (binding instanceof IFunctionBinding) { ITypeBinding declaringType= ((IFunctionBinding) binding).getDeclaringClass(); return addStaticImport(getRawQualifiedName(declaringType), binding.getName(), false, context); } } throw new IllegalArgumentException("Binding must be a static field or method."); //$NON-NLS-1$ } /** * Adds a new static import to the rewriter's record and returns a reference that can be used in the code. The reference will * be fully qualified if an import conflict prevented the import or unqualified if the import succeeded or was already * existing. * <p> * No imports are added for members that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param declaringTypeName The qualified name of the static's member declaring type * @param simpleName the simple name of the member; either a field or a method name. * @param isField <code>true</code> specifies that the member is a field, <code>false</code> if it is a * method. * @return returns either the simple member name if the import was successful or else the qualified name if * an import conflict prevented the import. */ public String addStaticImport(String declaringTypeName, String simpleName, boolean isField) { return addStaticImport(declaringTypeName, simpleName, isField, this.defaultContext); } /** * Adds a new static import to the rewriter's record and returns a reference that can be used in the code. The reference will * be fully qualified if an import conflict prevented the import or unqualified if the import succeeded or was already * existing. * <p> * No imports are added for members that are already known. If a import for a type is recorded to be removed, this record is discarded instead. * </p> * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been added. * </p> * @param declaringTypeName The qualified name of the static's member declaring type * @param simpleName the simple name of the member; either a field or a method name. * @param isField <code>true</code> specifies that the member is a field, <code>false</code> if it is a * method. * @param context an optional context that knows about members visible in the current scope or <code>null</code> * to use the default context only using the available imports. * @return returns either the simple member name if the import was successful or else the qualified name if * an import conflict prevented the import. */ public String addStaticImport(String declaringTypeName, String simpleName, boolean isField, ImportRewriteContext context) { if (declaringTypeName.indexOf('.') == -1) { return declaringTypeName + '.' + simpleName; } if (context == null) { context= this.defaultContext; } int kind= isField ? ImportRewriteContext.KIND_STATIC_FIELD : ImportRewriteContext.KIND_STATIC_METHOD; int res= context.findInContext(declaringTypeName, simpleName, kind); if (res == ImportRewriteContext.RES_NAME_CONFLICT) { return declaringTypeName + '.' + simpleName; } if (res == ImportRewriteContext.RES_NAME_UNKNOWN) { addEntry(STATIC_PREFIX + declaringTypeName + '.' + simpleName); } return simpleName; } private String internalAddImport(String fullTypeName, String packageName,ImportRewriteContext context) { String importName=(this.isImportMatchesType)? fullTypeName : packageName; int idx= importName.lastIndexOf('.'); String typeContainerName, typeName; if (idx != -1) { typeContainerName= importName.substring(0, idx); typeName= importName.substring(idx + 1); } else { typeContainerName= ""; //$NON-NLS-1$ typeName= importName; } if (typeContainerName.length() == 0 && PrimitiveType.toCode(typeName) != null) { return fullTypeName; } if (context == null) context= this.defaultContext; int res= context.findInContext(typeContainerName, typeName, ImportRewriteContext.KIND_TYPE); if (res == ImportRewriteContext.RES_NAME_CONFLICT) { return fullTypeName; } if (res == ImportRewriteContext.RES_NAME_UNKNOWN) { addEntry(NORMAL_PREFIX + importName); } return fullTypeName; } private void addEntry(String entry) { this.existingImports.add(entry); if (this.removedImports != null) { if (this.removedImports.remove(entry)) { return; } } if (this.addedImports == null) { this.addedImports= new ArrayList(); } this.addedImports.add(entry); } private boolean removeEntry(String entry) { if (this.existingImports.remove(entry)) { if (this.addedImports != null) { if (this.addedImports.remove(entry)) { return true; } } if (this.removedImports == null) { this.removedImports= new ArrayList(); } this.removedImports.add(entry); return true; } return false; } /** * Records to remove a import. No remove is recorded if no such import exists or if such an import is recorded * to be added. In that case the record of the addition is discarded. * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that an import has been removed. * </p> * @param qualifiedName The import name to remove. * @return <code>true</code> is returned of an import of the given name could be found. */ public boolean removeImport(String qualifiedName) { return removeEntry(NORMAL_PREFIX + qualifiedName); } /** * Records to remove a static import. No remove is recorded if no such import exists or if such an import is recorded * to be added. In that case the record of the addition is discarded. * <p> * The content of the javaScript unit itself is actually not modified * in any way by this method; rather, the rewriter just records that a new import has been removed. * </p> * @param qualifiedName The import name to remove. * @return <code>true</code> is returned of an import of the given name could be found. */ public boolean removeStaticImport(String qualifiedName) { return removeEntry(STATIC_PREFIX + qualifiedName); } private static String getRawName(ITypeBinding normalizedBinding) { return normalizedBinding.getTypeDeclaration().getName(); } private static String getRawQualifiedName(ITypeBinding normalizedBinding) { return normalizedBinding.getTypeDeclaration().getQualifiedName(); } /** * Converts all modifications recorded by this rewriter into an object representing the corresponding text * edits to the source code of the rewrite's javaScript unit. The javaScript unit itself is not modified. * <p> * Calling this methods does not discard the modifications on record. Subsequence modifications are added * to the ones already on record. If this method is called again later, the resulting text edit object will accurately * reflect the net cumulative affect of all those changes. * </p> * @param monitor the progress monitor or <code>null</code> * @return text edit object describing the changes to the document corresponding to the changes * recorded by this rewriter * @throws CoreException the exception is thrown if the rewrite fails. */ public final TextEdit rewriteImports(IProgressMonitor monitor) throws CoreException { if (monitor == null) { monitor= new NullProgressMonitor(); } try { monitor.beginTask(Messages.bind(Messages.importRewrite_processDescription), 2); if (!hasRecordedChanges()) { this.createdImports= CharOperation.NO_STRINGS; this.createdStaticImports= CharOperation.NO_STRINGS; return new MultiTextEdit(); } JavaScriptUnit usedAstRoot= this.astRoot; if (usedAstRoot == null) { ASTParser parser= ASTParser.newParser(AST.JLS3); parser.setSource(this.compilationUnit); parser.setFocalPosition(0); // reduced AST parser.setResolveBindings(false); usedAstRoot= (JavaScriptUnit) parser.createAST(new SubProgressMonitor(monitor, 1)); } ImportRewriteAnalyzer computer= new ImportRewriteAnalyzer(this.compilationUnit, usedAstRoot, this.importOrder, this.importOnDemandThreshold, this.staticImportOnDemandThreshold, this.restoreExistingImports, this.importRewriteExtension); computer.setFilterImplicitImports(this.filterImplicitImports); if (this.addedImports != null) { for (int i= 0; i < this.addedImports.size(); i++) { String curr= (String) this.addedImports.get(i); computer.addImport(curr.substring(1), STATIC_PREFIX == curr.charAt(0)); } } if (this.removedImports != null) { for (int i= 0; i < this.removedImports.size(); i++) { String curr= (String) this.removedImports.get(i); computer.removeImport(curr.substring(1), STATIC_PREFIX == curr.charAt(0)); } } TextEdit result= computer.getResultingEdits(new SubProgressMonitor(monitor, 1)); this.createdImports= computer.getCreatedImports(); this.createdStaticImports= computer.getCreatedStaticImports(); return result; } finally { monitor.done(); } } /** * Returns all new non-static imports created by the last invocation of {@link #rewriteImports(IProgressMonitor)} * or <code>null</code> if these methods have not been called yet. * <p> * Note that this list doesn't need to be the same as the added imports (see {@link #getAddedImports()}) as * implicit imports are not created and some imports are represented by on-demand imports instead. * </p> * @return the created imports */ public String[] getCreatedImports() { return this.createdImports; } /** * Returns all new static imports created by the last invocation of {@link #rewriteImports(IProgressMonitor)} * or <code>null</code> if these methods have not been called yet. * <p> * Note that this list doesn't need to be the same as the added static imports ({@link #getAddedStaticImports()}) as * implicit imports are not created and some imports are represented by on-demand imports instead. * </p * @return the created imports */ public String[] getCreatedStaticImports() { return this.createdStaticImports; } /** * Returns all non-static imports that are recorded to be added. * * @return the imports recorded to be added. */ public String[] getAddedImports() { return filterFromList(this.addedImports, NORMAL_PREFIX); } /** * Returns all static imports that are recorded to be added. * * @return the static imports recorded to be added. */ public String[] getAddedStaticImports() { return filterFromList(this.addedImports, STATIC_PREFIX); } /** * Returns all non-static imports that are recorded to be removed. * * @return the imports recorded to be removed. */ public String[] getRemovedImports() { return filterFromList(this.removedImports, NORMAL_PREFIX); } /** * Returns all static imports that are recorded to be removed. * * @return the static imports recorded to be removed. */ public String[] getRemovedStaticImports() { return filterFromList(this.removedImports, STATIC_PREFIX); } /** * Returns <code>true</code> if imports have been recorded to be added or removed. * @return boolean returns if any changes to imports have been recorded. */ public boolean hasRecordedChanges() { if (!writeImports) return false; return !this.restoreExistingImports || (this.addedImports != null && !this.addedImports.isEmpty()) || (this.removedImports != null && !this.removedImports.isEmpty()); } private static String[] filterFromList(List imports, char prefix) { if (imports == null) { return CharOperation.NO_STRINGS; } ArrayList res= new ArrayList(); for (int i= 0; i < imports.size(); i++) { String curr= (String) imports.get(i); if (prefix == curr.charAt(0)) { res.add(curr.substring(1)); } } return (String[]) res.toArray(new String[res.size()]); } public boolean isImportMatchesType() { return isImportMatchesType; } }