spoon.reflect.declaration.CtElement Maven / Gradle / Ivy
/*
* SPDX-License-Identifier: (MIT OR CECILL-C)
*
* Copyright (C) 2006-2019 INRIA and contributors
*
* Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
*/
package spoon.reflect.declaration;
import spoon.compiler.Environment;
import spoon.processing.FactoryAccessor;
import spoon.reflect.code.CtComment;
import spoon.reflect.cu.SourcePosition;
import spoon.reflect.cu.SourcePositionHolder;
import spoon.reflect.path.CtPath;
import spoon.reflect.path.CtRole;
import spoon.reflect.reference.CtTypeReference;
import spoon.reflect.visitor.CtVisitable;
import spoon.reflect.visitor.Filter;
import spoon.reflect.visitor.Root;
import spoon.reflect.visitor.chain.CtQueryable;
import spoon.support.DerivedProperty;
import spoon.reflect.annotations.PropertyGetter;
import spoon.reflect.annotations.PropertySetter;
import spoon.support.Experimental;
import spoon.support.sniper.internal.ElementSourceFragment;
import java.io.Serializable;
import java.lang.annotation.Annotation;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import static spoon.reflect.path.CtRole.ANNOTATION;
import static spoon.reflect.path.CtRole.COMMENT;
import static spoon.reflect.path.CtRole.IS_IMPLICIT;
import static spoon.reflect.path.CtRole.POSITION;
/**
* This interface is the root interface for the metamodel elements (any program
* element).
*/
@Root
public interface CtElement extends FactoryAccessor, CtVisitable, Cloneable, CtQueryable, Serializable, SourcePositionHolder {
/**
* Searches for an annotation of the given class that annotates the
* current element.
*
* When used with a shadow element, this method might return an empty list even on an annotated element
* because annotations without a RUNTIME retention policy are lost after compilation.
*
* WARNING: this method uses a class loader proxy, which is costly.
* Use {@link #getAnnotation(CtTypeReference)} preferably.
*
*
* NOTE: before using an annotation proxy, you have to make sure that all
* the types referenced by the annotation have been compiled and are in the
* classpath so that accessed values can be converted into the actual types.
*
* @param
* the annotation's type
* @param annotationType
* the annotation's class
* @return if found, returns a proxy for this annotation
*/
@PropertyGetter(role = ANNOTATION)
A getAnnotation(Class annotationType);
/**
* Gets the annotation element for a given annotation type.
*
* When used with a shadow element, this method might return an empty list even on an annotated element
* because annotations without a RUNTIME retention policy are lost after compilation.
*
* @param annotationType
* the annotation type
* @return the annotation if this element is annotated by one annotation of
* the given type
*/
@PropertyGetter(role = ANNOTATION)
CtAnnotation getAnnotation(
CtTypeReference annotationType);
/**
* @return true if the element is annotated by the given annotation type.
*
* @param annotationType
* the annotation type
*/
@DerivedProperty
boolean hasAnnotation(Class annotationType);
/**
* Returns the annotations that are present on this element.
*
* For sake of encapsulation, the returned list is unmodifiable.
*/
@PropertyGetter(role = ANNOTATION)
List> getAnnotations();
/**
* Returns the text of the documentation ("javadoc") comment of this
* element. It contains the text of Javadoc together with the tags.
*
* If one only wants only the text without the tag, one can call `getComments().get(0).getContent()`
*
* If one wants to analyze the tags, one can call `getComments().get(0).asJavaDoc().getTags()`
*
* See also {@link #getComments()}.and {@link spoon.reflect.code.CtJavaDoc}
*/
@DerivedProperty
String getDocComment();
/**
* Build a short representation of any element.
*/
@DerivedProperty
String getShortRepresentation();
/**
* Gets the position of this element in input source files
*
* @return Source file and line number of this element.
* It never returns null. Use {@link SourcePosition#isValidPosition()}
* to detect whether return instance contains start/end indexes.
*/
@PropertyGetter(role = POSITION)
@Override
SourcePosition getPosition();
/**
* Replaces this element by another one.
*/
void replace(CtElement element);
/**
* Replaces this element by several elements.
* If `elements` contains one single element, it is equivalent to {@link #replace(CtElement)}.
* If `elements` is empty, it is equivalent to {@link #delete()}.
*/
void replace(Collection elements);
/**
* Add an annotation for this element
*
* @param annotation
* @return true if this element changed as a result of the call
*/
@PropertySetter(role = ANNOTATION)
E addAnnotation(CtAnnotation extends Annotation> annotation);
/**
* Remove an annotation for this element
*
* @param annotation
* @return true if this element changed as a result of the call
*/
@PropertySetter(role = ANNOTATION)
boolean removeAnnotation(CtAnnotation extends Annotation> annotation);
/**
* Sets the text of the documentation ("javadoc") comment of this
* declaration. This API will set the content of the first javadoc
* {@link CtComment} or create a new javadoc {@link CtComment} if
* no javadoc {@link CtComment} is available on this object.
*/
E setDocComment(String docComment);
/**
* Sets the position in the Java source file. Note that this information is
* used to feed the line numbers in the generated bytecode if any (which is
* useful for debugging).
*
* @param position
* of this element in the input source files
*/
@PropertySetter(role = POSITION)
E setPosition(SourcePosition position);
/**
* Gets the child elements annotated with the given annotation type's
* instances.
*
* @param
* the element's type
* @param annotationType
* the annotation type
* @return all the child elements annotated with an instance of the given
* annotation type
*/
List getAnnotatedChildren(
Class extends Annotation> annotationType);
/**
* Returns true if this element is not present in the code (automatically added by the
* Java compiler or inferred when the model is built).
* Consequently, implicit elements are not pretty-printed and have no position.
*/
@PropertyGetter(role = IS_IMPLICIT)
boolean isImplicit();
/**
* Sets this element to be implicit.
*/
@PropertySetter(role = IS_IMPLICIT)
E setImplicit(boolean b);
/**
* Calculates and returns the set of all the types referenced by this
* element (and sub-elements in the AST).
*/
@DerivedProperty
Set> getReferencedTypes();
/**
* Returns all the children elements recursively matching the filter.
* If the receiver (this) matches the filter, it is also returned
*/
List getElements(Filter filter);
/**
* Sets the position of this element and all its children element. Note that
* this information is used to feed the line numbers in the generated
* bytecode if any (which is useful for debugging).
*
* @param position
* of this element and all children in the input source file
*/
E setPositions(SourcePosition position);
/**
* Sets the annotations for this element.
*/
@PropertySetter(role = ANNOTATION)
E setAnnotations(List> annotation);
/**
* Gets the parent of current reference.
*
* @throws ParentNotInitializedException
* when the parent of this element is not initialized
*/
@DerivedProperty
CtElement getParent() throws ParentNotInitializedException;
/**
* Gets the first parent that matches the given type.
*
* @return the nearest matching parent; null if no match is found or this element has no parent
*/
P getParent(Class
parentType);
/**
* Gets the first parent that matches the filter.
*
* @return the nearest matching parent; null if no match is found or this element has no parent
*/
E getParent(Filter filter);
/**
* Manually sets the parent element of the current element.
*
* @param parent parent reference.
* @param this element's type
* @return this element
*/
E setParent(CtElement parent);
/**
* Tells if this parent has been initialized.
*/
boolean isParentInitialized();
/**
* Tells if the given element is a direct or indirect parent.
*/
boolean hasParent(CtElement candidate);
/**
* Calculates and sets all the parents below this element. This function can
* be called to check and fix parents after manipulating the model.
*/
void updateAllParentsBelow();
/**
* @return the {@link CtRole} of the parent's attribute where this element is used.
* It returns the primary role. For example ((CtMethod) method).getRoleInParent() returns {@link CtRole#TYPE_MEMBER}.
*
* Returns null if parent doesn't contain this element as direct children or if this element has no parent.
*/
CtRole getRoleInParent();
/**
* Deletes the element. For instance, delete a statement from its containing block. Warning: it may result in an incorrect AST, use at your own risk.
*/
void delete();
/**
* Saves a bunch of metadata inside an Element
*/
E setAllMetadata(Map metadata);
/**
* Saves metadata inside an Element.
*/
E putMetadata(String key, Object val);
/**
* Retrieves metadata stored in an element. Returns null if it does not exist.
*/
Object getMetadata(String key);
/**
* Retrieves all metadata stored in an element.
*/
Map getAllMetadata();
/**
* Returns the metadata keys stored in an element.
*/
Set getMetadataKeys();
/**
* Set the comment list
*/
@PropertySetter(role = COMMENT)
E setComments(List comments);
/**
* The list of comments
* @return the list of comment
*/
@PropertyGetter(role = COMMENT)
List getComments();
/**
* Add a comment to the current element
* element.addComment(element.getFactory().Code().createComment("comment", CtComment.CommentType.INLINE)
* @param comment the comment
*/
@PropertySetter(role = COMMENT)
E addComment(CtComment comment);
/**
* Remove a comment
* @param comment the comment to remove
*/
@PropertySetter(role = COMMENT)
E removeComment(CtComment comment);
/**
* Clone the element which calls this method in a new object.
*
* Note that that references are kept as is, and thus, so if you clone whole classes
* or methods, some parts of the cloned element (eg executable references) may still point to the initial element.
* In this case, consider using methods {@link spoon.refactoring.Refactoring#copyType(CtType)} and {@link spoon.refactoring.Refactoring#copyMethod(CtMethod)} instead which does additional work beyond cloning.
*/
CtElement clone();
/**
* @return a a single value (eg a CtElement), List, Set or Map depending on this `element` and `role`. Returned collections are read-only.
* @param role the role of the returned attribute with respect to this element.
*
* For instance, "klass.getValueByRole(CtRole.METHOD)" returns a list of methods.
*
* See {@link spoon.reflect.meta.impl.RoleHandlerHelper} for more advanced methods.
*/
T getValueByRole(CtRole role);
/**
* Sets a field according to a role.
* @param role the role of the field to be set
* @param value to be assigned to this field.
*/
E setValueByRole(CtRole role, T value);
/**
* Return the path from the model root to this CtElement, eg `.spoon.test.path.Foo.foo#body#statement[index=0]`
*/
CtPath getPath();
/**
* Returns an iterator over this CtElement's descendants.
* @return An iterator over this CtElement's descendants.
*/
Iterator descendantIterator();
/**
* Returns an Iterable instance of this CtElement, allowing for dfs traversal of its descendants.
* @return an Iterable object that allows iterating through this CtElement's descendants.
*/
Iterable asIterable();
/**
* @return a list of CtElement containing the element's direct children.
*/
List getDirectChildren();
/**
* @return the source code of this element according to the setting of {@link Environment#getPrettyPrintingMode()}.
*
*/
String toString();
/**
* @return the most straightforward and explicit version of this element.
*/
String toStringDebug();
/**
* @return the source code of this element with the pretty-printing rules of Spoon
* Warning: this is not side-effect free, this triggers some {@link spoon.reflect.visitor.ImportAnalyzer} which would change the model: add/remove imports, change the value `implicit` of some model elements, etc.
*/
@Experimental
String prettyprint();
// overriding the return type
@Override
ElementSourceFragment getOriginalSourceFragment();
}