com.github.buzztaiki.jenova.InterfaceMethod.java Source code

Java tutorial

Introduction

Here is the source code for com.github.buzztaiki.jenova.InterfaceMethod.java

Source

/*
 * Copyright (C) 2012  Taiki Sugawara
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package com.github.buzztaiki.jenova;

import com.google.common.base.Equivalence;
import com.google.common.base.Function;
import com.google.common.base.Preconditions;
import com.google.common.base.Predicates;
import com.google.common.collect.Iterables;
import com.sun.tools.javac.code.Scope;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.code.Types;
import com.sun.tools.javac.model.FilteredMemberList;
import com.sun.tools.javac.model.JavacElements;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.tree.TreeMaker;
import com.sun.tools.javac.util.Context;
import com.sun.tools.javac.util.List;
import com.sun.tools.javac.util.ListBuffer;
import com.sun.tools.javac.util.Name;

public class InterfaceMethod {
    private final Symbol.TypeSymbol clazz;
    private final Symbol.MethodSymbol method;
    private final JavacElements elems;
    private final TreeMaker maker;
    private final Types types;

    public InterfaceMethod(Context context, Symbol.TypeSymbol clazz) {
        this.clazz = clazz;
        this.method = findMethod(context, clazz);
        this.elems = JavacElements.instance(context);
        this.maker = TreeMaker.instance(context);
        this.types = Types.instance(context);
    }

    static Symbol.MethodSymbol findMethod(Context context, Symbol.TypeSymbol clazz) {
        for (Symbol.MethodSymbol method : methods(clazz)) {
            if (!baseMethod(context, method))
                return method;
        }
        throw new IllegalArgumentException(clazz + " doesn't have interface method.");
    }

    static boolean baseMethod(Context context, Symbol.MethodSymbol method) {
        JavacElements elems = JavacElements.instance(context);
        Types types = Types.instance(context);
        for (Symbol.MethodSymbol baseMethod : methods(elems.getTypeElement("java.lang.Object"))) {
            if (sameMethod(types, baseMethod, method))
                return true;
        }
        return false;
    }

    private static boolean sameMethod(Types types, Symbol.MethodSymbol a, Symbol.MethodSymbol b) {
        return a.flatName().equals(b.flatName())
                && sameType(types).pairwise().equivalent(types(a.params()), types(b.params()));
    }

    private static Equivalence<Type> sameType(final Types types) {
        return new Equivalence<Type>() {
            @Override
            protected boolean doEquivalent(Type a, Type b) {
                return types.isSameType(a, b);
            }

            @Override
            protected int doHash(Type t) {
                return Types.hashCode(t);
            }
        };
    }

    private static Iterable<Symbol.MethodSymbol> methods(Symbol.TypeSymbol clazz) {
        return Iterables.filter(clazz.members().getElements(), Symbol.MethodSymbol.class);
    }

    private static Iterable<Type> types(Iterable<? extends Symbol> syms) {
        return Iterables.transform(syms, new Function<Symbol, Type>() {
            @Override
            public Type apply(Symbol sym) {
                return sym.asType();
            }
        });
    }

    public List<JCTree.JCExpression> getParamTypes(List<JCTree.JCExpression> typeArgs) {
        Preconditions.checkArgument(typeArgs.size() == clazz.getTypeParameters().size(),
                "typeArgs length for %s should be %s but takes %s", getClazz(), clazz.getTypeParameters().size(),
                typeArgs.size());

        Iterable<Type> typeParams = types(clazz.getTypeParameters());
        Equivalence<Type> sameType = sameType(types);
        ListBuffer<JCTree.JCExpression> res = new ListBuffer<JCTree.JCExpression>();
        for (Type type : types(method.getParameters())) {
            int i = Iterables.indexOf(typeParams, sameType.equivalentTo(type));
            res.append(i >= 0 ? typeArgs.get(i) : maker.Type(type));
        }
        return res.toList();
    }

    public JCTree.JCExpression getReturnType(List<JCTree.JCExpression> typeArgs) {
        Preconditions.checkArgument(typeArgs.size() == clazz.getTypeParameters().size(),
                "typeArgs length for %s should be %s but takes %s", getClazz(), clazz.getTypeParameters().size(),
                typeArgs.size());

        Type ret = method.getReturnType();
        int i = 0;
        for (Type type : types(clazz.getTypeParameters())) {
            if (types.isSameType(ret, type))
                return typeArgs.get(i);
            i++;
        }
        return maker.Type(ret);
    }

    public Symbol.MethodSymbol getMethod() {
        return method;
    }

    public Symbol.TypeSymbol getClazz() {
        return clazz;
    }
}