/* @VaadinApache2LicenseForJavaFiles@ */ package com.vaadin.terminal.gwt.client.ui; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; import com.google.gwt.event.shared.GwtEvent; import com.google.gwt.event.shared.HandlerManager; import com.google.web.bindery.event.shared.HandlerRegistration; import com.vaadin.shared.communication.ClientRpc; import com.vaadin.shared.communication.SharedState; import com.vaadin.terminal.gwt.client.ApplicationConnection; import com.vaadin.terminal.gwt.client.ServerConnector; import com.vaadin.terminal.gwt.client.Util; import com.vaadin.terminal.gwt.client.VConsole; import com.vaadin.terminal.gwt.client.communication.StateChangeEvent; import com.vaadin.terminal.gwt.client.communication.StateChangeEvent.StateChangeHandler; /** * An abstract implementation of Connector. * * @author Vaadin Ltd * @version @VERSION@ * @since 7.0.0 * */ public abstract class AbstractConnector implements ServerConnector, StateChangeHandler { private ApplicationConnection connection; private String id; private HandlerManager handlerManager; private Map> rpcImplementations; private final boolean debugLogging = false; private SharedState state; private ServerConnector parent; /** * Temporary storage for last enabled state to be able to see if it has * changed. Can be removed once we are able to listen specifically for * enabled changes in the state. Widget.isEnabled() cannot be used as all * Widgets do not implement HasEnabled */ private boolean lastEnabledState = true; private List children; /* * (non-Javadoc) * * @see com.vaadin.terminal.gwt.client.VPaintable#getConnection() */ @Override public final ApplicationConnection getConnection() { return connection; } /* * (non-Javadoc) * * @see com.vaadin.terminal.gwt.client.Connector#getId() */ @Override public String getConnectorId() { return id; } /** * Called once by the framework to initialize the connector. *

* Note that the shared state is not yet available when this method is * called. *

* Connector classes should override {@link #init()} instead of this method. */ @Override public final void doInit(String connectorId, ApplicationConnection connection) { this.connection = connection; id = connectorId; addStateChangeHandler(this); init(); } /** * Called when the connector has been initialized. Override this method to * perform initialization of the connector. */ // FIXME: It might make sense to make this abstract to force users to // use init instead of constructor, where connection and id has not yet been // set. protected void init() { } /** * Registers an implementation for a server to client RPC interface. * * Multiple registrations can be made for a single interface, in which case * all of them receive corresponding RPC calls. * * @param rpcInterface * RPC interface * @param implementation * implementation that should receive RPC calls * @param * The type of the RPC interface that is being registered */ protected void registerRpc(Class rpcInterface, T implementation) { String rpcInterfaceId = rpcInterface.getName().replaceAll("\\$", "."); if (null == rpcImplementations) { rpcImplementations = new HashMap>(); } if (null == rpcImplementations.get(rpcInterfaceId)) { rpcImplementations.put(rpcInterfaceId, new ArrayList()); } rpcImplementations.get(rpcInterfaceId).add(implementation); } /** * Unregisters an implementation for a server to client RPC interface. * * @param rpcInterface * RPC interface * @param implementation * implementation to unregister */ protected void unregisterRpc(Class rpcInterface, T implementation) { String rpcInterfaceId = rpcInterface.getName().replaceAll("\\$", "."); if (null != rpcImplementations && null != rpcImplementations.get(rpcInterfaceId)) { rpcImplementations.get(rpcInterfaceId).remove(implementation); } } @Override public Collection getRpcImplementations( String rpcInterfaceId) { if (null == rpcImplementations) { return Collections.emptyList(); } return (Collection) rpcImplementations.get(rpcInterfaceId); } @Override public void fireEvent(GwtEvent event) { if (handlerManager != null) { handlerManager.fireEvent(event); } } protected HandlerManager ensureHandlerManager() { if (handlerManager == null) { handlerManager = new HandlerManager(this); } return handlerManager; } @Override public HandlerRegistration addStateChangeHandler(StateChangeHandler handler) { return ensureHandlerManager() .addHandler(StateChangeEvent.TYPE, handler); } @Override public void onStateChanged(StateChangeEvent stateChangeEvent) { if (debugLogging) { VConsole.log("State change event for " + Util.getConnectorString(stateChangeEvent.getConnector()) + " received by " + Util.getConnectorString(this)); } updateEnabledState(isEnabled()); } /* * (non-Javadoc) * * @see com.vaadin.terminal.gwt.client.ServerConnector#onUnregister() */ @Override public void onUnregister() { if (debugLogging) { VConsole.log("Unregistered connector " + Util.getConnectorString(this)); } } /** * Returns the shared state object for this connector. * * Override this method to define the shared state type for your connector. * * @return the current shared state (never null) */ @Override public SharedState getState() { if (state == null) { state = createState(); } return state; } /** * Creates a state object with default values for this connector. The * created state object must be compatible with the return type of * {@link #getState()}. The default implementation creates a state object * using GWT.create() using the defined return type of {@link #getState()}. * * @return A new state object */ protected SharedState createState() { return ConnectorStateFactory.createState(getClass()); } @Override public ServerConnector getParent() { return parent; } @Override public void setParent(ServerConnector parent) { this.parent = parent; } @Override public List getChildren() { if (children == null) { return Collections.emptyList(); } return children; } @Override public void setChildren(List children) { this.children = children; } @Override public boolean isEnabled() { if (!getState().isEnabled()) { return false; } if (getParent() == null) { return true; } else { return getParent().isEnabled(); } } @Override public void updateEnabledState(boolean enabledState) { if (lastEnabledState == enabledState) { return; } lastEnabledState = enabledState; for (ServerConnector c : getChildren()) { // Update children as they might be affected by the enabled state of // their parent c.updateEnabledState(c.isEnabled()); } } }