diff options
author | Henri Sara <hesara@vaadin.com> | 2015-10-11 07:41:07 -0700 |
---|---|---|
committer | Vaadin Code Review <review@vaadin.com> | 2015-10-23 11:33:42 +0000 |
commit | e776732e571846e566f36490043780f2d8f078a5 (patch) | |
tree | 9aac1d1df5cfb3522d6d9960f58028795ae2675e /server | |
parent | 332fa13334ea81bc8d6f56985ce26233fcd6651e (diff) | |
download | vaadin-framework-e776732e571846e566f36490043780f2d8f078a5.tar.gz vaadin-framework-e776732e571846e566f36490043780f2d8f078a5.zip |
Make Navigator more extensible
The Spring, CDI etc. add-ons need hook points that permit switching
contexts at the right time. This was very hard to do due to the
monolithic nature of navigateTo() - beforeViewChange() is too early
and can be cancelled, afterViewChange() is too late.
Change-Id: I66ffbafe1597b782b4feaf2ebd0de3ca1941a9ae
Diffstat (limited to 'server')
-rw-r--r-- | server/src/com/vaadin/navigator/Navigator.java | 103 |
1 files changed, 91 insertions, 12 deletions
diff --git a/server/src/com/vaadin/navigator/Navigator.java b/server/src/com/vaadin/navigator/Navigator.java index bd2b5711f8..049c29423d 100644 --- a/server/src/com/vaadin/navigator/Navigator.java +++ b/server/src/com/vaadin/navigator/Navigator.java @@ -551,15 +551,77 @@ public class Navigator implements Serializable { protected void navigateTo(View view, String viewName, String parameters) { ViewChangeEvent event = new ViewChangeEvent(this, currentView, view, viewName, parameters); - if (!fireBeforeViewChange(event)) { + boolean navigationAllowed = beforeViewChange(event); + if (!navigationAllowed) { // #10901. Revert URL to previous state if back-button navigation // was canceled - if (currentNavigationState != null) { - getStateManager().setState(currentNavigationState); - } + revertNavigation(); return; } + updateNavigationState(event); + + if (getDisplay() != null) { + getDisplay().showView(view); + } + + switchView(event); + + view.enter(event); + + fireAfterViewChange(event); + } + + /** + * Check whether view change is allowed by view change listeners ( + * {@link ViewChangeListener#beforeViewChange(ViewChangeEvent)}). + * + * This method can be overridden to extend the behavior, and should not be + * called directly except by {@link #navigateTo(View, String, String)}. + * + * @since 7.6 + * @param event + * the event to fire as the before view change event + * @return true if view change is allowed + */ + protected boolean beforeViewChange(ViewChangeEvent event) { + return fireBeforeViewChange(event); + } + + /** + * Revert the changes to the navigation state. When navigation fails, this + * method can be called by {@link #navigateTo(View, String, String)} to + * revert the URL fragment to point to the previous view to which navigation + * succeeded. + * + * This method should only be called by + * {@link #navigateTo(View, String, String)}. Normally it should not be + * overridden, but can be by frameworks that need to hook into view change + * cancellations of this type. + * + * @since 7.6 + */ + protected void revertNavigation() { + if (currentNavigationState != null) { + getStateManager().setState(currentNavigationState); + } + } + + /** + * Update the internal state of the navigator (parameters, previous + * successful URL fragment navigated to) when navigation succeeds. + * + * Normally this method should not be overridden nor called directly from + * application code, but it can be called by a custom implementation of + * {@link #navigateTo(View, String, String)}. + * + * @since 7.6 + * @param event + * a view change event with details of the change + */ + protected void updateNavigationState(ViewChangeEvent event) { + String viewName = event.getViewName(); + String parameters = event.getParameters(); if (null != viewName && getStateManager() != null) { String navigationState = viewName; if (!parameters.isEmpty()) { @@ -570,15 +632,23 @@ public class Navigator implements Serializable { currentNavigationState = navigationState; } } + } - if (display != null) { - display.showView(view); - } - - view.enter(event); - currentView = view; - - fireAfterViewChange(event); + /** + * Update the internal state of the navigator to reflect the actual + * switching of views. + * + * This method should only be called by + * {@link #navigateTo(View, String, String)} between showing the view and + * calling {@link View#enter(ViewChangeEvent)}. If this method is + * overridden, the overriding version must call the super method. + * + * @since 7.6 + * @param event + * a view change event with details of the change + */ + protected void switchView(ViewChangeEvent event) { + currentView = event.getNewView(); } /** @@ -643,6 +713,15 @@ public class Navigator implements Serializable { } /** + * Return the currently active view. + * + * @return current view + */ + public View getCurrentView() { + return currentView; + } + + /** * Fires an event after the current view has changed. * <p> * Listeners are called in registration order. |