com.querydsl.codegen.DefaultProjectionSerializer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of querydsl-codegen Show documentation
Show all versions of querydsl-codegen Show documentation
codegen module for querydsl
The newest version!
/*
* Copyright 2015, The Querydsl Team (http://www.querydsl.com/team)
*
* 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 com.querydsl.codegen;
import static com.querydsl.codegen.utils.Symbols.THIS_ESCAPE;
import com.querydsl.codegen.utils.CodeWriter;
import com.querydsl.codegen.utils.model.ClassType;
import com.querydsl.codegen.utils.model.Constructor;
import com.querydsl.codegen.utils.model.Parameter;
import com.querydsl.codegen.utils.model.Type;
import com.querydsl.codegen.utils.model.TypeCategory;
import com.querydsl.codegen.utils.model.TypeExtends;
import com.querydsl.codegen.utils.model.Types;
import com.querydsl.core.types.ConstructorExpression;
import com.querydsl.core.types.Expression;
import com.querydsl.core.types.dsl.NumberExpression;
import jakarta.inject.Inject;
import jakarta.inject.Named;
import java.io.IOException;
import java.lang.annotation.Annotation;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
/**
* {@code ProjectionSerializer} is a {@link Serializer} implementation for projection types
*
* @author tiwe
*/
public class DefaultProjectionSerializer implements ProjectionSerializer {
private final Class extends Annotation> generatedAnnotationClass;
private final TypeMappings typeMappings;
/**
* Create a new {@code ProjectionSerializer} instance
*
* @param typeMappings type mappings to be used
*/
public DefaultProjectionSerializer(TypeMappings typeMappings) {
this(typeMappings, GeneratedAnnotationResolver.resolveDefault());
}
/**
* Create a new {@code ProjectionSerializer} instance
*
* @param typeMappings type mappings to be used
* @param generatedAnnotationClass the fully qualified class name of the Single-Element
* Annotation (with {@code String} element) to be used on the generated classes.
* @see Single-Element
* Annotation
*/
@Inject
public DefaultProjectionSerializer(
TypeMappings typeMappings,
@Named(CodegenModule.GENERATED_ANNOTATION_CLASS)
Class extends Annotation> generatedAnnotationClass) {
this.typeMappings = typeMappings;
this.generatedAnnotationClass = generatedAnnotationClass;
}
protected void intro(EntityType model, CodeWriter writer) throws IOException {
var simpleName = model.getSimpleName();
var queryType = typeMappings.getPathType(model, model, false);
// package
if (!queryType.getPackageName().isEmpty()) {
writer.packageDecl(queryType.getPackageName());
}
// imports
imports(model, writer);
Set sizes = new HashSet<>();
var constructors = new ArrayList<>(model.getConstructors());
Collections.sort(constructors);
for (Constructor c : constructors) {
sizes.add(c.getParameters().size());
}
if (sizes.size() != constructors.size()) {
writer.imports(Expression.class);
}
// javadoc
writer.javadoc(queryType + " is a Querydsl Projection type for " + simpleName);
writer.suppressWarnings(THIS_ESCAPE);
writer.line("@", generatedAnnotationClass.getSimpleName(), "(\"", getClass().getName(), "\")");
// class header
// writer.suppressWarnings("serial");
Type superType = new ClassType(TypeCategory.SIMPLE, ConstructorExpression.class, model);
writer.beginClass(queryType, superType);
writer.privateStaticFinal(Types.LONG_P, "serialVersionUID", model.hashCode() + "L");
}
protected void imports(EntityType model, CodeWriter writer) throws IOException {
writer.imports(NumberExpression.class.getPackage());
writer.imports(ConstructorExpression.class, generatedAnnotationClass);
}
protected void outro(EntityType model, CodeWriter writer) throws IOException {
writer.end();
}
@Override
public void serialize(
final EntityType model, SerializerConfig serializerConfig, CodeWriter writer)
throws IOException {
// intro
intro(model, writer);
var localName = writer.getRawName(model);
Set sizes = new HashSet<>();
var constructors = new ArrayList<>(model.getConstructors());
Collections.sort(constructors);
for (Constructor c : constructors) {
final var asExpr = sizes.add(c.getParameters().size());
// begin
writer.beginConstructor(
c.getParameters(),
(Parameter p) -> {
Type type;
if (!asExpr) {
type = typeMappings.getExprType(p.getType(), model, false, false, true);
} else if (p.getType().isFinal()) {
type = new ClassType(Expression.class, p.getType());
} else {
type = new ClassType(Expression.class, new TypeExtends(p.getType()));
}
return new Parameter(p.getName(), type);
});
// body
writer.beginLine("super(" + writer.getClassConstant(localName));
// TODO: Fix for Scala (Array[Class])
writer.append(", new Class>[]{");
var first = true;
for (Parameter p : c.getParameters()) {
if (!first) {
writer.append(", ");
}
parameterType(writer, p);
first = false;
}
writer.append("}");
for (Parameter p : c.getParameters()) {
writer.append(", ");
parameter(writer, p);
}
// end
writer.append(");\n");
writer.end();
}
// outro
outro(model, writer);
}
protected void parameterType(CodeWriter writer, Parameter p) throws IOException {
if (Types.PRIMITIVES.containsKey(p.getType())) {
var primitive = Types.PRIMITIVES.get(p.getType());
writer.append(writer.getClassConstant(primitive.getFullName()));
} else {
writer.append(writer.getClassConstant(writer.getRawName(p.getType())));
}
}
protected void parameter(CodeWriter writer, Parameter p) throws IOException {
writer.append(p.getName());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy