Java tutorial
/** * HostedJavaMethod.java * (c) Radim Loskot and Radek Burget, 2013-2014 * * ScriptBox is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * ScriptBox is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with ScriptBox. If not, see <http://www.gnu.org/licenses/>. * */ package com.jsen.javascript.java; import java.lang.reflect.Method; import java.util.HashSet; import java.util.Set; import javax.script.ScriptException; import org.apache.commons.lang3.ClassUtils; import org.mozilla.javascript.BaseFunction; import org.mozilla.javascript.ConsString; import org.mozilla.javascript.Context; import org.mozilla.javascript.Function; import org.mozilla.javascript.FunctionObject; import org.mozilla.javascript.Scriptable; import org.mozilla.javascript.TopLevel; import org.mozilla.javascript.TopLevel.Builtins; import org.mozilla.javascript.Undefined; import org.mozilla.javascript.WrappedException; import com.jsen.core.exceptions.FunctionException; import com.jsen.core.exceptions.UnknownException; import com.jsen.core.reflect.ClassFunction; import com.jsen.core.reflect.FunctionMember; import com.jsen.core.reflect.InvocableMember; import com.jsen.javascript.JavaScriptEngine; /** * Creates function scope for native Java method - wraps the native Java method * and makes it callable from the JavaScript. * * @author Radim Loskot * @version 0.9 * @since 0.9 - 21.4.2014 */ public class HostedJavaMethod extends FunctionObject { public final static Method FUNCTION_METHOD; static { Method functionMethod = null; try { functionMethod = HostedJavaMethod.class.getDeclaredMethod("invoke", Context.class, Scriptable.class, Object[].class, Function.class); } catch (NoSuchMethodException e) { } FUNCTION_METHOD = functionMethod; } private static final long serialVersionUID = -5644060115581311028L; private Object object; private Set<FunctionMember> objectFunctions; /** * Constructs new hosted Java method for given parent scope, of the passed object and with the passed overloaded function members. * * @param scope Scope to become the parent scope of this function. * @param object Object with the wrapped method. * @param objectFunctions Overloaded function members which should have the same name. */ public HostedJavaMethod(Scriptable scope, Object object, Set<? extends FunctionMember> objectFunctions) { super(objectFunctions.iterator().next().getMember().getName(), FUNCTION_METHOD, scope); this.object = object; this.objectFunctions = new HashSet<FunctionMember>(); this.objectFunctions.addAll(objectFunctions); TopLevel topLevel = JavaScriptEngine.getObjectTopLevel(scope); Scriptable builtinFunction = topLevel.getBuiltinCtor(Builtins.Function); setPrototype(builtinFunction); } /** * Constructs new hosted Java method for given parent scope, of the passed object and with the passed function member. * * @param scope Scope to become the parent scope of this function. * @param object Object with the wrapped method. * @param objectFunction Function member to be wrapped. */ public HostedJavaMethod(Scriptable scope, Object object, final FunctionMember objectFunction) { this(scope, object, new HashSet<FunctionMember>() { private static final long serialVersionUID = -7257365623995694177L; { add(objectFunction); } }); } /** * Returns attached object overloaded function members. * * @return All associated function members. */ public Set<? extends FunctionMember> getAttachedObjectFunctions() { return objectFunctions; } /** * Adds new overloaded member into this wrapped function object. * * @param objectFunction New function member to be attached. */ public void attachObjectFunction(FunctionMember objectFunction) { this.objectFunctions.add(objectFunction); } @Override public Object call(Context cx, Scriptable scope, Scriptable thisObj, Object[] args) { InvocableMember<?> nearestInvocable = getNearestObjectFunction(args, objectFunctions); if (nearestInvocable == null) { throw new FunctionException("Unable to match nearest function"); } FunctionMember nearestFunctionObject = (FunctionMember) nearestInvocable; Method functionMethod = nearestFunctionObject.getMember(); Class<?> returnType = functionMethod.getReturnType(); Class<?> expectedTypes[] = functionMethod.getParameterTypes(); Object[] castedArgs = castArgs(expectedTypes, args); try { Object returned = functionMethod.invoke(object, castedArgs); return (returnType == Void.class) ? Undefined.instance : returned; } catch (Exception e) { throw new UnknownException("Unable to invoke function " + nearestFunctionObject.getName(), e); } } /** * Invokes given function object. * * @param funObj Function object to be invoked. * @see Function#call(Context, Scriptable, Scriptable, Object[]) */ public static Object invoke(Context cx, Scriptable thisObj, Object[] args, Function funObj) { if (funObj instanceof HostedJavaMethod) { ((HostedJavaMethod) funObj).call(cx, thisObj, thisObj, args); } throw new FunctionException("Function object must be of class HostedJavaMethod"); } /** * Returns the nearest function - searches inside set of functions and * tries to match the appropriate function according to given arguments. * * @param args Arguments that are passed in the function call. * @param invocableMembers Set of the functions. * @return Function member if there was any that matched the passed arguments. */ public static InvocableMember<?> getNearestObjectFunction(Object[] args, Set<? extends InvocableMember<?>> invocableMembers) { Class<?> argsTypes[] = new Class<?>[args.length]; for (int i = 0; i < args.length; i++) { argsTypes[i] = (args[i] != null) ? args[i].getClass() : null; } for (InvocableMember<?> invocableMember : invocableMembers) { Class<?> methodTypes[] = invocableMember.getParameterTypes(); Object[] castedArgs = castArgs(methodTypes, args); if (castedArgs == null) { continue; } boolean isAssignable = ClassFunction.isAssignableTypes(castedArgs, methodTypes); if (isAssignable) { return invocableMember; } } return null; } /** * Casts the given arguments into given expected types. * * @param expectedTypes Types into which should be casted the given arguments. * @param args Arguments to be casted. * @return Array of the casted arguments if casting was successful, otherwise null. */ public static Object[] castArgs(Class<?>[] expectedTypes, Object... args) { if (expectedTypes != null && args != null) { if (expectedTypes.length <= args.length + 1 && expectedTypes.length > 0) { Class<?> lastType = expectedTypes[expectedTypes.length - 1]; if (lastType.isArray()) { Class<?> arrayType = lastType.getComponentType(); boolean maybeVarargs = true; if (expectedTypes.length == args.length) { Object lastArg = args[args.length - 1]; Class<?> lastArgClass = (lastArg != null) ? lastArg.getClass() : null; maybeVarargs = lastArgClass != null && !ClassUtils.isAssignable(lastArgClass, lastType); } if (maybeVarargs) { for (int i = expectedTypes.length - 1; i < args.length; i++) { if (args[i] == null) { continue; } Class<?> argType = args[i].getClass(); if (!ClassUtils.isAssignable(argType, arrayType)) { maybeVarargs = false; break; } } if (maybeVarargs) { Object[] oldArgs = args; args = new Object[expectedTypes.length]; for (int i = 0; i < expectedTypes.length - 1; i++) { args[i] = oldArgs[i]; } Object[] varargs = new Object[oldArgs.length - expectedTypes.length + 1]; for (int i = expectedTypes.length - 1; i < oldArgs.length; i++) { varargs[i - expectedTypes.length + 1] = oldArgs[i]; } args[expectedTypes.length - 1] = varargs; } } } } if (expectedTypes.length == args.length) { Object[] castedArgs = new Object[args.length]; for (int i = 0; i < args.length; i++) { Object arg = args[i]; Class<?> expectedType = expectedTypes[i]; if (arg == null) { castedArgs[i] = null; } else if (arg == Undefined.instance) { castedArgs[i] = null; } else if (arg instanceof ConsString) { castedArgs[i] = ((ConsString) arg).toString(); } else if (arg instanceof Double && (expectedType.equals(Integer.class) || expectedType.equals(int.class) || expectedType.equals(Long.class) || expectedType.equals(long.class))) { castedArgs[i] = ((Double) arg).intValue(); } else { castedArgs[i] = JavaScriptEngine.jsToJava(arg); //castedArgs[i] = Context.jsToJava(castedArgs[i], expectedType); } castedArgs[i] = HostedJavaObject.wrap(expectedTypes[i], castedArgs[i]); } return castedArgs; } } return null; } public static String getFunctionName(Function function) { if (function instanceof BaseFunction) { return "function " + ((BaseFunction) function).getFunctionName() + "()"; } else { return "function ()"; } } /** * Calls JavaScript native function. * * @param function Function to be called. * @param args Call arguments. */ public static void call(final Function function, final Object... args) { Scriptable scope = function.getParentScope(); ObjectTopLevel topLevel = JavaScriptEngine.getObjectTopLevel(scope); if (topLevel != null) { Context cx = topLevel.getBrowserScriptEngine().enterContext(); try { function.call(cx, scope, scope, args); } catch (Exception ex) { try { JavaScriptEngine.throwWrappedScriptException(ex); } catch (ScriptException e) { throw new WrappedException(e); } } finally { Context.exit(); } } } }