Lately I have been looking into JSF and especially the Myfaces implementation as a development framework for ALUI portlets, Similar to ASP.NET, I personally do think JSF is also an awesome and powerful framework to develop all sort of cool portlets and applications. But similar to ASP.NET portlets, JSF portlets have some problems when integrated in ALUI portal.
Mainly, the problems lie in the fact that those 2 frameworks are component oriented: We don't have a lot of control on the rendition of those control, and sometimes, the way they render is not fully compatible with the way ALUI portal behave:
- Some urls in generated javascript are not always successfully gatewayed.
- The portal does not enforce generated Html components to have unique IDs throughout the portal pages (i.e. 2 portlets on the same page can possibly have 2 form tags with same ids...which would very likely mess with the functionality of the portlets)
- If you don't enable ALUI inline refresh, the portlets will go in the "Gateway" space (or the maximized state) on any navigation event. Although that can be useful in certain case, it won't to perform portlet-to-portlet communication (since the other portlets disappear in this maximized state).
- If you DO enable inline refresh, you will hit other problems due to the fact that the Javascript layer added to the "inline-refresh" enabled portlet can interfere with the good behavior of some of the JSF/DOTNET components...
Ok...having painted such a "Dantesque" picture of the portlet development within ALUI portal, I now want to reassure you: Because ALUI is so extensible and powerful, all those problems can be resolved, and that's the goal of this post: Showing you some of the trick to make it all work.
For you DotNet users, you are in luck :) BEA introduced the "BEA AquaLogic .NET Application Accelerator" to offer a remedy to all the problems above. Since it is all there at http://www.bea.com/framework.jsp?CNT=index.htm&FP=/content/products/more/accelerator/ I won't talk to much about it.
But what about you poor JSF users? Well unfortunately for you (and I), we will have to get to work :)
But before getting to work, some of you might say: What about the "Java Portlet Tool" (http://dev2dev.bea.com/pub/a/2006/05/java-portlet-tools.html)? Well I looked at that quite thoroughly. I must admit that it is a pretty involved and clever piece of code...and it makes things work a little better on the JSF side. I especially do like the bean support for IDK functionality. Unfortunately, the following points are making me very hesitant and uneasy about this framework:
- Unlike "BEA AquaLogic .NET Application Accelerator", the "Java Portlet Tool" (which is written by a BEA resource) is not supported by BEA. (Even though I could go with that - after all since the Java portlet tool code is provided, we could eventually support it ourselves - I would not recommend it to my clients who pay a lot of money to get support),
- In addition to that, the java portlet tool, which rely on the modification of the pretty involved inline refresh jsxml javascript libraries (PTXML.js), seems to have been developed and tested with Plumtree 5.x, and does not seem to have been upgraded for 6.x versions of the portal. (thus, it might or might not work with 6.x versions)
- Finally, the java portlet tool project has not been modified since 2006, which makes me think there are not a lot of people using it, or keeping it up-to-date.
Okay...so no supported portal framework for JSF...but still I want to use inline refresh...
Does it mean that I have to develop my JAVA portlets the way I use to develop them 5 years ago, without the popular frameworks out-there? Well I personally refuse to do that, and you should too :)
So it is time to have some fun finally!
JSF Navigation
A major problem I found with the ALUI inline refresh and JSF is that HTTP POSTs with JSF navigation is not working. JSF framework navigation (similarly to struts etc...) is controlled by the framework itself, based on the navigation rules you define in the JSF config file. (faces-config.xml). In other words, when the form is posted back to the server, the framework will deal with the navigation based on the action define on the command button that triggered the HTTP POST. Nothing unusual until now, and such a navigation behavior will work fine if you don't enable inline refresh...but the next page will be displayed in the maximized state we want to avoid.
When you enable inline refresh though, the ALUI portal will intercept the button submit action and perform its own submit action using its AJAX/Javascript framework (PTXML.js). On the html side, you will see:
<form id="viewaddress" name="viewaddress" onsubmit="pt_280.formRefresh(this); return false;" method="post"
We can particularly notice how the submit is intercepted: the OnSubmit event handlers of the form tag does a "return false", which is the equivalent of saying: the submit button has never been clicked, the form has never been submitted. The "pt280.formRefresh(this);" is the statement that will post the form using an AJAX call. How this works is interesting: Using JavaScript to go through the DOM, the portal reconstructs the post request using all the input fields in the form (which would have been posted to the server). But the problem is: during that reconstruction, the <input type="button" /> and <input type="submit" /> are purposely left out. Although this usually makes sense in a standard web application (those buttons don't have a particular meaning on the server side, they are just useful to trigger a post back to the server), it does not make sense in the JSF world: the button name/value pair should be posted back to the server because the navigation rules won't happen otherwise. The result is that with inline refresh, navigation initiated by a form post is NOT working.
That is inconvenient...but we can correct that easily: if we create a hidden input field that has the same name and value as the button clicked, then this hidden field WILL be included in the posted request and the navigation will then work! Great! But you don't want to manually add yourself the hidden field in the form...for the simple reason that you want the value posted only when a particular button is clicked.
So the simple fix is to use javascript to dynamically add the input field in the form:
<pt:namespace pt:token="PORTLET_ID" xmlns:pt='http://www.plumtree.com/xmlschemas/ptui/' />
function setHiddenInputPORTLET_ID(form, name, value){
var newInput = document.createElement('input');
newInput.setAttribute('type','hidden');
newInput.setAttribute('name',name);
newInput.setAttribute('value',value);
form.appendChild(newInput);
}<input id="viewaddress:_idJsp12" name="viewaddress:_idJsp12" type="submit" value="Change"
onclick="if(typeof window.oamSetHiddenInputPORTLET_ID!='undefined'){setHiddenInputPORTLET_ID(this.form,'viewaddress:_idJsp12','Change');}" />
That way, when you click the button, the equivalent hidden input is added to the form, which will then be effectively posted by the "pt280.formRefresh(this);" call. You noticed that I also use the ALUI adaptive tag to generate a unique token (portlet id) and thus make the JavaScript function "setHiddenInput" unique (that way, no interferences can occur if multiple portlets are on the same page)
In addition, you certainly don't want to force your developers to create that by hand for every buttons, (as well as introducing in your presentation code such a specific bug fix). So in order to be the least intrusive possible, I chose to customize a little bit the rendering of the <h:commandButton> element, which is the one responsible for postback and navigation:
<h:panelGroup>
<h:commandButton action="editaddress" value="Change"></h:commandButton>
</h:panelGroup>
To do that, you just have to create a Class that extends the HtmlButtonRenderer class, and overrides 2 methods:
- encodeEnd to generate the javascript function setHiddenInput281()
- buildOnClick so add the call to the generated function setHiddenInput281()
That way, your JSF page are exactly the same...but the rendering of it is a little different when viewed through the portal. You can find the full custom renderer here.
That fix alone will take care of a good majority of the problems you would have had with using JSF and inline refresh inside ALUI. On a next post, I will show you how to deal with form ids when several JSF portlets are on the same page.
So long!
What are your thoughts on how to solve the "back button" problem in regards to inline refresh...for example, a portlet hosting an external application.
ReplyDeleteFor a link, the inline refresh hijacks it similarly to the way you described in your post. What would be your approach to storing a history to allow users to get back using the browser controls?
Hi Steve, I have similar problem as you mentioned i.e. "back button" problem if i select "inline refresh" for a portlet. Please help me if you find any solution for this. My email address is sandeep.mann@gmail.com ..I appreciate your help.. Thanks
ReplyDeletesandeep
Is there a way to call another function before "pt_280.formRefresh(this); " in the onSubmit event dynamically.
ReplyDeleteI want to be able to control the form submission depending upon some forma validation. but seems like the form is posted by alui javascript no matter what. any ideas ?
Namit,
ReplyDeleteYou should make sure you are gatewaying the links.
For example, if your web service url is: http://localhost:7001/Namit
You'll need to gateway
http://localhost:7001/Namit/
and check the "Inline Refresh" button
on the "HTTP Configuration" screen.
Thanks Steve,
ReplyDeleteI have already done both the settings. But I am still facing the same issue.
I need to click the Button 2 times then it is going to the next page.
First time it is giving some java script error " Object Required".
This is my First JSP code where I want the navigation to be done on clicking the button:
html
head
script type="javascript"
function setHiddenInputPORTLET_ID(form, name, value){
var newInput = document.createElement('input');
newInput.setAttribute('type','hidden');
newInput.setAttribute('name',name);
newInput.setAttribute('value',value);
form.appendChild(newInput);
}
/script
titleNavigation Page/title/head
body
h:form id="form1"
h:commandButton immediate="true" id="source" value="second page" action="secondpage" onclick="setHiddenInputPORTLET_ID(this.form,this.id,this.id)"
/
/h:form
/body
/html
NKK...
ReplyDeleteYou are probably very close to a solution. Have you checked the other boxes? "Use hosted display mode on gateway pages", "Transform JavaScript files" "Transform CSS files". Typically I keep these unchecked.
I looked over your code, and it looks like you do some things a little differently than Fabien describes. You pass in this.id twice. It may be what you wanted to do or it may be a mistake. I would also recommend you use Firebug with Firefox to get a good idea of where your Javascript error lies. You also left out the pt:namespace pt:token="PORTLET_ID" xmlns:pt='http://www.plumtree.com/xmlschemas/ptui/' / line.
It's kind of hard to debug remotely, but if you want to get a larger audience for your question, I would try the Oracle WebCenter Interaction forums:
http://forums.oracle.com/forums/forum.jspa?forumID=576
They are very helpful over there.
You could also try starting with Fabien's code, make sure it works, and then slightly adjust it until it fits what you're trying to do.