Posts Tagged Faces Navigation
How to personalise the URLs with Faces Navigation?
This is an important question if you want to have search engine friendly URLs on your website!
But unfortunately, the solution is not straightforward with JavaServer Faces (JSF).
First of all, for security reason JSF doesn’t allow you to use the GET method for your forms. I didn’t really understand why but this is very (too) restrictive!
Can you imagine Google doing the same thing? We would have the same URL for every search terms http://www.google.co.uk/search
instead of something like http://www.google.co.uk/search?hl=en&safe=off&esrch=FT1&q=test&meta=&aq=f&aqi=g10&aql=&oq=&gs_rfai=
.
It wouldn’t be very easy to share a search page with a friend if Google was using the POST method.
So the question is how to get around this JSF limitation?
Let’s take a look at how would look our faces-navigation.xml
file for a search page:
1 | < navigation-case > |
2 | < from-action >#{searchBean.searchAction}</ from-action > |
3 | < from-outcome >success</ from-outcome > |
4 | < to-view-id >/search.xhtml</ to-view-id > |
5 | < redirect /> |
6 | </ navigation-case > |
In this example, all the JSF elements calling the action searchBean.searchAction
will be redirected to the search.xhtml
page.
But, how are we going to get the search parameters into the URL?
Ideally, it would be great to be able to do something like the following:
1 | < navigation-case > |
2 | < from-action >#{searchBean.searchAction}</ from-action > |
3 | < from-outcome >success</ from-outcome > |
4 | < to-view-id >/search.xhtml?q=#{param.q}</ to-view-id > |
5 | < redirect /> |
6 | </ navigation-case > |
This solution would allow us to inject EL expressions into the URL before the page is redirected to the destination page.
In order to do this, we need to create our own view handler and register it to our application.
The code below is the view handler class which also includes some comments:
01 | package com.logikdev.gui.handler; |
02 |
03 | import javax.el.ExpressionFactory; |
04 | import javax.el.ValueExpression; |
05 | import javax.faces.application.ViewHandler; |
06 | import javax.faces.context.FacesContext; |
07 |
08 | import com.sun.facelets.FaceletViewHandler; |
09 |
10 | /** |
11 | * Overrides the Facelet view handler to support EL expressions in URLs. |
12 | * @author Stéphane Moreau |
13 | */ |
14 | public class DynamicViewHandler extends FaceletViewHandler { |
15 |
16 | public DynamicViewHandler(ViewHandler parent) { |
17 | super (parent); |
18 | } |
19 | |
20 | /* (non-Javadoc) |
21 | * @see com.sun.facelets.FaceletViewHandler#getActionURL(javax.faces.context.FacesContext, java.lang.String) |
22 | */ |
23 | @Override |
24 | public String getActionURL(FacesContext context, String viewId) { |
25 | String queryString = null ; |
26 |
27 | // Replace the EL expressions in the URL |
28 | ExpressionFactory expressionFactory = context.getApplication().getExpressionFactory(); |
29 | ValueExpression valueExpression = expressionFactory.createValueExpression(context.getELContext(), viewId, String. class ); |
30 | String result = (String) valueExpression.getValue(context.getELContext()); |
31 | |
32 | // Separate the query string from the URL |
33 | int dotIndex = result.lastIndexOf( '.' ); |
34 | int questionMarkIndex = result.indexOf( '?' ); |
35 | if (questionMarkIndex != - 1 ) { |
36 | queryString = result.substring(questionMarkIndex, dotIndex); |
37 | viewId = result.substring( 0 , questionMarkIndex) + result.substring(dotIndex); |
38 | } |
39 | |
40 | // Call the parent without the query string |
41 | result = super .getActionURL(context, viewId); |
42 | |
43 | // Put back the query string at the end of the URL |
44 | if (queryString != null ) { |
45 | result += queryString; |
46 | } |
47 | |
48 | return result; |
49 | } |
50 |
51 | } |
And the following is the code to put in the faces-config.xml
file in order to register the newly created view handler to the application:
1 | < application > |
2 | < variable-resolver >org.springframework.web.jsf.DelegatingVariableResolver</ variable-resolver > |
3 | < view-handler >com.logikdev.gui.handler.DynamicViewHandler</ view-handler > |
4 | </ application > |
One last thing!
This view handler also has a limitation which I wasn’t able to fix. The file extension has to ALWAYS be placed at the end of the to-view-id
URL! The view handler will then put it back before the question mark.
For example:
1 | < navigation-case > |
2 | < from-action >#{searchBean.searchAction}</ from-action > |
3 | < from-outcome >success</ from-outcome > |
4 | <!-- The extension has to be at the end --> |
5 | < to-view-id >/search?q=#{param.q}.xhtml</ to-view-id > |
6 | < redirect /> |
7 | </ navigation-case > |
If you perform a search on ‘jsf’ with the above navigation rule, the user will be redirected to the page /search.xhtml?q=jsf
.
EL expressions, Faces Navigation, Java, JSF, SEO, view handler
Share this
Archives
- March 2014 (2)
- September 2013 (1)
- August 2013 (1)
- July 2013 (1)
- June 2013 (1)
- February 2013 (1)
- January 2013 (1)
- December 2012 (1)
- September 2012 (1)
- August 2012 (1)
- June 2012 (1)
- May 2012 (1)
- April 2012 (3)
- February 2012 (1)
- December 2011 (3)
- November 2011 (2)
- October 2011 (2)
- September 2011 (2)
- July 2011 (1)
- June 2011 (2)
- May 2011 (2)
- April 2011 (2)
- March 2011 (2)
- February 2011 (2)
- January 2011 (2)
- December 2010 (1)
- November 2010 (2)
- October 2010 (3)
- September 2010 (1)
- August 2010 (2)
- July 2010 (1)
- June 2010 (2)
- May 2010 (3)
- March 2010 (2)
- February 2010 (3)
- January 2010 (2)
- December 2009 (3)
- November 2009 (3)
Tags
Ajax Amazon S3 Arch Linux backup bash benchmarking Bug cron D-Link decryption DNS-313 DNS-323 encryption Html Internet Explorer IOzone Java JavaScript javax.crypto JSF Linux locale Mac OS X Monitoring MyFaces MySQL OAuth onclick OpenSolaris PHP Raspberry Pi RichFaces s3sync Samba shell SQL Server Tomcat Twitter Twitter4J UTF-8 VBScript virtual storage pool Windows Zabbix zpool