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

net.minecraftforge.gdi.transformer.DefaultMethodsTransformer.groovy Maven / Gradle / Ivy

Go to download

Groovy Compiler Plugin to improve building Groovy based DSLs, like those used in Gradle.

The newest version!
/*
 * Copyright (c) Forge Development LLC and contributors
 * SPDX-License-Identifier: LGPL-2.1-only
 */

package net.minecraftforge.gdi.transformer

import groovy.transform.CompileStatic
import groovy.transform.Trait
import groovyjarjarasm.asm.Opcodes
import org.codehaus.groovy.ast.*
import org.codehaus.groovy.ast.expr.Expression
import org.codehaus.groovy.ast.expr.VariableExpression
import org.codehaus.groovy.ast.stmt.Statement
import org.codehaus.groovy.ast.tools.GeneralUtils
import org.codehaus.groovy.control.CompilePhase
import org.codehaus.groovy.control.SourceUnit
import org.codehaus.groovy.transform.AbstractASTTransformation
import org.codehaus.groovy.transform.GroovyASTTransformation
import org.codehaus.groovy.transform.sc.StaticCompileTransformation

import java.util.stream.Collectors
import java.util.stream.Stream

@CompileStatic
@GroovyASTTransformation(phase = CompilePhase.CANONICALIZATION)
class DefaultMethodsTransformer extends AbstractASTTransformation implements Opcodes {
    private static final ClassNode TRAIT = ClassHelper.make(Trait)
    private static final ClassNode CSTATIC = ClassHelper.make(CompileStatic)
    @Override
    void visit(ASTNode[] astNodes, SourceUnit sourceUnit) {
        this.init(astNodes, sourceUnit)
        if (astNodes[1] !instanceof InnerClassNode) return
        final traitHelper = (InnerClassNode)astNodes[1]
        if (!(traitHelper.name.endsWith('Trait$Helper'))) {
            addError('@DefaultMethods can only be applied on interfaces!', traitHelper)
            return
        }
        final clazz = traitHelper.outerClass
        new ArrayList(clazz.methods).each {
            final method = traitHelper.methods.find { m -> m.name == it.name && m.parameters.drop(1) == it.parameters && m.parameters[0].type == clazz }
            if (method === null) return

            clazz.removeMethod(it)
            final Expression call = GeneralUtils.callX(traitHelper, it.name, GeneralUtils.args(
                    Stream.concat(
                            Stream.of(VariableExpression.THIS_EXPRESSION),
                            Stream.of(it.parameters)
                                    .map { GeneralUtils.varX(it) },
                    )
                        .collect(Collectors.toList())
            ))

            final mtd = clazz.addMethod(
                    it.name, ACC_PUBLIC,
                    it.returnType, it.parameters,
                    it.exceptions,
                    (Statement) (it.returnType == ClassHelper.VOID_TYPE ? GeneralUtils.stmt(call) : GeneralUtils.returnS(call))
            )
            method.annotations.each{ mtd.addAnnotation(it) }
            mtd.genericsTypes = method.genericsTypes
            final ann = new AnnotationNode(CSTATIC)
            mtd.addAnnotation(ann)
            new StaticCompileTransformation().visit(new ASTNode[] {
                ann, mtd
            }, sourceUnit)
        }

        clazz.annotations.removeIf { it.classNode == TRAIT }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy