Este contenido no está disponible en el idioma seleccionado.
1.7.2. Understanding Nested Conversations
The following code shows the behavior of the hotel booking application with entended behavior for nested conversations. Again, think of the code as a set of steps to be read in sequence.
Example 1.31. RoomPreferenceAction.java
The hotel instance is injected from the conversation context. The hotel is loaded through an extended persistence context so that the entity remains managed throughout the conversation. This allows us to lazily load the availableRooms through an @Factory method by simply walking the association.
| |
When @Begin(nested=true) is encountered, a nested conversation is pushed onto the conversation stack. When executing within a nested conversation, components still have access to all outer conversation state, but setting any values in the nested conversation’s state container does not affect the outer conversation. In addition, nested conversations can exist concurrently stacked on the same outer conversation, allowing independent state for each.
| |
The roomSelection is outjected to the conversation based on the @DataModelSelection . Note that because the nested conversation has an independent context, the roomSelection is only set into the new nested conversation. Should the user select a different preference in another window or tab a new nested conversation would be started.
| |
The @End annotation pops the conversation stack and resumes the outer conversation. The roomSelection is destroyed along with the conversation context.
|
When we begin a nested conversation, it is pushed onto the conversation stack. In the
nestedbooking
example, the conversation stack consists of the external long-running conversation (the booking) and each of the nested conversations (room selections).
Example 1.32. rooms.xhtml
When requested from EL, the #{availableRooms} are loaded by the @Factory method defined in RoomPreferenceAction . The @Factory method will only be executed once to load the values into the current context as a @DataModel instance.
| |
Invoking the #{roomPreference.selectPreference} action results in the row being selected and set into the @DataModelSelection . This value is then outjected to the nested conversation context.
| |
Revising the dates simply returns to the /book.xhtml . Note that we have not yet nested a conversation (no room preference has been selected), so the current conversation can simply be resumed. The <s:button> component simply propagates the current conversation when displaying the /book.xhtml view.
|
Now that you have seen how to nest a conversation, the following code shows how we can confirm the booking of a selected room by extending the behavior of the
HotelBookingAction
.
Example 1.33. HotelBookingAction.java
Annotating an action with @End(root=true) ends the root conversation which effectively destroys the entire conversation stack. When any conversation is ended, its nested conversations are ended as well. As the root is the conversation that started it all, this is a simple way to destroy and release all state associated with a workspace once the booking is confirmed.
| |
The roomSelection is only associated with the booking on user confirmation. While outjecting values to the nested conversation context will not impact the outer conversation, any objects injected from the outer conversation are injected by reference. This means that any changing to these objects will be reflected in the parent conversation as well as other concurrent nested conversations.
| |
By simply annotating the cancellation action with @End(root=true, beforeRedirect=true) we can easily destroy and release all state associated with the workspace prior to redirecting the user back to the hotel selection view.
|
Feel free to deploy the application and test it yourself. Open many windows or tabs, and attempt combinations of various hotel and room preferences. Confirming a booking will always result in the correct hotel and room preference with the nested conversation model.