boot/src/main/java/org/netbeans/html/boot/impl/JavaScriptProcesor.java
author Jaroslav Tulach <jtulach@netbeans.org>
Fri, 15 Jan 2016 13:05:42 +0100
changeset 1043 b189d001b9bd
parent 1042 e633fed12064
permissions -rw-r--r--
#257579: Erase the parameter types before computing the signature
jaroslav@165
     1
/**
jaroslav@358
     2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
jaroslav@165
     3
 *
jaroslav@551
     4
 * Copyright 2013-2014 Oracle and/or its affiliates. All rights reserved.
jaroslav@165
     5
 *
jaroslav@358
     6
 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
jaroslav@358
     7
 * Other names may be trademarks of their respective owners.
jaroslav@165
     8
 *
jaroslav@358
     9
 * The contents of this file are subject to the terms of either the GNU
jaroslav@358
    10
 * General Public License Version 2 only ("GPL") or the Common
jaroslav@358
    11
 * Development and Distribution License("CDDL") (collectively, the
jaroslav@358
    12
 * "License"). You may not use this file except in compliance with the
jaroslav@358
    13
 * License. You can obtain a copy of the License at
jaroslav@358
    14
 * http://www.netbeans.org/cddl-gplv2.html
jaroslav@358
    15
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
jaroslav@358
    16
 * specific language governing permissions and limitations under the
jaroslav@358
    17
 * License.  When distributing the software, include this License Header
jaroslav@358
    18
 * Notice in each file and include the License file at
jaroslav@358
    19
 * nbbuild/licenses/CDDL-GPL-2-CP.  Oracle designates this
jaroslav@358
    20
 * particular file as subject to the "Classpath" exception as provided
jaroslav@358
    21
 * by Oracle in the GPL Version 2 section of the License file that
jaroslav@358
    22
 * accompanied this code. If applicable, add the following below the
jaroslav@358
    23
 * License Header, with the fields enclosed by brackets [] replaced by
jaroslav@358
    24
 * your own identifying information:
jaroslav@358
    25
 * "Portions Copyrighted [year] [name of copyright owner]"
jaroslav@358
    26
 *
jaroslav@358
    27
 * Contributor(s):
jaroslav@358
    28
 *
jaroslav@358
    29
 * The Original Software is NetBeans. The Initial Developer of the Original
jaroslav@551
    30
 * Software is Oracle. Portions Copyright 2013-2014 Oracle. All Rights Reserved.
jaroslav@358
    31
 *
jaroslav@358
    32
 * If you wish your version of this file to be governed by only the CDDL
jaroslav@358
    33
 * or only the GPL Version 2, indicate your decision by adding
jaroslav@358
    34
 * "[Contributor] elects to include this software in this distribution
jaroslav@358
    35
 * under the [CDDL or GPL Version 2] license." If you do not indicate a
jaroslav@358
    36
 * single choice of license, a recipient has the option to distribute
jaroslav@358
    37
 * your version of this file under either the CDDL, the GPL Version 2 or
jaroslav@358
    38
 * to extend the choice of license to its licensees as provided above.
jaroslav@358
    39
 * However, if you add GPL Version 2 code and therefore, elected the GPL
jaroslav@358
    40
 * Version 2 license, then the option applies only if the new code is
jaroslav@358
    41
 * made subject to such option by the copyright holder.
jaroslav@165
    42
 */
jaroslav@362
    43
package org.netbeans.html.boot.impl;
jaroslav@165
    44
jaroslav@186
    45
import java.io.IOException;
jaroslav@459
    46
import java.io.OutputStream;
jaroslav@459
    47
import java.io.OutputStreamWriter;
jaroslav@459
    48
import java.io.PrintWriter;
jaroslav@186
    49
import java.io.Writer;
jtulach@959
    50
import java.util.Arrays;
jaroslav@165
    51
import java.util.Collections;
jaroslav@186
    52
import java.util.HashMap;
jaroslav@165
    53
import java.util.HashSet;
jaroslav@165
    54
import java.util.List;
jaroslav@184
    55
import java.util.Map;
jaroslav@165
    56
import java.util.Set;
jaroslav@186
    57
import java.util.TreeMap;
jaroslav@165
    58
import javax.annotation.processing.AbstractProcessor;
jaroslav@165
    59
import javax.annotation.processing.Completion;
jaroslav@165
    60
import javax.annotation.processing.Completions;
jaroslav@165
    61
import javax.annotation.processing.Messager;
jaroslav@165
    62
import javax.annotation.processing.Processor;
jaroslav@165
    63
import javax.annotation.processing.RoundEnvironment;
jtulach@327
    64
import javax.lang.model.SourceVersion;
jaroslav@165
    65
import javax.lang.model.element.AnnotationMirror;
jaroslav@165
    66
import javax.lang.model.element.Element;
jaroslav@165
    67
import javax.lang.model.element.ElementKind;
jaroslav@165
    68
import javax.lang.model.element.ExecutableElement;
jaroslav@186
    69
import javax.lang.model.element.Modifier;
jaroslav@509
    70
import javax.lang.model.element.Name;
jaroslav@186
    71
import javax.lang.model.element.PackageElement;
jaroslav@165
    72
import javax.lang.model.element.TypeElement;
jaroslav@165
    73
import javax.lang.model.element.VariableElement;
jaroslav@185
    74
import javax.lang.model.type.ArrayType;
jaroslav@185
    75
import javax.lang.model.type.ExecutableType;
jaroslav@185
    76
import javax.lang.model.type.TypeKind;
jaroslav@185
    77
import javax.lang.model.type.TypeMirror;
jtulach@1043
    78
import javax.lang.model.util.Types;
jaroslav@165
    79
import javax.tools.Diagnostic;
jaroslav@348
    80
import javax.tools.FileObject;
jaroslav@348
    81
import javax.tools.StandardLocation;
jaroslav@165
    82
import net.java.html.js.JavaScriptBody;
jaroslav@165
    83
import net.java.html.js.JavaScriptResource;
jaroslav@165
    84
import org.openide.util.lookup.ServiceProvider;
jaroslav@165
    85
jaroslav@165
    86
/**
jaroslav@165
    87
 *
jtulach@790
    88
 * @author Jaroslav Tulach
jaroslav@165
    89
 */
jaroslav@165
    90
@ServiceProvider(service = Processor.class)
jaroslav@165
    91
public final class JavaScriptProcesor extends AbstractProcessor {
jtulach@959
    92
    private final Map<String,Map<String,ExecutableElement>> javacalls =
jaroslav@186
    93
        new HashMap<String,Map<String,ExecutableElement>>();
jtulach@959
    94
    private final Map<String,Set<TypeElement>> bodies =
jaroslav@459
    95
        new HashMap<String, Set<TypeElement>>();
jtulach@959
    96
jaroslav@165
    97
    @Override
jaroslav@165
    98
    public Set<String> getSupportedAnnotationTypes() {
jaroslav@165
    99
        Set<String> set = new HashSet<String>();
jaroslav@165
   100
        set.add(JavaScriptBody.class.getName());
jaroslav@165
   101
        set.add(JavaScriptResource.class.getName());
jaroslav@165
   102
        return set;
jaroslav@165
   103
    }
jtulach@959
   104
jaroslav@165
   105
    @Override
jaroslav@165
   106
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
jaroslav@165
   107
        final Messager msg = processingEnv.getMessager();
jaroslav@165
   108
        for (Element e : roundEnv.getElementsAnnotatedWith(JavaScriptBody.class)) {
jaroslav@165
   109
            if (e.getKind() != ElementKind.METHOD && e.getKind() != ElementKind.CONSTRUCTOR) {
jaroslav@165
   110
                continue;
jaroslav@165
   111
            }
jaroslav@165
   112
            ExecutableElement ee = (ExecutableElement)e;
jaroslav@165
   113
            List<? extends VariableElement> params = ee.getParameters();
jtulach@959
   114
jaroslav@165
   115
            JavaScriptBody jsb = e.getAnnotation(JavaScriptBody.class);
jaroslav@165
   116
            if (jsb == null) {
jaroslav@165
   117
                continue;
jaroslav@459
   118
            } else {
jaroslav@459
   119
                Set<TypeElement> classes = this.bodies.get(findPkg(e));
jaroslav@459
   120
                if (classes == null) {
jaroslav@459
   121
                    classes = new HashSet<TypeElement>();
jaroslav@459
   122
                    bodies.put(findPkg(e), classes);
jaroslav@459
   123
                }
jaroslav@459
   124
                Element t = e.getEnclosingElement();
jaroslav@459
   125
                while (!t.getKind().isClass() && !t.getKind().isInterface()) {
jaroslav@459
   126
                    t = t.getEnclosingElement();
jaroslav@459
   127
                }
jaroslav@459
   128
                classes.add((TypeElement)t);
jaroslav@165
   129
            }
jaroslav@165
   130
            String[] arr = jsb.args();
jaroslav@165
   131
            if (params.size() != arr.length) {
jaroslav@165
   132
                msg.printMessage(Diagnostic.Kind.ERROR, "Number of args arguments does not match real arguments!", e);
jaroslav@165
   133
            }
jtulach@959
   134
            for (int i = 0; i < arr.length; i++) {
jtulach@959
   135
                if (!params.get(i).getSimpleName().toString().equals(arr[i])) {
jtulach@959
   136
                    msg.printMessage(Diagnostic.Kind.WARNING, "Actual method parameter names and args ones " + Arrays.toString(arr) + " differ", e);
jtulach@959
   137
                }
jtulach@959
   138
            }
jaroslav@570
   139
            if (!jsb.wait4js() && ee.getReturnType().getKind() != TypeKind.VOID) {
jaroslav@570
   140
                msg.printMessage(Diagnostic.Kind.ERROR, "Methods that don't wait for JavaScript to finish must return void!", e);
jaroslav@570
   141
            }
jaroslav@165
   142
            if (!jsb.javacall() && jsb.body().contains(".@")) {
jaroslav@165
   143
                msg.printMessage(Diagnostic.Kind.WARNING, "Usage of .@ usually requires javacall=true", e);
jaroslav@165
   144
            }
jaroslav@184
   145
            if (jsb.javacall()) {
jaroslav@184
   146
                JsCallback verify = new VerifyCallback(e);
jaroslav@273
   147
                try {
jaroslav@273
   148
                    verify.parse(jsb.body());
jaroslav@273
   149
                } catch (IllegalStateException ex) {
jaroslav@273
   150
                    msg.printMessage(Diagnostic.Kind.ERROR, ex.getLocalizedMessage(), e);
jaroslav@273
   151
                }
jaroslav@184
   152
            }
jaroslav@165
   153
        }
jaroslav@348
   154
        for (Element e : roundEnv.getElementsAnnotatedWith(JavaScriptResource.class)) {
jaroslav@348
   155
            JavaScriptResource r = e.getAnnotation(JavaScriptResource.class);
jaroslav@348
   156
            if (r == null) {
jaroslav@348
   157
                continue;
jaroslav@348
   158
            }
jaroslav@348
   159
            final String res;
jaroslav@348
   160
            if (r.value().startsWith("/")) {
jaroslav@676
   161
                res = r.value().substring(1);
jaroslav@348
   162
            } else {
jaroslav@348
   163
                res = findPkg(e).replace('.', '/') + "/" + r.value();
jaroslav@348
   164
            }
jtulach@959
   165
jaroslav@348
   166
            try {
jaroslav@348
   167
                FileObject os = processingEnv.getFiler().getResource(StandardLocation.SOURCE_PATH, "", res);
jaroslav@348
   168
                os.openInputStream().close();
jaroslav@348
   169
            } catch (IOException ex1) {
jaroslav@348
   170
                try {
jaroslav@348
   171
                    FileObject os2 = processingEnv.getFiler().getResource(StandardLocation.CLASS_OUTPUT, "", res);
jaroslav@348
   172
                    os2.openInputStream().close();
jaroslav@348
   173
                } catch (IOException ex2) {
jaroslav@676
   174
                    try {
jaroslav@676
   175
                        FileObject os3 = processingEnv.getFiler().getResource(StandardLocation.CLASS_PATH, "", res);
jaroslav@676
   176
                        os3.openInputStream().close();
jaroslav@676
   177
                    } catch (IOException ex3) {
jaroslav@676
   178
                        msg.printMessage(Diagnostic.Kind.ERROR, "Cannot find resource " + res, e);
jaroslav@676
   179
                    }
jaroslav@348
   180
                }
jaroslav@348
   181
            }
jtulach@959
   182
jaroslav@531
   183
            boolean found = false;
jaroslav@531
   184
            for (Element mthod : e.getEnclosedElements()) {
jaroslav@531
   185
                if (mthod.getKind() != ElementKind.METHOD) {
jaroslav@531
   186
                    continue;
jaroslav@531
   187
                }
jaroslav@531
   188
                if (mthod.getAnnotation(JavaScriptBody.class) != null) {
jaroslav@531
   189
                    found = true;
jaroslav@531
   190
                    break;
jaroslav@531
   191
                }
jaroslav@531
   192
            }
jaroslav@531
   193
            if (!found) {
jaroslav@531
   194
                msg.printMessage(Diagnostic.Kind.ERROR, "At least one method needs @JavaScriptBody annotation. "
jaroslav@531
   195
                    + "Otherwise it is not guaranteed the resource will ever be loaded,", e
jaroslav@531
   196
                );
jaroslav@531
   197
            }
jaroslav@348
   198
        }
jaroslav@348
   199
jaroslav@186
   200
        if (roundEnv.processingOver()) {
jaroslav@186
   201
            generateCallbackClass(javacalls);
jaroslav@459
   202
            generateJavaScriptBodyList(bodies);
jaroslav@186
   203
            javacalls.clear();
jaroslav@186
   204
        }
jaroslav@165
   205
        return true;
jaroslav@165
   206
    }
jaroslav@165
   207
jaroslav@165
   208
    @Override
jtulach@959
   209
    public Iterable<? extends Completion> getCompletions(Element e,
jaroslav@165
   210
        AnnotationMirror annotation, ExecutableElement member, String userText
jaroslav@165
   211
    ) {
jaroslav@165
   212
        StringBuilder sb = new StringBuilder();
jaroslav@165
   213
        if (e.getKind() == ElementKind.METHOD && member.getSimpleName().contentEquals("args")) {
jaroslav@165
   214
            ExecutableElement ee = (ExecutableElement) e;
jaroslav@165
   215
            String sep = "";
jaroslav@165
   216
            sb.append("{ ");
jaroslav@165
   217
            for (VariableElement ve : ee.getParameters()) {
jaroslav@165
   218
                sb.append(sep).append('"').append(ve.getSimpleName())
jaroslav@165
   219
                    .append('"');
jaroslav@165
   220
                sep = ", ";
jaroslav@165
   221
            }
jaroslav@165
   222
            sb.append(" }");
jaroslav@165
   223
            return Collections.nCopies(1, Completions.of(sb.toString()));
jaroslav@165
   224
        }
jaroslav@165
   225
        return null;
jaroslav@165
   226
    }
jaroslav@165
   227
jaroslav@184
   228
    private class VerifyCallback extends JsCallback {
jaroslav@184
   229
        private final Element e;
jaroslav@184
   230
        public VerifyCallback(Element e) {
jaroslav@184
   231
            this.e = e;
jaroslav@184
   232
        }
jaroslav@184
   233
jaroslav@184
   234
        @Override
jaroslav@184
   235
        protected CharSequence callMethod(String ident, String fqn, String method, String params) {
jaroslav@185
   236
            final TypeElement type = processingEnv.getElementUtils().getTypeElement(fqn);
jaroslav@185
   237
            if (type == null) {
jtulach@959
   238
                processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,
jaroslav@184
   239
                    "Callback to non-existing class " + fqn, e
jaroslav@184
   240
                );
jaroslav@185
   241
                return "";
jaroslav@185
   242
            }
jaroslav@186
   243
            ExecutableElement found = null;
jaroslav@185
   244
            StringBuilder foundParams = new StringBuilder();
jaroslav@185
   245
            for (Element m : type.getEnclosedElements()) {
jaroslav@185
   246
                if (m.getKind() != ElementKind.METHOD) {
jaroslav@185
   247
                    continue;
jaroslav@185
   248
                }
jaroslav@185
   249
                if (m.getSimpleName().contentEquals(method)) {
jaroslav@188
   250
                    String paramTypes = findParamTypes((ExecutableElement)m);
jaroslav@185
   251
                    if (paramTypes.equals(params)) {
jaroslav@186
   252
                        found = (ExecutableElement) m;
jaroslav@185
   253
                        break;
jaroslav@185
   254
                    }
jaroslav@185
   255
                    foundParams.append(paramTypes).append("\n");
jaroslav@185
   256
                }
jaroslav@185
   257
            }
jaroslav@185
   258
            if (found == null) {
jaroslav@185
   259
                if (foundParams.length() == 0) {
jtulach@959
   260
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,
jaroslav@185
   261
                        "Callback to class " + fqn + " with unknown method " + method, e
jaroslav@185
   262
                    );
jaroslav@185
   263
                } else {
jtulach@959
   264
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,
jtulach@959
   265
                        "Callback to " + fqn + "." + method + " with wrong parameters: " +
jaroslav@185
   266
                        params + ". Only known parameters are " + foundParams, e
jaroslav@185
   267
                    );
jaroslav@185
   268
                }
jaroslav@186
   269
            } else {
jaroslav@186
   270
                Map<String,ExecutableElement> mangledOnes = javacalls.get(findPkg(e));
jaroslav@186
   271
                if (mangledOnes == null) {
jaroslav@186
   272
                    mangledOnes = new TreeMap<String, ExecutableElement>();
jaroslav@186
   273
                    javacalls.put(findPkg(e), mangledOnes);
jaroslav@186
   274
                }
jaroslav@188
   275
                String mangled = JsCallback.mangle(fqn, method, findParamTypes(found));
jaroslav@186
   276
                mangledOnes.put(mangled, found);
jaroslav@184
   277
            }
jaroslav@184
   278
            return "";
jaroslav@184
   279
        }
jaroslav@185
   280
jaroslav@188
   281
        private String findParamTypes(ExecutableElement method) {
jaroslav@185
   282
            ExecutableType t = (ExecutableType) method.asType();
jaroslav@185
   283
            StringBuilder sb = new StringBuilder();
jaroslav@188
   284
            sb.append('(');
jaroslav@185
   285
            for (TypeMirror tm : t.getParameterTypes()) {
jaroslav@185
   286
                if (tm.getKind().isPrimitive()) {
jaroslav@185
   287
                    switch (tm.getKind()) {
jaroslav@185
   288
                        case INT: sb.append('I'); break;
jaroslav@185
   289
                        case BOOLEAN: sb.append('Z'); break;
jaroslav@185
   290
                        case BYTE: sb.append('B'); break;
jaroslav@185
   291
                        case CHAR: sb.append('C'); break;
jaroslav@185
   292
                        case SHORT: sb.append('S'); break;
jaroslav@185
   293
                        case DOUBLE: sb.append('D'); break;
jaroslav@185
   294
                        case FLOAT: sb.append('F'); break;
jaroslav@185
   295
                        case LONG: sb.append('J'); break;
jaroslav@185
   296
                        default:
jaroslav@185
   297
                            throw new IllegalStateException("Uknown " + tm.getKind());
jaroslav@185
   298
                    }
jaroslav@185
   299
                } else {
jaroslav@185
   300
                    while (tm.getKind() == TypeKind.ARRAY) {
jaroslav@185
   301
                        sb.append('[');
jaroslav@185
   302
                        tm = ((ArrayType)tm).getComponentType();
jaroslav@185
   303
                    }
jaroslav@185
   304
                    sb.append('L');
jtulach@1043
   305
                    Types tu = processingEnv.getTypeUtils();
jtulach@1043
   306
                    Element elm = tu.asElement(tu.erasure(tm));
jtulach@933
   307
                    dumpElems(sb, elm, ';');
jaroslav@185
   308
                }
jaroslav@185
   309
            }
jaroslav@188
   310
            sb.append(')');
jaroslav@185
   311
            return sb.toString();
jaroslav@185
   312
        }
jaroslav@184
   313
    }
jtulach@959
   314
jtulach@933
   315
    private static void dumpElems(StringBuilder sb, Element e, char after) {
jtulach@933
   316
        if (e == null) {
jtulach@933
   317
            return;
jtulach@933
   318
        }
jtulach@933
   319
        if (e.getKind() == ElementKind.PACKAGE) {
jtulach@933
   320
            PackageElement pe = (PackageElement) e;
jtulach@933
   321
            sb.append(pe.getQualifiedName().toString().replace('.', '/')).append('/');
jtulach@933
   322
            return;
jtulach@933
   323
        }
jtulach@933
   324
        Element p = e.getEnclosingElement();
jtulach@933
   325
        dumpElems(sb, p, '$');
jtulach@933
   326
        sb.append(e.getSimpleName());
jtulach@933
   327
        sb.append(after);
jtulach@933
   328
    }
jtulach@959
   329
jaroslav@459
   330
    private void generateJavaScriptBodyList(Map<String,Set<TypeElement>> bodies) {
jaroslav@509
   331
        if (bodies.isEmpty()) {
jaroslav@509
   332
            return;
jaroslav@509
   333
        }
jaroslav@509
   334
        try {
jaroslav@509
   335
            FileObject all = processingEnv.getFiler().createResource(
jtulach@959
   336
                StandardLocation.CLASS_OUTPUT, "", "META-INF/net.java.html.js.classes"
jaroslav@509
   337
            );
jaroslav@509
   338
            PrintWriter wAll = new PrintWriter(new OutputStreamWriter(
jaroslav@509
   339
                all.openOutputStream(), "UTF-8"
jaroslav@509
   340
            ));
jaroslav@509
   341
            for (Map.Entry<String, Set<TypeElement>> entry : bodies.entrySet()) {
jaroslav@509
   342
                String pkg = entry.getKey();
jaroslav@509
   343
                Set<TypeElement> classes = entry.getValue();
jaroslav@509
   344
jaroslav@459
   345
                FileObject out = processingEnv.getFiler().createResource(
jaroslav@459
   346
                    StandardLocation.CLASS_OUTPUT, pkg, "net.java.html.js.classes",
jaroslav@459
   347
                    classes.iterator().next()
jaroslav@459
   348
                );
jaroslav@459
   349
                OutputStream os = out.openOutputStream();
jaroslav@459
   350
                try {
jaroslav@459
   351
                    PrintWriter w = new PrintWriter(new OutputStreamWriter(os, "UTF-8"));
jaroslav@459
   352
                    for (TypeElement type : classes) {
jaroslav@509
   353
                        final Name bn = processingEnv.getElementUtils().getBinaryName(type);
jaroslav@509
   354
                        w.println(bn);
jaroslav@509
   355
                        wAll.println(bn);
jaroslav@459
   356
                    }
jaroslav@459
   357
                    w.flush();
jaroslav@459
   358
                    w.close();
jaroslav@509
   359
                } catch (IOException x) {
jaroslav@509
   360
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, "Failed to write to " + entry.getKey() + ": " + x.toString());
jaroslav@459
   361
                } finally {
jaroslav@459
   362
                    os.close();
jaroslav@459
   363
                }
jaroslav@459
   364
            }
jaroslav@509
   365
            wAll.close();
jaroslav@509
   366
        } catch (IOException x) {
jaroslav@509
   367
            processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, "Failed to write to " + "META-INF/net.java.html.js.classes: " + x.toString());
jaroslav@459
   368
        }
jaroslav@459
   369
    }
jtulach@959
   370
jaroslav@186
   371
    private void generateCallbackClass(Map<String,Map<String, ExecutableElement>> process) {
jaroslav@186
   372
        for (Map.Entry<String, Map<String, ExecutableElement>> pkgEn : process.entrySet()) {
jaroslav@186
   373
            String pkgName = pkgEn.getKey();
jaroslav@186
   374
            Map<String, ExecutableElement> map = pkgEn.getValue();
jaroslav@186
   375
            StringBuilder source = new StringBuilder();
jaroslav@186
   376
            source.append("package ").append(pkgName).append(";\n");
jaroslav@188
   377
            source.append("public final class $JsCallbacks$ {\n");
jaroslav@288
   378
            source.append("  static final $JsCallbacks$ VM = new $JsCallbacks$(null);\n");
jtulach@838
   379
            source.append("  private final org.netbeans.html.boot.spi.Fn.Presenter p;\n");
jaroslav@288
   380
            source.append("  private $JsCallbacks$ last;\n");
jtulach@838
   381
            source.append("  private $JsCallbacks$(org.netbeans.html.boot.spi.Fn.Presenter p) {\n");
jaroslav@288
   382
            source.append("    this.p = p;\n");
jaroslav@288
   383
            source.append("  }\n");
jaroslav@288
   384
            source.append("  final $JsCallbacks$ current() {\n");
jtulach@838
   385
            source.append("    org.netbeans.html.boot.spi.Fn.Presenter now = org.netbeans.html.boot.spi.Fn.activePresenter();\n");
jaroslav@288
   386
            source.append("    if (now == p) return this;\n");
jaroslav@288
   387
            source.append("    if (last != null && now == last.p) return last;\n");
jaroslav@288
   388
            source.append("    return last = new $JsCallbacks$(now);\n");
jaroslav@288
   389
            source.append("  }\n");
jaroslav@186
   390
            for (Map.Entry<String, ExecutableElement> entry : map.entrySet()) {
jaroslav@186
   391
                final String mangled = entry.getKey();
jaroslav@186
   392
                final ExecutableElement m = entry.getValue();
jtulach@911
   393
                generateMethod(false, m, source, mangled);
jtulach@911
   394
                generateMethod(true, m, source, "raw$" + mangled);
jaroslav@186
   395
            }
jaroslav@186
   396
            source.append("}\n");
jaroslav@186
   397
            final String srcName = pkgName + ".$JsCallbacks$";
jaroslav@186
   398
            try {
jaroslav@186
   399
                Writer w = processingEnv.getFiler().createSourceFile(srcName,
jaroslav@186
   400
                    map.values().toArray(new Element[map.size()])
jaroslav@186
   401
                ).openWriter();
jaroslav@186
   402
                w.write(source.toString());
jaroslav@186
   403
                w.close();
jaroslav@186
   404
            } catch (IOException ex) {
jaroslav@186
   405
                processingEnv.getMessager().printMessage(
jaroslav@186
   406
                    Diagnostic.Kind.ERROR, "Can't write " + srcName + ": " + ex.getMessage()
jaroslav@186
   407
                );
jaroslav@186
   408
            }
jaroslav@186
   409
        }
jaroslav@186
   410
    }
jaroslav@609
   411
jtulach@911
   412
    private void generateMethod(boolean selfObj, final ExecutableElement m, StringBuilder source, final String mangled) {
jtulach@911
   413
        final boolean isStatic = m.getModifiers().contains(Modifier.STATIC);
jtulach@911
   414
        if (isStatic && selfObj) {
jtulach@911
   415
            return;
jtulach@911
   416
        }
jtulach@911
   417
        final TypeElement selfType = (TypeElement)m.getEnclosingElement();
jtulach@1043
   418
        Types tu = processingEnv.getTypeUtils();
jtulach@959
   419
jtulach@911
   420
        source.append("\n  public java.lang.Object ")
jtulach@911
   421
                .append(mangled)
jtulach@911
   422
                .append("(");
jtulach@959
   423
jtulach@911
   424
        String sep = "";
jtulach@911
   425
        StringBuilder convert = new StringBuilder();
jtulach@911
   426
        if (!isStatic) {
jtulach@911
   427
            if (selfObj) {
jtulach@1042
   428
                source.append("java.lang.Object self");
jtulach@911
   429
                convert.append("    if (p instanceof org.netbeans.html.boot.spi.Fn.FromJavaScript) {\n");
jtulach@911
   430
                convert.append("      self").
jtulach@911
   431
                        append(" = ((org.netbeans.html.boot.spi.Fn.FromJavaScript)p).toJava(self").
jtulach@911
   432
                        append(");\n");
jtulach@911
   433
                convert.append("    }\n");
jtulach@911
   434
            } else {
jtulach@911
   435
                source.append(selfType.getQualifiedName());
jtulach@911
   436
                source.append(" self");
jtulach@911
   437
            }
jtulach@911
   438
            sep = ", ";
jtulach@911
   439
        }
jtulach@959
   440
jtulach@911
   441
        int cnt = 0;
jtulach@911
   442
        for (VariableElement ve : m.getParameters()) {
jtulach@911
   443
            source.append(sep);
jtulach@911
   444
            ++cnt;
jtulach@911
   445
            final TypeMirror t = ve.asType();
jtulach@913
   446
            if (!t.getKind().isPrimitive() && !"java.lang.String".equals(t.toString())) { // NOI18N
jtulach@1042
   447
                source.append("java.lang.Object");
jtulach@911
   448
                convert.append("    if (p instanceof org.netbeans.html.boot.spi.Fn.FromJavaScript) {\n");
jtulach@911
   449
                convert.append("      arg").append(cnt).
jtulach@911
   450
                        append(" = ((org.netbeans.html.boot.spi.Fn.FromJavaScript)p).toJava(arg").append(cnt).
jtulach@911
   451
                        append(");\n");
jtulach@911
   452
                convert.append("    }\n");
jtulach@911
   453
            } else {
jtulach@911
   454
                source.append(t);
jtulach@911
   455
            }
jtulach@911
   456
            source.append(" arg").append(cnt);
jtulach@911
   457
            sep = ", ";
jtulach@911
   458
        }
jtulach@911
   459
        source.append(") throws Throwable {\n");
jtulach@911
   460
        source.append(convert);
jtulach@911
   461
        if (useTryResources()) {
jtulach@911
   462
            source.append("    try (java.io.Closeable a = org.netbeans.html.boot.spi.Fn.activate(p)) { \n");
jtulach@911
   463
        } else {
jtulach@911
   464
            source.append("    java.io.Closeable a = org.netbeans.html.boot.spi.Fn.activate(p); try {\n");
jtulach@911
   465
        }
jtulach@911
   466
        source.append("    ");
jtulach@911
   467
        if (m.getReturnType().getKind() != TypeKind.VOID) {
jtulach@1042
   468
            source.append("java.lang.Object $ret = ");
jtulach@911
   469
        }
jtulach@911
   470
        if (isStatic) {
jtulach@911
   471
            source.append(((TypeElement)m.getEnclosingElement()).getQualifiedName());
jtulach@911
   472
            source.append('.');
jtulach@911
   473
        } else {
jtulach@911
   474
            if (selfObj) {
jtulach@911
   475
                source.append("((");
jtulach@911
   476
                source.append(selfType.getQualifiedName());
jtulach@911
   477
                source.append(")self).");
jtulach@911
   478
            } else {
jtulach@911
   479
                source.append("self.");
jtulach@911
   480
            }
jtulach@911
   481
        }
jtulach@911
   482
        source.append(m.getSimpleName());
jtulach@911
   483
        source.append("(");
jtulach@911
   484
        cnt = 0;
jtulach@911
   485
        sep = "";
jtulach@911
   486
        for (VariableElement ve : m.getParameters()) {
jtulach@911
   487
            source.append(sep);
jtulach@1043
   488
            source.append("(").append(tu.erasure(ve.asType()));
jtulach@911
   489
            source.append(")arg").append(++cnt);
jtulach@911
   490
            sep = ", ";
jtulach@911
   491
        }
jtulach@911
   492
        source.append(");\n");
jtulach@911
   493
        if (m.getReturnType().getKind() == TypeKind.VOID) {
jtulach@911
   494
            source.append("    return null;\n");
jtulach@911
   495
        } else {
jtulach@911
   496
            source.append("    if (p instanceof org.netbeans.html.boot.spi.Fn.ToJavaScript) {\n");
jtulach@911
   497
            source.append("      $ret = ((org.netbeans.html.boot.spi.Fn.ToJavaScript)p).toJavaScript($ret);\n");
jtulach@911
   498
            source.append("    }\n");
jtulach@911
   499
            source.append("    return $ret;\n");
jtulach@911
   500
        }
jtulach@911
   501
        if (useTryResources()) {
jtulach@911
   502
            source.append("    }\n");
jtulach@911
   503
        } else {
jtulach@959
   504
jtulach@911
   505
            source.append("    } finally {\n");
jtulach@911
   506
            source.append("      a.close();\n");
jtulach@911
   507
            source.append("    }\n");
jtulach@911
   508
        }
jtulach@911
   509
        source.append("  }\n");
jtulach@911
   510
    }
jtulach@911
   511
jaroslav@609
   512
    private boolean useTryResources() {
jaroslav@609
   513
        try {
jaroslav@609
   514
            return processingEnv.getSourceVersion().compareTo(SourceVersion.RELEASE_7) >= 0;
jaroslav@609
   515
        } catch (LinkageError err) {
jaroslav@609
   516
            // can happen when running on JDK6
jaroslav@609
   517
            return false;
jaroslav@609
   518
        }
jaroslav@609
   519
    }
jtulach@959
   520
jaroslav@186
   521
    private static String findPkg(Element e) {
jaroslav@186
   522
        while (e.getKind() != ElementKind.PACKAGE) {
jaroslav@186
   523
            e = e.getEnclosingElement();
jaroslav@186
   524
        }
jaroslav@186
   525
        return ((PackageElement)e).getQualifiedName().toString();
jaroslav@186
   526
    }
jtulach@959
   527
jaroslav@165
   528
}