/* @VaadinApache2LicenseForJavaFiles@ */ package com.vaadin.terminal; import java.io.Serializable; import java.lang.reflect.Method; import java.util.Arrays; import java.util.HashMap; import java.util.HashSet; import java.util.Map; import java.util.Set; import com.vaadin.external.json.JSONArray; import com.vaadin.external.json.JSONException; import com.vaadin.shared.JavaScriptConnectorState; import com.vaadin.terminal.gwt.client.JavaScriptConnectorHelper; import com.vaadin.tools.ReflectTools; import com.vaadin.ui.AbstractJavaScriptComponent; import com.vaadin.ui.JavaScript.JavaScriptCallbackRpc; import com.vaadin.ui.JavaScriptCallback; /** * Internal helper class used to implement functionality common to * {@link AbstractJavaScriptComponent} and {@link AbstractJavaScriptExtension}. * Corresponding support in client-side code is in * {@link JavaScriptConnectorHelper}. *

* You should most likely no use this class directly. * * @author Vaadin Ltd * @version @VERSION@ * @since 7.0.0 */ public class JavaScriptCallbackHelper implements Serializable { private static final Method CALL_METHOD = ReflectTools.findMethod( JavaScriptCallbackRpc.class, "call", String.class, JSONArray.class); private AbstractClientConnector connector; private Map callbacks = new HashMap(); private JavaScriptCallbackRpc javascriptCallbackRpc; public JavaScriptCallbackHelper(AbstractClientConnector connector) { this.connector = connector; } public void registerCallback(String functionName, JavaScriptCallback javaScriptCallback) { callbacks.put(functionName, javaScriptCallback); JavaScriptConnectorState state = getConnectorState(); if (state.getCallbackNames().add(functionName)) { connector.requestRepaint(); } ensureRpc(); } private JavaScriptConnectorState getConnectorState() { JavaScriptConnectorState state = (JavaScriptConnectorState) connector .getState(); return state; } private void ensureRpc() { if (javascriptCallbackRpc == null) { javascriptCallbackRpc = new JavaScriptCallbackRpc() { @Override public void call(String name, JSONArray arguments) { JavaScriptCallback callback = callbacks.get(name); try { callback.call(arguments); } catch (JSONException e) { throw new IllegalArgumentException(e); } } }; connector.registerRpc(javascriptCallbackRpc); } } public void invokeCallback(String name, Object... arguments) { if (callbacks.containsKey(name)) { throw new IllegalStateException( "Can't call callback " + name + " on the client because a callback with the same name is registered on the server."); } JSONArray args = new JSONArray(Arrays.asList(arguments)); connector.addMethodInvocationToQueue( JavaScriptCallbackRpc.class.getName(), CALL_METHOD, new Object[] { name, args }); connector.requestRepaint(); } public void registerRpc(Class rpcInterfaceType) { if (rpcInterfaceType == JavaScriptCallbackRpc.class) { // Ignore return; } Map> rpcInterfaces = getConnectorState() .getRpcInterfaces(); String interfaceName = rpcInterfaceType.getName(); if (!rpcInterfaces.containsKey(interfaceName)) { Set methodNames = new HashSet(); for (Method method : rpcInterfaceType.getMethods()) { methodNames.add(method.getName()); } rpcInterfaces.put(interfaceName, methodNames); connector.requestRepaint(); } } }