mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2024-12-31 12:24:10 -08:00
Adds support of native JavaScript interaction
This commit is contained in:
parent
b621b0524b
commit
ae2e669ec3
|
@ -422,7 +422,7 @@ public class TString extends TObject implements TSerializable, TComparable<TStri
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(TObject obj) {
|
public static TString valueOf(TObject obj) {
|
||||||
return obj != null ? obj.toString0() : TString.wrap("null");
|
return obj != null ? TString.wrap(obj.toString()) : TString.wrap("null");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(char[] data) {
|
public static TString valueOf(char[] data) {
|
||||||
|
@ -450,19 +450,19 @@ public class TString extends TObject implements TSerializable, TComparable<TStri
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(int i) {
|
public static TString valueOf(int i) {
|
||||||
return new TStringBuilder().append(i).toString0();
|
return TString.wrap(new TStringBuilder().append(i).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(long l) {
|
public static TString valueOf(long l) {
|
||||||
return new TStringBuilder().append(l).toString0();
|
return TString.wrap(new TStringBuilder().append(l).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(float f) {
|
public static TString valueOf(float f) {
|
||||||
return new TStringBuilder().append(f).toString0();
|
return TString.wrap(new TStringBuilder().append(f).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TString valueOf(double d) {
|
public static TString valueOf(double d) {
|
||||||
return new TStringBuilder().append(d).toString0();
|
return TString.wrap(new TStringBuilder().append(d).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -17,11 +17,9 @@ package org.teavm.dependency;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.concurrent.*;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import org.teavm.common.ConcurrentCachedMapper;
|
import java.util.concurrent.ConcurrentMap;
|
||||||
import org.teavm.common.FiniteExecutor;
|
import org.teavm.common.*;
|
||||||
import org.teavm.common.Mapper;
|
|
||||||
import org.teavm.common.SimpleFiniteExecutor;
|
|
||||||
import org.teavm.common.ConcurrentCachedMapper.KeyListener;
|
import org.teavm.common.ConcurrentCachedMapper.KeyListener;
|
||||||
import org.teavm.model.*;
|
import org.teavm.model.*;
|
||||||
|
|
||||||
|
|
|
@ -44,9 +44,9 @@ public class JavascriptBuilder {
|
||||||
private Map<String, String> exportedClasses = new HashMap<>();
|
private Map<String, String> exportedClasses = new HashMap<>();
|
||||||
|
|
||||||
JavascriptBuilder(ClassHolderSource classSource, ClassLoader classLoader, FiniteExecutor executor) {
|
JavascriptBuilder(ClassHolderSource classSource, ClassLoader classLoader, FiniteExecutor executor) {
|
||||||
this.classSource = classSource;
|
this.classSource = new JavascriptProcessedClassSource(classSource);
|
||||||
this.classLoader = classLoader;
|
this.classLoader = classLoader;
|
||||||
dependencyChecker = new DependencyChecker(classSource, classLoader, executor);
|
dependencyChecker = new DependencyChecker(this.classSource, classLoader, executor);
|
||||||
this.executor = executor;
|
this.executor = executor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -15,12 +15,10 @@
|
||||||
*/
|
*/
|
||||||
package org.teavm.javascript;
|
package org.teavm.javascript;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.*;
|
||||||
import java.util.List;
|
import org.teavm.javascript.ni.*;
|
||||||
import java.util.Map;
|
|
||||||
import org.teavm.javascript.ni.JSObject;
|
|
||||||
import org.teavm.model.*;
|
import org.teavm.model.*;
|
||||||
import org.teavm.model.instructions.InvokeInstruction;
|
import org.teavm.model.instructions.*;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -29,6 +27,8 @@ import org.teavm.model.instructions.InvokeInstruction;
|
||||||
class JavascriptNativeProcessor {
|
class JavascriptNativeProcessor {
|
||||||
private ClassHolderSource classSource;
|
private ClassHolderSource classSource;
|
||||||
private Map<String, Boolean> knownJavaScriptClasses = new HashMap<>();
|
private Map<String, Boolean> knownJavaScriptClasses = new HashMap<>();
|
||||||
|
private Program program;
|
||||||
|
private List<Instruction> replacement = new ArrayList<>();
|
||||||
|
|
||||||
public JavascriptNativeProcessor(ClassHolderSource classSource) {
|
public JavascriptNativeProcessor(ClassHolderSource classSource) {
|
||||||
this.classSource = classSource;
|
this.classSource = classSource;
|
||||||
|
@ -36,6 +36,7 @@ class JavascriptNativeProcessor {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void processProgram(Program program) {
|
public void processProgram(Program program) {
|
||||||
|
this.program = program;
|
||||||
for (int i = 0; i < program.basicBlockCount(); ++i) {
|
for (int i = 0; i < program.basicBlockCount(); ++i) {
|
||||||
BasicBlock block = program.basicBlockAt(i);
|
BasicBlock block = program.basicBlockAt(i);
|
||||||
List<Instruction> instructions = block.getInstructions();
|
List<Instruction> instructions = block.getInstructions();
|
||||||
|
@ -44,10 +45,219 @@ class JavascriptNativeProcessor {
|
||||||
if (!(insn instanceof InvokeInstruction)) {
|
if (!(insn instanceof InvokeInstruction)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
InvokeInstruction invoke = (InvokeInstruction)insn;
|
||||||
|
if (!isJavaScriptClass(invoke.getMethod().getClassName())) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
replacement.clear();
|
||||||
|
MethodHolder method = getMethod(invoke.getMethod());
|
||||||
|
if (method.getAnnotations().get(JSProperty.class.getName()) != null) {
|
||||||
|
if (isProperGetter(method.getDescriptor())) {
|
||||||
|
String propertyName = method.getName().charAt(0) == 'i' ? cutPrefix(method.getName(), 2) :
|
||||||
|
cutPrefix(method.getName(), 3);
|
||||||
|
Variable result = invoke.getReceiver() != null ? program.createVariable() : null;
|
||||||
|
addPropertyGet(propertyName, invoke.getInstance(), result);
|
||||||
|
if (result != null) {
|
||||||
|
result = unwrap(result, method.getResultType());
|
||||||
|
copyVar(result, invoke.getReceiver());
|
||||||
|
}
|
||||||
|
} else if (isProperSetter(method.getDescriptor())) {
|
||||||
|
Variable wrapped = wrap(invoke.getArguments().get(3), method.parameterType(2));
|
||||||
|
addPropertySet(cutPrefix(method.getName(), 3), invoke.getArguments().get(0), wrapped);
|
||||||
|
} else {
|
||||||
|
throw new RuntimeException("Method " + invoke.getMethod() + " is not " +
|
||||||
|
"a proper native JavaScript property declaration");
|
||||||
|
}
|
||||||
|
} else if (method.getAnnotations().get(JSIndexer.class.getName()) != null) {
|
||||||
|
if (isProperGetIndexer(method.getDescriptor())) {
|
||||||
|
Variable result = invoke.getReceiver() != null ? program.createVariable() : null;
|
||||||
|
addIndexerGet(invoke.getInstance(), wrap(invoke.getArguments().get(0),
|
||||||
|
method.parameterType(0)), result);
|
||||||
|
if (result != null) {
|
||||||
|
result = unwrap(result, method.getResultType());
|
||||||
|
copyVar(result, invoke.getReceiver());
|
||||||
|
}
|
||||||
|
} else if (isProperSetIndexer(method.getDescriptor())) {
|
||||||
|
Variable index = wrap(invoke.getArguments().get(0), method.parameterType(0));
|
||||||
|
Variable value = wrap(invoke.getArguments().get(1), method.parameterType(1));
|
||||||
|
addIndexerSet(invoke.getInstance(), index, value);
|
||||||
|
} else {
|
||||||
|
throw new RuntimeException("Method " + invoke.getMethod() + " is not " +
|
||||||
|
"a proper native JavaScript indexer declaration");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!isSupportedType(method.getResultType())) {
|
||||||
|
throw new RuntimeException("Method " + invoke.getMethod() + " is not " +
|
||||||
|
"a proper native JavaScript method declaration");
|
||||||
|
}
|
||||||
|
for (ValueType arg : method.getParameterTypes()) {
|
||||||
|
if (!isSupportedType(arg)) {
|
||||||
|
throw new RuntimeException("Method " + invoke.getMethod() + " is not " +
|
||||||
|
"a proper native JavaScript method declaration");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Variable result = invoke.getReceiver() != null ? program.createVariable() : null;
|
||||||
|
InvokeInstruction newInvoke = new InvokeInstruction();
|
||||||
|
ValueType[] signature = new ValueType[method.parameterCount() + 3];
|
||||||
|
Arrays.fill(signature, ValueType.object(JSObject.class.getName()));
|
||||||
|
newInvoke.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("invoke",
|
||||||
|
signature)));
|
||||||
|
newInvoke.setType(InvocationType.SPECIAL);
|
||||||
|
newInvoke.setReceiver(result);
|
||||||
|
newInvoke.getArguments().add(invoke.getInstance());
|
||||||
|
newInvoke.getArguments().add(addStringWrap(addString(method.getName())));
|
||||||
|
for (int k = 0; k < invoke.getArguments().size(); ++k) {
|
||||||
|
Variable arg = wrap(invoke.getArguments().get(k), method.parameterType(k));
|
||||||
|
newInvoke.getArguments().add(arg);
|
||||||
|
}
|
||||||
|
replacement.add(newInvoke);
|
||||||
|
if (result != null) {
|
||||||
|
result = unwrap(result, method.getResultType());
|
||||||
|
copyVar(result, invoke.getReceiver());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
block.getInstructions().set(j, replacement.get(0));
|
||||||
|
block.getInstructions().addAll(j + 1, replacement.subList(1, replacement.size()));
|
||||||
|
j += replacement.size() - 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addPropertyGet(String propertyName, Variable instance, Variable receiver) {
|
||||||
|
Variable nameVar = addStringWrap(addString(propertyName));
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("get",
|
||||||
|
ValueType.object(JSObject.class.getName()), ValueType.object(JSObject.class.getName()),
|
||||||
|
ValueType.object(JSObject.class.getName()))));
|
||||||
|
insn.setReceiver(receiver);
|
||||||
|
insn.getArguments().add(instance);
|
||||||
|
insn.getArguments().add(nameVar);
|
||||||
|
replacement.add(insn);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addPropertySet(String propertyName, Variable instance, Variable value) {
|
||||||
|
Variable nameVar = addStringWrap(addString(propertyName));
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("set",
|
||||||
|
ValueType.object(JSObject.class.getName()), ValueType.object(JSObject.class.getName()),
|
||||||
|
ValueType.VOID)));
|
||||||
|
insn.getArguments().add(instance);
|
||||||
|
insn.getArguments().add(nameVar);
|
||||||
|
insn.getArguments().add(value);
|
||||||
|
replacement.add(insn);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addIndexerGet(Variable array, Variable index, Variable receiver) {
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("get",
|
||||||
|
ValueType.object(JSObject.class.getName()), ValueType.object(JSObject.class.getName()),
|
||||||
|
ValueType.object(JSObject.class.getName()))));
|
||||||
|
insn.setReceiver(receiver);
|
||||||
|
insn.getArguments().add(array);
|
||||||
|
insn.getArguments().add(index);
|
||||||
|
replacement.add(insn);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addIndexerSet(Variable array, Variable index, Variable value) {
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("set",
|
||||||
|
ValueType.object(JSObject.class.getName()), ValueType.object(JSObject.class.getName()),
|
||||||
|
ValueType.object(JSObject.class.getName()), ValueType.VOID)));
|
||||||
|
insn.getArguments().add(array);
|
||||||
|
insn.getArguments().add(index);
|
||||||
|
insn.getArguments().add(value);
|
||||||
|
replacement.add(insn);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void copyVar(Variable a, Variable b) {
|
||||||
|
AssignInstruction insn = new AssignInstruction();
|
||||||
|
insn.setAssignee(a);
|
||||||
|
insn.setReceiver(b);
|
||||||
|
replacement.add(insn);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Variable addStringWrap(Variable var) {
|
||||||
|
return wrap(var, ValueType.object("java.lang.String"));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Variable addString(String str) {
|
||||||
|
Variable var = program.createVariable();
|
||||||
|
StringConstantInstruction nameInsn = new StringConstantInstruction();
|
||||||
|
nameInsn.setReceiver(var);
|
||||||
|
nameInsn.setConstant(str);
|
||||||
|
replacement.add(nameInsn);
|
||||||
|
return var;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Variable unwrap(Variable var, ValueType type) {
|
||||||
|
if (type instanceof ValueType.Primitive) {
|
||||||
|
switch (((ValueType.Primitive)type).getKind()) {
|
||||||
|
case BOOLEAN:
|
||||||
|
return unwrap(var, "unwrapBoolean", ValueType.BOOLEAN);
|
||||||
|
case BYTE:
|
||||||
|
return unwrap(var, "unwrapByte", ValueType.BYTE);
|
||||||
|
case SHORT:
|
||||||
|
return unwrap(var, "unwrapShort", ValueType.SHORT);
|
||||||
|
case INTEGER:
|
||||||
|
return unwrap(var, "unwrapShort", ValueType.INTEGER);
|
||||||
|
case CHARACTER:
|
||||||
|
return unwrap(var, "unwrapCharacter", ValueType.CHARACTER);
|
||||||
|
case DOUBLE:
|
||||||
|
return unwrap(var, "unwrapDouble", ValueType.DOUBLE);
|
||||||
|
case FLOAT:
|
||||||
|
return unwrap(var, "unwrapFloat", ValueType.FLOAT);
|
||||||
|
case LONG:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if (type instanceof ValueType.Object) {
|
||||||
|
String className = ((ValueType.Object)type).getClassName();
|
||||||
|
if (className.equals(JSObject.class.getName())) {
|
||||||
|
return var;
|
||||||
|
} else if (className.equals("java.lang.String")) {
|
||||||
|
return unwrap(var, "unwrapString", ValueType.object("java.lang.String"));
|
||||||
|
} else {
|
||||||
|
Variable result = program.createVariable();
|
||||||
|
CastInstruction castInsn = new CastInstruction();
|
||||||
|
castInsn.setReceiver(result);
|
||||||
|
castInsn.setValue(var);
|
||||||
|
castInsn.setTargetType(type);
|
||||||
|
replacement.add(castInsn);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new IllegalArgumentException("Unsupported type: " + type);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Variable unwrap(Variable var, String methodName, ValueType resultType) {
|
||||||
|
Variable result = program.createVariable();
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor(methodName,
|
||||||
|
resultType)));
|
||||||
|
insn.getArguments().add(var);
|
||||||
|
insn.setReceiver(result);
|
||||||
|
insn.setType(InvocationType.SPECIAL);
|
||||||
|
replacement.add(insn);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Variable wrap(Variable var, ValueType type) {
|
||||||
|
if (type instanceof ValueType.Object) {
|
||||||
|
String className = ((ValueType.Object)type).getClassName();
|
||||||
|
if (!className.equals("java.lang.String")) {
|
||||||
|
return var;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Variable result = program.createVariable();
|
||||||
|
InvokeInstruction insn = new InvokeInstruction();
|
||||||
|
insn.setMethod(new MethodReference(JS.class.getName(), new MethodDescriptor("wrap", type,
|
||||||
|
ValueType.object(JSObject.class.getName()))));
|
||||||
|
insn.getArguments().add(var);
|
||||||
|
insn.setReceiver(result);
|
||||||
|
insn.setType(InvocationType.SPECIAL);
|
||||||
|
replacement.add(insn);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
private boolean isJavaScriptClass(String className) {
|
private boolean isJavaScriptClass(String className) {
|
||||||
Boolean known = knownJavaScriptClasses.get(className);
|
Boolean known = knownJavaScriptClasses.get(className);
|
||||||
if (known == null) {
|
if (known == null) {
|
||||||
|
@ -69,4 +279,89 @@ class JavascriptNativeProcessor {
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private MethodHolder getMethod(MethodReference ref) {
|
||||||
|
ClassHolder cls = classSource.getClassHolder(ref.getClassName());
|
||||||
|
MethodHolder method = cls.getMethod(ref.getDescriptor());
|
||||||
|
if (method != null) {
|
||||||
|
return method;
|
||||||
|
}
|
||||||
|
for (String iface : cls.getInterfaces()) {
|
||||||
|
method = getMethod(new MethodReference(iface, ref.getDescriptor()));
|
||||||
|
if (method != null) {
|
||||||
|
return method;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isProperGetter(MethodDescriptor desc) {
|
||||||
|
if (desc.parameterCount() > 0 || !isSupportedType(desc.getResultType())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (desc.getResultType().equals(ValueType.BOOLEAN)) {
|
||||||
|
if (isProperPrefix(desc.getName(), "is")) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isProperPrefix(desc.getName(), "get");
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isProperSetter(MethodDescriptor desc) {
|
||||||
|
if (desc.parameterCount() != 1 || !isSupportedType(desc.parameterType(0)) ||
|
||||||
|
desc.getResultType() != ValueType.VOID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return isProperPrefix(desc.getName(), "set");
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isProperPrefix(String name, String prefix) {
|
||||||
|
if (!name.startsWith(prefix) || name.length() == prefix.length()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
char c = name.charAt(prefix.length());
|
||||||
|
return Character.isUpperCase(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isProperGetIndexer(MethodDescriptor desc) {
|
||||||
|
return desc.parameterCount() == 1 && isSupportedType(desc.parameterType(0)) &&
|
||||||
|
isSupportedType(desc.getResultType());
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isProperSetIndexer(MethodDescriptor desc) {
|
||||||
|
return desc.parameterCount() == 2 && isSupportedType(desc.parameterType(0)) &&
|
||||||
|
isSupportedType(desc.parameterType(0)) && desc.getResultType() == ValueType.VOID;
|
||||||
|
}
|
||||||
|
|
||||||
|
private String cutPrefix(String name, int prefixLength) {
|
||||||
|
if (name.length() == prefixLength + 1) {
|
||||||
|
return name.substring(prefixLength).toLowerCase();
|
||||||
|
}
|
||||||
|
char c = name.charAt(prefixLength + 1);
|
||||||
|
if (Character.isUpperCase(c)) {
|
||||||
|
return name.substring(prefixLength);
|
||||||
|
}
|
||||||
|
return Character.toLowerCase(name.charAt(prefixLength)) + name.substring(prefixLength + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isSupportedType(ValueType type) {
|
||||||
|
if (type == ValueType.VOID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (type instanceof ValueType.Primitive) {
|
||||||
|
switch (((ValueType.Primitive)type).getKind()) {
|
||||||
|
case LONG:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} else if (type instanceof ValueType.Array) {
|
||||||
|
return isSupportedType(((ValueType.Array)type).getItemType());
|
||||||
|
} else if (type instanceof ValueType.Object) {
|
||||||
|
String typeName = ((ValueType.Object)type).getClassName();
|
||||||
|
return typeName.equals("java.lang.String") || isJavaScriptClass(typeName);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2014 Alexey Andreev.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.teavm.javascript;
|
||||||
|
|
||||||
|
import org.teavm.model.ClassHolder;
|
||||||
|
import org.teavm.model.ClassHolderSource;
|
||||||
|
import org.teavm.model.MethodHolder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Alexey Andreev
|
||||||
|
*/
|
||||||
|
public class JavascriptProcessedClassSource implements ClassHolderSource {
|
||||||
|
private ClassHolderSource innerSource;
|
||||||
|
|
||||||
|
public JavascriptProcessedClassSource(ClassHolderSource innerSource) {
|
||||||
|
this.innerSource = innerSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ClassHolder getClassHolder(String name) {
|
||||||
|
ClassHolder cls = innerSource.getClassHolder(name);
|
||||||
|
if (cls != null) {
|
||||||
|
transformClass(cls);
|
||||||
|
}
|
||||||
|
return cls;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void transformClass(ClassHolder cls) {
|
||||||
|
JavascriptNativeProcessor processor = new JavascriptNativeProcessor(innerSource);
|
||||||
|
for (MethodHolder method : cls.getMethods()) {
|
||||||
|
if (method.getProgram() != null) {
|
||||||
|
processor.processProgram(method.getProgram());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
178
teavm-core/src/main/java/org/teavm/javascript/ni/JS.java
Normal file
178
teavm-core/src/main/java/org/teavm/javascript/ni/JS.java
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2014 Alexey Andreev.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.teavm.javascript.ni;
|
||||||
|
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Alexey Andreev
|
||||||
|
*/
|
||||||
|
public final class JS {
|
||||||
|
private JS() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public static JSType getType(JSObject obj) {
|
||||||
|
switch (unwrapString(getTypeName(obj))) {
|
||||||
|
case "boolean":
|
||||||
|
return JSType.OBJECT;
|
||||||
|
case "number":
|
||||||
|
return JSType.NUMBER;
|
||||||
|
case "string":
|
||||||
|
return JSType.STRING;
|
||||||
|
case "function":
|
||||||
|
return JSType.FUNCTION;
|
||||||
|
case "object":
|
||||||
|
return JSType.OBJECT;
|
||||||
|
case "undefined":
|
||||||
|
return JSType.UNDEFINED;
|
||||||
|
}
|
||||||
|
throw new AssertionError("Unexpected type");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static native <T extends JSObject> JSArray<T> createArray(int size);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject getTypeName(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject getGlobal();
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject wrap(String str);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject wrap(char c);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject wrap(int num);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject wrap(float num);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject wrap(double num);
|
||||||
|
|
||||||
|
public static <T extends JSObject> JSArray<T> wrap(T[] array) {
|
||||||
|
JSArray<T> result = createArray(array.length);
|
||||||
|
for (int i = 0; i < array.length; ++i) {
|
||||||
|
result.set(i, array[i]);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T extends JSObject> JSArray<JSArray<T>> wrap(T[][] array) {
|
||||||
|
JSArray<JSArray<T>> result = createArray(array.length);
|
||||||
|
for (int i = 0; i < array.length; ++i) {
|
||||||
|
result.set(i, wrap(array[i]));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T extends JSObject> JSArray<JSArray<JSArray<T>>> wrap(T[][][] array) {
|
||||||
|
JSArray<JSArray<JSArray<T>>> result = createArray(array.length);
|
||||||
|
for (int i = 0; i < array.length; ++i) {
|
||||||
|
result.set(i, wrap(array[i]));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native boolean unwrapBoolean(JSObject obj);
|
||||||
|
|
||||||
|
public static byte unwrapByte(JSObject obj) {
|
||||||
|
return (byte)unwrapInt(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static short unwrapShort(JSObject obj) {
|
||||||
|
return (short)unwrapInt(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native int unwrapInt(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native float unwrapFloat(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native double unwrapDouble(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native String unwrapString(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native char unwrapCharacter(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native boolean isUndefined(JSObject obj);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c,
|
||||||
|
JSObject d);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c,
|
||||||
|
JSObject d, JSObject e);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c,
|
||||||
|
JSObject d, JSObject e, JSObject f);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c,
|
||||||
|
JSObject d, JSObject e, JSObject f, JSObject g);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject invoke(JSObject instance, JSObject method, JSObject a, JSObject b, JSObject c,
|
||||||
|
JSObject d, JSObject e, JSObject f, JSObject g, JSObject h);
|
||||||
|
|
||||||
|
public static <T extends JSObject> Iterable<T> iterate(final JSArray<T> array) {
|
||||||
|
return new Iterable<T>() {
|
||||||
|
@Override public Iterator<T> iterator() {
|
||||||
|
return new Iterator<T>() {
|
||||||
|
int index = 0;
|
||||||
|
@Override public boolean hasNext() {
|
||||||
|
return index < array.getLength();
|
||||||
|
}
|
||||||
|
@Override public T next() {
|
||||||
|
return array.get(index++);
|
||||||
|
}
|
||||||
|
@Override public void remove() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native JSObject get(JSObject instance, JSObject index);
|
||||||
|
|
||||||
|
@GeneratedBy(JSNativeGenerator.class)
|
||||||
|
public static native void set(JSObject instance, JSObject index, JSObject obj);
|
||||||
|
}
|
|
@ -19,12 +19,13 @@ package org.teavm.javascript.ni;
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public interface JSArray<T extends JSObject> extends JSObject, Iterable<T> {
|
public interface JSArray<T extends JSObject> extends JSObject {
|
||||||
@JSProperty
|
@JSProperty
|
||||||
int getLength();
|
int getLength();
|
||||||
|
|
||||||
@Override
|
@JSIndexer
|
||||||
T get(int index);
|
T get(int index);
|
||||||
|
|
||||||
<S extends T> JSArray<S> cast();
|
@JSIndexer
|
||||||
|
void set(int index, T value);
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,7 +24,7 @@ public interface JSFunction extends JSObject {
|
||||||
int getLength();
|
int getLength();
|
||||||
|
|
||||||
@JSProperty
|
@JSProperty
|
||||||
JSString getName();
|
String getName();
|
||||||
|
|
||||||
JSObject call(JSObject thisArg);
|
JSObject call(JSObject thisArg);
|
||||||
|
|
||||||
|
|
|
@ -15,10 +15,16 @@
|
||||||
*/
|
*/
|
||||||
package org.teavm.javascript.ni;
|
package org.teavm.javascript.ni;
|
||||||
|
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public interface JSString extends JSObject {
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target(ElementType.METHOD)
|
||||||
|
public @interface JSIndexer {
|
||||||
}
|
}
|
|
@ -24,7 +24,7 @@ import org.teavm.model.MethodReference;
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public class JSRootNativeGenerator implements Generator {
|
public class JSNativeGenerator implements Generator {
|
||||||
@Override
|
@Override
|
||||||
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef)
|
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
@ -42,17 +42,53 @@ public class JSRootNativeGenerator implements Generator {
|
||||||
writer.append("return ").append(context.getParameterName(1)).append(";").softNewLine();
|
writer.append("return ").append(context.getParameterName(1)).append(";").softNewLine();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case "get":
|
||||||
|
writer.append("return ").append(context.getParameterName(1)).append("[")
|
||||||
|
.append(context.getParameterName(2)).append("];").softNewLine();
|
||||||
|
break;
|
||||||
|
case "set":
|
||||||
|
writer.append(context.getParameterName(1)).append("[").append(context.getParameterName(2))
|
||||||
|
.append("] = ").append(context.getParameterName(3)).softNewLine();
|
||||||
|
break;
|
||||||
|
case "invoke":
|
||||||
|
generateInvoke(context, writer, methodRef.parameterCount() - 2);
|
||||||
|
break;
|
||||||
|
case "isUndefined":
|
||||||
|
writer.append("return ").append(context.getParameterName(1)).append(" === undefined;");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (methodRef.getName().startsWith("unwrap")) {
|
||||||
|
if (methodRef.getDescriptor().getResultType().isObject("java.lang.String")) {
|
||||||
|
writer.append("return $rt_str(").append(context.getParameterName(1)).append(");")
|
||||||
|
.softNewLine();
|
||||||
|
} else {
|
||||||
|
writer.append("return ").append(context.getParameterName(1)).append(";").softNewLine();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void generateWrapString(GeneratorContext context, SourceWriter writer) throws IOException {
|
private void generateWrapString(GeneratorContext context, SourceWriter writer) throws IOException {
|
||||||
FieldReference charsField = new FieldReference("java.lang.String", "characters");
|
FieldReference charsField = new FieldReference("java.lang.String", "characters");
|
||||||
writer.append("var result = \"\";").softNewLine();
|
writer.append("var result = \"\";").softNewLine();
|
||||||
writer.append("var data = ").append(context.getParameterName(1))
|
writer.append("var data = ").append(context.getParameterName(1)).append('.')
|
||||||
.appendField(charsField).append(".data;").softNewLine();
|
.appendField(charsField).append(".data;").softNewLine();
|
||||||
writer.append("for (var i = 0; i < data.length; i = (i + 1) | 0) {").indent().softNewLine();
|
writer.append("for (var i = 0; i < data.length; i = (i + 1) | 0) {").indent().softNewLine();
|
||||||
writer.append("result += String.fromCharCode(data[i]);").softNewLine();
|
writer.append("result += String.fromCharCode(data[i]);").softNewLine();
|
||||||
writer.outdent().append("}").softNewLine();
|
writer.outdent().append("}").softNewLine();
|
||||||
writer.append("return result;").softNewLine();
|
writer.append("return result;").softNewLine();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void generateInvoke(GeneratorContext context, SourceWriter writer, int argNum) throws IOException {
|
||||||
|
writer.append("return ").append(context.getParameterName(1)).append("[")
|
||||||
|
.append(context.getParameterName(2)).append("](");
|
||||||
|
for (int i = 0; i < argNum; ++i) {
|
||||||
|
if (i > 0) {
|
||||||
|
writer.append(",").ws();
|
||||||
|
}
|
||||||
|
writer.append(context.getParameterName(i + 3));
|
||||||
|
}
|
||||||
|
writer.append(");").softNewLine();
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -20,78 +20,6 @@ package org.teavm.javascript.ni;
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public interface JSObject {
|
public interface JSObject {
|
||||||
@JSProperty
|
@Override
|
||||||
JSObject getPrototype();
|
String toString();
|
||||||
|
|
||||||
JSObject get(String name);
|
|
||||||
|
|
||||||
void set(String name, JSObject obj);
|
|
||||||
|
|
||||||
JSObject get(JSString name);
|
|
||||||
|
|
||||||
void set(JSString name, JSObject obj);
|
|
||||||
|
|
||||||
JSObject get(int index);
|
|
||||||
|
|
||||||
void set(int index, JSObject obj);
|
|
||||||
|
|
||||||
JSObject invoke(String method);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c, JSObject d);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f, JSObject g);
|
|
||||||
|
|
||||||
JSObject invoke(String method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f,
|
|
||||||
JSObject g, JSObject h);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c, JSObject d);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f,
|
|
||||||
JSObject g);
|
|
||||||
|
|
||||||
JSObject invoke(JSString method, JSObject a, JSObject b, JSObject c, JSObject d, JSObject e, JSObject f,
|
|
||||||
JSObject g, JSObject h);
|
|
||||||
|
|
||||||
@JSProperty
|
|
||||||
boolean hasOwnProperty(String property);
|
|
||||||
|
|
||||||
@JSProperty
|
|
||||||
boolean hasOwnProperty(JSString property);
|
|
||||||
|
|
||||||
boolean asBoolean();
|
|
||||||
|
|
||||||
int asInteger();
|
|
||||||
|
|
||||||
double asNumber();
|
|
||||||
|
|
||||||
String asString();
|
|
||||||
|
|
||||||
boolean isNull();
|
|
||||||
|
|
||||||
boolean isUndefined();
|
|
||||||
|
|
||||||
@JSMethod("toString")
|
|
||||||
JSString toJSString();
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,61 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2014 Alexey Andreev.
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
package org.teavm.javascript.ni;
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @author Alexey Andreev
|
|
||||||
*/
|
|
||||||
public final class JSRoot {
|
|
||||||
private JSRoot() {
|
|
||||||
}
|
|
||||||
|
|
||||||
public static JSType getType(JSObject obj) {
|
|
||||||
switch (getTypeName(obj).asString()) {
|
|
||||||
case "boolean":
|
|
||||||
return JSType.OBJECT;
|
|
||||||
case "number":
|
|
||||||
return JSType.NUMBER;
|
|
||||||
case "string":
|
|
||||||
return JSType.STRING;
|
|
||||||
case "function":
|
|
||||||
return JSType.FUNCTION;
|
|
||||||
case "object":
|
|
||||||
return JSType.OBJECT;
|
|
||||||
case "undefined":
|
|
||||||
return JSType.UNDEFINED;
|
|
||||||
}
|
|
||||||
throw new AssertionError("Unexpected type");
|
|
||||||
}
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSString getTypeName(JSObject obj);
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSObject getGlobal();
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSString wrap(String str);
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSNumber wrap(int num);
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSNumber wrap(float num);
|
|
||||||
|
|
||||||
@GeneratedBy(JSRootNativeGenerator.class)
|
|
||||||
public static native JSNumber wrap(double num);
|
|
||||||
}
|
|
|
@ -23,6 +23,19 @@
|
||||||
</parent>
|
</parent>
|
||||||
<artifactId>teavm-samples</artifactId>
|
<artifactId>teavm-samples</artifactId>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.teavm</groupId>
|
||||||
|
<artifactId>teavm-core</artifactId>
|
||||||
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.teavm</groupId>
|
||||||
|
<artifactId>teavm-dom</artifactId>
|
||||||
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
<plugins>
|
<plugins>
|
||||||
<plugin>
|
<plugin>
|
||||||
|
|
|
@ -15,22 +15,48 @@
|
||||||
*/
|
*/
|
||||||
package org.teavm.samples;
|
package org.teavm.samples;
|
||||||
|
|
||||||
|
import org.teavm.dom.core.Document;
|
||||||
|
import org.teavm.dom.core.Element;
|
||||||
|
import org.teavm.dom.core.Window;
|
||||||
|
import org.teavm.javascript.ni.JS;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public class HelloWorld {
|
public class HelloWorld {
|
||||||
|
private static Window window;
|
||||||
|
private static Document document;
|
||||||
|
private static Element body;
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
System.out.println("Hello, world!");
|
window = (Window)JS.getGlobal();
|
||||||
System.out.println("Here is the Fibonacci sequence:");
|
document = window.getDocument();
|
||||||
|
body = document.getDocumentElement().getElementsByTagName("body").item(0);
|
||||||
|
|
||||||
|
println("Hello, world!");
|
||||||
|
println("Here is the Fibonacci sequence:");
|
||||||
long a = 0;
|
long a = 0;
|
||||||
long b = 1;
|
long b = 1;
|
||||||
for (int i = 0; i < 70; ++i) {
|
for (int i = 0; i < 70; ++i) {
|
||||||
System.out.println(a);
|
println(a);
|
||||||
long c = a + b;
|
long c = a + b;
|
||||||
a = b;
|
a = b;
|
||||||
b = c;
|
b = c;
|
||||||
}
|
}
|
||||||
System.out.println("And so on...");
|
println("And so on...");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void println(Object obj) {
|
||||||
|
Element elem = document.createElement("div");
|
||||||
|
elem.appendChild(document.createTextNode(String.valueOf(obj)));
|
||||||
|
body.appendChild(elem);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void println(long val) {
|
||||||
|
Element elem = document.createElement("div");
|
||||||
|
elem.appendChild(document.createTextNode(String.valueOf(val)));
|
||||||
|
body.appendChild(elem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user