10

I have specific use case for JSF validation. For example I have an inputText field:

<p:inputText id="input" required="#{myBean.required}" value="#{myBean.value}" maxlength="20" disabled="#{myBean.disabled}">
  <p:ajax event="blur" process="@this" update="name" listener="#{myBean.listener}"/>
</p:inputText>

Value of input is number (in some cases it can also be a string, because this is part of composite component, but problem is better described if we assume this is a number). This input is part of the form, at the end of form I have submit button:

<p:commandButton value="Save" actionListener="#{myBean.save}"/>

What are my requests:

  1. When submit button is pressed all validation should be processed and this is OK, this works fine.
  2. When blur event is fired on input field if field is not empty a number validation should be processed, and this is also OK. At the end I update field with id name with some value.
  3. Now I have a problem. My third request is when input is empty validation on input should not be processed. This is special case in which I will clear field with id name. This is also case when i remove text which is already entered in input, remove focus from component (press TAB for example) and in that case AJAX request should also be processed and name input will also be cleared.

How I can disable validation of this input field in case when it is empty, and just for this ajax event?

2 Answers 2

13

Let the input's required attribute check if the save button is pressed or not (which can be identified by the presence of its client ID in the request parameter map as available by implicit EL variable #{param}).

<h:form>
    <p:inputText ... required="#{param.containsKey(save.clientId) and myBean.required}" />
    
    <p:commandButton binding="#{save}" ... />
</h:form>

(note: do not bind it to a bean property! the code is as-is)

This way it would only evaluate true when the save button is actually pressed.

Or, if you have problems with binding and/or don't have a problem hardcoding the button's client ID:

<h:form id="formId">
    <p:inputText ... required="#{param.containsKey('formId:buttonId') and myBean.required}" />
    
    <p:commandButton id="buttonId" ... />
</h:form>
10
  • You think that required validation will be skipped in that case? Why?
    – partlov
    Commented Feb 6, 2013 at 14:17
  • As long as you don't fill (and thus change) the input, the validation won't be fired on blur.
    – BalusC
    Commented Feb 6, 2013 at 14:22
  • Maybe I was not clear with specific use case in third request. I edited it. User can (probably accidentally) remove text and in that case name input will be cleared.
    – partlov
    Commented Feb 6, 2013 at 14:30
  • Oh, wouldn't have expected that. Well, just let required attribute check if save button is pressed. See edited answer.
    – BalusC
    Commented Feb 6, 2013 at 14:31
  • As I use @ViewScoped backing bean I would prefer not to use binding. Maybe some f:param can be used instead?
    – partlov
    Commented Feb 6, 2013 at 14:38
2

Just remove the required attribute as you accept the input if the input is empty. Then write a custom validator which accepts only empty input or numerical input.

<p:inputText id="input" value="#{myBean.value}" maxlength="20" disabled="#{myBean.disabled}" validator="customerNumericInputValidator">   <p:ajax event="blur" process="@this" update="name" listener="#{myBean.listener}"/> </p:inputText>
public class customerNumericInputValidator implements Validator {

    @Override
    public void validate(FacesContext facesContext, UIComponent uIComponent,
            Object object) throws ValidatorException {

        String number = (String) object;
        number = Strings.nullToEmpty(number).trim();

        //if the request is a full request then number can not be empty
        if(!FacesContext.getCurrentInstance().isPostback() && Strings.isNullOrEmpty(number))
        {
             FacesMessage message = new FacesMessage();
             message.setSummary(Messages.getMessage("error empty value"));
             message.setSeverity(FacesMessage.SEVERITY_ERROR);
             throw new ValidatorException(message);
        } 

        if(!Strings.isNullOrEmpty(number))
        { 
            if(!isNumber(number))
            {
               FacesMessage message = new FacesMessage();
               message.setSummary(Messages.getMessage("error not numerical value"));
               message.setSeverity(FacesMessage.SEVERITY_ERROR);
               throw new ValidatorException(message);
            }
        }
    }

}
8
  • But required validation should be processed in case when form is submitted with commandButton.
    – partlov
    Commented Feb 6, 2013 at 13:23
  • When form is submitted with commandButton your customValidator will be executed also and it will throw an exception if the entered value is not number; but it won't complain if the input is empty.
    – cubbuk
    Commented Feb 6, 2013 at 13:26
  • OK, in request 1 I wrote that when form is submitted with button, required must not be ignored! We can say that field is required when submitting with button, but should not be required when submitting with ajax on blur event.
    – partlov
    Commented Feb 6, 2013 at 13:29
  • Ok then you might check whether the request is an ajax request or not in your validator and process your input according to that. I updated my answer to show you an example.
    – cubbuk
    Commented Feb 6, 2013 at 13:37
  • As you can see, this Prmefaces commandButton is also AJAX :), so no it can't be processed like that. It should be little more robust component (it is, as I wrote, part of composite component).
    – partlov
    Commented Feb 6, 2013 at 13:42

Not the answer you're looking for? Browse other questions tagged or ask your own question.