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