I’m calling a @JSFunction annotated method of a ScriptableObject
The JavaScript file
Target = Packages.com.acme.rhino.Target;
function evaluate() {
var t = Target();
t.addModifier("foobar", 1);
return t;
}
The Java File
public class Target extends ScriptableObject {
private static final long serialVersionUID = 1L;
public List<Modifier> modifiers = new LinkedList<>();
@JSConstructor
public Target() {
}
@JSFunction
public void addModifier(final String message, final int value) {
modifiers.add(new Modifier(message, value));
}
public int getValue() {
int sum = 0;
for (final Modifier modifier : modifiers) {
sum += modifier.getValue();
}
return sum;
}
@Override
public String getClassName() {
return "Target";
}
}
But I get
org.mozilla.javascript.EcmaError: TypeError: Cannot find default value for object.
at org.mozilla.javascript.ScriptRuntime.constructError(ScriptRuntime.java:3687)
at org.mozilla.javascript.ScriptRuntime.constructError(ScriptRuntime.java:3665)
at org.mozilla.javascript.ScriptRuntime.typeError(ScriptRuntime.java:3693)
at org.mozilla.javascript.ScriptRuntime.typeError1(ScriptRuntime.java:3705)
at org.mozilla.javascript.ScriptableObject.getDefaultValue(ScriptableObject.java:976 )
at org.mozilla.javascript.ScriptableObject.getDefaultValue(ScriptableObject.java:895 )
at org.mozilla.javascript.ScriptRuntime.toString(ScriptRuntime.java:761)
at org.mozilla.javascript.ScriptRuntime.notFunctionError(ScriptRuntime.java:3774)
at org.mozilla.javascript.ScriptRuntime.getPropFunctionAndThisHelper(ScriptRuntime. java:2269)
at org.mozilla.javascript.ScriptRuntime.getPropFunctionAndThis(ScriptRuntime. java:2251)
at org.mozilla.javascript.optimizer.OptRuntime.callProp0(OptRuntime.java:83)
at org.mozilla.javascript.gen.script_5._c_evaluate_1(script:6)
at org.mozilla.javascript.gen.script_5.call(script)
at org.mozilla.javascript.ContextFactory.doTopCall(ContextFactory.java:394)
at org.mozilla.javascript.ScriptRuntime.doTopCall(ScriptRuntime.java:3091)
at org.mozilla.javascript.gen.script_5.call(script)
and don’t know where to go from there. When I don’t call addModifier method the given code works, and given the error notFunctionError down in the stack trace I think that Rhino doesn’t interpret the given method as a JavaScript Function.
- OSX 10.8.2
- Java 7
- Rhino 1.7R4
Complete Maven project that reproduces the error can be found here
tl;dr see these two alternatives.
The problem with the approach above is that
Target.prototypeis not properly set up in the script scope. See the staticScriptableObject.defineClass()method for details on how to properly define prototypes in a script scope.You have a couple alternatives for providing the
Targetconstructor to your scripts. The first alternative would be to always define theTargetconstructor for all scripts. This works well if you know ahead of time that you wantTargetto be globally available. This basically comes down to the following:If instead you want the script author to decide which constructors are necessary, the second alternative is to provide the
defineClassfunction to scripts. With this function, script authors can “import” any scriptable objects on their class path (which may be more than you want to allow). To provide thedefineClassfunctions to scripts, do the following after entering the context:And then, the JavaScript author makes use of the
Targetconstructor with the following:In both of the above branches, I’ve made a couple other changes that set you up better if you add more to the
Targetconstructor. The zero argument constructor doesn’t need the@JSConstructorannotation. If you later want to have a constructor that accepts arguments, this zero argument constructor will be used as the prototype constructor, and you can use the@JSConstructorannotation on a method that will be used to initialize your object. Depending on how you author this constructor method, it will become important to use thenewkeyword in your JavaScript.In short, the
Packages.com.acme...syntax is not useful for getting access toScriptableObjectconstructors from scripts.