/*

   Licensed to the Apache Software Foundation (ASF) under one or more
   contributor license agreements.  See the NOTICE file distributed with
   this work for additional information regarding copyright ownership.
   The ASF licenses this file to You 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.apache.batik.bridge;

import java.lang.ref.SoftReference;
import java.util.Map;
import java.util.WeakHashMap;

import org.apache.batik.dom.AbstractNode;
import org.apache.batik.script.ScriptEventWrapper;
import org.mozilla.javascript.Context;
import org.mozilla.javascript.ContextAction;
import org.mozilla.javascript.Function;
import org.mozilla.javascript.NativeJavaObject;
import org.mozilla.javascript.NativeObject;
import org.mozilla.javascript.Scriptable;
import org.mozilla.javascript.ScriptableObject;
import org.mozilla.javascript.Undefined;
import org.w3c.dom.events.Event;
import org.w3c.dom.events.EventListener;
import org.w3c.dom.events.EventTarget;

A class that wraps an EventTarget instance to expose it in the Rhino engine. Then calling addEventListener with a Rhino function as parameter should redirect the call to addEventListener with a Java function object calling the Rhino function. This class also allows to pass an ECMAScript (Rhino) object as a parameter instead of a function provided the fact that this object has a handleEvent method.
Author:Christophe Jolif
Version:$Id: EventTargetWrapper.java 1803263 2017-07-28 10:51:01Z ssteiner $
/** * A class that wraps an <code>EventTarget</code> instance to expose * it in the Rhino engine. Then calling <code>addEventListener</code> * with a Rhino function as parameter should redirect the call to * <code>addEventListener</code> with a Java function object calling * the Rhino function. * This class also allows to pass an ECMAScript (Rhino) object as * a parameter instead of a function provided the fact that this object * has a <code>handleEvent</code> method. * @author <a href="mailto:cjolif@ilog.fr">Christophe Jolif</a> * @version $Id: EventTargetWrapper.java 1803263 2017-07-28 10:51:01Z ssteiner $ */
class EventTargetWrapper extends NativeJavaObject {
The Java function object calling the Rhino function.
/** * The Java function object calling the Rhino function. */
static class FunctionEventListener implements EventListener { protected Function function; protected RhinoInterpreter interpreter; FunctionEventListener(Function f, RhinoInterpreter i) { function = f; interpreter = i; } public void handleEvent(Event evt) { Object event; if (evt instanceof ScriptEventWrapper) { event = ((ScriptEventWrapper) evt).getEventObject(); } else { event = evt; } interpreter.callHandler(function, event); } } static class HandleEventListener implements EventListener { public static final String HANDLE_EVENT = "handleEvent"; public Scriptable scriptable; public Object[] array = new Object[1]; public RhinoInterpreter interpreter; HandleEventListener(Scriptable s, RhinoInterpreter interpreter) { scriptable = s; this.interpreter = interpreter; } public void handleEvent(Event evt) { if (evt instanceof ScriptEventWrapper) { array[0] = ((ScriptEventWrapper) evt).getEventObject(); } else { array[0] = evt; } ContextAction handleEventAction = new ContextAction() { public Object run(Context cx) { ScriptableObject.callMethod (scriptable, HANDLE_EVENT, array); return null; } }; interpreter.call(handleEventAction); } } abstract static class FunctionProxy implements Function { protected Function delegate; public FunctionProxy(Function delegate) { this.delegate = delegate; } public Scriptable construct(Context cx, Scriptable scope, Object[] args) { return this.delegate.construct(cx, scope, args); } public String getClassName() { return this.delegate.getClassName(); } public Object get(String name, Scriptable start) { return this.delegate.get(name, start); } public Object get(int index, Scriptable start) { return this.delegate.get(index, start); } public boolean has(String name, Scriptable start) { return this.delegate.has(name, start); } public boolean has(int index, Scriptable start) { return this.delegate.has(index, start); } public void put(String name, Scriptable start, Object value) { this.delegate.put(name, start, value); } public void put(int index, Scriptable start, Object value) { this.delegate.put(index, start, value); } public void delete(String name) { this.delegate.delete(name); } public void delete(int index) { this.delegate.delete(index); } public Scriptable getPrototype() { return this.delegate.getPrototype(); } public void setPrototype(Scriptable prototype) { this.delegate.setPrototype(prototype); } public Scriptable getParentScope() { return this.delegate.getParentScope(); } public void setParentScope(Scriptable parent) { this.delegate.setParentScope(parent); } public Object[] getIds() { return this.delegate.getIds(); } public Object getDefaultValue(Class hint) { return this.delegate.getDefaultValue(hint); } public boolean hasInstance(Scriptable instance) { return this.delegate.hasInstance(instance); } }
This function proxy is delegating most of the job to the underlying NativeJavaMethod object through the FunctionProxy. However to allow user to specify "Function" or objects with an "handleEvent" method as parameter of "addEventListener" it redefines the call method to deal with these cases.
/** * This function proxy is delegating most of the job * to the underlying NativeJavaMethod object through * the FunctionProxy. However to allow user to specify * "Function" or objects with an "handleEvent" method * as parameter of "addEventListener" * it redefines the call method to deal with these * cases. */
static class FunctionAddProxy extends FunctionProxy { protected Map listenerMap; protected RhinoInterpreter interpreter; FunctionAddProxy(RhinoInterpreter interpreter, Function delegate, Map listenerMap) { super(delegate); this.listenerMap = listenerMap; this.interpreter = interpreter; } public Object call(Context ctx, Scriptable scope, Scriptable thisObj, Object[] args) { NativeJavaObject njo = (NativeJavaObject)thisObj; if (args[1] instanceof Function) { EventListener evtListener = null; SoftReference sr = (SoftReference)listenerMap.get(args[1]); if (sr != null) evtListener = (EventListener)sr.get(); if (evtListener == null) { evtListener = new FunctionEventListener ((Function)args[1], interpreter); listenerMap.put(args[1], new SoftReference(evtListener)); } // we need to marshall args Class[] paramTypes = { String.class, Function.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); ((EventTarget)njo.unwrap()).addEventListener ((String)args[0], evtListener, (Boolean) args[2]); return Undefined.instance; } if (args[1] instanceof NativeObject) { EventListener evtListener = null; SoftReference sr = (SoftReference)listenerMap.get(args[1]); if (sr != null) evtListener = (EventListener)sr.get(); if (evtListener == null) { evtListener = new HandleEventListener((Scriptable)args[1], interpreter); listenerMap.put(args[1], new SoftReference(evtListener)); } // we need to marshall args Class[] paramTypes = { String.class, Scriptable.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); ((EventTarget)njo.unwrap()).addEventListener ((String)args[0], evtListener, (Boolean) args[2]); return Undefined.instance; } return delegate.call(ctx, scope, thisObj, args); } } static class FunctionRemoveProxy extends FunctionProxy { public Map listenerMap; FunctionRemoveProxy(Function delegate, Map listenerMap) { super(delegate); this.listenerMap = listenerMap; } public Object call(Context ctx, Scriptable scope, Scriptable thisObj, Object[] args) { NativeJavaObject njo = (NativeJavaObject)thisObj; if (args[1] instanceof Function) { SoftReference sr = (SoftReference)listenerMap.get(args[1]); if (sr == null) return Undefined.instance; EventListener el = (EventListener)sr.get(); if (el == null) return Undefined.instance; // we need to marshall args Class[] paramTypes = { String.class, Function.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); ((EventTarget)njo.unwrap()).removeEventListener ((String)args[0], el, (Boolean) args[2]); return Undefined.instance; } if (args[1] instanceof NativeObject) { SoftReference sr = (SoftReference)listenerMap.get(args[1]); if (sr == null) return Undefined.instance; EventListener el = (EventListener)sr.get(); if (el == null) return Undefined.instance; // we need to marshall args Class[] paramTypes = { String.class, Scriptable.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); ((EventTarget)njo.unwrap()).removeEventListener ((String)args[0], el, (Boolean) args[2]); return Undefined.instance; } return delegate.call(ctx, scope, thisObj, args); } } static class FunctionAddNSProxy extends FunctionProxy { protected Map listenerMap; protected RhinoInterpreter interpreter; FunctionAddNSProxy(RhinoInterpreter interpreter, Function delegate, Map listenerMap) { super(delegate); this.listenerMap = listenerMap; this.interpreter = interpreter; } public Object call(Context ctx, Scriptable scope, Scriptable thisObj, Object[] args) { NativeJavaObject njo = (NativeJavaObject)thisObj; if (args[2] instanceof Function) { EventListener evtListener = new FunctionEventListener ((Function)args[2], interpreter); listenerMap.put(args[2], new SoftReference(evtListener)); // we need to marshall args Class[] paramTypes = { String.class, String.class, Function.class, Boolean.TYPE, Object.class }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); AbstractNode target = (AbstractNode) njo.unwrap(); target.addEventListenerNS ((String)args[0], (String)args[1], evtListener, (Boolean) args[3], args[4]); return Undefined.instance; } if (args[2] instanceof NativeObject) { EventListener evtListener = new HandleEventListener((Scriptable)args[2], interpreter); listenerMap.put(args[2], new SoftReference(evtListener)); // we need to marshall args Class[] paramTypes = { String.class, String.class, Scriptable.class, Boolean.TYPE, Object.class }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); AbstractNode target = (AbstractNode) njo.unwrap(); target.addEventListenerNS ((String)args[0], (String)args[1], evtListener, (Boolean) args[3], args[4]); return Undefined.instance; } return delegate.call(ctx, scope, thisObj, args); } } static class FunctionRemoveNSProxy extends FunctionProxy { protected Map listenerMap; FunctionRemoveNSProxy(Function delegate, Map listenerMap) { super(delegate); this.listenerMap = listenerMap; } public Object call(Context ctx, Scriptable scope, Scriptable thisObj, Object[] args) { NativeJavaObject njo = (NativeJavaObject)thisObj; if (args[2] instanceof Function) { SoftReference sr = (SoftReference)listenerMap.get(args[2]); if (sr == null) return Undefined.instance; EventListener el = (EventListener)sr.get(); if (el == null) return Undefined.instance; // we need to marshall args Class[] paramTypes = { String.class, String.class, Function.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); AbstractNode target = (AbstractNode) njo.unwrap(); target.removeEventListenerNS ((String)args[0], (String)args[1], el, (Boolean) args[3]); return Undefined.instance; } if (args[2] instanceof NativeObject) { SoftReference sr = (SoftReference)listenerMap.get(args[2]); if (sr == null) return Undefined.instance; EventListener el = (EventListener)sr.get(); if (el == null) return Undefined.instance; // we need to marshall args Class[] paramTypes = { String.class, String.class, Scriptable.class, Boolean.TYPE }; for (int i = 0; i < args.length; i++) args[i] = Context.jsToJava(args[i], paramTypes[i]); AbstractNode target = (AbstractNode) njo.unwrap(); target.removeEventListenerNS ((String)args[0], (String)args[1], el, (Boolean) args[3]); return Undefined.instance; } return delegate.call(ctx, scope, thisObj, args); } } // the keys are the underlying Java object, in order // to remove potential memory leaks use a WeakHashMap to allow // to collect entries as soon as the underlying Java object is // not available anymore. protected static WeakHashMap mapOfListenerMap; public static final String ADD_NAME = "addEventListener"; public static final String ADDNS_NAME = "addEventListenerNS"; public static final String REMOVE_NAME = "removeEventListener"; public static final String REMOVENS_NAME = "removeEventListenerNS"; protected RhinoInterpreter interpreter; EventTargetWrapper(Scriptable scope, EventTarget object, RhinoInterpreter interpreter) { super(scope, object, null); this.interpreter = interpreter; }
Overriden Rhino method.
/** * Overriden Rhino method. */
public Object get(String name, Scriptable start) { Object method = super.get(name, start); if (name.equals(ADD_NAME)) { // prevent creating a Map for all JavaScript objects // when we need it only from time to time... method = new FunctionAddProxy(interpreter, (Function)method, initMap()); } else if (name.equals(REMOVE_NAME)) { // prevent creating a Map for all JavaScript objects // when we need it only from time to time... method = new FunctionRemoveProxy ((Function)method, initMap()); } else if (name.equals(ADDNS_NAME)) { method = new FunctionAddNSProxy(interpreter, (Function) method, initMap()); } else if (name.equals(REMOVENS_NAME)) { method = new FunctionRemoveNSProxy((Function) method, initMap()); } return method; } // we have to store the listenerMap in a Map because // several EventTargetWrapper may be created for the exact // same underlying Java object. public Map initMap() { Map map = null; if (mapOfListenerMap == null) mapOfListenerMap = new WeakHashMap(10); if ((map = (Map)mapOfListenerMap.get(unwrap())) == null) { mapOfListenerMap.put(unwrap(), map = new WeakHashMap(2)); } return map; } }