使用JSF Validator

时间:2013-02-08 08:10:43

标签: jsf validation

我使用Prime faces 3.4.1作为组件框架,使用JSF 2.0作为服务器端框架

以下是我的要求

1)我有一个标签为“需要会议”的字段。然后我有SelectOneRadio有两个选项“No”和“Yes”,默认值为“No”。我正在使用JSF / HTML组件h:SelectOneRadio。

2)我有另一个字段,它是Calendar组件,这是一个primefaces日历组件。当用户选择“是”时,表示“需要会议”,用户应从日历控件中选择日期。

3)如果用户选择“是”并且未选择日期,则应显示验证消息,指示应选择日期。

我创建了一个Custom Validation组件并附加到SelectOneRadio,我可以在Custom validator中看到所选的值。现在,我尝试从Calendar组件中获取值,以通过UIComponent.getParent()。findCompoent(“rvmDate”)来检查值是否为空,我得到组件但我不知道如何检查日期组件是空的或包含任何值。

请帮助我获取用户选择的日期值。

请帮我解决这个问题。或者还有其他方法吗?请找到源代码。

XHTML页面

<h:selectOneRadio id="rvmMeetingOption"
    readonly="#{wipMB.rvmMeetingOptionReadOnly}"
    value="#{wipMB.requirementsMeeting}" 
    disabled="#{wipMB.rvmMeetingOptionDisabled}"
    validator="#{wipMB.validateRVMDate}"
    validatorMessage="Please enter RVM Date>

     <f:selectItem itemLabel="No" itemValue="0"></f:selectItem>
     <f:selectItem itemLabel="Yes" itemValue="1" ></f:selectItem>
     <f:attribute value="#{rvmDateComp}" name="rvmDateComp"></f:attribute>
</h:selectOneRadio>

<p:calendar id="rvmDate" 
        readonly="#{wipMB.rvmMeetingDateReadOnly}"
        disabled="#{wipMB.rvmMeetingDateDisabled}"
        readonlyInput="true"
        navigator="true" mode="popup" 
        pattern="dd/MM/yyyy" 
        value="#{wipMB.rvmDate}" 
        effect="explode" 
        yearRange="1900:2500" 
        style="margin-left:5px"
        binding="#{rvmDateComp}"
</p:calendar>
<p:message id="rvmDateMsg" for="rvmDate" display="both" ></p:message>

自定义验证器

public void validateRVMDate(FacesContext context, UIComponent component, Object value)
        throws ValidatorException
{
    String invalidDate;
    String rvmOption;
    Date rvmDate;
    String rvmDt = "";

    try
    {

        FacesContext fc = FacesContext.getCurrentInstance();

        rvmOption = value.toString();
        DateFormat formatter = new SimpleDateFormat("E MMM dd HH:mm:ss Z yyyy");


        UIInput rvmCompDt = (UIInput)component.getAttributes().get("rvmDateComp");

        rvmDateId = rvmCompDt.getId();
        rvmDt = rvmCompDt.getSubmittedValue() == null ? "" : rvmCompDt.getSubmittedValue().toString();


        if (rvmOption.equals("1") && rvmDt.isEmpty())
        {

            FacesMessage msg = new FacesMessage("RVM date is required");
            msg.setSeverity(FacesMessage.SEVERITY_ERROR);
            fc.addMessage("rvmDateMsg", msg);
            throw new ValidatorException(msg);

        }
    }

    catch (Exception ex)
    {
        String msg = ex.getMessage();
    }
}

3 个答案:

答案 0 :(得分:3)

在JSF中,每个组件都是它自己的一个小MVC堆栈;有Model(存储为值),Controller(组件对象)和View(渲染器)。验证器和控制器是架构的一部分,需要在模型和视图之间移动值。

虽然JSF Validators发挥了重要作用,但重要的只是INSIDE这个小MVC堆栈。它们不是为了“验证表格”而设计的,它们严格地用于“验证组件价值”。令人遗憾的是,“验证器”这个名称让每个参与JSF的人都认为,每次需要验证任何验证时,验证器都是解决方案。奇怪的是,转换器并没有被滥用。

在您的情况下,构建自定义验证器会产生一种奇怪的情况,其中:

  • 验证器和视图之间存在循环依赖关系,
  • 需要使用黑客(例如“立即”)和低级API,
  • 在不太可能的地方硬编码视图逻辑,
  • 需要更多的知识并且是片状的。例如,根据文档顺序中的单选按钮是在日历之前还是之后,从日历获取值的逻辑可能不同。

上述问题可以解决,但由于它们都是滥用JSF架构引起的,我认为重新思考这个问题会更好。由于您的验证涉及应用程序的流程,因此它非常适合于操作方法,其中所有复杂性将溶解为带有条件“addMessage”的单个简单“if”语句。

答案 1 :(得分:3)

首先需要从immediate="true"中删除<p:calendar>,否则在处理单选按钮时根本不会处理它。

然后,要检查字符串是空还是空,只需执行

String dt = (String) uiCalendar.getSubmittedValue();

if (dt == null || dt.isEmpty()) {
    // dt is null or empty. Throw validator exception depending on the 
    // current radio button value. Note: you should not catch it yourself!
}

请注意,这与JSF无关。它只是基本的Java。您if (dt == "")的初始尝试确实完全无效。 String是一个对象,而不是一个原始对象。 ==按引用比较对象,而不是按内部值比较。从技术上讲,您应该使用if (dt.equals("")),但isEmpty()更好。


对于具体问题,

无关,更简单的方法是只检查日历组件的required属性中的单选按钮值。首先通过binding将单选按钮组件绑定到视图中的变量,然后在UIInput#getValue()属性中引用其required方法。

<h:selectOneRadio id="rvmMeetingOption" binding="#{rvmMeetingOption}"
        readonly="#{wipMB.rvmMeetingOptionReadOnly}"
        value="#{wipMB.requirementsMeeting}" 
        disabled="#{wipMB.rvmMeetingOptionDisabled}">
    <f:selectItem itemLabel="No" itemValue="0"></f:selectItem>
    <f:selectItem itemLabel="Yes" itemValue="1" ></f:selectItem>
</h:selectOneRadio>

<p:calendar id="rvmDate" 
        readonly="#{wipMB.rvmMeetingDateReadOnly}"
        disabled="#{wipMB.rvmMeetingDateDisabled}"
        readonlyInput="true"
        navigator="true" mode="popup" 
        pattern="dd/MM/yyyy" 
        value="#{wipMB.rvmDate}" 
        effect="explode" 
        yearRange="1900:2500" 
        style="margin-left:5px"
        required="#{rvmMeetingOption.value == 1}">
</p:calendar>

答案 2 :(得分:0)

使用

UIInput uiCalendar = (UIInput) component.getParent().findComponent("rvmDate");
Date test = uiCalendar.getValue();
if(test==null){
throw ValidatorException
}
然后,

测试将填写日期,或者在日期字段

中未选择任何内容时将为空