| /******************************************************************************* |
| * 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.jdt.internal.codeassist; |
| |
| import java.util.*; |
| |
| import org.eclipse.jdt.core.Signature; |
| import org.eclipse.jdt.core.compiler.*; |
| import org.eclipse.jdt.internal.codeassist.impl.*; |
| import org.eclipse.jdt.internal.codeassist.select.*; |
| import org.eclipse.jdt.internal.compiler.*; |
| import org.eclipse.jdt.internal.compiler.env.*; |
| import org.eclipse.jdt.internal.compiler.ast.*; |
| import org.eclipse.jdt.internal.compiler.lookup.*; |
| import org.eclipse.jdt.internal.compiler.parser.*; |
| import org.eclipse.jdt.internal.compiler.parser.Scanner; |
| import org.eclipse.jdt.internal.compiler.problem.*; |
| import org.eclipse.jdt.internal.core.SearchableEnvironment; |
| import org.eclipse.jdt.internal.core.SelectionRequestor; |
| import org.eclipse.jdt.internal.core.SourceType; |
| import org.eclipse.jdt.internal.core.SourceTypeElementInfo; |
| import org.eclipse.jdt.internal.core.util.ASTNodeFinder; |
| |
| /** |
| * The selection engine is intended to infer the nature of a selected name in some |
| * source code. This name can be qualified. |
| * |
| * Selection is resolving context using a name environment (no need to search), assuming |
| * the source where selection occurred is correct and will not perform any completion |
| * attempt. If this was the desired behavior, a call to the CompletionEngine should be |
| * performed instead. |
| */ |
| public final class SelectionEngine extends Engine implements ISearchRequestor { |
| |
| public static boolean DEBUG = false; |
| public static boolean PERF = false; |
| |
| SelectionParser parser; |
| ISelectionRequestor requestor; |
| |
| boolean acceptedAnswer; |
| |
| private int actualSelectionStart; |
| private int actualSelectionEnd; |
| private char[] selectedIdentifier; |
| |
| private char[][][] acceptedClasses; |
| private int[] acceptedClassesModifiers; |
| private char[][][] acceptedInterfaces; |
| private int[] acceptedInterfacesModifiers; |
| private char[][][] acceptedEnums; |
| private int[] acceptedEnumsModifiers; |
| private char[][][] acceptedAnnotations; |
| private int[] acceptedAnnotationsModifiers; |
| int acceptedClassesCount; |
| int acceptedInterfacesCount; |
| int acceptedEnumsCount; |
| int acceptedAnnotationsCount; |
| |
| boolean noProposal = true; |
| IProblem problem = null; |
| |
| /** |
| * The SelectionEngine is responsible for computing the selected object. |
| * |
| * It requires a searchable name environment, which supports some |
| * specific search APIs, and a requestor to feed back the results to a UI. |
| * |
| * @param nameEnvironment org.eclipse.jdt.internal.core.SearchableEnvironment |
| * used to resolve type/package references and search for types/packages |
| * based on partial names. |
| * |
| * @param requestor org.eclipse.jdt.internal.codeassist.ISelectionRequestor |
| * since the engine might produce answers of various forms, the engine |
| * is associated with a requestor able to accept all possible completions. |
| * |
| * @param settings java.util.Map |
| * set of options used to configure the code assist engine. |
| */ |
| public SelectionEngine( |
| SearchableEnvironment nameEnvironment, |
| ISelectionRequestor requestor, |
| Map settings) { |
| |
| super(settings); |
| |
| this.requestor = requestor; |
| this.nameEnvironment = nameEnvironment; |
| |
| ProblemReporter problemReporter = |
| new ProblemReporter( |
| DefaultErrorHandlingPolicies.proceedWithAllProblems(), |
| this.compilerOptions, |
| new DefaultProblemFactory(Locale.getDefault())) { |
| |
| public IProblem createProblem( |
| char[] fileName, |
| int problemId, |
| String[] problemArguments, |
| String[] messageArguments, |
| int severity, |
| int problemStartPosition, |
| int problemEndPosition, |
| int lineNumber) { |
| IProblem pb = super.createProblem( |
| fileName, |
| problemId, |
| problemArguments, |
| messageArguments, |
| severity, |
| problemStartPosition, |
| problemEndPosition, |
| lineNumber); |
| if(SelectionEngine.this.problem == null && pb.isError() && (pb.getID() & IProblem.Syntax) == 0) { |
| SelectionEngine.this.problem = pb; |
| } |
| |
| return pb; |
| } |
| }; |
| this.lookupEnvironment = |
| new LookupEnvironment(this, this.compilerOptions, problemReporter, nameEnvironment); |
| this.parser = new SelectionParser(problemReporter); |
| } |
| |
| public void acceptType(char[] packageName, char[] typeName, int modifiers, AccessRestriction accessRestriction) { |
| if (CharOperation.equals(typeName, this.selectedIdentifier)) { |
| if(mustQualifyType(packageName, typeName)) { |
| int length = 0; |
| int kind = modifiers & (IConstants.AccInterface | IConstants.AccEnum | IConstants.AccAnnotation); |
| switch (kind) { |
| case IConstants.AccAnnotation: |
| case IConstants.AccAnnotation | IConstants.AccInterface: |
| char[][] acceptedAnnotation = new char[2][]; |
| acceptedAnnotation[0] = packageName; |
| acceptedAnnotation[1] = typeName; |
| |
| if(this.acceptedAnnotations == null) { |
| this.acceptedAnnotations = new char[10][][]; |
| this.acceptedAnnotationsModifiers = new int[10]; |
| this.acceptedAnnotationsCount = 0; |
| } |
| length = this.acceptedAnnotations.length; |
| if(length == this.acceptedAnnotationsCount) { |
| int newLength = (length + 1)* 2; |
| System.arraycopy(this.acceptedAnnotations, 0, this.acceptedAnnotations = new char[newLength][][], 0, length); |
| System.arraycopy(this.acceptedAnnotationsModifiers, 0, this.acceptedAnnotationsModifiers = new int[newLength], 0, length); |
| } |
| this.acceptedAnnotationsModifiers[this.acceptedAnnotationsCount] = modifiers; |
| this.acceptedAnnotations[this.acceptedAnnotationsCount++] = acceptedAnnotation; |
| break; |
| case IConstants.AccEnum: |
| char[][] acceptedEnum = new char[2][]; |
| acceptedEnum[0] = packageName; |
| acceptedEnum[1] = typeName; |
| |
| if(this.acceptedEnums == null) { |
| this.acceptedEnums = new char[10][][]; |
| this.acceptedEnumsModifiers = new int[10]; |
| this.acceptedEnumsCount = 0; |
| } |
| length = this.acceptedEnums.length; |
| if(length == this.acceptedEnumsCount) { |
| int newLength = (length + 1)* 2; |
| System.arraycopy(this.acceptedEnums, 0, this.acceptedEnums = new char[newLength][][], 0, length); |
| System.arraycopy(this.acceptedEnumsModifiers, 0, this.acceptedEnumsModifiers = new int[newLength], 0, length); |
| } |
| this.acceptedEnumsModifiers[this.acceptedEnumsCount] = modifiers; |
| this.acceptedEnums[this.acceptedEnumsCount++] = acceptedEnum; |
| break; |
| case IConstants.AccInterface: |
| char[][] acceptedInterface= new char[2][]; |
| acceptedInterface[0] = packageName; |
| acceptedInterface[1] = typeName; |
| |
| if(this.acceptedInterfaces == null) { |
| this.acceptedInterfaces = new char[10][][]; |
| this.acceptedInterfacesModifiers = new int[10]; |
| this.acceptedInterfacesCount = 0; |
| } |
| length = this.acceptedInterfaces.length; |
| if(length == this.acceptedInterfacesCount) { |
| int newLength = (length + 1)* 2; |
| System.arraycopy(this.acceptedInterfaces, 0, this.acceptedInterfaces = new char[newLength][][], 0, length); |
| System.arraycopy(this.acceptedInterfacesModifiers, 0, this.acceptedInterfacesModifiers = new int[newLength], 0, length); |
| } |
| this.acceptedInterfacesModifiers[this.acceptedInterfacesCount] = modifiers; |
| this.acceptedInterfaces[this.acceptedInterfacesCount++] = acceptedInterface; |
| break; |
| default: |
| char[][] acceptedClass = new char[2][]; |
| acceptedClass[0] = packageName; |
| acceptedClass[1] = typeName; |
| |
| if(this.acceptedClasses == null) { |
| this.acceptedClasses = new char[10][][]; |
| this.acceptedClassesModifiers = new int[10]; |
| this.acceptedClassesCount = 0; |
| } |
| length = this.acceptedClasses.length; |
| if(length == this.acceptedClassesCount) { |
| int newLength = (length + 1)* 2; |
| System.arraycopy(this.acceptedClasses, 0, this.acceptedClasses = new char[newLength][][], 0, length); |
| System.arraycopy(this.acceptedClassesModifiers, 0, this.acceptedClassesModifiers = new int[newLength], 0, length); |
| } |
| this.acceptedClassesModifiers[this.acceptedClassesCount] = modifiers; |
| this.acceptedClasses[this.acceptedClassesCount++] = acceptedClass; |
| break; |
| } |
| } else { |
| this.noProposal = false; |
| this.requestor.acceptType( |
| packageName, |
| typeName, |
| modifiers, |
| false, |
| null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| this.acceptedAnswer = true; |
| } |
| } |
| } |
| |
| /** |
| * One result of the search consists of a new package. |
| * @param packageName char[] |
| * |
| * NOTE - All package names are presented in their readable form: |
| * Package names are in the form "a.b.c". |
| * The default package is represented by an empty array. |
| */ |
| public void acceptPackage(char[] packageName) { |
| // implementation of interface method |
| } |
| |
| private void acceptQualifiedTypes() { |
| if(this.acceptedClasses != null){ |
| this.acceptedAnswer = true; |
| for (int i = 0; i < this.acceptedClassesCount; i++) { |
| this.noProposal = false; |
| this.requestor.acceptType( |
| this.acceptedClasses[i][0], |
| this.acceptedClasses[i][1], |
| this.acceptedClassesModifiers[i], |
| false, |
| null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedClasses = null; |
| this.acceptedClassesModifiers = null; |
| this.acceptedClassesCount = 0; |
| } |
| if(this.acceptedInterfaces != null){ |
| this.acceptedAnswer = true; |
| for (int i = 0; i < this.acceptedInterfacesCount; i++) { |
| this.noProposal = false; |
| this.requestor.acceptType( |
| this.acceptedInterfaces[i][0], |
| this.acceptedInterfaces[i][1], |
| this.acceptedInterfacesModifiers[i], |
| false, |
| null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedInterfaces = null; |
| this.acceptedInterfacesModifiers = null; |
| this.acceptedInterfacesCount = 0; |
| } |
| if(this.acceptedAnnotations != null){ |
| this.acceptedAnswer = true; |
| for (int i = 0; i < this.acceptedAnnotationsCount; i++) { |
| this.noProposal = false; |
| this.requestor.acceptType( |
| this.acceptedAnnotations[i][0], |
| this.acceptedAnnotations[i][1], |
| this.acceptedAnnotationsModifiers[i], |
| false, |
| null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedAnnotations = null; |
| this.acceptedAnnotationsModifiers = null; |
| this.acceptedAnnotationsCount = 0; |
| } |
| if(this.acceptedEnums != null){ |
| this.acceptedAnswer = true; |
| for (int i = 0; i < this.acceptedEnumsCount; i++) { |
| this.noProposal = false; |
| this.requestor.acceptType( |
| this.acceptedEnums[i][0], |
| this.acceptedEnums[i][1], |
| this.acceptedEnumsModifiers[i], |
| false, |
| null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedEnums = null; |
| this.acceptedEnumsModifiers = null; |
| this.acceptedEnumsCount = 0; |
| } |
| } |
| private boolean checkSelection( |
| char[] source, |
| int selectionStart, |
| int selectionEnd) { |
| |
| Scanner scanner = new Scanner(); |
| scanner.setSource(source); |
| |
| int lastIdentifierStart = -1; |
| int lastIdentifierEnd = -1; |
| char[] lastIdentifier = null; |
| int token; |
| |
| if(selectionStart > selectionEnd){ |
| |
| // compute start position of current line |
| int currentPosition = selectionStart - 1; |
| int nextCharacterPosition = selectionStart; |
| char currentCharacter = ' '; |
| try { |
| lineLoop: while(currentPosition > 0){ |
| |
| if(source[currentPosition] == '\\' && source[currentPosition+1] == 'u') { |
| int pos = currentPosition + 2; |
| int c1 = 0, c2 = 0, c3 = 0, c4 = 0; |
| while (source[pos] == 'u') { |
| pos++; |
| } |
| if ((c1 = Character.getNumericValue(source[pos++])) > 15 |
| || c1 < 0 |
| || (c2 = Character.getNumericValue(source[pos++])) > 15 |
| || c2 < 0 |
| || (c3 = Character.getNumericValue(source[pos++])) > 15 |
| || c3 < 0 |
| || (c4 = Character.getNumericValue(source[pos++])) > 15 |
| || c4 < 0) { |
| return false; |
| } else { |
| currentCharacter = (char) (((c1 * 16 + c2) * 16 + c3) * 16 + c4); |
| nextCharacterPosition = pos; |
| } |
| } else { |
| currentCharacter = source[currentPosition]; |
| nextCharacterPosition = currentPosition+1; |
| } |
| |
| switch(currentCharacter) { |
| case '\r': |
| case '\n': |
| case '/': |
| case '"': |
| case '\'': |
| break lineLoop; |
| } |
| currentPosition--; |
| } |
| } catch (ArrayIndexOutOfBoundsException e) { |
| return false; |
| } |
| |
| // compute start and end of the last token |
| scanner.resetTo(nextCharacterPosition, selectionEnd + 1 == source.length ? selectionEnd : selectionEnd + 1); |
| do { |
| try { |
| token = scanner.getNextToken(); |
| } catch (InvalidInputException e) { |
| return false; |
| } |
| switch (token) { |
| case TerminalTokens.TokenNamethis: |
| case TerminalTokens.TokenNamesuper: |
| case TerminalTokens.TokenNameIdentifier: |
| if (scanner.startPosition <= selectionStart && selectionStart <= scanner.currentPosition) { |
| lastIdentifierStart = scanner.startPosition; |
| lastIdentifierEnd = scanner.currentPosition - 1; |
| lastIdentifier = scanner.getCurrentTokenSource(); |
| } |
| break; |
| } |
| } while (token != TerminalTokens.TokenNameEOF); |
| } else { |
| scanner.resetTo(selectionStart, selectionEnd); |
| |
| boolean expectingIdentifier = true; |
| try { |
| do { |
| token = scanner.getNextToken(); |
| |
| switch (token) { |
| case TerminalTokens.TokenNamethis : |
| case TerminalTokens.TokenNamesuper : |
| case TerminalTokens.TokenNameIdentifier : |
| if (!expectingIdentifier) |
| return false; |
| lastIdentifier = scanner.getCurrentTokenSource(); |
| lastIdentifierStart = scanner.startPosition; |
| lastIdentifierEnd = scanner.currentPosition - 1; |
| if(lastIdentifierEnd > selectionEnd) { |
| lastIdentifierEnd = selectionEnd; |
| lastIdentifier = CharOperation.subarray(lastIdentifier, 0,lastIdentifierEnd - lastIdentifierStart + 1); |
| } |
| |
| expectingIdentifier = false; |
| break; |
| case TerminalTokens.TokenNameDOT : |
| if (expectingIdentifier) |
| return false; |
| expectingIdentifier = true; |
| break; |
| case TerminalTokens.TokenNameEOF : |
| if (expectingIdentifier) |
| return false; |
| break; |
| case TerminalTokens.TokenNameLESS : |
| if(!checkTypeArgument(scanner)) |
| return false; |
| break; |
| case TerminalTokens.TokenNameAT: |
| if(scanner.startPosition != scanner.initialPosition) |
| return false; |
| break; |
| default : |
| return false; |
| } |
| } while (token != TerminalTokens.TokenNameEOF); |
| } catch (InvalidInputException e) { |
| return false; |
| } |
| } |
| if (lastIdentifierStart > 0) { |
| this.actualSelectionStart = lastIdentifierStart; |
| this.actualSelectionEnd = lastIdentifierEnd; |
| this.selectedIdentifier = lastIdentifier; |
| return true; |
| } |
| return false; |
| } |
| private boolean checkTypeArgument(Scanner scanner) throws InvalidInputException { |
| int depth = 1; |
| int token; |
| StringBuffer buffer = new StringBuffer(); |
| do { |
| token = scanner.getNextToken(); |
| |
| switch(token) { |
| case TerminalTokens.TokenNameLESS : |
| depth++; |
| buffer.append(scanner.getCurrentTokenSource()); |
| break; |
| case TerminalTokens.TokenNameGREATER : |
| depth--; |
| buffer.append(scanner.getCurrentTokenSource()); |
| break; |
| case TerminalTokens.TokenNameRIGHT_SHIFT : |
| depth-=2; |
| buffer.append(scanner.getCurrentTokenSource()); |
| break; |
| case TerminalTokens.TokenNameUNSIGNED_RIGHT_SHIFT : |
| depth-=3; |
| buffer.append(scanner.getCurrentTokenSource()); |
| break; |
| case TerminalTokens.TokenNameextends : |
| case TerminalTokens.TokenNamesuper : |
| buffer.append(' '); |
| buffer.append(scanner.getCurrentTokenSource()); |
| buffer.append(' '); |
| break; |
| case TerminalTokens.TokenNameCOMMA : |
| if(depth == 1) { |
| int length = buffer.length(); |
| char[] typeRef = new char[length]; |
| buffer.getChars(0, length, typeRef, 0); |
| try { |
| Signature.createTypeSignature(typeRef, true); |
| buffer = new StringBuffer(); |
| } catch(IllegalArgumentException e) { |
| return false; |
| } |
| } |
| break; |
| default : |
| buffer.append(scanner.getCurrentTokenSource()); |
| break; |
| |
| } |
| if(depth < 0) { |
| return false; |
| } |
| } while (depth != 0 && token != TerminalTokens.TokenNameEOF); |
| |
| if(depth == 0) { |
| int length = buffer.length() - 1; |
| char[] typeRef = new char[length]; |
| buffer.getChars(0, length, typeRef, 0); |
| try { |
| Signature.createTypeSignature(typeRef, true); |
| return true; |
| } catch(IllegalArgumentException e) { |
| return false; |
| } |
| } |
| |
| return false; |
| } |
| |
| public AssistParser getParser() { |
| return this.parser; |
| } |
| |
| /* |
| * Returns whether the given binding is a local/anonymous reference binding, or if its declaring class is |
| * local. |
| */ |
| private boolean isLocal(ReferenceBinding binding) { |
| if(binding instanceof ParameterizedTypeBinding) { |
| return isLocal(((ParameterizedTypeBinding)binding).type); |
| } |
| if (!(binding instanceof SourceTypeBinding)) return false; |
| if (binding instanceof LocalTypeBinding) return true; |
| if (binding instanceof MemberTypeBinding) { |
| return isLocal(((MemberTypeBinding)binding).enclosingType); |
| } |
| return false; |
| } |
| |
| /** |
| * Ask the engine to compute the selection at the specified position |
| * of the given compilation unit. |
| |
| * @param sourceUnit org.eclipse.jdt.internal.compiler.env.ICompilationUnit |
| * the source of the current compilation unit. |
| * |
| * @param selectionSourceStart int |
| * @param selectionSourceEnd int |
| * a range in the source where the selection is. |
| */ |
| public void select( |
| ICompilationUnit sourceUnit, |
| int selectionSourceStart, |
| int selectionSourceEnd) { |
| |
| char[] source = sourceUnit.getContents(); |
| |
| if(DEBUG) { |
| System.out.print("SELECTION IN "); //$NON-NLS-1$ |
| System.out.print(sourceUnit.getFileName()); |
| System.out.print(" FROM "); //$NON-NLS-1$ |
| System.out.print(selectionSourceStart); |
| System.out.print(" TO "); //$NON-NLS-1$ |
| System.out.println(selectionSourceEnd); |
| System.out.println("SELECTION - Source :"); //$NON-NLS-1$ |
| System.out.println(source); |
| } |
| if (!checkSelection(source, selectionSourceStart, selectionSourceEnd)) { |
| return; |
| } |
| if (DEBUG) { |
| System.out.print("SELECTION - Checked : \""); //$NON-NLS-1$ |
| System.out.print(new String(source, actualSelectionStart, actualSelectionEnd-actualSelectionStart+1)); |
| System.out.println('"'); |
| } |
| try { |
| this.acceptedAnswer = false; |
| CompilationResult result = new CompilationResult(sourceUnit, 1, 1, this.compilerOptions.maxProblemsPerUnit); |
| CompilationUnitDeclaration parsedUnit = |
| this.parser.dietParse(sourceUnit, result, this.actualSelectionStart, this.actualSelectionEnd); |
| |
| if (parsedUnit != null) { |
| if(DEBUG) { |
| System.out.println("SELECTION - Diet AST :"); //$NON-NLS-1$ |
| System.out.println(parsedUnit.toString()); |
| } |
| |
| // scan the package & import statements first |
| if (parsedUnit.currentPackage instanceof SelectionOnPackageReference) { |
| char[][] tokens = |
| ((SelectionOnPackageReference) parsedUnit.currentPackage).tokens; |
| this.noProposal = false; |
| this.requestor.acceptPackage(CharOperation.concatWith(tokens, '.')); |
| return; |
| } |
| ImportReference[] imports = parsedUnit.imports; |
| if (imports != null) { |
| for (int i = 0, length = imports.length; i < length; i++) { |
| ImportReference importReference = imports[i]; |
| if (importReference instanceof SelectionOnImportReference) { |
| char[][] tokens = ((SelectionOnImportReference) importReference).tokens; |
| this.noProposal = false; |
| this.requestor.acceptPackage(CharOperation.concatWith(tokens, '.')); |
| this.nameEnvironment.findTypes(CharOperation.concatWith(tokens, '.'), this); |
| |
| if(importReference.isStatic()) { |
| this.lookupEnvironment.buildTypeBindings(parsedUnit, null /*no access restriction*/); |
| if ((this.unitScope = parsedUnit.scope) != null) { |
| int tokenCount = tokens.length; |
| char[] lastToken = tokens[tokenCount - 1]; |
| char[][] qualifierTokens = CharOperation.subarray(tokens, 0, tokenCount - 1); |
| |
| Binding binding = this.unitScope.getTypeOrPackage(qualifierTokens); |
| if(binding != null && binding instanceof ReferenceBinding) { |
| ReferenceBinding ref = (ReferenceBinding) binding; |
| selectStaticFieldFromStaticImport(parsedUnit, lastToken, ref); |
| selectStaticMethodFromStaticImport(parsedUnit, lastToken, ref); |
| } |
| } |
| } |
| |
| // accept qualified types only if no unqualified type was accepted |
| if(!this.acceptedAnswer) { |
| acceptQualifiedTypes(); |
| if (!this.acceptedAnswer) { |
| this.nameEnvironment.findTypes(this.selectedIdentifier, this); |
| // try with simple type name |
| if(!this.acceptedAnswer) { |
| acceptQualifiedTypes(); |
| } |
| } |
| } |
| if(this.noProposal && this.problem != null) { |
| this.requestor.acceptError(this.problem); |
| } |
| return; |
| } |
| } |
| } |
| if (parsedUnit.types != null) { |
| if(selectDeclaration(parsedUnit)) |
| return; |
| this.lookupEnvironment.buildTypeBindings(parsedUnit, null /*no access restriction*/); |
| if ((this.unitScope = parsedUnit.scope) != null) { |
| try { |
| this.lookupEnvironment.completeTypeBindings(parsedUnit, true); |
| parsedUnit.scope.faultInTypes(); |
| ASTNode node = parseBlockStatements(parsedUnit, selectionSourceStart); |
| if(DEBUG) { |
| System.out.println("SELECTION - AST :"); //$NON-NLS-1$ |
| System.out.println(parsedUnit.toString()); |
| } |
| parsedUnit.resolve(); |
| if (node != null) { |
| selectLocalDeclaration(node); |
| } |
| } catch (SelectionNodeFound e) { |
| if (e.binding != null) { |
| if(DEBUG) { |
| System.out.println("SELECTION - Selection binding:"); //$NON-NLS-1$ |
| System.out.println(e.binding.toString()); |
| } |
| // if null then we found a problem in the selection node |
| selectFrom(e.binding, parsedUnit, e.isDeclaration); |
| } |
| } |
| } |
| } |
| } |
| // only reaches here if no selection could be derived from the parsed tree |
| // thus use the selected source and perform a textual type search |
| if (!this.acceptedAnswer) { |
| this.nameEnvironment.findTypes(this.selectedIdentifier, this); |
| |
| // accept qualified types only if no unqualified type was accepted |
| if(!this.acceptedAnswer) { |
| acceptQualifiedTypes(); |
| } |
| } |
| if(this.noProposal && this.problem != null) { |
| this.requestor.acceptError(this.problem); |
| } |
| } catch (IndexOutOfBoundsException e) { // work-around internal failure - 1GEMF6D |
| if(DEBUG) { |
| System.out.println("Exception caught by SelectionEngine:"); //$NON-NLS-1$ |
| e.printStackTrace(System.out); |
| } |
| } catch (AbortCompilation e) { // ignore this exception for now since it typically means we cannot find java.lang.Object |
| if(DEBUG) { |
| System.out.println("Exception caught by SelectionEngine:"); //$NON-NLS-1$ |
| e.printStackTrace(System.out); |
| } |
| } finally { |
| reset(); |
| } |
| } |
| |
| private void selectStaticFieldFromStaticImport(CompilationUnitDeclaration parsedUnit, char[] lastToken, ReferenceBinding ref) { |
| int fieldLength = lastToken.length; |
| FieldBinding[] fields = ref.fields(); |
| next : for (int j = 0; j < fields.length; j++) { |
| FieldBinding field = fields[j]; |
| |
| if (fieldLength > field.name.length) |
| continue next; |
| |
| if (field.isSynthetic()) |
| continue next; |
| |
| if (!field.isStatic()) |
| continue next; |
| |
| if (!CharOperation.equals(lastToken, field.name, true)) |
| continue next; |
| |
| this.selectFrom(field, parsedUnit, false); |
| } |
| } |
| |
| private void selectStaticMethodFromStaticImport(CompilationUnitDeclaration parsedUnit, char[] lastToken, ReferenceBinding ref) { |
| int methodLength = lastToken.length; |
| MethodBinding[] methods = ref.methods(); |
| next : for (int j = 0; j < methods.length; j++) { |
| MethodBinding method = methods[j]; |
| |
| if (method.isSynthetic()) continue next; |
| |
| if (method.isDefaultAbstract()) continue next; |
| |
| if (method.isConstructor()) continue next; |
| |
| if (!method.isStatic()) continue next; |
| |
| if (methodLength > method.selector.length) |
| continue next; |
| |
| if (!CharOperation.equals(lastToken, method.selector, true)) |
| continue next; |
| |
| this.selectFrom(method, parsedUnit, false); |
| } |
| } |
| |
| private void selectFrom(Binding binding, CompilationUnitDeclaration parsedUnit, boolean isDeclaration) { |
| if(binding instanceof TypeVariableBinding) { |
| TypeVariableBinding typeVariableBinding = (TypeVariableBinding) binding; |
| Binding enclosingElement = typeVariableBinding.declaringElement; |
| this.noProposal = false; |
| |
| if(enclosingElement instanceof SourceTypeBinding) { |
| SourceTypeBinding enclosingType = (SourceTypeBinding) enclosingElement; |
| this.requestor.acceptTypeParameter( |
| enclosingType.qualifiedPackageName(), |
| enclosingType.qualifiedSourceName(), |
| typeVariableBinding.sourceName(), |
| false, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } else if(enclosingElement instanceof MethodBinding) { |
| MethodBinding enclosingMethod = (MethodBinding) enclosingElement; |
| |
| this.requestor.acceptMethodTypeParameter( |
| enclosingMethod.declaringClass.qualifiedPackageName(), |
| enclosingMethod.declaringClass.qualifiedSourceName(), |
| enclosingMethod.isConstructor() |
| ? enclosingMethod.declaringClass.sourceName() |
| : enclosingMethod.selector, |
| enclosingMethod.sourceStart(), |
| enclosingMethod.sourceEnd(), |
| typeVariableBinding.sourceName(), |
| false, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedAnswer = true; |
| } else if (binding instanceof ReferenceBinding) { |
| ReferenceBinding typeBinding = (ReferenceBinding) binding; |
| |
| if(typeBinding instanceof ProblemReferenceBinding) { |
| typeBinding = ((ProblemReferenceBinding) typeBinding).original; |
| } |
| if (typeBinding == null) return; |
| if (isLocal(typeBinding) && this.requestor instanceof SelectionRequestor) { |
| this.noProposal = false; |
| ((SelectionRequestor)this.requestor).acceptLocalType(typeBinding); |
| } else { |
| this.noProposal = false; |
| |
| this.requestor.acceptType( |
| typeBinding.qualifiedPackageName(), |
| typeBinding.qualifiedSourceName(), |
| typeBinding.modifiers, |
| false, |
| typeBinding.computeUniqueKey(), |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedAnswer = true; |
| } else |
| if (binding instanceof MethodBinding) { |
| MethodBinding methodBinding = (MethodBinding) binding; |
| TypeBinding[] parameterTypes = methodBinding.original().parameters; |
| int length = parameterTypes.length; |
| char[][] parameterPackageNames = new char[length][]; |
| char[][] parameterTypeNames = new char[length][]; |
| String[] parameterSignatures = new String[length]; |
| for (int i = 0; i < length; i++) { |
| parameterPackageNames[i] = parameterTypes[i].qualifiedPackageName(); |
| parameterTypeNames[i] = parameterTypes[i].qualifiedSourceName(); |
| parameterSignatures[i] = new String(getSignature(parameterTypes[i])).replace('/', '.'); |
| } |
| this.noProposal = false; |
| ReferenceBinding declaringClass = methodBinding.declaringClass; |
| if (isLocal(declaringClass) && this.requestor instanceof SelectionRequestor) { |
| ((SelectionRequestor)this.requestor).acceptLocalMethod(methodBinding); |
| } else { |
| this.requestor.acceptMethod( |
| declaringClass.qualifiedPackageName(), |
| declaringClass.qualifiedSourceName(), |
| declaringClass.enclosingType() == null ? null : new String(getSignature(declaringClass.enclosingType())), |
| methodBinding.isConstructor() |
| ? declaringClass.sourceName() |
| : methodBinding.selector, |
| parameterPackageNames, |
| parameterTypeNames, |
| parameterSignatures, |
| methodBinding.isConstructor(), |
| isDeclaration, |
| methodBinding.computeUniqueKey(), |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedAnswer = true; |
| } else |
| if (binding instanceof FieldBinding) { |
| FieldBinding fieldBinding = (FieldBinding) binding; |
| ReferenceBinding declaringClass = fieldBinding.declaringClass; |
| if (declaringClass != null) { // arraylength |
| this.noProposal = false; |
| if (isLocal(declaringClass) && this.requestor instanceof SelectionRequestor) { |
| ((SelectionRequestor)this.requestor).acceptLocalField(fieldBinding); |
| } else { |
| this.requestor.acceptField( |
| declaringClass.qualifiedPackageName(), |
| declaringClass.qualifiedSourceName(), |
| fieldBinding.name, |
| false, |
| fieldBinding.computeUniqueKey(), |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| } |
| this.acceptedAnswer = true; |
| } |
| } else |
| if (binding instanceof LocalVariableBinding) { |
| if (this.requestor instanceof SelectionRequestor) { |
| ((SelectionRequestor)this.requestor).acceptLocalVariable((LocalVariableBinding)binding); |
| this.acceptedAnswer = true; |
| } else { |
| // open on the type of the variable |
| selectFrom(((LocalVariableBinding) binding).type, parsedUnit, false); |
| } |
| } else |
| if (binding instanceof ArrayBinding) { |
| selectFrom(((ArrayBinding) binding).leafComponentType, parsedUnit, false); |
| // open on the type of the array |
| } else |
| if (binding instanceof PackageBinding) { |
| PackageBinding packageBinding = (PackageBinding) binding; |
| this.noProposal = false; |
| this.requestor.acceptPackage(packageBinding.readableName()); |
| this.acceptedAnswer = true; |
| } else |
| if(binding instanceof BaseTypeBinding) { |
| this.acceptedAnswer = true; |
| } |
| } |
| |
| /* |
| * Checks if a local declaration got selected in this method/initializer/field. |
| */ |
| private void selectLocalDeclaration(ASTNode node) { |
| // the selected identifier is not identical to the parser one (equals but not identical), |
| // for traversing the parse tree, the parser assist identifier is necessary for identitiy checks |
| final char[] assistIdentifier = this.getParser().assistIdentifier(); |
| if (assistIdentifier == null) return; |
| |
| class Visitor extends ASTVisitor { |
| public boolean visit(ConstructorDeclaration constructorDeclaration, ClassScope scope) { |
| if (constructorDeclaration.selector == assistIdentifier){ |
| if (constructorDeclaration.binding != null) { |
| throw new SelectionNodeFound(constructorDeclaration.binding); |
| } else { |
| if (constructorDeclaration.scope != null) { |
| throw new SelectionNodeFound(new MethodBinding(constructorDeclaration.modifiers, constructorDeclaration.selector, null, null, null, constructorDeclaration.scope.referenceType().binding)); |
| } |
| } |
| } |
| return true; |
| } |
| public boolean visit(FieldDeclaration fieldDeclaration, MethodScope scope) { |
| if (fieldDeclaration.name == assistIdentifier){ |
| throw new SelectionNodeFound(fieldDeclaration.binding); |
| } |
| return true; |
| } |
| public boolean visit(TypeDeclaration localTypeDeclaration, BlockScope scope) { |
| if (localTypeDeclaration.name == assistIdentifier) { |
| throw new SelectionNodeFound(localTypeDeclaration.binding); |
| } |
| return true; |
| } |
| public boolean visit(TypeDeclaration memberTypeDeclaration, ClassScope scope) { |
| if (memberTypeDeclaration.name == assistIdentifier) { |
| throw new SelectionNodeFound(memberTypeDeclaration.binding); |
| } |
| return true; |
| } |
| public boolean visit(MethodDeclaration methodDeclaration, ClassScope scope) { |
| if (methodDeclaration.selector == assistIdentifier){ |
| if (methodDeclaration.binding != null) { |
| throw new SelectionNodeFound(methodDeclaration.binding); |
| } else { |
| if (methodDeclaration.scope != null) { |
| throw new SelectionNodeFound(new MethodBinding(methodDeclaration.modifiers, methodDeclaration.selector, null, null, null, methodDeclaration.scope.referenceType().binding)); |
| } |
| } |
| } |
| return true; |
| } |
| public boolean visit(TypeDeclaration typeDeclaration, CompilationUnitScope scope) { |
| if (typeDeclaration.name == assistIdentifier) { |
| throw new SelectionNodeFound(typeDeclaration.binding); |
| } |
| return true; |
| } |
| } |
| |
| if (node instanceof AbstractMethodDeclaration) { |
| ((AbstractMethodDeclaration)node).traverse(new Visitor(), (ClassScope)null); |
| } else { |
| ((FieldDeclaration)node).traverse(new Visitor(), (MethodScope)null); |
| } |
| } |
| |
| /** |
| * Asks the engine to compute the selection of the given type |
| * from the source type. |
| * |
| * @param sourceType org.eclipse.jdt.internal.compiler.env.ISourceType |
| * a source form of the current type in which code assist is invoked. |
| * |
| * @param typeName char[] |
| * a type name which is to be resolved in the context of a compilation unit. |
| * NOTE: the type name is supposed to be correctly reduced (no whitespaces, no unicodes left) |
| * |
| * @param topLevelTypes SourceTypeElementInfo[] |
| * a source form of the top level types of the compilation unit in which code assist is invoked. |
| |
| * @param searchInEnvironment |
| * if <code>true</code> and no selection could be found in context then search type in environment. |
| */ |
| public void selectType(ISourceType sourceType, char[] typeName, SourceTypeElementInfo[] topLevelTypes, boolean searchInEnvironment) { |
| try { |
| this.acceptedAnswer = false; |
| |
| // find the outer most type |
| ISourceType outerType = sourceType; |
| ISourceType parent = sourceType.getEnclosingType(); |
| while (parent != null) { |
| outerType = parent; |
| parent = parent.getEnclosingType(); |
| } |
| // compute parse tree for this most outer type |
| CompilationResult result = new CompilationResult(outerType.getFileName(), 1, 1, this.compilerOptions.maxProblemsPerUnit); |
| if (!(sourceType instanceof SourceTypeElementInfo)) return; |
| SourceType typeHandle = (SourceType) ((SourceTypeElementInfo)sourceType).getHandle(); |
| int flags = SourceTypeConverter.FIELD_AND_METHOD | SourceTypeConverter.MEMBER_TYPE; |
| if (typeHandle.isAnonymous() || typeHandle.isLocal()) |
| flags |= SourceTypeConverter.LOCAL_TYPE; |
| CompilationUnitDeclaration parsedUnit = |
| SourceTypeConverter.buildCompilationUnit( |
| topLevelTypes, |
| flags, |
| this.parser.problemReporter(), |
| result); |
| |
| if (parsedUnit != null && parsedUnit.types != null) { |
| if(DEBUG) { |
| System.out.println("SELECTION - Diet AST :"); //$NON-NLS-1$ |
| System.out.println(parsedUnit.toString()); |
| } |
| // find the type declaration that corresponds to the original source type |
| TypeDeclaration typeDecl = new ASTNodeFinder(parsedUnit).findType(typeHandle); |
| |
| if (typeDecl != null) { |
| |
| // add fake field with the type we're looking for |
| // note: since we didn't ask for fields above, there is no field defined yet |
| FieldDeclaration field = new FieldDeclaration(); |
| int dot; |
| if ((dot = CharOperation.lastIndexOf('.', typeName)) == -1) { |
| this.selectedIdentifier = typeName; |
| field.type = new SelectionOnSingleTypeReference(typeName, -1); |
| // position not used |
| } else { |
| char[][] previousIdentifiers = CharOperation.splitOn('.', typeName, 0, dot); |
| char[] selectionIdentifier = |
| CharOperation.subarray(typeName, dot + 1, typeName.length); |
| this.selectedIdentifier = selectionIdentifier; |
| field.type = |
| new SelectionOnQualifiedTypeReference( |
| previousIdentifiers, |
| selectionIdentifier, |
| new long[previousIdentifiers.length + 1]); |
| } |
| field.name = "<fakeField>".toCharArray(); //$NON-NLS-1$ |
| typeDecl.fields = new FieldDeclaration[] { field }; |
| |
| // build bindings |
| this.lookupEnvironment.buildTypeBindings(parsedUnit, null /*no access restriction*/); |
| if ((this.unitScope = parsedUnit.scope) != null) { |
| try { |
| // build fields |
| // note: this builds fields only in the parsed unit (the buildFieldsAndMethods flag is not passed along) |
| this.lookupEnvironment.completeTypeBindings(parsedUnit, true); |
| |
| // resolve |
| parsedUnit.scope.faultInTypes(); |
| parsedUnit.resolve(); |
| } catch (SelectionNodeFound e) { |
| if (e.binding != null) { |
| if(DEBUG) { |
| System.out.println("SELECTION - Selection binding :"); //$NON-NLS-1$ |
| System.out.println(e.binding.toString()); |
| } |
| // if null then we found a problem in the selection node |
| selectFrom(e.binding, parsedUnit, e.isDeclaration); |
| } |
| } |
| } |
| } |
| } |
| // only reaches here if no selection could be derived from the parsed tree |
| // thus use the selected source and perform a textual type search |
| if (!this.acceptedAnswer && searchInEnvironment) { |
| if (this.selectedIdentifier != null) { |
| this.nameEnvironment.findTypes(typeName, this); |
| |
| // accept qualified types only if no unqualified type was accepted |
| if(!this.acceptedAnswer) { |
| acceptQualifiedTypes(); |
| } |
| } |
| } |
| if(this.noProposal && this.problem != null) { |
| this.requestor.acceptError(this.problem); |
| } |
| } catch (AbortCompilation e) { // ignore this exception for now since it typically means we cannot find java.lang.Object |
| } finally { |
| reset(); |
| } |
| } |
| |
| // Check if a declaration got selected in this unit |
| private boolean selectDeclaration(CompilationUnitDeclaration compilationUnit){ |
| |
| // the selected identifier is not identical to the parser one (equals but not identical), |
| // for traversing the parse tree, the parser assist identifier is necessary for identitiy checks |
| char[] assistIdentifier = this.getParser().assistIdentifier(); |
| if (assistIdentifier == null) return false; |
| |
| ImportReference currentPackage = compilationUnit.currentPackage; |
| char[] packageName = currentPackage == null ? new char[0] : CharOperation.concatWith(currentPackage.tokens, '.'); |
| // iterate over the types |
| TypeDeclaration[] types = compilationUnit.types; |
| for (int i = 0, length = types == null ? 0 : types.length; i < length; i++){ |
| if(selectDeclaration(types[i], assistIdentifier, packageName)) |
| return true; |
| } |
| return false; |
| } |
| |
| // Check if a declaration got selected in this type |
| private boolean selectDeclaration(TypeDeclaration typeDeclaration, char[] assistIdentifier, char[] packageName){ |
| |
| if (typeDeclaration.name == assistIdentifier){ |
| char[] qualifiedSourceName = null; |
| |
| TypeDeclaration enclosingType = typeDeclaration; |
| while(enclosingType != null) { |
| qualifiedSourceName = CharOperation.concat(enclosingType.name, qualifiedSourceName, '.'); |
| enclosingType = enclosingType.enclosingType; |
| } |
| char[] uniqueKey = typeDeclaration.binding != null ? typeDeclaration.binding.computeUniqueKey() : null; |
| |
| this.requestor.acceptType( |
| packageName, |
| qualifiedSourceName, |
| typeDeclaration.modifiers, |
| true, |
| uniqueKey, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| |
| this.noProposal = false; |
| return true; |
| } |
| TypeDeclaration[] memberTypes = typeDeclaration.memberTypes; |
| for (int i = 0, length = memberTypes == null ? 0 : memberTypes.length; i < length; i++){ |
| if(selectDeclaration(memberTypes[i], assistIdentifier, packageName)) |
| return true; |
| } |
| FieldDeclaration[] fields = typeDeclaration.fields; |
| for (int i = 0, length = fields == null ? 0 : fields.length; i < length; i++){ |
| if (fields[i].name == assistIdentifier){ |
| char[] qualifiedSourceName = null; |
| |
| TypeDeclaration enclosingType = typeDeclaration; |
| while(enclosingType != null) { |
| qualifiedSourceName = CharOperation.concat(enclosingType.name, qualifiedSourceName, '.'); |
| enclosingType = enclosingType.enclosingType; |
| } |
| FieldDeclaration field = fields[i]; |
| this.requestor.acceptField( |
| packageName, |
| qualifiedSourceName, |
| field.name, |
| true, |
| field.binding != null ? field.binding.computeUniqueKey() : null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| |
| this.noProposal = false; |
| return true; |
| } |
| } |
| AbstractMethodDeclaration[] methods = typeDeclaration.methods; |
| for (int i = 0, length = methods == null ? 0 : methods.length; i < length; i++){ |
| AbstractMethodDeclaration method = methods[i]; |
| |
| if (method.selector == assistIdentifier){ |
| char[] qualifiedSourceName = null; |
| |
| TypeDeclaration enclosingType = typeDeclaration; |
| while(enclosingType != null) { |
| qualifiedSourceName = CharOperation.concat(enclosingType.name, qualifiedSourceName, '.'); |
| enclosingType = enclosingType.enclosingType; |
| } |
| |
| this.requestor.acceptMethod( |
| packageName, |
| qualifiedSourceName, |
| null, // SelectionRequestor does not need of declaring type signature for method declaration |
| method.selector, |
| null, // SelectionRequestor does not need of parameters type for method declaration |
| null, // SelectionRequestor does not need of parameters type for method declaration |
| null, // SelectionRequestor does not need of parameters type for method declaration |
| method.isConstructor(), |
| true, |
| method.binding != null ? method.binding.computeUniqueKey() : null, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| |
| this.noProposal = false; |
| return true; |
| } |
| |
| TypeParameter[] methodTypeParameters = method.typeParameters(); |
| for (int j = 0, length2 = methodTypeParameters == null ? 0 : methodTypeParameters.length; j < length2; j++){ |
| TypeParameter methodTypeParameter = methodTypeParameters[j]; |
| |
| if(methodTypeParameter.name == assistIdentifier) { |
| char[] qualifiedSourceName = null; |
| |
| TypeDeclaration enclosingType = typeDeclaration; |
| while(enclosingType != null) { |
| qualifiedSourceName = CharOperation.concat(enclosingType.name, qualifiedSourceName, '.'); |
| enclosingType = enclosingType.enclosingType; |
| } |
| |
| this.requestor.acceptMethodTypeParameter( |
| packageName, |
| qualifiedSourceName, |
| method.selector, |
| method.sourceStart, |
| method.sourceEnd, |
| methodTypeParameter.name, |
| true, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| |
| this.noProposal = false; |
| return true; |
| } |
| } |
| } |
| |
| TypeParameter[] typeParameters = typeDeclaration.typeParameters; |
| for (int i = 0, length = typeParameters == null ? 0 : typeParameters.length; i < length; i++){ |
| TypeParameter typeParameter = typeParameters[i]; |
| if(typeParameter.name == assistIdentifier) { |
| char[] qualifiedSourceName = null; |
| |
| TypeDeclaration enclosingType = typeDeclaration; |
| while(enclosingType != null) { |
| qualifiedSourceName = CharOperation.concat(enclosingType.name, qualifiedSourceName, '.'); |
| enclosingType = enclosingType.enclosingType; |
| } |
| |
| this.requestor.acceptTypeParameter( |
| packageName, |
| qualifiedSourceName, |
| typeParameter.name, |
| true, |
| this.actualSelectionStart, |
| this.actualSelectionEnd); |
| |
| this.noProposal = false; |
| return true; |
| } |
| } |
| |
| return false; |
| } |
| } |