3

我有这个f:viewParam,我尝试绑定 validate 并将 a 转换userIdPlayer,我得到了意想不到的结果。

<f:metadata>
    <f:viewParam name="userId" value="#{myBean.selectedPlayer}" converter="pConverter"
         converterMessage="Bad Request. Unknown User" required="true"
         requiredMessage="Bad Request. Please use a link from within the system" />

</f:metadata>
<h:body>
    <p:messages id="msgs"/>        
    <h:form>
        <ul>
            <li><a href="index2.xhtml?userId=1">Harry</a></li>
            <li><a href="index2.xhtml?userId=2">Tom</a></li>
            <li><a href="index2.xhtml?userId=3">Peter</a></li>
        </ul>            
    </h:form>
    <h:form>
        <h:panelGrid columns="2" rendered="#{not empty myBean.selectedPlayer}">

            <h:outputText value="Id: #{myBean.selectedPlayer.id}"/>

            <h:outputText value="Name: #{myBean.selectedPlayer.name}"/>

        </h:panelGrid>
    </h:form>
    <h:form id="testForm">
        <h:inputText value="#{myBean.text}"/>
        <p:commandButton value="Switch" update=":msgs testForm"/>
        <h:outputText value="#{myBean.text}" rendered="#{not empty myBean.text}"/>
    </h:form>    
</h:body>

我的转换器看起来像这样

@FacesConverter(value="pConverter")
public class PConverter implements Converter {
private static final List<Player> playerList;
static{
    playerList = new ArrayList<Player>();        
    playerList.add(new Player(1, "Harry"));
    playerList.add(new Player(2, "Tom"));
    playerList.add(new Player(3, "Peter"));
}

@Override
public Object getAsObject(FacesContext fc, UIComponent uic, String value) {
    if(value == null || !value.matches("\\d+")){
        return null;
    }
    long id = Long.parseLong(value);
    for(Player p : playerList){
        if(p.getId() == id){
            return p;
        }
    }
    throw new ConverterException(new FacesMessage("Unknown userId: " + value));

}

@Override
public String getAsString(FacesContext fc, UIComponent uic, Object value) {
    if(!(value instanceof Player) || value == null){
        return null;
    }
    return String.valueOf(((Player)value).getId());
}
}

当我单击三个链接(Harry、Tom、Peter)时,转换器工作得很好。它转换 id 并将player回绑定到我的托管 bean。然后我在文本框中输入一些东西,然后单击Switch,第一次它工作正常,我输入的内容出现在按钮旁边,但后来我更改了我输入的内容,Switch再次单击,然后出现错误消息Bad Request. Please use a link from within the system,这是错误消息为. required_ f:viewParam如果我把 f:viewParam 拿出来,那么一切正常。令人惊讶的是,如果我从 f:viewParam 切换到 o:viewParam (OmniFaces),那么效果很好。

4

1 回答 1

5

这是因为<f:viewParam>在每个 HTTP 请求上运行,也在回发上。它在您的情况下适用于普通的 GET 链接,因为您在链接中准确地传递了该参数。对于 POST 表单,它会失败,因为您没有在按钮中传递该参数。因此它出现null在请求参数映射中,required验证器启动,因此出现此验证错误。

为了保持<f:viewParam required="true">对 POST 表单的满意,您基本上需要<f:param>在命令按钮/链接中保留初始请求参数。

<p:commandButton value="Switch" update=":msgs testForm">
    <f:param name="userId" value="#{param.userId}" />
</p:commandButton>

OmniFaces<o:viewParam>旨在与视图范围的 bean 结合使用,在isRequired()getter 中有一个额外的检查(源代码在这里):

@Override
public boolean isRequired() {
    // The request parameter get lost on postbacks, however it's already present in the view scoped bean.
    // So we can safely skip the required validation on postbacks.
    return !FacesContext.getCurrentInstance().isPostback() && super.isRequired();
}

因此,这会required在每次回发时跳过验证器(此外,由于其无状态性质,它还会在每次回发时跳过设置模型值)。这就是为什么您看不到验证错误并且您仍然拥有正确的模型值(不会在每次回发时重置)的原因。

于 2012-05-10T02:18:47.590 回答