2

我有以下页面/bean结构(使用myfaces 2.0.11 + tomcat 6)

我有一个复选框,当检查了他旁边的bean h:inputtext(连接到我的bean中的变量)时,请启用该复选框,并且当不选中复选框时,输入将被禁用,并且我有一个提交按钮,将它们提交给两者(整个表单)Integer

这是代码

<h:form prependId="false">
    <h:selectBooleanCheckbox id="my_len" value="#{myBean.myLenBool}">
        <f:ajax render="my_len_input_wrapper"/>
    </h:selectBooleanCheckbox>
    <h:panelGroup id="my_len_input_wrapper">
        <h:inputText value="#{myBean.myLen}" id="my_len_input"
            disabled="#{not myBean.myLenBool}" required="#{myBean.myLenBool}">
            <f:validateLongRange minimum="1"/>
        </h:inputText>
        <h:message for="my_len_input"/>
    </h:panelGroup> 

        <h:commandButton action="#{myBean.submit}" value="submit">
            <f:ajax render="@form" execute="@form"></f:ajax>
        </h:commandButton>
</h:form>

豆码

import javax.faces.bean.ManagedBean;
import javax.faces.bean.SessionScoped;

@ManagedBean
@SessionScoped
public class myBean {
    Integer myLen;
    boolean myLenBool;

    public Integer getMyLen() {
        return myLen;
    }

    public void setMyLen(Integer myLen) {
        this.myLen = myLen;
    }

    public boolean isMyLenBool() {
        return myLenBool;
    }

    public void setMyLenBool(boolean myLenBool) {
        this.myLenBool = myLenBool;
    }

    public void submit() {
        // submit
    }

}

场景如下

1)选中复选框(将启用输入) 2)输入无效值(例如 0.5),其无效原因myLenInteger 3)点击提交 -> 将显示错误消息h:message导致转换错误 4)取消选中复选框(它将禁用输入文本)5)点击提交<---表单未提交导致转换错误?!?!

所以问题是:如何提交带有转换错误的禁用输入的表单???

到目前为止,我发现的唯一解决方案是编写自己的自定义converter,如果该字段被禁用,则忽略转换

@FacesConverter("APCustomConverter")
public class APCustomConverter extends IntegerConverter{

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String value) {

        if (component.getAttributes().get("disabled") != null && component.getAttributes().get("disabled").equals(true)) {
            return null;
        }
        Object retValue = super.getAsObject(context, component, value);
        return retValue;
    }
}

希望有比我更好的解决方案(使用CustomConverter),

有点离题/不太离题:

这个转换错误最终让我遇到了一个非常烦人的场景:当我使用只有 render="@form" 的丢弃按钮时,复选框的状态和输入出现错误,以至于在单击丢弃后 -> 复选框仍然处于选中状态(虽然它不应该导致表单没有真正提交)并且输入被禁用而不是只读导致真正的复选框值是错误的,并且当我再次点击提交时,复选框变得真正被选中,但输入自己得到一个空值(所有这会导致服务器上出现空指针异常),所以最终我不得不在丢弃按钮中使用<f:actionListener type="org.omnifaces.eventlistener.ResetInputAjaxActionListener" />omnifaces 。

4

1 回答 1

3

使用Mojarra 2.1.26MyFaces 2.0.11测试。取消选中该复选框时,Mojarra 会更新h:inputText并将其留空,这是预期的行为,因为该值从未达到模型(验证错误)。但是,MyFaces 仅将其更新为禁用模式,将旧的输入值保留在那里。

这似乎是一个 MyFaces 问题,即使在最新的(2.1.12-2.0.18)分支版本中也没有解决。实际上,如果您想根据其状态跳过某些元素的转换/验证,则可以编写自定义转换器/验证器,但在您的情况下,问题与 MyFaces 的 ajax 周期有关,它应该像 Mojarra 的一样工作做。

作为解决方案,您有三种可能的选择:

  • 如果可能,切换到 Mojarra 实现,这将使问题消失。
  • 编写一些 JS 代码,以便在未选中复选框时重置输入值。
  • 使用您当前的解决方案,在被禁用的情况下跳过服务器端的元素转换。
于 2013-10-15T09:35:09.200 回答