diff options
-rw-r--r-- | client/src/com/vaadin/client/ApplicationConnection.java | 137 |
1 files changed, 131 insertions, 6 deletions
diff --git a/client/src/com/vaadin/client/ApplicationConnection.java b/client/src/com/vaadin/client/ApplicationConnection.java index a4de5dd4ad..61d9792b82 100644 --- a/client/src/com/vaadin/client/ApplicationConnection.java +++ b/client/src/com/vaadin/client/ApplicationConnection.java @@ -36,6 +36,11 @@ import com.google.gwt.core.client.JsArray; import com.google.gwt.core.client.JsArrayString; import com.google.gwt.core.client.Scheduler; import com.google.gwt.core.client.Scheduler.ScheduledCommand; +import com.google.gwt.event.shared.EventBus; +import com.google.gwt.event.shared.EventHandler; +import com.google.gwt.event.shared.GwtEvent; +import com.google.gwt.event.shared.HandlerRegistration; +import com.google.gwt.event.shared.SimpleEventBus; import com.google.gwt.http.client.Request; import com.google.gwt.http.client.RequestBuilder; import com.google.gwt.http.client.RequestCallback; @@ -232,6 +237,87 @@ public class ApplicationConnection { /** The max timeout the rendering phase may be suspended */ private static final int MAX_SUSPENDED_TIMEOUT = 5000; + /** Event bus for communication events */ + private EventBus eventBus = GWT.create(SimpleEventBus.class); + + /** + * The communication handler methods are called at certain points during + * communication with the server. This allows for making add-ons that keep + * track of different aspects of the communication. + */ + public interface CommunicationHandler extends EventHandler { + void onRequestStarted(RequestStartedEvent e); + + void onRequestEnded(RequestEndedEvent e); + + void onResponseReceived(ResponseReceivedEvent e); + } + + public static class RequestStartedEvent extends + GwtEvent<CommunicationHandler> { + public static Type<CommunicationHandler> TYPE = new Type<CommunicationHandler>(); + + @Override + public com.google.gwt.event.shared.GwtEvent.Type<CommunicationHandler> getAssociatedType() { + return TYPE; + } + + @Override + protected void dispatch(CommunicationHandler handler) { + handler.onRequestStarted(this); + } + } + + public static class RequestEndedEvent extends + GwtEvent<CommunicationHandler> { + public static Type<CommunicationHandler> TYPE = new Type<CommunicationHandler>(); + + @Override + public com.google.gwt.event.shared.GwtEvent.Type<CommunicationHandler> getAssociatedType() { + return TYPE; + } + + @Override + protected void dispatch(CommunicationHandler handler) { + handler.onRequestEnded(this); + } + } + + public static class ResponseReceivedEvent extends + GwtEvent<CommunicationHandler> { + public static Type<CommunicationHandler> TYPE = new Type<CommunicationHandler>(); + + @Override + public com.google.gwt.event.shared.GwtEvent.Type<CommunicationHandler> getAssociatedType() { + return TYPE; + } + + @Override + protected void dispatch(CommunicationHandler handler) { + handler.onResponseReceived(this); + } + } + + /** + * Allows custom handling of communication errors. + */ + public interface CommunicationErrorDelegate { + /** + * Called when a communication error has occurred. Returning + * <code>true</code> from this method suppresses error handling. + * + * @param details + * A string describing the error. + * @param statusCode + * The HTTP status code (e.g. 404, etc). + * @return true if the error reporting should be suppressed, false to + * perform normal error reporting. + */ + public boolean onError(String details, int statusCode); + } + + private CommunicationErrorDelegate communicationErrorDelegate = null; + public static class MultiStepDuration extends Duration { private int previousStep = elapsedMillis(); @@ -562,7 +648,14 @@ public class ApplicationConnection { RequestCallback requestCallback = new RequestCallback() { @Override public void onError(Request request, Throwable exception) { - showCommunicationError(exception.getMessage(), -1); + handleCommunicationError(exception.getMessage(), -1); + } + + private void handleCommunicationError(String details, + int statusCode) { + if (!handleErrorInDelegate(details, statusCode)) { + showCommunicationError(details, statusCode); + } endRequest(); } @@ -577,10 +670,9 @@ public class ApplicationConnection { switch (statusCode) { case 0: - showCommunicationError( + handleCommunicationError( "Invalid status code 0 (server down?)", statusCode); - endRequest(); return; case 401: @@ -623,9 +715,8 @@ public class ApplicationConnection { } else if ((statusCode / 100) == 5) { // Something's wrong on the server, there's nothing the // client can do except maybe try again. - showCommunicationError("Server error. Error code: " + handleCommunicationError("Server error. Error code: " + statusCode, statusCode); - endRequest(); return; } @@ -677,7 +768,7 @@ public class ApplicationConnection { /** * Handles received UIDL JSON text, parsing it, and passing it on to the - * appropriate handlers, while logging timiing information. + * appropriate handlers, while logging timing information. * * @param jsonText * @param statusCode @@ -875,6 +966,7 @@ public class ApplicationConnection { // First one kicks in at 300ms } loadTimer.schedule(300); + eventBus.fireEvent(new RequestStartedEvent()); } protected void endRequest() { @@ -907,6 +999,7 @@ public class ApplicationConnection { } } }); + eventBus.fireEvent(new RequestEndedEvent()); } /** @@ -1060,6 +1153,8 @@ public class ApplicationConnection { } VConsole.log("Handling message from server"); + eventBus.fireEvent(new ResponseReceivedEvent()); + // Handle redirect if (json.containsKey("redirect")) { String url = json.getValueMap("redirect").getString("url"); @@ -2853,4 +2948,34 @@ public class ApplicationConnection { pendingUIDLMessages.clear(); } + private boolean handleErrorInDelegate(String details, int statusCode) { + if (communicationErrorDelegate == null) { + return false; + } + return communicationErrorDelegate.onError(details, statusCode); + } + + /** + * Sets the delegate that is called whenever a communication error occurrs. + * + * @param delegate + * the delegate. + */ + public void setCommunicationErrorDelegate( + CommunicationErrorDelegate delegate) { + communicationErrorDelegate = delegate; + } + + public void setApplicationRunning(boolean running) { + applicationRunning = running; + } + + public boolean isApplicationRunning() { + return applicationRunning; + } + + public <H extends EventHandler> HandlerRegistration addHandler( + GwtEvent.Type<H> type, H handler) { + return eventBus.addHandler(type, handler); + } } |