Communicating with JSF Bean through Javascript using jsf.ajax.request

In web development, many a times we need a way to interact with the server side from the client tier. One standard way which I've been using is through the use of Ajax, and when it comes to JSF and in particular JSF 2.0, Ajax is the standard here too. But sometimes we might want to make Ajax calls to the server side and have the results returned and use it the way we want.

JSF 2.0 already gives standard ways of doing what I am trying to show here, but I found this way quite helpful sometimes e.g. if I have two or more forms and from one of the forms I want to keep on saving the data as I change but I dont to keep that functionality just applied to that form alone but make it generic.

Other case could be that if you have Custom component written say in JSP or using JSF 2.0 Custom Component Standard way but say if you have basic HTTP elements used in those custom components, then you might want to have a way to communicate back to the JSF Bean and use the data as if you using it through JSF.

Other case could be that if you have created custom component using jQuery JS library alone, and in that case also it's highly likely that we would need data coming from server and might also need to send back.

Here I will take a very basic XHTML file which has two input elements in one form and there is one more form which has one simple Html button and one JSF hidden component.




<ui:composition lang="en"
    xmlns="http://www.w3.org/1999/xhtml"
    xmlns:f="http://java.sun.com/jsf/core"
    xmlns:c="http://java.sun.com/jsp/jstl/core"
    xmlns:h="http://java.sun.com/jsf/html"
    xmlns:ui="http://java.sun.com/jsf/facelets"
    xmlns:p="http://primefaces.org/ui"
    xmlns:my="http://java.sun.com/jsf/composite/mycomponents"
    xmlns:util="http://java.sun.com/jsf/composite/util">

<h:form id="mainForm">
       <h:inputText id="id" value=""/>
<h:inputText id="outText22" value="" />
    </h:form>

<h:form prependId="false" id="hiddenForm">
  <input type="button" id="SubmitHiddenFormButtonId" value="SubmitHiddenForm" onclick="submitHiddenForm();"/>
  <h:inputHidden id="idHidden1" value="#{register.hiddenValue}" valueChangeListener="#{register.valueChangeListener}"/>
</h:form>

</ui:composition>

And above Xhtml file is included in the below Xhtml file


<html lang="en"
    xmlns="http://www.w3.org/1999/xhtml"
    xmlns:f="http://java.sun.com/jsf/core"
    xmlns:c="http://java.sun.com/jsp/jstl/core"
    xmlns:h="http://java.sun.com/jsf/html"
    xmlns:ui="http://java.sun.com/jsf/facelets"
    xmlns:p="http://primefaces.org/ui"
    xmlns:my="http://java.sun.com/jsf/composite/mycomponents"
    xmlns:util="http://java.sun.com/jsf/composite/util">
   
    <ui:debug hotkey="x" />
    <h:head>
        <f:facet name="first">
            <meta http-equiv="X-UA-Compatible" content="EmulateIE8" />
            <meta content='text/html; charset=UTF-8' http-equiv="Content-Type"/>
        </f:facet>
       
        <link type="text/css" rel="stylesheet" href="#{request.contextPath}/css/default.css" />
        <link type="text/css" rel="stylesheet" href="#{request.contextPath}/css/syntaxhighlighter/syntaxhighlighter.css" />
<h:outputScript name="jsf.js" library="javax.faces"/>
<h:outputScript library="javascript" name="utils.js" target="head"/>
    </h:head>
    <h:body>
    <ui:include src="ui/test.xhtml"/>
</h:body>

</html>


Idea is very simple, so basically if I click the button it will make JS call to "submitHiddenForm" function, which will in turn call jsf.ajax.request (JSF standard) making call to JSF Bean.

The JS file would be placed under resources/javascript folder as JSF standard states.




function handleEvent(data){
    if(data.status === "complete"){
    var returnedTxt = data.responseText;
    document.getElementById("mainForm:outText22").value = returnedTxt;
    }
    }
   
    function submitHiddenForm(){
    document.getElementById("idHidden1").value = Math.random();
    jsf.ajax.request("idHidden1",
null,
{render: null, onevent: handleEvent, "a":112222, "b": 222});
    }

And the JSF Bean has the method implemented as:


public void valueChangeListener(ValueChangeEvent e) {
        System.out.println(" valueChangeListener() ");
        Map<String, String> reqParams = FacesContext.getCurrentInstance()
           .getExternalContext().getRequestParameterMap();
FacesContext context = FacesContext.getCurrentInstance();
System.out.println(" reqParams.get(a) "+reqParams.get("a"));
System.out.println(" reqParams.get(b) "+reqParams.get("b"));
try{
HttpServletResponse response = (HttpServletResponse)context.getExternalContext().getResponse();
ServletOutputStream out = response.getOutputStream();
out.println("SUCCESS");
out.flush();
}catch(Exception ex){
ex.printStackTrace();
}
context.responseComplete();
    }



In the above JSF bean method, I am using Servlet's outputstream to write the data I want as if it's any other Servlet or JSP output that could be used on the client side when the response is written.

NOTE: When I used servlet output stream to write the data, in that case on the client side in the "handleEvent" method, the "data" object has two statues only which are "begin" and "complete". But if I use  standard JSF facescontext to write message back then the statuses available are "begin", "complete" and "success"/"error" (did not try this, it could be failure also I guess).


FacesMessage message = new FacesMessage("Registration succesful!");
        FacesContext.getCurrentInstance().addMessage(null, message);

But even if I have two statuses when I use servlet output stream, I think I can handle the output accordingly.

So that's it, using above approach it gives some liberty to use Ajax communication with server side and without loosing much of JSF 2.0 benefits.
Continue Reading

Customizing Primefaces 3: Extending Renderkit

Working on a current Project needed a very rich UI in terms of custom behaviors e.g. showing a "* Required" image icon next to a UIInput component or showing two columns format for Radio buttons along with some info help text to be shown underneath each Radio button option.


Choosing Primefaces 3 for this POC, as the UI component library has come from a long drive other *faces libraries. There were couple of things I loved about Primefaces (but please let me know if I missed any other features), but I think biggest
1. one I liked was the concept of "WidgetVar" on the UIComponents. It gives you the control right back on the UI/Client side without making a Server roundtrip which I guess I didnt see in other famous *faces libraries ( may be I only missed).
2. Second is obvious JSF 2 features.
3. And trust me so thin and rich set of  components. ( hats off to Cagatay. Guys, trust me I'm not an agent for Primefaces, though looks good Job ;-) )

But I'm still learning something new about JSF 2/Primefaces/RIA everyday.

The WidgetVar helped me (I'll soon be updating that code as well) doing client side Tab navigation, client side button toggling, dialog box show/hide.

Here I'll try to show the Custom rendering I was able to achieve by extending Primefaces renderkit. First case was to show "* Required" image icon next to certain UIInput components ( I'll try to customize it further).

I've just taken Primefaces showcase application and extended that for my POC.

Note: Looks like I'm not seeing an option to attach WAR file for this project. Please let me know in case you need the code for this sample project.








And to achieve that, I had extend org.primefaces.component.selectonemenu.SelectOneMenuRenderer and with below config in faces-config.xml.

<renderer>
   <component-family>org.primefaces.component</component-family>
   <renderer-type>org.primefaces.component.SelectOneMenuRenderer</renderer-type>
   <renderer-class>com.sample.eduadmin.extendedTags.ExSelectOneMenuRenderer</renderer-class>
  </renderer>

And here is the code snippet from the ExSelectOneMenuRenderer.

public void encodeEnd(FacesContext context, UIComponent component)
    throws IOException {
        super.encodeEnd(context, component);   
        System.out.println(" ExSelectOneMenuRenderer encodeEnd() this "+this);        
        try{
            FacesContext facesContext = FacesContext.getCurrentInstance ();  
//            ServletContext servletContext = (ServletContext) facesContext.getExternalContext ().getContext(); 
            if(ApplicationUtils.getInstance().get(component.getClientId()) == null){
                System.out.println(" ApplicationUtils.getInstance().get(component.getClientId()) "+
                        ApplicationUtils.getInstance().get(component.getClientId()));
              
                ResponseWriter writer = context.getResponseWriter();                           
                writer.startElement("div", component);       
                writer.writeAttribute("id", component.getClientId()+":"+"selectMenuCstDivId", "id");                
               
                writer.writeAttribute("class", "click_here_required successIconCss", null);
                writer.startElement("span", null);     
                writer.endElement("span");               
                writer.endElement("div");
                 
                ApplicationUtils.getInstance().put(component.getClientId(), "true");
            }else{
               
            }
        }catch(Exception ex){    
            ex.printStackTrace();
        }


And below is the one for Two columns render layout for Radio buttons. I tried different use cases and wanted to make sure that even if there is big text in radio buttons, the rendering still should look good. Good that it worked.

<h:panelGrid columns="1">   
            <h:panelGroup>     
                <h:outputLabel for="edLevelRadio" value="#{i18n['edLevels']}" />        
            </h:panelGroup>
        </h:panelGrid>   
       
        <h:panelGrid columns="4" width="100%" border="1">       
                <p:selectOneRadio id="edLevelRadio"     
                         required="true"
                         layout="twoColumns" 
                         value="#{institutionBean.edLevelId}">
                     <f:selectItems value="#{institutionBean.edLevelsList}"/>                
                </p:selectOneRadio>
        </h:panelGrid> 

public class CustomSelectItem extends SelectItem {
   
    public CustomSelectItem(String value, String label, String subLabel){
        super(value,label);
        this.subLabel = subLabel;
    }
   
    //private String itemDesc
    private String subLabel;

    public String getSubLabel() {
        return subLabel;
    }

    public void setSubLabel(String subLabel) {
        this.subLabel = subLabel;
    }
   
}

public List<SelectItem> getEdLevelsList(){  
        if (_edLevelsList == null) {  
            _edLevelsList = new ArrayList<SelectItem>();       
           
            _edLevelsList.add(new CustomSelectItem("item1",  
                    "This is item1 from ItemsSet which contains Items This is item1 from ItemsSet <br/> which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item1)</b>")); 
            _edLevelsList.add(new CustomSelectItem("item2",
                    "This is item2 from ItemsSet which contains Items This is item1 from ItemsSet <br/> which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item2)</b>"));       
            _edLevelsList.add(new CustomSelectItem("item3",
                    "This is item3 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item3)</b>"));   
            _edLevelsList.add(new CustomSelectItem("item4",
                    "This is item4 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item4)</b>"));
            _edLevelsList.add(new CustomSelectItem("item5",   
                    "This is item5 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item5)</b>")); 
            _edLevelsList.add(new CustomSelectItem("item6", 
                    "This is item6 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items <br/> This is item1 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item6)</b>"));         
            _edLevelsList.add(new CustomSelectItem("item7",    
                    "This is item7 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item7)</b>"));    
            _edLevelsList.add(new CustomSelectItem("item8",
                    "This is item8 from ItemsSet which contains Items", 
                    "<b>((Custom Text) Help Info: This is the help info for Item8)</b>"));     
            _edLevelsList.add(new CustomSelectItem("item9", 
                    "This is item9 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item9)</b>"));   
            _edLevelsList.add(new CustomSelectItem("item10",
                    "This is item10 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item10)</b>"));
            _edLevelsList.add(new CustomSelectItem("item11",
                    "This is item11 from ItemsSet which contains Items",
                    "<b>((Custom Text) Help Info: This is the help info for Item11)</b>"));
        }                
        return _edLevelsList;                        
    }

And for this I had to extend org.primefaces.component.selectoneradio.SelectOneRadioRenderer and put the custom class in the same package as the method "encodeSelectItems" is protected but I'd to override that method.


protected void encodeSelectItems(FacesContext context, SelectOneRadio radio) throws IOException {
}

<renderer>
   <component-family>org.primefaces.component</component-family>
   <renderer-type>org.primefaces.component.SelectOneRadioRenderer</renderer-type>
   <renderer-class>org.primefaces.component.selectoneradio.ExSelectOneRadioRenderer</renderer-class>
  </renderer>

And the code snippet ( this was a little painful).

protected void encodeSelectItems(FacesContext context, SelectOneRadio radio) throws IOException {
        try{
            System.out.println(" ExSelectOneRadioRenderer encodeSelectItems() ");
            ResponseWriter writer = context.getResponseWriter();
            List<SelectItem> selectItems = getSelectItems(context, radio);
            Converter converter = getConverter(context, radio);
            Object value = radio.getValue();
            String layout = radio.getLayout();
            boolean pageDirectionTwoColumns = layout != null && layout.equals("twoColumns");
            boolean pageDirection = layout != null && layout.equals("pageDirection");
   
            if(pageDirectionTwoColumns){   
                Iterator<SelectItem> selectItemsItr = selectItems.iterator();   
                while(selectItemsItr.hasNext()){
                      writer.startElement("tr", null);
                     
                          SelectItem selectItem1 = null;
                          SelectItem selectItem2 = null;
                         
                          if(selectItemsItr.hasNext()){
                              writer.startElement("td", null);
                              writer.startElement("table", null);
                             
                                  selectItem1 = selectItemsItr.next();
                                  Object itemValue1 = selectItem1.getValue();
                                  String itemLabel1 = selectItem1.getLabel();
                                 
                                  writer.startElement("tr", null);
                                  encodeOption(context, radio, value, converter, itemLabel1, itemValue1);
                                  writer.endElement("tr");
                                 
                                  CustomSelectItem cstselectItem1 = (CustomSelectItem)selectItem1;
                                 
                                  if(cstselectItem1.getSubLabel() != null){
                                      writer.startElement("tr", null);
                                      String itemSubLabel1 = cstselectItem1.getSubLabel();
                                     
                                      writer.startElement("td", null);
                                      writer.endElement("td");
                                     
                                      writer.startElement("td", null);
                                      writer.write(itemSubLabel1);
                                      writer.endElement("td");
                                     
                                      writer.endElement("tr");
                                  }
                              writer.endElement("table");  
                              writer.endElement("td");
                          }  
                          if(selectItemsItr.hasNext()){ 
                              writer.startElement("td", null);
                              writer.startElement("table", null);
                                  selectItem2 = selectItemsItr.next();
                                  Object itemValue2 = selectItem2.getValue();
                                  String itemLabel2 = selectItem2.getLabel();
                                 
                                  writer.startElement("tr", null);
                                  encodeOption(context, radio, value, converter, itemLabel2, itemValue2);
                                  writer.endElement("tr");
                                   
                                  CustomSelectItem cstselectItem2 = (CustomSelectItem)selectItem2;
                                  if(cstselectItem2.getSubLabel() != null){ 
                                     
                                      writer.startElement("tr", null);
                                        String itemSubLabel2 = cstselectItem2.getSubLabel(); 
                                       
                                       writer.startElement("td", null);
                                      writer.endElement("td");
                                     
                                      writer.startElement("td", null);
                                      writer.write(itemSubLabel2);
                                      writer.endElement("td");
                                      
                                      writer.endElement("tr");      
                                  } 
                              writer.endElement("table");        
                              writer.endElement("td");   
                          }
                      writer.endElement("tr");
                }  
            }else{  
                for(SelectItem selectItem : selectItems) {                
                    Object itemValue = selectItem.getValue();
                    String itemLabel = selectItem.getLabel();
                    if(pageDirection) { 
                        writer.startElement("tr", null);
                    }   
                    encodeOption(context, radio, value, converter, itemLabel, itemValue);  
                    if(pageDirection) {   
                        writer.endElement("tr");
                    }
                }
            }
        }catch(Exception ex){
            ex.printStackTrace();    
        }      
    }

protected void encodeOption(FacesContext context, UIInput component, Object componentValue, Converter converter, String label, Object value) throws IOException {
        ResponseWriter writer = context.getResponseWriter();
        SelectOneRadio radio = (SelectOneRadio) component;
        String formattedValue = getOptionAsString(context, component, converter, value);
        String clientId = component.getClientId(context);
        String containerClientId = component.getContainerClientId(context);
        boolean checked = componentValue != null && componentValue.equals(value);
        boolean disabled = radio.isDisabled();
 
        writer.startElement("td", null);
        writer.writeAttribute("valign", "top", null); // custom code
 
        String styleClass = "ui-radiobutton ui-widget";
        if(disabled) {
            styleClass += " ui-state-disabled";
        }

        writer.startElement("div", null);
        writer.writeAttribute("class", styleClass, null);

        encodeOptionInput(context, radio, clientId, containerClientId, checked, disabled, label, formattedValue);
        encodeOptionOutput(context, radio, checked);

        writer.endElement("div");
        writer.endElement("td");

        writer.startElement("td", null);
        encodeOptionLabel(context, radio, containerClientId, label);
        writer.endElement("td");
    }

I wanted to share this with you all as I found it a little tough to find good example showing how to extend Renderkits in JSF. I'll still try to keep this blog up to date based on my work and updates from you in case my information is misleading.

Thanks for reading...
Continue Reading