
com.globalmentor.java.model.ModelElements Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of globalmentor-java-model Show documentation
Show all versions of globalmentor-java-model Show documentation
GlobalMentor Java library for working with the Java model and facilitating annotation processing.
The newest version!
/*
* Copyright © 2023 GlobalMentor, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.globalmentor.java.model;
import static com.globalmentor.java.Conditions.*;
import static com.globalmentor.java.Objects.*;
import java.lang.annotation.Annotation;
import java.util.Optional;
import java.util.stream.Stream;
import javax.annotation.*;
import javax.lang.model.element.*;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.util.*;
/**
* Utilities for working with Java model {@link Element} and related classes.
* @author Garret Wilson
* @see Elements
* @see Types
* @see ModelTypes
*/
public class ModelElements {
/**
* Finds a type element from a class if the type element is uniquely determinable in the environment.
* @implSpec This implementation delegates to {@link #findTypeElementForCanonicalName(Elements, CharSequence)}.
* @param elements The element utilities.
* @param clazz The class for which a type element is to be found.
* @return The type element for the class, which will not be present if no type element can be uniquely determined.
* @see javax.annotation.processing.ProcessingEnvironment#getElementUtils()
* @see Class#getCanonicalName()
*/
public static Optional findTypeElementForClass(@Nonnull final Elements elements, @Nonnull final Class> clazz) { //TODO create module-related variations as well
return findTypeElementForCanonicalName(elements, clazz.getCanonicalName());
}
/**
* Finds a type element given its canonical name if the type element is uniquely determinable in the environment.
* @apiNote This method functions identically to {@link Elements#getTypeElement(CharSequence)} except that it returns an {@link Optional} and never
* null
.
* @implSpec This implementation delegates to {@link Elements#getTypeElement(CharSequence)}.
* @param elements The element utilities.
* @param canonicalName The canonical name of the type element to return.
* @return The named type element, which will not be present if no type element can be uniquely determined.
* @see javax.annotation.processing.ProcessingEnvironment#getElementUtils()
*/
public static Optional findTypeElementForCanonicalName(@Nonnull final Elements elements, @Nonnull final CharSequence canonicalName) { //TODO create module-related variations as well
return Optional.ofNullable(elements.getTypeElement(canonicalName));
}
/**
* Retrieves an annotation mirror from a type element for annotations of a particular type.
* @implSpec This implementation does not check for repeated annotations.
* @param typeElement The type element representing the type potentially annotated with the specified annotation.
* @param annotationClass The type of annotation to find.
* @return The mirrors for the annotation annotating the indicated type, if any.
*/
public static Optional extends AnnotationMirror> findElementAnnotationMirrorForClass(@Nonnull final TypeElement typeElement,
@Nonnull final Class extends Annotation> annotationClass) {
return elementEnnotationMirrorsForClass(typeElement, annotationClass).findAny();
}
/**
* Retrieves all the annotation mirrors from a type element for annotations of a particular type.
* @param typeElement The type element representing the type potentially annotated with the specified annotation.
* @param annotationClass The type of annotation to find.
* @return The mirrors for the annotation(s) annotating the indicated type, if any.
*/
public static Stream extends AnnotationMirror> elementEnnotationMirrorsForClass(@Nonnull final TypeElement typeElement,
@Nonnull final Class extends Annotation> annotationClass) {
final String canonicalName = annotationClass.getCanonicalName();
checkArgument(canonicalName != null, "Annotation class `%s` has no canonical name.", annotationClass.getName()); //check for completeness; not realistically possible: an annotation cannot be defined as an anonymous inner class
return typeElement.getAnnotationMirrors().stream().filter(annotationMirror -> {
final Element annotationElement = annotationMirror.getAnnotationType().asElement();
assert annotationElement instanceof TypeElement : "An annotation mirror type's element should always be a `TypeElement`.";
return ((TypeElement)annotationElement).getQualifiedName().contentEquals(canonicalName);
});
}
/**
* Returns all interfaces of a type element annotated with the given annotation.
* @param types The type utilities.
* @param typeElement The type element representing the type potentially having an interface annotated with the specified annotation.
* @param annotationClass The type of annotation to look for.
* @return The interfaces of the type element which are in turn annotated with the given annotation.
*/
public static Stream elementInterfacesAnnotatedWith(@Nonnull final Types types, @Nonnull final TypeElement typeElement,
@Nonnull final Class extends Annotation> annotationClass) {
return typeElement.getInterfaces().stream().flatMap(asInstances(DeclaredType.class))
.filter(interfaceType -> findElementAnnotationMirrorForClass((TypeElement)interfaceType.asElement(), annotationClass).isPresent());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy