All Downloads are FREE. Search and download functionalities are using the official Maven repository.

tech.deplant.javapoet.ParameterSpec Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2015 Square, 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
 *
 * http://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 tech.deplant.javapoet;

import javax.lang.model.SourceVersion;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.VariableElement;
import java.io.IOException;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Set;

/**
 * A generated parameter declaration.
 */
public final class ParameterSpec {
	public final String name;
	public final List annotations;
	public final Set modifiers;
	public final TypeName type;
	public final CodeBlock javadoc;

	private ParameterSpec(Builder builder) {
		this.name = Util.checkNotNull(builder.name, "parameterName == null");
		this.annotations = Util.immutableList(builder.annotations);
		this.modifiers = Util.immutableSet(builder.modifiers);
		this.type = Util.checkNotNull(builder.type, "type == null");
		this.javadoc = builder.javadoc.build();
	}

	public static ParameterSpec get(VariableElement element) {
		Util.checkArgument(element.getKind().equals(ElementKind.PARAMETER), "element is not a parameter");

		TypeName type = TypeName.get(element.asType());
		String name = element.getSimpleName().toString();
		// Copying parameter annotations can be incorrect so we're deliberately not including them.
		// See https://github.com/square/javapoet/issues/482.
		return ParameterSpec.builder(type, name)
		                    .addModifiers(element.getModifiers())
		                    .build();
	}

	static List parametersOf(ExecutableElement method) {
		List result = new ArrayList<>();
		for (VariableElement parameter : method.getParameters()) {
			result.add(ParameterSpec.get(parameter));
		}
		return result;
	}

	private static boolean isValidParameterName(String name) {
		// Allow "this" for explicit receiver parameters
		// See https://docs.oracle.com/javase/specs/jls/se8/html/jls-8.html#jls-8.4.1.
		if (name.endsWith(".this")) {
			return SourceVersion.isIdentifier(name.substring(0, name.length() - ".this".length()));
		}
		return name.equals("this") || SourceVersion.isName(name);
	}

	public static Builder builder(TypeName type, String name, Modifier... modifiers) {
		Util.checkNotNull(type, "type == null");
		Util.checkArgument(isValidParameterName(name), "not a valid parameterName: %s", name);
		return new Builder(type, name)
				.addModifiers(modifiers);
	}

	public static Builder builder(Type type, String name, Modifier... modifiers) {
		return builder(TypeName.get(type), name, modifiers);
	}

	public boolean hasModifier(Modifier modifier) {
		return this.modifiers.contains(modifier);
	}

	void emit(CodeWriter codeWriter, boolean varargs) throws IOException {
		codeWriter.emitAnnotations(this.annotations, true);
		codeWriter.emitModifiers(this.modifiers);
		if (varargs) {
			TypeName.asArray(this.type).emit(codeWriter, true);
		} else {
			this.type.emit(codeWriter);
		}
		codeWriter.emit(" $L", this.name);
	}

	@Override
	public int hashCode() {
		return toString().hashCode();
	}

	@Override
	public boolean equals(Object o) {
		if (this == o) {
			return true;
		}
		if (o == null) {
			return false;
		}
		if (getClass() != o.getClass()) {
			return false;
		}
		return toString().equals(o.toString());
	}

	@Override
	public String toString() {
		StringBuilder out = new StringBuilder();
		try {
			CodeWriter codeWriter = new CodeWriter(out);
			emit(codeWriter, false);
			return out.toString();
		} catch (IOException e) {
			throw new AssertionError();
		}
	}

	public Builder toBuilder() {
		return toBuilder(this.type, this.name);
	}

	Builder toBuilder(TypeName type, String name) {
		Builder builder = new Builder(type, name);
		builder.annotations.addAll(this.annotations);
		builder.modifiers.addAll(this.modifiers);
		return builder;
	}

	public static final class Builder {
		public final List annotations = new ArrayList<>();
		public final List modifiers = new ArrayList<>();
		private final TypeName type;
		private final String name;
		private final CodeBlock.Builder javadoc = CodeBlock.builder();

		private Builder(TypeName type, String name) {
			this.type = type;
			this.name = name;
		}

		public Builder addJavadoc(String format, Object... args) {
			this.javadoc.add(format, args);
			return this;
		}

		public Builder addJavadoc(CodeBlock block) {
			this.javadoc.add(block);
			return this;
		}

		public Builder addAnnotations(Iterable annotationSpecs) {
			Util.checkArgument(annotationSpecs != null, "annotationSpecs == null");
			for (AnnotationSpec annotationSpec : annotationSpecs) {
				this.annotations.add(annotationSpec);
			}
			return this;
		}

		public Builder addAnnotation(AnnotationSpec annotationSpec) {
			this.annotations.add(annotationSpec);
			return this;
		}

		public Builder addAnnotation(ClassName annotation) {
			this.annotations.add(AnnotationSpec.builder(annotation).build());
			return this;
		}

		public Builder addAnnotation(Class annotation) {
			return addAnnotation(ClassName.get(annotation));
		}

		public Builder addModifiers(Modifier... modifiers) {
			Collections.addAll(this.modifiers, modifiers);
			return this;
		}

		public Builder addModifiers(Iterable modifiers) {
			Util.checkNotNull(modifiers, "modifiers == null");
			for (Modifier modifier : modifiers) {
				if (!modifier.equals(Modifier.FINAL)) {
					throw new IllegalStateException("unexpected parameter modifier: " + modifier);
				}
				this.modifiers.add(modifier);
			}
			return this;
		}

		public ParameterSpec build() {
			return new ParameterSpec(this);
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy