The problem I am having is with the PrimesFaces 3.4.1 calendar. When using the popup date picker activated either through the button or on input field focus you can only select valid dates which work fine, happy days!
The issues comes when you manually add a date into the input field, if you add an invalid date the PrimeFaces calendar component takes its best guess at converting this into a valid date and then sending it, meaning that back-end validation is a no go. Some interesting translations below:
- 30/02/2012 becomes 2/6/2014
- 322/05/2012 becomes 5/10/2038
- 01/14/2012 becomes 4/1/2012
To recreate this madness have a look at the PrimeFaces Calendar Showcase.
I have seen solution around using the readOnlyInput='true'
attribute but that only seems to prevent letters being entered in the field not number or slashes. Below is one instance of the calendar I have implemented:
<p:calendar id="fldDateOfBirth"
value="#{pc_CreateUser.user.dateOfBirth}"
binding="#{pc_CreateUser.dobComp}"
navigator="true"
pattern="dd/MM/yyyy"
maxlength="10"
yearRange="-100"
validator="#{pc_CreateUser.validateDOB}"
title="#{msg.user_date_format_default_tip}"
converterMessage="#{msg.user_error_dob_invalid}"
readOnlyInput="true"
showOn="button" />
Solution wise I am open to any suggestions:
- Is this a common issues in PrimeFaces? Is there a trick I can use to
fix it?
- Could I use JavaScript to validate the date before it's sent or to
block all user input entirely?
- Anything else I haven't thought of!
Thanks in advance, this has been causing me issues for weeks!
The <p:calendar>
uses under the covers SimpleDateFormat
which in turn uses by default lenient parsing, causing the overflowed values to roll over into the next date metric level. E.g. 32 January would become 1 February, etc.
In plain Java terms, this can be turned off by DateFormat#setLenient()
, passing false
. See also among others this question: validating a date using dateformat.
In JSF terms, you basically need to provide a custom converter which uses a non-lenient DateFormat
. Fortunately, standard JSF already provides such one out the box in flavor of <f:convertDateTime>
, so you could just make use of it directly.
<p:calendar ...>
<f:convertDateTime pattern="dd/MM/yyyy" />
</p:calendar>
In faces-config.xml add this
<converter>
<converter-id>localDateConverter</converter-id>
<converter-class>com.utility.LocalDateConverter</converter-class>
</converter>
In the above class i.e LocaldateConverter add this below code
/**
* @param facesContext .
* @param uiComponent .
* @param input .
* @return Object .
*/
@Override
public Object getAsObject(final FacesContext facesContext, final UIComponent uiComponent, final String input) {
if (StringUtils.isBlank(input)) {
return null;
}
final String componentPattern = (String) uiComponent.getAttributes().get("datePattern");
final String patternToUse = componentPattern != null ? componentPattern : CommonConstants.OUTPUT_DATE_FORMAT;
try {
final DateFormat fmt = new SimpleDateFormat(patternToUse);
Date convertedDate = new java.sql.Date(fmt.parse(input).getTime());
return convertedDate;
} catch (Exception e) {
throw new ConverterException(new FacesMessage(FacesMessage.SEVERITY_ERROR, "Invalid Date Format", null));
}
}
/**
* @param facesContext .
* @param uiComponent .
* @param obj .
* @return String .
*/
@Override
public String getAsString(final FacesContext facesContext, final UIComponent uiComponent, final Object obj) {
if (obj==null) {
return null;
}
final Date date = (Date) obj;
return date.toString();
}