aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/src/com/vaadin/server/AbstractClientConnector.java31
-rw-r--r--server/src/com/vaadin/server/ClientConnector.java86
-rw-r--r--server/src/com/vaadin/server/DragAndDropService.java16
-rw-r--r--server/src/com/vaadin/ui/Component.java8
-rw-r--r--server/tests/src/com/vaadin/tests/server/clientconnector/AttachDetachListeners.java180
5 files changed, 308 insertions, 13 deletions
diff --git a/server/src/com/vaadin/server/AbstractClientConnector.java b/server/src/com/vaadin/server/AbstractClientConnector.java
index 45c6da7e77..8b5bd78d01 100644
--- a/server/src/com/vaadin/server/AbstractClientConnector.java
+++ b/server/src/com/vaadin/server/AbstractClientConnector.java
@@ -92,6 +92,30 @@ public abstract class AbstractClientConnector implements ClientConnector,
private ErrorHandler errorHandler = null;
+ @Override
+ public void addAttachListener(AttachListener listener) {
+ addListener(AttachEvent.ATTACH_EVENT_IDENTIFIER, AttachEvent.class,
+ listener, AttachListener.attachMethod);
+ }
+
+ @Override
+ public void removeAttachListener(AttachListener listener) {
+ removeListener(AttachEvent.ATTACH_EVENT_IDENTIFIER, AttachEvent.class,
+ listener);
+ }
+
+ @Override
+ public void addDetachListener(DetachListener listener) {
+ addListener(DetachEvent.DETACH_EVENT_IDENTIFIER, DetachEvent.class,
+ listener, DetachListener.detachMethod);
+ }
+
+ @Override
+ public void removeDetachListener(DetachListener listener) {
+ removeListener(DetachEvent.DETACH_EVENT_IDENTIFIER, DetachEvent.class,
+ listener);
+ }
+
/**
* @deprecated As of 7.0.0, use {@link #markAsDirty()} instead
*/
@@ -568,10 +592,11 @@ public abstract class AbstractClientConnector implements ClientConnector,
getUI().getConnectorTracker().registerConnector(this);
+ fireEvent(new AttachEvent(this));
+
for (ClientConnector connector : getAllChildrenIterable(this)) {
connector.attach();
}
-
}
/**
@@ -588,6 +613,8 @@ public abstract class AbstractClientConnector implements ClientConnector,
connector.detach();
}
+ fireEvent(new DetachEvent(this));
+
getUI().getConnectorTracker().unregisterConnector(this);
}
@@ -950,7 +977,6 @@ public abstract class AbstractClientConnector implements ClientConnector,
if (eventRouter != null) {
eventRouter.fireEvent(event);
}
-
}
/*
@@ -971,5 +997,4 @@ public abstract class AbstractClientConnector implements ClientConnector,
public void setErrorHandler(ErrorHandler errorHandler) {
this.errorHandler = errorHandler;
}
-
}
diff --git a/server/src/com/vaadin/server/ClientConnector.java b/server/src/com/vaadin/server/ClientConnector.java
index 440f282508..4a09158da5 100644
--- a/server/src/com/vaadin/server/ClientConnector.java
+++ b/server/src/com/vaadin/server/ClientConnector.java
@@ -16,7 +16,11 @@
package com.vaadin.server;
import java.io.IOException;
+import java.io.Serializable;
+import java.lang.reflect.Method;
import java.util.Collection;
+import java.util.EventListener;
+import java.util.EventObject;
import java.util.List;
import org.json.JSONException;
@@ -27,6 +31,7 @@ import com.vaadin.shared.communication.SharedState;
import com.vaadin.ui.Component;
import com.vaadin.ui.ComponentContainer;
import com.vaadin.ui.UI;
+import com.vaadin.util.ReflectTools;
/**
* Interface implemented by all connectors that are capable of communicating
@@ -37,6 +42,81 @@ import com.vaadin.ui.UI;
*
*/
public interface ClientConnector extends Connector {
+
+ public static abstract class ConnectorEvent extends EventObject {
+ public ConnectorEvent(ClientConnector source) {
+ super(source);
+ }
+
+ public ClientConnector getConnector() {
+ return (ClientConnector) getSource();
+ }
+ }
+
+ /**
+ * Event fired after a connector is attached to the application.
+ */
+ public static class AttachEvent extends ConnectorEvent {
+ public static final String ATTACH_EVENT_IDENTIFIER = "clientConnectorAttach";
+
+ public AttachEvent(ClientConnector source) {
+ super(source);
+ }
+ }
+
+ /**
+ * Interface for listening {@link DetachEvent connector detach events}.
+ *
+ */
+ public static interface AttachListener extends EventListener, Serializable {
+ public static final Method attachMethod = ReflectTools.findMethod(
+ AttachListener.class, "attach", AttachEvent.class);
+
+ /**
+ * Called when a AttachListener is notified of a AttachEvent.
+ *
+ * @param event
+ * The attach event that was fired.
+ */
+ public void attach(AttachEvent event);
+ }
+
+ /**
+ * Event fired before a connector is detached from the application.
+ */
+ public static class DetachEvent extends ConnectorEvent {
+ public static final String DETACH_EVENT_IDENTIFIER = "clientConnectorDetach";
+
+ public DetachEvent(ClientConnector source) {
+ super(source);
+ }
+ }
+
+ /**
+ * Interface for listening {@link DetachEvent connector detach events}.
+ *
+ */
+ public static interface DetachListener extends EventListener, Serializable {
+ public static final Method detachMethod = ReflectTools.findMethod(
+ DetachListener.class, "detach", DetachEvent.class);
+
+ /**
+ * Called when a DetachListener is notified of a DetachEvent.
+ *
+ * @param event
+ * The detach event that was fired.
+ */
+ public void detach(DetachEvent event);
+ }
+
+ public void addAttachListener(AttachListener listener);
+
+ public void removeAttachListener(AttachListener listener);
+
+ public void addDetachListener(DetachListener listener);
+
+ public void removeDetachListener(DetachListener listener);
+
/**
* An error event for connector related errors. Use {@link #getConnector()}
* to find the connector where the error occurred or {@link #getComponent()}
@@ -174,13 +254,13 @@ public interface ClientConnector extends Connector {
public void attach();
/**
- * Notifies the component that it is detached from the application.
+ * Notifies the connector that it is detached from the application.
*
* <p>
* The caller of this method is {@link #setParent(ClientConnector)} if the
* parent is in the application. When the parent is detached from the
- * application it is its response to call {@link #detach()} for all the
- * children and to detach itself from the terminal.
+ * application it is its responsibility to call {@link #detach()} for each
+ * of its children.
* </p>
*/
public void detach();
diff --git a/server/src/com/vaadin/server/DragAndDropService.java b/server/src/com/vaadin/server/DragAndDropService.java
index 0fa39fa0e8..529cce670e 100644
--- a/server/src/com/vaadin/server/DragAndDropService.java
+++ b/server/src/com/vaadin/server/DragAndDropService.java
@@ -358,4 +358,20 @@ public class DragAndDropService implements VariableOwner, ClientConnector {
public void setErrorHandler(ErrorHandler errorHandler) {
this.errorHandler = errorHandler;
}
+
+ @Override
+ public void addAttachListener(AttachListener listener) {
+ }
+
+ @Override
+ public void removeAttachListener(AttachListener listener) {
+ }
+
+ @Override
+ public void addDetachListener(DetachListener listener) {
+ }
+
+ @Override
+ public void removeDetachListener(DetachListener listener) {
+ }
}
diff --git a/server/src/com/vaadin/ui/Component.java b/server/src/com/vaadin/ui/Component.java
index 85140a8351..afde1cdbdf 100644
--- a/server/src/com/vaadin/ui/Component.java
+++ b/server/src/com/vaadin/ui/Component.java
@@ -18,7 +18,6 @@ package com.vaadin.ui;
import java.io.Serializable;
import java.util.EventListener;
-import java.util.EventObject;
import java.util.Locale;
import com.vaadin.event.FieldEvents;
@@ -580,11 +579,6 @@ public interface Component extends ClientConnector, Sizeable, Serializable {
* application, the {@code attach()} is called immediately from
* {@link #setParent(Component)}.
* </p>
- * <p>
- * This method must call {@link UI#componentAttached(Component)} to let the
- * UI know that a new Component has been attached.
- * </p>
- *
*
* <pre>
* public class AttachExample extends CustomComponent {
@@ -735,7 +729,7 @@ public interface Component extends ClientConnector, Sizeable, Serializable {
* @see Component.Listener
*/
@SuppressWarnings("serial")
- public static class Event extends EventObject {
+ public static class Event extends ConnectorEvent {
/**
* Constructs a new event with the specified source component.
diff --git a/server/tests/src/com/vaadin/tests/server/clientconnector/AttachDetachListeners.java b/server/tests/src/com/vaadin/tests/server/clientconnector/AttachDetachListeners.java
new file mode 100644
index 0000000000..f5b940d97b
--- /dev/null
+++ b/server/tests/src/com/vaadin/tests/server/clientconnector/AttachDetachListeners.java
@@ -0,0 +1,180 @@
+/*
+@VaadinApache2LicenseForJavaFiles@
+ */
+
+package com.vaadin.tests.server.clientconnector;
+
+import org.easymock.EasyMock;
+import org.easymock.IArgumentMatcher;
+import org.easymock.IMocksControl;
+import org.junit.Before;
+import org.junit.Test;
+
+import com.vaadin.server.ClientConnector.AttachEvent;
+import com.vaadin.server.ClientConnector.AttachListener;
+import com.vaadin.server.ClientConnector.ConnectorEvent;
+import com.vaadin.server.ClientConnector.DetachEvent;
+import com.vaadin.server.ClientConnector.DetachListener;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.server.VaadinService;
+import com.vaadin.server.VaadinSession;
+import com.vaadin.ui.Component;
+import com.vaadin.ui.CssLayout;
+import com.vaadin.ui.Label;
+import com.vaadin.ui.Layout;
+import com.vaadin.ui.UI;
+
+public class AttachDetachListeners {
+
+ private IMocksControl control;
+
+ private VaadinSession session;
+ private UI ui;
+ private Layout content;
+ private Component component;
+
+ AttachListener attachListener;
+ DetachListener detachListener;
+
+ @Before
+ public void setUp() {
+ control = EasyMock.createStrictControl();
+
+ session = new VaadinSession(control.createMock(VaadinService.class));
+
+ ui = new UI() {
+ @Override
+ protected void init(VaadinRequest request) {
+ }
+ };
+ content = new CssLayout();
+ component = new Label();
+
+ attachListener = control.createMock(AttachListener.class);
+ detachListener = control.createMock(DetachListener.class);
+ }
+
+ @Test
+ public void attachListeners_setSessionLast() {
+ setupAttachListeners();
+
+ ui.setContent(content);
+ content.addComponent(component);
+ ui.setSession(session);
+
+ control.verify();
+ }
+
+ @Test
+ public void attachListeners_setSessionFirst() {
+ setupAttachListeners();
+
+ ui.setSession(session);
+ ui.setContent(content);
+ content.addComponent(component);
+
+ control.verify();
+ }
+
+ @Test
+ public void attachListeners_setSessionBetween() {
+ setupAttachListeners();
+
+ ui.setContent(content);
+ ui.setSession(session);
+ content.addComponent(component);
+
+ control.verify();
+ }
+
+ @Test
+ public void detachListeners_setSessionNull() {
+ setupDetachListeners();
+
+ ui.setContent(content);
+ content.addComponent(component);
+ ui.setSession(null);
+
+ control.verify();
+ }
+
+ @Test
+ public void detachListeners_removeComponent() {
+ setupDetachListeners();
+
+ ui.setContent(content);
+ content.addComponent(component);
+ content.removeAllComponents();
+ ui.setSession(null);
+
+ control.verify();
+ }
+
+ @Test
+ public void detachListeners_setContentNull() {
+ setupDetachListeners();
+
+ ui.setContent(content);
+ content.addComponent(component);
+ ui.setContent(null);
+ ui.setSession(null);
+
+ control.verify();
+ }
+
+ public static class EventEquals<E extends ConnectorEvent> implements
+ IArgumentMatcher {
+
+ private E expected;
+
+ public EventEquals(E expected) {
+ this.expected = expected;
+ }
+
+ @Override
+ public void appendTo(StringBuffer buffer) {
+ buffer.append("EventEquals(");
+ buffer.append("expected " + expected.getClass().getSimpleName()
+ + " with connector " + expected.getConnector());
+ buffer.append(")");
+ }
+
+ @Override
+ public boolean matches(Object argument) {
+ return expected.getClass().isInstance(argument)
+ && ((ConnectorEvent) argument).getConnector() == expected
+ .getConnector();
+ }
+ }
+
+ public static <E extends ConnectorEvent> E eventEquals(E expected) {
+ EasyMock.reportMatcher(new EventEquals<E>(expected));
+ return null;
+ }
+
+ private void setupDetachListeners() {
+ detachListener.detach(eventEquals(new DetachEvent(component)));
+ detachListener.detach(eventEquals(new DetachEvent(content)));
+ detachListener.detach(eventEquals(new DetachEvent(ui)));
+
+ control.replay();
+
+ ui.addDetachListener(detachListener);
+ content.addDetachListener(detachListener);
+ component.addDetachListener(detachListener);
+
+ ui.setSession(session);
+ }
+
+ private void setupAttachListeners() {
+ attachListener.attach(eventEquals(new AttachEvent(ui)));
+ attachListener.attach(eventEquals(new AttachEvent(content)));
+ attachListener.attach(eventEquals(new AttachEvent(component)));
+
+ control.replay();
+
+ ui.addAttachListener(attachListener);
+ content.addAttachListener(attachListener);
+ component.addAttachListener(attachListener);
+ }
+}