8

我想处理这个表单(valueChangeListener 在实际情况下无效)。

这是后面的bean:

public class TestBean extends PrivateBaseBean implements Serializable {
private List<String> strings;

@PostConstruct
public void init() {
    strings = new ArrayList<String>();
    strings.add("");
    strings.add("");
    strings.add("");
}

public void saveAction(ActionEvent event) {

    StringBuilder textToShowInMessage = new StringBuilder();
    for (String string : strings) {
        textToShowInMessage.append(string);
        textToShowInMessage.append("; ");
    }
    FacesMessage msg = new FacesMessage(super.getBundle().getString(
            textToShowInMessage.toString()), "");

    FacesContext.getCurrentInstance().addMessage(null, msg);
}

getters... setters...

一个观点:

....
<h:form>
<ui:repeat var="string" value="#{testBean.strings}">
    <h:inputText value="#{string}" />
    <br />
</ui:repeat>
<p:commandButton value="#{msg.save}"
actionListener="#{testBean.saveAction}" icon="ui-icon-disk"
        update="@form" />
</h:form>
...

当表单被处理后,bean 字符串列表总是为空。

如何在没有任何值更改侦听器的情况下处理表单输入的内部迭代?

有一些截图: 调试代码

浏览器中的表格

同样的问题发生在 action 或 actionListener on

4

2 回答 2

13

您的问题与 PrimeFaces<p:commandButton>的行为无关,而是与使用<ui:repeat>标签时隐含的范围界定问题有关。

首先,让我们离开你的例子。基本上,你有

<ui:repeat value="#{bean.strings}" var="s">
    <h:inputText value="#{s}"/>
</ui:repeat>

有后盾List<String> strings

罪魁祸首在这里:value="#{s}"。导出的<ui:repeat>变量s仅在其循环内可见,并且它不绑定到任何托管 bean 的属性,而仅绑定到局部变量。换句话说,它不像人们所期望的那样s被束缚/等于,bean.strings[index]并且正如我们所看到的,它不知道它的起源。所以基本上,你是单方面的关系:来自 bean 的值被正确地打印在你的输入中,但相反的情况没有发生。

解决方法

解决方法#1:包装类/模型对象

可以通过为您的类使用包装器对象来克服这种情况。如果是字符串,它可能是“简单可变字符串”,如下所示:

public class MString {
    private String string;//getter+setter+constructor
}

在这种情况下,迭代将按预期工作:

<ui:repeat value="#{bean.mstrings}" var="ms">
    <h:inputText value="#{ms.string}"/>
</ui:repeat>

有后盾List<MString> mstrings

请注意,如果您有模型类,例如User,并且将在类中更改其属性,<ui:repeat>则该类本身将有效地成为包装器,以便适​​当地设置属性。

解决方法 #2:链式属性访问

另一种解决方法是直接从<h:inputText>标签内访问集合中的元素。这样,任何此类属性都将通过访问 bean,然后是集合,然后在所需索引处设置属性来设置。太长了,但就是这样。至于如何问题,<ui:repeat>提供了一个导出的当前迭代状态变量,varStatus将用于访问托管 bean 中的数组/集合。

在这种情况下,迭代也将按预期工作:

<ui:repeat value="#{bean.strings}" var="s" varStatus="status">
    <h:inputText value="#{bean.strings[status.index]}"/>
</ui:repeat>

有普通的靠山List<String> strings

于 2013-05-14T18:59:00.637 回答
2

我的解决方案直接从页面获取值:

<ui:repeat id="repeat" value="#{bean.strings}" var="s" varStatus="status">
    <h:inputText id="x" value="#{s.field}"/>
    <h:commandLink style="margin: .5em 0" styleClass="commandLink" actionListener="#{bean.save(status.index)}" value="#{bundle.Send}"/>
</ui:repeat>

保存方法:

public void save(String rowid) {
        String jsParam = Util.getJsParam("repeat:" + rowid + ":x");
        System.out.println("jsParam: " + jsParam); //persist...
    }

getJsParam 方法:

public static String getJsParam(String paramName) {
    javax.faces.context.FacesContext jsf = javax.faces.context.FacesContext.getCurrentInstance();
    Map<String, String> requestParameterMap = jsf.getExternalContext().getRequestParameterMap();
    String paramValue = requestParameterMap.get(paramName);
    if (paramValue != null) {
        paramValue = paramValue.trim();
        if (paramValue.length() == 0) {
            paramValue = null;
        }
    }
    return paramValue;
}
于 2013-12-12T09:47:11.253 回答