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

org.jibx.schema.codegen.ClassBuilder Maven / Gradle / Ivy

/*
 * Copyright (c) 2007-2009, Dennis M. Sosnoski. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 * 
 * Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 * disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 * following disclaimer in the documentation and/or other materials provided with the distribution. Neither the name of
 * JiBX nor the names of its contributors may be used to endorse or promote products derived from this software without
 * specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package org.jibx.schema.codegen;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;

import org.eclipse.jdt.core.dom.AST;
import org.eclipse.jdt.core.dom.ASTNode;
import org.eclipse.jdt.core.dom.AbstractTypeDeclaration;
import org.eclipse.jdt.core.dom.AnonymousClassDeclaration;
import org.eclipse.jdt.core.dom.ArrayAccess;
import org.eclipse.jdt.core.dom.ArrayCreation;
import org.eclipse.jdt.core.dom.BodyDeclaration;
import org.eclipse.jdt.core.dom.CastExpression;
import org.eclipse.jdt.core.dom.ClassInstanceCreation;
import org.eclipse.jdt.core.dom.EnumConstantDeclaration;
import org.eclipse.jdt.core.dom.EnumDeclaration;
import org.eclipse.jdt.core.dom.FieldDeclaration;
import org.eclipse.jdt.core.dom.InfixExpression;
import org.eclipse.jdt.core.dom.Javadoc;
import org.eclipse.jdt.core.dom.MethodDeclaration;
import org.eclipse.jdt.core.dom.MethodInvocation;
import org.eclipse.jdt.core.dom.Modifier;
import org.eclipse.jdt.core.dom.Name;
import org.eclipse.jdt.core.dom.ParameterizedType;
import org.eclipse.jdt.core.dom.PrefixExpression;
import org.eclipse.jdt.core.dom.StringLiteral;
import org.eclipse.jdt.core.dom.TagElement;
import org.eclipse.jdt.core.dom.TextElement;
import org.eclipse.jdt.core.dom.Type;
import org.eclipse.jdt.core.dom.TypeDeclaration;
import org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.eclipse.jdt.core.dom.InfixExpression.Operator;
import org.jibx.runtime.Utility;

/**
 * Builder for a class definition. This wraps the AST with convenience methods and added control information.
 */
public class ClassBuilder
{
    /** Source file containing this class. */
    private final SourceBuilder m_source;
    
    /** Type declaration for class. */
    private final ASTNode m_class;
    
    /** Fields added to class. */
    private final ArrayList m_fields;
    
    /** Methods added to class. */
    private final ArrayList m_methods;
    
    /** Builders for inner classes of this class. */
    private final ArrayList m_innerBuilders;
    
    /** Directly-added inner classes of this class. */
    private final ArrayList m_innerClasses;
    
    /**
     * Constructor.
     * 
     * @param clas
     * @param source 
     */
    ClassBuilder(AbstractTypeDeclaration clas, SourceBuilder source) {
        m_source = source;
        m_class = clas;
        m_fields = new ArrayList();
        m_methods = new ArrayList();
        m_innerBuilders = new ArrayList();
        m_innerClasses = new ArrayList();
    }
    
    /**
     * Constructor for an inner class.
     * 
     * @param clas
     * @param outer
     */
    ClassBuilder(AbstractTypeDeclaration clas, ClassBuilder outer) {
        this(clas, outer.m_source);
        outer.m_innerBuilders.add(this);
    }
    
    /**
     * Constructor for an anonymous inner class.
     * 
     * @param clas
     * @param outer
     */
    public ClassBuilder(AnonymousClassDeclaration clas, ClassBuilder outer) {
        m_source = outer.m_source;
        m_class = clas;
        m_fields = null;
        m_methods = null;
        m_innerBuilders = null;
        m_innerClasses = null;
    }
    
    /**
     * AST access for related classes.
     *
     * @return AST
     */
    AST getAST() {
        return m_source.getAST();
    }

    /**
     * Add separately-constructed field declaration.
     *
     * @param field
     */
    public void addField(FieldDeclaration field) {
        m_fields.add(ASTNode.copySubtree(getAST(), field));
    }

    /**
     * Add separately-constructed method declaration.
     *
     * @param method
     */
    public void addMethod(MethodDeclaration method) {
        m_methods.add(ASTNode.copySubtree(getAST(), method));
    }

    /**
     * Add separately-constructed inner class declaration.
     *
     * @param type
     */
    public void addType(TypeDeclaration type) {
        m_innerClasses.add(ASTNode.copySubtree(getAST(), type));
    }
    
    /**
     * Set the superclass for this class.
     *
     * @param name
     */
    public void setSuperclass(String name) {
        if (m_class instanceof TypeDeclaration) {
            ((TypeDeclaration)m_class).setSuperclassType(createType(name));
        } else {
            throw new IllegalStateException("Internal error - can't set superclass on an enum or other special type");
        }
    }
    
    /**
     * Create type name.
     *
     * @param type fully qualified type name
     * @return name
     */
    Name createTypeName(String type) {
        return m_source.createTypeName(type);
    }
    
    /**
     * Clone an AST node. The cloned node will have no parent.
     *
     * @param node
     * @return clone
     */
    public ASTNode clone(ASTNode node) {
        return ASTNode.copySubtree(m_source.getAST(), node);
    }
    
    /**
     * Create type definition. This uses the supplied type name, which may include array suffixes, to construct the
     * actual type definition.
     *
     * @param type fully qualified type name, or primitive type name
     * @return constructed typed definition
     */
    public Type createType(String type) {
        return m_source.createType(type);
    }
    
    /**
     * Create a parameterized type. Both the type itself and the parameter type are given as names in this variation.
     *
     * @param type fully qualified type name
     * @param param fully qualified parameter type name
     * @return type
     */
    public Type createParameterizedType(String type, String param) {
        return m_source.createParameterizedType(type, param);
    }
    
    /**
     * Create a parameterized type. The type itself is given as a name, while the parameter type is an actual type in
     * this variation.
     *
     * @param type fully qualified type name
     * @param param type parameter
     * @return type
     */
    public Type createParameterizedType(String type, Type param) {
        ParameterizedType ptype = getAST().newParameterizedType(createType(type));
        ptype.typeArguments().add(param);
        return ptype;
    }
    
    /**
     * Set source comment for this class.
     *
     * @param text comment text
     */
    public void addSourceComment(String text) {
        if (m_class instanceof AbstractTypeDeclaration) {
            Javadoc javadoc = getAST().newJavadoc();
            TextElement element = getAST().newTextElement();
            element.setText(text);
            TagElement tag = getAST().newTagElement();
            tag.fragments().add(element);
            javadoc.tags().add(tag);
            ((AbstractTypeDeclaration)m_class).setJavadoc(javadoc);
        } else {
            throw new IllegalStateException("Internal error - cannot add JavaDoc to non-class type");
        }
    }
    
    /**
     * Set the abstract flag for this class.
     */
    public void setAbstract() {
        if (m_class instanceof TypeDeclaration) {
            Modifier modifier = m_source.getAST().newModifier(Modifier.ModifierKeyword.ABSTRACT_KEYWORD);
            ((TypeDeclaration)m_class).modifiers().add(modifier);
        } else {
            // should not be possible, but just in case of added types in future
            throw new IllegalStateException("Internal error - abstract not supported for class type");
        }
    }
    
    /**
     * Get the interfaces implemented by this class.
     *
     * @return interface names
     */
    public String[] getInterfaces() {
        if (m_class instanceof TypeDeclaration) {
            List types = ((TypeDeclaration)m_class).superInterfaceTypes();
            ArrayList names = new ArrayList();
            for (Iterator iter = types.iterator(); iter.hasNext();) {
                Type type = (Type)iter.next();
                names.add(type.toString());
            }
            return (String[])names.toArray(new String[names.size()]);
        } else if (m_class instanceof EnumDeclaration) {
            return Utility.EMPTY_STRING_ARRAY;
        } else {
            // should not be possible, but just in case of added types in future
            throw new IllegalStateException("Internal error - interface not supported for class type");
        }
    }
    
    /**
     * Get the fields defined in this class.
     *
     * @return fields
     */
    public FieldDeclaration[] getFields() {
        return (FieldDeclaration[])m_fields.toArray(new FieldDeclaration[m_fields.size()]);
    }
    
    /**
     * Get the methods defined in this class.
     *
     * @return methods
     */
    public MethodDeclaration[] getMethods() {
        return (MethodDeclaration[])m_methods.toArray(new MethodDeclaration[m_methods.size()]);
    }

    /**
     * Add an interface to this class definition.
     *
     * @param type interface type
     */
    public void addInterface(String type) {
        if (m_class instanceof TypeDeclaration) {
            ((TypeDeclaration)m_class).superInterfaceTypes().add(m_source.createType(type));
        } else if (m_class instanceof EnumDeclaration) {
            ((EnumDeclaration)m_class).superInterfaceTypes().add(m_source.createType(type));
        } else {
            // should not be possible, but just in case of added types in future
            throw new IllegalStateException("Internal error - interface not supported for class type");
        }
    }
    
    /**
     * Add JavaDoc to a declaration.
     *
     * @param doc documentation text, or null if none
     * @param decl
     */
    public void addJavaDoc(String doc, BodyDeclaration decl) {
        if (doc != null) {
            Javadoc javadoc = getAST().newJavadoc();
            TextElement element = getAST().newTextElement();
            element.setText(doc);
            TagElement tag = getAST().newTagElement();
            tag.fragments().add(element);
            javadoc.tags().add(tag);
            decl.setJavadoc(javadoc);
        }
    }

    /**
     * Add a constant to a Java 5 enum definition. This method is used for enums which use the name as the value.
     *
     * @param value 
     * @param doc documentation text, or null if none
     */
    public void addEnumConstant(String value, String doc) {
        if (m_class instanceof EnumDeclaration) {
            EnumConstantDeclaration enumdecl = getAST().newEnumConstantDeclaration();
            enumdecl.setName(getAST().newSimpleName(value));
            addJavaDoc(doc, enumdecl);
            ((EnumDeclaration)m_class).enumConstants().add(enumdecl);
        } else {
            // should not be possible, but just in case of added types in future
            throw new IllegalStateException("Internal error - cannot add constant to class type");
        }
    }

    /**
     * Add a constant to a Java 5 enum definition. This method is used for enums which have a value separate from the
     * name.
     *
     * @param name
     * @param doc documentation text, or null if none
     * @param value
     */
    public void addEnumConstant(String name, String doc, String value) {
        if (m_class instanceof EnumDeclaration) {
            EnumConstantDeclaration enumdecl = getAST().newEnumConstantDeclaration();
            enumdecl.setName(getAST().newSimpleName(name));
            StringLiteral strlit = getAST().newStringLiteral();
            strlit.setLiteralValue(value);
            enumdecl.arguments().add(strlit);
            addJavaDoc(doc, enumdecl);
            ((EnumDeclaration)m_class).enumConstants().add(enumdecl);
        } else {
            // should not be possible, but just in case of added types in future
            throw new IllegalStateException("Internal error - cannot add constant to class type");
        }
    }
    
    /**
     * Create new instance of array type.
     *
     * @param type base type name
     * @return array creation
     */
    public NewArrayBuilder newArrayBuilder(String type) {
        ArrayCreation create = getAST().newArrayCreation();
        create.setType(getAST().newArrayType(m_source.createType(type)));
        return new NewArrayBuilder(this, create);
    }

    /**
     * Build new instance creator of type using a no-argument constructor.
     *
     * @param type actual type
     * @return instance creation
     */
    public NewInstanceBuilder newInstance(Type type) {
        ClassInstanceCreation create = getAST().newClassInstanceCreation();
        create.setType(type);
        return new NewInstanceBuilder(this, create);
    }

    /**
     * Build new instance creator of type using a no-argument constructor.
     *
     * @param type base type name
     * @return instance creation
     */
    public NewInstanceBuilder newInstance(String type) {
        return newInstance(m_source.createType(type));
    }

    /**
     * Build new instance creator of a simple type using a constructor that takes a single string value.
     *
     * @param type simple type name
     * @param value string value to be passed to constructor
     * @return instance creation
     */
    public NewInstanceBuilder newInstanceFromString(String type, String value) {
        ClassInstanceCreation create = getAST().newClassInstanceCreation();
        create.setType(createType(type));
        StringLiteral literal = getAST().newStringLiteral();
        literal.setLiteralValue(value);
        create.arguments().add(literal);
        return new NewInstanceBuilder(this, create);
    }

    /**
     * Build new instance creator of a simple type using a constructor that takes a pair of string values.
     *
     * @param type simple type name
     * @param value1 first string value to be passed to constructor
     * @param value2 second string value to be passed to constructor
     * @return instance creation
     */
    public NewInstanceBuilder newInstanceFromStrings(String type, String value1, String value2) {
        ClassInstanceCreation create = getAST().newClassInstanceCreation();
        create.setType(createType(type));
        StringLiteral literal = getAST().newStringLiteral();
        literal.setLiteralValue(value1);
        create.arguments().add(literal);
        literal = getAST().newStringLiteral();
        literal.setLiteralValue(value2);
        create.arguments().add(literal);
        return new NewInstanceBuilder(this, create);
    }

    /**
     * Add field declaration.
     *
     * @param name field name
     * @param type field type
     * @return field builder
     */
    public FieldBuilder addField(String name, Type type) {
        VariableDeclarationFragment vfrag = getAST().newVariableDeclarationFragment();
        vfrag.setName(getAST().newSimpleName(name));
        FieldDeclaration fdecl = getAST().newFieldDeclaration(vfrag);
        fdecl.setType(type);
        m_fields.add(fdecl);
        return new FieldBuilder(this, fdecl);
    }

    /**
     * Add field declaration.
     *
     * @param name field name
     * @param type type name
     * @return field builder
     */
    public FieldBuilder addField(String name, String type) {
        return addField(name, m_source.createType(type));
    }

    /**
     * Add int field declaration with constant initialization.
     *
     * @param name variable name
     * @param value initial value
     * @return field builder
     */
    public FieldBuilder addIntField(String name, String value) {
        FieldBuilder field = addField(name, "int");
        field.setNumberInitializer(value);
        return field;
    }
    
    /**
     * Add constructor declaration.
     *
     * @param name simple class name
     * @return constructor builder
     */
    public MethodBuilder addConstructor(String name) {
        MethodDeclaration constr = getAST().newMethodDeclaration();
        constr.setName(getAST().newSimpleName(name));
        constr.setConstructor(true);
        m_methods.add(constr);
        return new MethodBuilder(this, constr);
    }
    
    /**
     * Add method declaration.
     *
     * @param name
     * @param type
     * @return method builder
     */
    public MethodBuilder addMethod(String name, Type type) {
        MethodDeclaration meth = getAST().newMethodDeclaration();
        meth.setName(getAST().newSimpleName(name));
        meth.setConstructor(false);
        meth.setReturnType2(type);
        if (m_class instanceof AnonymousClassDeclaration) {
            ((AnonymousClassDeclaration)m_class).bodyDeclarations().add(meth);        
        } else {
            m_methods.add(meth);
        }
        return new MethodBuilder(this, meth);
    }
    
    /**
     * Add method declaration.
     *
     * @param name
     * @param type fully qualified type name or primitive type name, with optional array suffixes
     * @return method builder
     */
    public MethodBuilder addMethod(String name, String type) {
        return addMethod(name, m_source.createType(type));
    }
    
    /**
     * Create internal member method call builder. 
     *
     * @param mname method name
     * @return builder
     */
    public InvocationBuilder createMemberMethodCall(String mname) {
        MethodInvocation methcall = getAST().newMethodInvocation();
        methcall.setName(getAST().newSimpleName(mname));
        return new InvocationBuilder(this, methcall);
    }
    
    /**
     * Create internal static method call builder. 
     *
     * @param mname method name
     * @return builder
     */
    public InvocationBuilder createLocalStaticMethodCall(String mname) {
        MethodInvocation methcall = getAST().newMethodInvocation();
        methcall.setName(getAST().newSimpleName(mname));
        return new InvocationBuilder(this, methcall);
    }
    
    /**
     * Create a static method call builder. 
     *
     * @param cname fully qualified class name
     * @param mname method name
     * @return builder
     */
    public InvocationBuilder createStaticMethodCall(String cname, String mname) {
        MethodInvocation methcall = getAST().newMethodInvocation();
        methcall.setExpression(getAST().newName(cname));
        methcall.setName(getAST().newSimpleName(mname));
        return new InvocationBuilder(this, methcall);
    }
    
    /**
     * Create a static method call builder.
     *
     * @param fname fully-qualified class and method name
     * @return builder
     */
    public InvocationBuilder createStaticMethodCall(String fname) {
        int split = fname.lastIndexOf('.');
        return createStaticMethodCall(fname.substring(0, split), fname.substring(split+1));
    }
    
    /**
     * Create method call builder on a local variable or field value. 
     *
     * @param name local variable or field name
     * @param mname method name
     * @return builder
     */
    public InvocationBuilder createNormalMethodCall(String name, String mname) {
        MethodInvocation methcall = getAST().newMethodInvocation();
        methcall.setExpression(getAST().newSimpleName(name));
        methcall.setName(getAST().newSimpleName(mname));
        return new InvocationBuilder(this, methcall);
    }
    
    /**
     * Create method call builder on the reference result of an expression. 
     *
     * @param expr instance expression
     * @param mname method name
     * @return builder
     */
    public InvocationBuilder createExpressionMethodCall(ExpressionBuilderBase expr, String mname) {
        MethodInvocation methcall = getAST().newMethodInvocation();
        methcall.setExpression(expr.getExpression());
        methcall.setName(getAST().newSimpleName(mname));
        return new InvocationBuilder(this, methcall);
    }
    
    /**
     * Build general infix expression.
     *
     * @param op operator
     * @return expression
     */
    public InfixExpressionBuilder buildInfix(Operator op) {
        InfixExpression infixex = getAST().newInfixExpression();
        infixex.setOperator(op);
        return new InfixExpressionBuilder(this, infixex);
    }
    
    /**
     * Build infix expression involving a local variable or field name as the left operand.
     *
     * @param name local variable or field name
     * @param op operator
     * @return expression
     */
    public InfixExpressionBuilder buildNameOp(String name, Operator op) {
        InfixExpression infixex = getAST().newInfixExpression();
        infixex.setOperator(op);
        return new InfixExpressionBuilder(this, infixex, getAST().newSimpleName(name));
    }
    
    /**
     * Build a string concatenation expression starting from from a string literal.
     *
     * @param text literal text
     * @return string concatenation expression
     */
    public InfixExpressionBuilder buildStringConcatenation(String text) {
        InfixExpression expr = getAST().newInfixExpression();
        StringLiteral strlit = getAST().newStringLiteral();
        strlit.setLiteralValue(text);
        expr.setOperator(Operator.PLUS);
        return new InfixExpressionBuilder(this, expr, strlit);
    }
    
    /**
     * Build a preincrement expression using a local variable or field name as the operand.
     *
     * @param name local variable or field name
     * @return expression
     */
    public PrefixExpressionBuilder buildPreincrement(String name) {
        PrefixExpression prefixex = getAST().newPrefixExpression();
        prefixex.setOperator(PrefixExpression.Operator.INCREMENT);
        return new PrefixExpressionBuilder(this, prefixex, getAST().newSimpleName(name));
    }
    
    /**
     * Build a cast expression.
     *
     * @param type result type
     * @return expression
     */
    public CastBuilder buildCast(Type type) {
        CastExpression castex = getAST().newCastExpression();
        castex.setType(type);
        return new CastBuilder(this, castex);
    }
    
    /**
     * Build a cast expression.
     *
     * @param type result type
     * @return expression
     */
    public CastBuilder buildCast(String type) {
        return buildCast(createType(type));
    }
    
    /**
     * Build array access expression for a named array variable and named index variable.
     *
     * @param aname
     * @param iname
     * @return array access
     */
    public ArrayAccessBuilder buildArrayIndexAccess(String aname, String iname) {
        ArrayAccess access = getAST().newArrayAccess();
        access.setArray(getAST().newSimpleName(aname));
        access.setIndex(getAST().newSimpleName(iname));
        return new ArrayAccessBuilder(this, access);
    }
    
    /**
     * Create a new block.
     *
     * @return block builder
     */
    public BlockBuilder newBlock() {
        return new BlockBuilder(this, getAST().newBlock());
    }
    
    /**
     * Finish building the source file data structures.
     */
    public void finish() {
        if (m_class instanceof AbstractTypeDeclaration) {
            List decls = ((AbstractTypeDeclaration)m_class).bodyDeclarations();
            decls.addAll(m_fields);
            decls.addAll(m_methods);
            for (int i = 0; i < m_innerBuilders.size(); i++) {
                ClassBuilder builder = (ClassBuilder)m_innerBuilders.get(i);
                builder.finish();
                decls.add(builder.m_class);
            }
            decls.addAll(m_innerClasses);
        }
    }
    
    /**
     * Get a sorted array of the field names and types defined in this class.
     *
     * @return sorted pairs
     */
    public StringPair[] getSortedFields() {
        StringPair[] pairs = new StringPair[m_fields.size()];
        for (int i = 0; i < m_fields.size(); i++) {
            FieldDeclaration field = (FieldDeclaration)m_fields.get(i);
            String name = ((VariableDeclarationFragment)field.fragments().get(0)).getName().toString();
            pairs[i] = new StringPair(name, field.getType().toString());
        }
        Arrays.sort(pairs);
        return pairs;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy