Ответ 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>
.