emul/mini/src/main/java/org/apidesign/bck2brwsr/emul/reflect/AnnotationImpl.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Mon, 04 Feb 2013 23:18:51 +0100
branchreflection
changeset 662 7832188e26b8
parent 654 26a86cc00224
child 663 09225928b9fb
permissions -rw-r--r--
Can obtain values of inner annotation types
     1 /**
     2  * Back 2 Browser Bytecode Translator
     3  * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
     4  *
     5  * This program is free software: you can redistribute it and/or modify
     6  * it under the terms of the GNU General Public License as published by
     7  * the Free Software Foundation, version 2 of the License.
     8  *
     9  * This program is distributed in the hope that it will be useful,
    10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    12  * GNU General Public License for more details.
    13  *
    14  * You should have received a copy of the GNU General Public License
    15  * along with this program. Look for COPYING file in the top folder.
    16  * If not, see http://opensource.org/licenses/GPL-2.0.
    17  */
    18 package org.apidesign.bck2brwsr.emul.reflect;
    19 
    20 import java.lang.annotation.Annotation;
    21 import java.lang.reflect.Method;
    22 import java.lang.reflect.Modifier;
    23 import org.apidesign.bck2brwsr.core.JavaScriptBody;
    24 
    25 /**
    26  *
    27  * @author Jaroslav Tulach <jtulach@netbeans.org>
    28  */
    29 public final class AnnotationImpl implements Annotation {
    30     private final Class<? extends Annotation> type;
    31 
    32     public AnnotationImpl(Class<? extends Annotation> type) {
    33         this.type = type;
    34     }
    35     
    36     public Class<? extends Annotation> annotationType() {
    37         return type;
    38     }
    39 
    40     @JavaScriptBody(args = { "a", "n", "arr", "values" }, body = ""
    41         + "function f(v, p, c) {\n"
    42         + "  var val = v;\n"
    43         + "  var prop = p;\n"
    44         + "  var clazz = c;\n"
    45         + "  return function() {\n"
    46         + "    if (clazz == null) return val[prop];\n"
    47         + "    return CLS.prototype.c__Ljava_lang_Object_2Ljava_lang_Class_2Ljava_lang_Object_2(clazz, val[prop]);\n"
    48         + "  };\n"
    49         + "}\n"
    50         + "for (var i = 0; i < arr.length; i += 3) {\n"
    51         + "  var m = arr[i];\n"
    52         + "  var p = arr[i + 1];\n"
    53         + "  var c = arr[i + 2];\n"
    54         + "  a[m] = new f(values, p, c);\n"
    55         + "}\n"
    56         + "a['$instOf_' + n] = true;\n"
    57         + "return a;"
    58     )
    59     private static native <T extends Annotation> T create(
    60         AnnotationImpl a, String n, Object[] methodsAndProps, Object values
    61     );
    62     
    63     private static Object c(Class<? extends Annotation> a, Object v) {
    64         return create(a, v);
    65     }
    66     
    67     public static <T extends Annotation> T create(Class<T> annoClass, Object values) {
    68         return create(new AnnotationImpl(annoClass), 
    69             annoClass.getName().replace('.', '_'), 
    70             findProps(annoClass), values
    71         );
    72     }
    73 
    74     public static Annotation[] create(Object anno) {
    75         String[] names = findNames(anno);
    76         Annotation[] ret = new Annotation[names.length];
    77         for (int i = 0; i < names.length; i++) {
    78             String annoNameSlash = names[i].substring(1, names[i].length() - 1);
    79             Class<? extends Annotation> annoClass;
    80             try {
    81                 annoClass = (Class<? extends Annotation>)Class.forName(annoNameSlash.replace('/', '.'));
    82             } catch (ClassNotFoundException ex) {
    83                 throw new IllegalStateException("Can't find annotation class " + annoNameSlash);
    84             }
    85             ret[i] = create(
    86                 new AnnotationImpl(annoClass), 
    87                 annoNameSlash.replace('/', '_'),
    88                 findProps(annoClass),
    89                 findData(anno, names[i])
    90             );
    91         }
    92         return ret;
    93     }
    94     @JavaScriptBody(args = "anno", body =
    95           "var arr = new Array();"
    96         + "var props = Object.getOwnPropertyNames(anno);\n"
    97         + "for (var i = 0; i < props.length; i++) {\n"
    98         + "  var p = props[i];\n"
    99         + "  arr.push(p);"
   100         + "}"
   101         + "return arr;"
   102     )
   103     private static native String[] findNames(Object anno);
   104 
   105     @JavaScriptBody(args={ "anno", "p"}, body="return anno[p];")
   106     private static native Object findData(Object anno, String p);
   107 
   108     private static Object[] findProps(Class<?> annoClass) {
   109         final Method[] marr = MethodImpl.findMethods(annoClass, Modifier.PUBLIC);
   110         Object[] arr = new Object[marr.length * 3];
   111         int pos = 0;
   112         for (Method m : marr) {
   113             arr[pos++] = MethodImpl.toSignature(m);
   114             arr[pos++] = m.getName();
   115             arr[pos++] = m.getReturnType().isAnnotation() ? m.getReturnType() : null;
   116         }
   117         return arr;
   118     }
   119 }