emul/mini/src/main/java/org/apidesign/bck2brwsr/emul/reflect/AnnotationImpl.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 05 Feb 2013 00:00:22 +0100
branchreflection
changeset 665 799a8cfefe35
parent 663 09225928b9fb
permissions -rw-r--r--
Returning proper runtime representation for inner annotation arrays
     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(val, prop, clazz) {\n"
    42         + "  return function() {\n"
    43         + "    if (clazz == null) return val[prop];\n"
    44         + "    if (clazz.isArray__Z()) {\n"
    45         + "      var valarr = val[prop];\n"
    46         + "      var cmp = clazz.getComponentType__Ljava_lang_Class_2();\n"
    47         + "      var retarr = vm.java_lang_reflect_Array(false).newInstance__Ljava_lang_Object_2Ljava_lang_Class_2I(cmp, valarr.length);\n"
    48         + "      for (var i = 0; i < valarr.length; i++) {\n"
    49         + "        retarr[i] = CLS.prototype.c__Ljava_lang_Object_2Ljava_lang_Class_2Ljava_lang_Object_2(cmp, valarr[i]);\n"
    50         + "      }\n"
    51         + "      return retarr;\n"
    52         + "    }\n"
    53         + "    return CLS.prototype.c__Ljava_lang_Object_2Ljava_lang_Class_2Ljava_lang_Object_2(clazz, val[prop]);\n"
    54         + "  };\n"
    55         + "}\n"
    56         + "for (var i = 0; i < arr.length; i += 3) {\n"
    57         + "  var m = arr[i];\n"
    58         + "  var p = arr[i + 1];\n"
    59         + "  var c = arr[i + 2];\n"
    60         + "  a[m] = f(values, p, c);\n"
    61         + "}\n"
    62         + "a['$instOf_' + n] = true;\n"
    63         + "return a;"
    64     )
    65     private static native <T extends Annotation> T create(
    66         AnnotationImpl a, String n, Object[] methodsAndProps, Object values
    67     );
    68     
    69     private static Object c(Class<? extends Annotation> a, Object v) {
    70         return create(a, v);
    71     }
    72     
    73     public static <T extends Annotation> T create(Class<T> annoClass, Object values) {
    74         return create(new AnnotationImpl(annoClass), 
    75             annoClass.getName().replace('.', '_'), 
    76             findProps(annoClass), values
    77         );
    78     }
    79 
    80     public static Annotation[] create(Object anno) {
    81         String[] names = findNames(anno);
    82         Annotation[] ret = new Annotation[names.length];
    83         for (int i = 0; i < names.length; i++) {
    84             String annoNameSlash = names[i].substring(1, names[i].length() - 1);
    85             Class<? extends Annotation> annoClass;
    86             try {
    87                 annoClass = (Class<? extends Annotation>)Class.forName(annoNameSlash.replace('/', '.'));
    88             } catch (ClassNotFoundException ex) {
    89                 throw new IllegalStateException("Can't find annotation class " + annoNameSlash);
    90             }
    91             ret[i] = create(
    92                 new AnnotationImpl(annoClass), 
    93                 annoNameSlash.replace('/', '_'),
    94                 findProps(annoClass),
    95                 findData(anno, names[i])
    96             );
    97         }
    98         return ret;
    99     }
   100     @JavaScriptBody(args = "anno", body =
   101           "var arr = new Array();"
   102         + "var props = Object.getOwnPropertyNames(anno);\n"
   103         + "for (var i = 0; i < props.length; i++) {\n"
   104         + "  var p = props[i];\n"
   105         + "  arr.push(p);"
   106         + "}"
   107         + "return arr;"
   108     )
   109     private static native String[] findNames(Object anno);
   110 
   111     @JavaScriptBody(args={ "anno", "p"}, body="return anno[p];")
   112     private static native Object findData(Object anno, String p);
   113 
   114     private static Object[] findProps(Class<?> annoClass) {
   115         final Method[] marr = MethodImpl.findMethods(annoClass, Modifier.PUBLIC);
   116         Object[] arr = new Object[marr.length * 3];
   117         int pos = 0;
   118         for (Method m : marr) {
   119             arr[pos++] = MethodImpl.toSignature(m);
   120             arr[pos++] = m.getName();
   121             final Class<?> rt = m.getReturnType();
   122             if (rt.isArray()) {
   123                 arr[pos++] = rt.getComponentType().isAnnotation() ? rt : null;
   124             } else {
   125                 arr[pos++] = rt.isAnnotation() ? rt : null;
   126             }
   127         }
   128         return arr;
   129     }
   130 }