Подтвердить что ты не робот

JSF не поддерживает проверку кросс-поля, есть ли обходной путь?

JSF 2.0 позволяет только проверять ввод в одном поле, например, проверять, имеет ли он определенную длину. Это не позволяет вам иметь форму, которая гласит: "введите город и штат или введите только почтовый индекс".

Как вы обошли это? Меня интересуют только ответы, связанные с фазой проверки JSF. Мне не интересно размещать логику проверки в Managed Beans.

4b9b3361

Ответ 1

Самый простой пользовательский подход, который я видел и использовал до сих пор, заключается в создании поля <h:inputHidden> с <f:validator>, в котором вы ссылаетесь на все задействованные компоненты как <f:attribute>. Если вы объявите его перед компонентами, подлежащими проверке, вы можете получить представленные значения внутри валидатора UIInput#getSubmittedValue().

например.

<h:form>
    <h:inputHidden id="foo" value="true">
        <f:validator validatorId="fooValidator" />
        <f:attribute name="input1" value="#{input1}" />
        <f:attribute name="input2" value="#{input2}" />
        <f:attribute name="input3" value="#{input3}" />
    </h:inputHidden>
    <h:inputText binding="#{input1}" value="#{bean.input1}" />
    <h:inputText binding="#{input2}" value="#{bean.input2}" />
    <h:inputText binding="#{input3}" value="#{bean.input3}" />
    <h:commandButton value="submit" action="#{bean.submit}" />
    <h:message for="foo" />
</h:form>

(обратите внимание на value="true" на скрытый ввод, фактическое значение фактически не имеет значения, но имейте в виду, что валидатор не обязательно будет запущен, когда он будет пустым или пустым, в зависимости от версии и конфигурации JSF )

с

@FacesValidator(value="fooValidator")
public class FooValidator implements Validator {

    @Override
    public void validate(FacesContext context, UIComponent component, Object value) throws ValidatorException {
        UIInput input1 = (UIInput) component.getAttributes().get("input1");
        UIInput input2 = (UIInput) component.getAttributes().get("input2");
        UIInput input3 = (UIInput) component.getAttributes().get("input3");
        // ...

        Object value1 = input1.getSubmittedValue();
        Object value2 = input2.getSubmittedValue();
        Object value3 = input3.getSubmittedValue();
        // ...
    }

}

Если вы объявляете <h:inputHidden> после подлежащих проверке компонентов, тогда значения задействованных компонентов уже конвертируются и проверяются, и вы должны получить их UIInput#getValue() или, возможно, UIInput#getLocalValue() (в случае, если UIInput не isValid()).

См. также:


В качестве альтернативы вы можете использовать для этого сторонние теги/компоненты. RichFaces, например, имеет тег <rich:graphValidator> для это Seam3 имеет <s:validateForm> для этого, и OmniFaces имеет несколько стандартных <o:validateXxx> компонентов для этого, которые представлены здесь здесь. OmniFaces использует компонентный подход, посредством которого задание выполняется в UIComponent#processValidators(). Он также позволяет настраивать его таким образом, чтобы вышеуказанное можно было выполнить, как показано ниже:

<h:form>
    <o:validateMultiple id="foo" components="input1 input2 input3" validator="#{fooValidator}" />
    <h:inputText id="input1" value="#{bean.input1}" />
    <h:inputText id="input2" value="#{bean.input2}" />
    <h:inputText id="input3" value="#{bean.input3}" />
    <h:commandButton value="submit" action="#{bean.submit}" />
    <h:message for="foo" />
</h:form>

с

@ManagedBean
@RequestScoped
public class FooValidator implements MultiFieldValidator {

    @Override
    public boolean validateValues(FacesContext context, List<UIInput> components, List<Object> values) {
        // ...
    }
}

Единственное отличие состоит в том, что он возвращает boolean и что сообщение должно быть указано как message атрибут в <o:validateMultiple>.