ide/editor/src/main/java/org/apidesign/bck2brwsr/ide/editor/JSEmbeddingProvider.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 13 Feb 2013 16:52:50 +0100
branchide
changeset 723 5d1c3bc8d297
parent 720 d5c565eb0986
permissions -rw-r--r--
Adding missing license
     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.ide.editor;
    19 
    20 import com.sun.source.tree.AnnotationTree;
    21 import com.sun.source.tree.AssignmentTree;
    22 import com.sun.source.tree.CompilationUnitTree;
    23 import com.sun.source.tree.ExpressionTree;
    24 import com.sun.source.tree.LiteralTree;
    25 import com.sun.source.tree.MethodTree;
    26 import com.sun.source.util.SourcePositions;
    27 import com.sun.source.util.TreePath;
    28 import com.sun.source.util.TreePathScanner;
    29 import com.sun.source.util.Trees;
    30 import java.io.IOException;
    31 import java.util.ArrayList;
    32 import java.util.Collection;
    33 import java.util.Collections;
    34 import java.util.List;
    35 import java.util.concurrent.atomic.AtomicBoolean;
    36 import javax.lang.model.element.TypeElement;
    37 import javax.swing.text.Document;
    38 import org.netbeans.api.editor.mimelookup.MimeRegistration;
    39 import org.netbeans.api.java.source.CompilationInfo;
    40 import org.netbeans.api.java.source.JavaParserResultTask;
    41 import org.netbeans.api.java.source.JavaSource;
    42 import org.netbeans.api.lexer.Language;
    43 import org.netbeans.api.lexer.TokenHierarchy;
    44 import org.netbeans.api.lexer.TokenSequence;
    45 import org.netbeans.modules.parsing.api.Snapshot;
    46 import org.netbeans.modules.parsing.spi.Parser;
    47 import org.netbeans.modules.parsing.spi.Scheduler;
    48 import org.netbeans.modules.parsing.spi.SchedulerEvent;
    49 import org.netbeans.modules.parsing.spi.SchedulerTask;
    50 import org.netbeans.modules.parsing.spi.TaskFactory;
    51 import org.openide.util.Exceptions;
    52 
    53 /**
    54  *
    55  * @author Tomas Zezula
    56  */
    57 public final class JSEmbeddingProvider extends JavaParserResultTask<Parser.Result> {
    58 
    59     private static final int PRIORITY = 1000;
    60     private static final String JS_ANNOTATION = "org.apidesign.bck2brwsr.core.JavaScriptBody";  //NOI18N
    61     private static final String BODY = "body";                            //NOI18N
    62     private static final String JAVA_MIME_TYPE = "text/x-java";           //NOI18N
    63     private static final String JAVASCRIPT_MIME_TYPE = "text/javascript"; //NOI18N
    64     private final AtomicBoolean canceled = new AtomicBoolean();
    65 
    66     private JSEmbeddingProvider() {
    67         super(JavaSource.Phase.ELEMENTS_RESOLVED);
    68     }
    69 
    70     @Override
    71     public int getPriority() {
    72         return PRIORITY;
    73     }
    74 
    75     @Override
    76     public void cancel() {
    77         canceled.set(true);
    78     }
    79 
    80     @Override
    81     public Class<? extends Scheduler> getSchedulerClass() {
    82         return Scheduler.EDITOR_SENSITIVE_TASK_SCHEDULER;
    83     }
    84 
    85     @Override
    86     public void run(Parser.Result t, SchedulerEvent se) {
    87         canceled.set(false);
    88         final CompilationInfo ci = CompilationInfo.get(t);
    89         final CompilationUnitTree cu = ci.getCompilationUnit();
    90         final Trees trees = ci.getTrees();
    91         final SourcePositions sp = trees.getSourcePositions();
    92         final Finder f = new Finder(trees);
    93         final List<LiteralTree> result = new ArrayList<LiteralTree>();
    94         f.scan(cu, result);
    95         if (!result.isEmpty()) {
    96             try {
    97                 final TokenHierarchy<Document> tk = TokenHierarchy.get(ci.getDocument());
    98                 final Language<?> java = Language.find(JAVA_MIME_TYPE);
    99                 final Language<?> javaScript = Language.find(JAVASCRIPT_MIME_TYPE);
   100                 if (java != null && javaScript != null) {
   101                     final TokenSequence<?> seq = tk.tokenSequence(java);
   102                     if (seq != null) {
   103                         for (LiteralTree lt : result) {
   104                             final int start = (int) sp.getStartPosition(cu, lt);
   105                             final int end = (int) sp.getEndPosition(cu, lt);
   106                             seq.move(start);
   107                             while (seq.moveNext() && seq.offset() < end) {
   108                                 seq.createEmbedding(javaScript, 1, 1, true);
   109                             }
   110                         }
   111                     }
   112                 }
   113             } catch (IOException ioe) {
   114                 Exceptions.printStackTrace(ioe);
   115             }
   116         }
   117     }
   118 
   119 
   120 
   121 
   122     private static final class Finder extends TreePathScanner<Void, List<? super LiteralTree>> {
   123 
   124         private final Trees trees;
   125         private CompilationUnitTree cu;
   126         private boolean inEmbedding;
   127 
   128         Finder(final Trees trees) {
   129             this.trees = trees;
   130         }
   131 
   132         @Override
   133         public Void visitCompilationUnit(
   134                 final CompilationUnitTree unit,
   135                 final List p) {
   136             this.cu = unit;
   137             return super.visitCompilationUnit(unit, p);
   138         }
   139 
   140 
   141 
   142         @Override
   143         public Void visitMethod(
   144                 final MethodTree m,
   145                 final List<? super LiteralTree> p) {
   146             for (AnnotationTree a : m.getModifiers().getAnnotations()) {
   147                 final TypeElement ae = (TypeElement) trees.getElement(TreePath.getPath(cu, a.getAnnotationType()));
   148                 if (ae != null && JS_ANNOTATION.contentEquals(ae.getQualifiedName())) {
   149                     final List<? extends ExpressionTree> args =  a.getArguments();
   150                     for (ExpressionTree kvp : args) {
   151                         if (kvp instanceof AssignmentTree) {
   152                             final AssignmentTree assignemt = (AssignmentTree) kvp;
   153                             if (BODY.equals(assignemt.getVariable().toString())) {
   154                                 inEmbedding = true;
   155                                 try {
   156                                     scan(assignemt.getExpression(), p);
   157                                 } finally {
   158                                     inEmbedding = false;
   159                                 }
   160                             }
   161                         }
   162                     }
   163                 }
   164             }
   165             return null;
   166         }
   167 
   168         @Override
   169         public Void visitLiteral(LiteralTree node, List<? super LiteralTree> p) {
   170             if (inEmbedding) {
   171                 p.add(node);
   172             }
   173             return super.visitLiteral(node, p);
   174         }
   175 
   176     }
   177 
   178     @MimeRegistration(
   179             service = TaskFactory.class,
   180             mimeType = JAVA_MIME_TYPE)
   181     public static final class Factory extends TaskFactory {
   182         @Override
   183         public Collection<? extends SchedulerTask> create(Snapshot snpsht) {
   184             return Collections.singleton(new JSEmbeddingProvider());
   185         }
   186     }
   187 
   188 }