Abstract
A question that often comes up is when and where to use a particular scope in JSF. This has become even muddier with the addition of CDI support in JSF. In the upcoming release of Java EE 7, there will be even more scopes to consider. This example code demonstrates using@ViewScoped
and @ConversationScoped
.
Explanation
@ViewScoped
If you want to have the information on the page to update, and you are not navigating away from the page; a clear choice is the use of @ViewScoped. This is particularly true when the page is heavily dependent on AJAX. This allows partial-page submission for validation for example without losing the data contained on the page.@ConversationScoped
A @ConversationScoped scoped bean is for a long running process where there is a definite beginning and end. A perfect example is a wizard, or checking out of an online store.The conversation begins when you click "Pay" and ends when you are given your receipt, or order information. Another example would be a complex series of wizards where there are multiple pages to complete the process. The point to keep in mind is that you have a clear bounds to start and end with.
Description
In this example code, I have a list of family members which are in different scopes depending on which example you pick. In the @ViewScoped bean you can delete people from the list and it maintains the list of changes unless you navigate away from the page. Then the list is reset back to the original values. In the @ConversationScoped bean, once you navigate to the page, a conversation begins. This will allow you to modify the list of names, and navigate to different pages and back without losing your data. Once you click on the End button, the conversation ends.Code
The project was developed using NetBeans 7 IDE and Apache Maven. The project was tested on GlassFish 3.1.2.The code for the project can be downloaded here: scope-examples.zip
viewscope.xhtml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 | <? xml version = '1.0' encoding = 'UTF-8' ?> <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> > < h:head > < title >Family Names - @ViewScoped</ title > </ h:head > < h:body > < h1 > Family Names - @ViewScoped</ h1 > < p > You can delete elements from the table below. The page will reload on submit, or you can use the command button to force a reload of the page. The elements which you delete will remain deleted. </ p > < p > If you choose to navigate back to the index page, or conversation scope and return, the list of family members will contain the original complete list. </ p > < h:form > < h:dataTable id = "namesTable" value = "#{viewScopeBean.names}" var = "name" > < f:facet name = "header" > < h:outputText value = "Family Members" /> </ f:facet > < h:column > < f:facet name = "header" > < h:outputText value = "Name" /> </ f:facet > < h:outputText value = "#{name}" /> </ h:column > < h:column > < h:commandButton value = "Delete" action = "#{viewScopeBean.delete(name)}" /> </ h:column > < f:facet name = "footer" > < h:outputText value = "Count: #{viewScopeBean.names.size()}" /> </ f:facet > </ h:dataTable > < h:panelGroup > < h:commandButton value = "Reload" /> < h:commandButton value = "Index" action = "index" /> < h:commandButton action = "conversation.xhtml" value = "Conversation Scope" /> </ h:panelGroup > </ h:form > </ h:body > </ html > |
conversation.xhtml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 | <? xml version = '1.0' encoding = 'UTF-8' ?> <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> > < h:head > < title >Family Members - @ConversationScoped</ title > </ h:head > < h:body > < h1 > Family Members - @ConversationScoped</ h1 > < p > You can delete elements from the table below. The page will reload on submit, or you can use the command button to force a reload of the page. The elements which you delete will remain deleted. </ p > < p > If you choose to navigate back to the index page, or view scope and return, the list of family members will remain in the previous state. If you press the command button to end the conversation, the list will return to the original state. </ p > < h:form > < h:dataTable id = "namesTable" value = "#{conversationBean.names}" var = "name" > < f:facet name = "header" > < h:outputText value = "Family Members" /> </ f:facet > < h:column > < f:facet name = "header" > < h:outputText value = "Name" /> </ f:facet > < h:outputText value = "#{name}" /> </ h:column > < h:column > < h:commandButton value = "Delete" action = "#{conversationBean.delete(name)}" /> </ h:column > < f:facet name = "footer" > < h:outputText value = "Count: #{conversationBean.names.size()}" /> </ f:facet > </ h:dataTable > < h:panelGroup > < h:commandButton value = "Reload" /> < h:commandButton action = "#{conversationBean.destroy()}" value = "End" /> < h:commandButton value = "Index" action = "index.xhtml" /> < h:commandButton value = "View Scoped" action = "viewscope.xhtml" /> </ h:panelGroup > </ h:form > </ h:body > </ html > |
ViewScopeBean.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 | package com.bluelotussoftware.example.jsf; import java.io.Serializable; import java.util.ArrayList; import java.util.List; import javax.annotation.PostConstruct; import javax.faces.bean.ManagedBean; import javax.faces.bean.ViewScoped; /** * * @author John Yeary * @version 1.0 */ @ManagedBean @ViewScoped public class ViewScopeBean implements Serializable { private static final long serialVersionUID = 3152248670694285690L; private List<string> names; public ViewScopeBean() { names = new ArrayList<string>(); } @PostConstruct private void initialize() { names.add( "John" ); names.add( "Ethan" ); names.add( "Sean" ); names.add( "Patty" ); names.add( "Java" ); names.add( "Duke" ); } public List<string> getNames() { return names; } public void setNames(List<string> names) { this .names = names; } public void delete(String name) { names.remove(name); } } |
ConversationBean.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 | package com.bluelotussoftware.example.cdi; import java.io.Serializable; import java.util.ArrayList; import java.util.List; import javax.annotation.ManagedBean; import javax.annotation.PostConstruct; import javax.enterprise.context.Conversation; import javax.enterprise.context.ConversationScoped; import javax.faces.event.ActionEvent; import javax.inject.Inject; import javax.inject.Named; /** * * @author John Yeary * @version 1.0 */ @Named @ManagedBean @ConversationScoped public class ConversationBean implements Serializable { private static final long serialVersionUID = 5649508572742937677L; private List<string> names; @Inject private Conversation conversation; public ConversationBean() { names = new ArrayList<string>(); } @PostConstruct private void initialize() { conversation.begin(); names.add( "John" ); names.add( "Ethan" ); names.add( "Sean" ); names.add( "Patty" ); names.add( "Java" ); names.add( "Duke" ); } public List<string> getNames() { return names; } public void setNames(List<string> names) { this .names = names; } public void delete(String name) { names.remove(name); } public String destroy() { conversation.end(); return "index" ; } public void destroy(ActionEvent event) { conversation.end(); } } |
1 comments :
I was asked by a friend why I have @Named and @ManagedBean in the @ConversationScoped bean.
I replied that a @Named bean is a @ManagedBean, but that a @ManagedBean is not a @Named bean.
You can choose to expose the CDI bean out to the JSF layer with a @Named annotation, but at some point you could decide to turn it off.
This may change it JSF 2.2 when it is released.
Post a Comment