org.eclipse.jdt.core.BindingKey Maven / Gradle / Ivy
Show all versions of spotless-ext-greclipse Show documentation
/*******************************************************************************
* Copyright (c) 2005, 2011 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.core;
import org.eclipse.jdt.internal.compiler.lookup.TypeConstants;
import org.eclipse.jdt.internal.core.util.KeyKind;
import org.eclipse.jdt.internal.core.util.KeyToSignature;
/**
* Utility class to decode or create a binding key.
*
* This class is not intended to be subclassed by clients.
*
*
* @see org.eclipse.jdt.core.dom.IBinding#getKey()
* @since 3.1
*/
public final class BindingKey {
private String key;
/**
* Creates a new binding key.
*
* @param key the key to decode
*/
public BindingKey(String key) {
this.key = key;
}
/**
* Creates a new array type binding key from the given type binding key and the given array dimension.
*
* For example:
*
*
* createArrayTypeBindingKey("Ljava/lang/Object;", 1) -> "[Ljava/lang/Object;"
* createArrayTypeBindingKey("I", 2) -> "[[I"
*
*
*
*
* @param typeKey the binding key of the given type
* @param arrayDimension the given array dimension
* @return a new array type binding key
*/
public static String createArrayTypeBindingKey(String typeKey, int arrayDimension) {
// Note this implementation is heavily dependent on ArrayTypeBinding#computeUniqueKey()
StringBuffer buffer = new StringBuffer();
while (arrayDimension-- > 0)
buffer.append('[');
buffer.append(typeKey);
return buffer.toString();
}
/**
* Creates a new parameterized type binding key from the given generic type binding key and the given argument type binding keys.
* If the argument type keys array is empty, then a raw type binding key is created.
*
* For example:
*
*
* createParameterizedTypeBindingKey(
* "Ljava/util/Map<TK;TV;>;",
* new String[] {"Ljava/lang/String;", "Ljava/lang/Object;"}) ->
* "Ljava/util/Map<Ljava/lang/String;Ljava/lang/Object;>;"
* createParameterizedTypeBindingKey(
* "Ljava/util/List<TE;>;", new String[] {}) ->
* "Ljava/util/List<>;"
*
*
*
*
* @param genericTypeKey the binding key of the generic type
* @param argumentTypeKeys the possibly empty list of binding keys of argument types
* @return a new parameterized type binding key
*/
public static String createParameterizedTypeBindingKey(String genericTypeKey, String[] argumentTypeKeys) {
// Note this implementation is heavily dependent on ParameterizedTypeBinding#computeUniqueKey() and its subclasses
StringBuffer buffer = new StringBuffer();
buffer.append(Signature.getTypeErasure(genericTypeKey));
buffer.insert(buffer.length()-1, '<');
for (int i = 0, length = argumentTypeKeys.length; i < length; i++) {
String argumentTypeKey = argumentTypeKeys[i];
buffer.insert(buffer.length()-1, argumentTypeKey);
}
buffer.insert(buffer.length()-1, '>');
return buffer.toString();
}
/**
* Creates a new type binding key from the given type name. The type name must be either
* a fully qualified name, an array type name or a primitive type name.
* If the type name is fully qualified, then it is expected to be dot-based.
* Note that inner types, generic types and parameterized types are not supported.
*
* For example:
*
*
* createTypeBindingKey("int") -> "I"
* createTypeBindingKey("java.lang.String") -> "Ljava/lang/String;"
* createTypeBindingKey("boolean[]") -> "[Z"
*
*
*
*
* @param typeName the possibly qualified type name
* @return a new type binding key
*/
public static String createTypeBindingKey(String typeName) {
// Note this implementation is heavily dependent on TypeBinding#computeUniqueKey() and its subclasses
return Signature.createTypeSignature(typeName.replace('.', '/'), true/*resolved*/);
}
/**
* Creates a new type variable binding key from the given type variable name and the given declaring key.
* The declaring key can either be a type binding key or a method binding key.
*
* For example:
*
*
* createTypeVariableBindingKey("T", "Ljava/util/List<TE;>;") ->
* "Ljava/util/List<TE;>;:TT;"
* createTypeVariableBindingKey("SomeTypeVariable", "Lp/X;.foo()V") ->
* "Lp/X;.foo()V:TSomeTypeVariable;"
*
*
*
*
* @param typeVariableName the name of the given type variable
* @param declaringKey the binding key of the type or method the type variable belongs to
* @return a new type variable binding key
*/
public static String createTypeVariableBindingKey(String typeVariableName, String declaringKey) {
// Note this implementation is heavily dependent on TypeVariableBinding#computeUniqueKey()
StringBuffer buffer = new StringBuffer();
buffer.append(declaringKey);
buffer.append(':');
buffer.append('T');
buffer.append(typeVariableName);
buffer.append(';');
return buffer.toString();
}
/**
* Creates a new wildcard type binding key from the given type binding key and the given wildcard kind
* (one of {@link Signature#C_STAR}, {@link Signature#C_SUPER}, or {@link Signature#C_EXTENDS}.
* If the wildcard is {@link Signature#C_STAR}, the given type binding key is ignored.
*
* For example:
*
*
* createWilcardTypeBindingKey(null, Signature.C_STAR) -> "*"
* createWilcardTypeBindingKey("Ljava/util/List<TE;>;",
* Signature.C_SUPER) -> "-Ljava/util/List<TE;>;"
* createWilcardTypeBindingKey("Ljava/util/ArrayList;", Signature.C_EXTENDS) ->
* "+Ljava/util/ArrayList;"
*
*
*
*
* @param typeKey the binding key of the given type
* @param kind one of {@link Signature#C_STAR}, {@link Signature#C_SUPER}, or {@link Signature#C_EXTENDS}
* @return a new wildcard type binding key
* @deprecated This method is missing crucial information necessary for proper wildcard binding key creation.
* @see org.eclipse.jdt.core.BindingKey#createWildcardTypeBindingKey(String, char, String, int)
*/
public static String createWilcardTypeBindingKey(String typeKey, char kind) {
// Note this implementation is supposed to closely follow the behavior in WildcardBinding#computeUniqueKey()
// but it doesn't and hence the deprecation. See https://bugs.eclipse.org/bugs/show_bug.cgi?id=234609
switch (kind) {
case Signature.C_STAR:
return "*"; //$NON-NLS-1$
case Signature.C_SUPER:
return '-' + typeKey;
case Signature.C_EXTENDS:
return '+' + typeKey;
}
return null;
}
/**
* Creates a new wildcard type binding key from the given generic type binding key, the given wildcard
* kind (one of {@link Signature#C_STAR}, {@link Signature#C_SUPER}, or {@link Signature#C_EXTENDS}
* the given bound type binding key and the given rank. If the wildcard kind is {@link Signature#C_STAR},
* the given bound type binding key is ignored.
*
* For example:
*
*
* createWildcardTypeBindingKey("Ljava/util/ArrayList;", Signature.C_STAR, null, 0) -> "Ljava/util/ArrayList;{0}*"
* createWildcardTypeBindingKey("Ljava/util/ArrayList;", Signature.C_SUPER, "Ljava/lang/String;", 0) -> "Ljava/util/ArrayList;{0}-Ljava/lang/String;"
* createWildcardTypeBindingKey("Ljava/util/HashMap;", Signature.C_EXTENDS, "Ljava/lang/String;", 1) ->
* "Ljava/util/HashMap;{1}+Ljava/lang/String;"
*
*
*
*
* @param genericTypeKey the binding key of the generic type
* @param boundKind one of {@link Signature#C_STAR}, {@link Signature#C_SUPER}, or {@link Signature#C_EXTENDS}
* @param boundTypeKey the binding key of the bounding type.
* @param rank the relative position of this wild card type in the parameterization of the generic type.
* @return a new wildcard type binding key
* @since 3.5
*/
public static String createWildcardTypeBindingKey(String genericTypeKey, char boundKind, String boundTypeKey, int rank) {
// Note this implementation is heavily dependent on WildcardBinding#computeUniqueKey()
String wildCardKey;
switch (boundKind) {
case Signature.C_STAR:
wildCardKey = new String(TypeConstants.WILDCARD_STAR);
break;
case Signature.C_SUPER:
wildCardKey = new String(TypeConstants.WILDCARD_MINUS) + boundTypeKey;
break;
case Signature.C_EXTENDS:
wildCardKey = new String(TypeConstants.WILDCARD_PLUS) + boundTypeKey;
break;
default:
return null;
}
return genericTypeKey + '{' + rank + '}' + wildCardKey;
}
/**
* Returns the binding key of the declaring type of the element represented by this binding key. If the binding key
* does not represent a member or if the member doesn't have a declaring type, returns null
.
*
*
* Note that only binding keys for references to methods and fields
* are fully supported. The binding keys for declarations will not have type parameters.
*
* @return the type binding key or null
* @since 3.7.1
*/
public BindingKey getDeclaringType() {
int end = this.key.lastIndexOf(Signature.C_DOT);
if (end == -1) {
end = this.key.lastIndexOf(Signature.C_DOLLAR); // for inner types
if (end == -1) return null;
}
KeyKind kind = new KeyKind(this.key);
kind.parse();
if ((kind.flags & KeyKind.F_LOCAL_VAR) != 0) {
// declaring type for locals doesn't make sense, hence return null.
return null;
}
String typeKey = this.key.substring(0, end);
if (typeKey.charAt(typeKey.length()-1) != Signature.C_SEMICOLON) {
typeKey += Signature.C_SEMICOLON;
}
return new BindingKey(typeKey);
}
/**
* Returns the thrown exception signatures of the element represented by this binding key.
* If this binding key does not represent a method or does not throw any exception,
* returns an empty array.
*
* @return the thrown exceptions signatures
* @since 3.3
*/
public String[] getThrownExceptions() {
KeyToSignature keyToSignature = new KeyToSignature(this.key, KeyToSignature.THROWN_EXCEPTIONS);
keyToSignature.parse();
return keyToSignature.getThrownExceptions();
}
/**
* Returns the type argument signatures of the element represented by this binding key.
* If this binding key doesn't represent a parameterized type or a parameterized method,
* returns an empty array.
*
* @return the type argument signatures
*/
public String[] getTypeArguments() {
KeyToSignature keyToSignature = new KeyToSignature(this.key, KeyToSignature.TYPE_ARGUMENTS);
keyToSignature.parse();
return keyToSignature.getTypeArguments();
}
/**
* Returns whether this binding key represents a raw type.
*
* @return whether this binding key represents a raw type
*/
public boolean isRawType() {
KeyKind kind = new KeyKind(this.key);
kind.parse();
return (kind.flags & KeyKind.F_RAW_TYPE) != 0;
}
/**
* Returns whether this binding key represents a parameterized type, or if its declaring type is a parameterized type.
*
* @return whether this binding key represents a parameterized type
*/
public boolean isParameterizedType() {
KeyKind kind = new KeyKind(this.key);
kind.parse();
return (kind.flags & KeyKind.F_PARAMETERIZED_TYPE) != 0;
}
/**
* Returns whether this binding key represents a parameterized method, or if its declaring method is a parameterized method.
*
* @return whether this binding key represents a parameterized method
*/
public boolean isParameterizedMethod() {
KeyKind kind = new KeyKind(this.key);
kind.parse();
return (kind.flags & KeyKind.F_PARAMETERIZED_METHOD) != 0;
}
/**
* Transforms this binding key into a resolved signature.
* If this binding key represents a field, the returned signature is
* the field type's signature.
*
* @return the resolved signature for this binding key
* @see Signature
* @since 3.2
*/
public String toSignature() {
KeyToSignature keyToSignature = new KeyToSignature(this.key, KeyToSignature.SIGNATURE);
keyToSignature.parse();
return keyToSignature.signature.toString();
}
/* (non-Javadoc)
* @see java.lang.Object#toString()
*/
public String toString() {
return this.key;
}
}