物理上只有一个 UIInput
组件的状态会根据UIRepeat
. 它仅通过其客户端 ID 而不带UIRepeat
索引可用:(findComponent("formId:inputId")
索引UIRepeat
仅在客户端有意义)。但是,当在这种方式的上下文之外以编程方式访问组件时UIRepeat
,它确实会返回一个看似空的状态。
为了UIInput
在所有这些状态下访问组件UIRepeat
并收集它们的值,您需要UIComponent#visitTree()
在UIRepeat
.
这是一个启动示例:
<ui:repeat value="#{bean.items}" var="item">
<f:event type="postValidate" listener="#{bean.validateOrder}" />
<h:inputText value="#{item.value}" />
</ui:repeat>
使用这种validateOrder()
方法(同样,只是一个启动示例,这种方法天真地假设转发器中只有一个UIInput
组件):
@SuppressWarnings("rawtypes")
public void validateOrder(ComponentSystemEvent event) {
final FacesContext context = FacesContext.getCurrentInstance();
final List<Comparable> values = new ArrayList<Comparable>();
event.getComponent().visitTree(VisitContext.createVisitContext(context), new VisitCallback() {
@Override
public VisitResult visit(VisitContext context, UIComponent target) {
if (target instanceof UIInput) {
values.add((Comparable) ((UIInput) target).getValue());
}
return VisitResult.ACCEPT;
}
});
boolean ordered = new ArrayList<Comparable>(new TreeSet<Comparable>(values)).equals(values);
if (!ordered) {
event.getComponent().visitTree(VisitContext.createVisitContext(context), new VisitCallback() {
@Override
public VisitResult visit(VisitContext context, UIComponent target) {
if (target instanceof UIInput) {
((UIInput) target).setValid(false);
}
return VisitResult.ACCEPT;
}
});
context.validationFailed();
context.addMessage(null, new FacesMessage("Values are not in order!"));
}
}
请注意,它会访问树两次;第一次收集值,第二次将这些输入标记为无效。还要注意,这个非常具体的要求不能用标准的 JSF 验证器来完成。你不能附加一个<f:validator>
on <ui:repeat>
。附加它在<h:inputText>
理论上是可能的,但它会导致同一个验证器运行的次数与重复项的数量一样多,这是没有意义的。此外,验证者需要以这种方式考虑getSubmittedValue()
vs。getValue()
OmniFaces 有一个<o:validateOrder>
组件,它在固定组件上做类似的事情,但它不是为在动态重复组件中使用而设计的。