我想处理这个表单(valueChangueListener在实际情况下无效) .
这是后 beans :
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}" />
</ui:repeat>
<p:commandButton value="#{msg.save}"
actionListener="#{testBean.saveAction}" icon="ui-icon-disk"
update="@form" />
</h:form>
...
在后面的bean字符串列表中处理表单时,始终为空 .
如何处理表单intput的内部迭代,没有任何值的changue监听器?
有一些截图:
action或actionListener on会出现同样的问题
2 回答
您的问题与PrimeFaces
<p:commandButton>
的行为无关,而是与使用<ui:repeat>
标记时隐含的范围问题有关 .首先,让我们离开你的榜样 . 基本上,你有
支持
List<String> strings
.罪魁祸首在这里: value="#" . 由
<ui:repeat>
变量s
导出的内容仅在其循环中可见,并且不绑定到任何托管bean的属性,而是仅绑定到局部变量 . 换句话说,s
并不像人们所期望的那样绑定/等于bean.strings[index]
,并且我们看不出它来自何处 . 所以基本上,你处于单边关系之中:来自bean的值正确地打印在你的输入中,但反过来却没有发生 .解决方法
Workaround #1: wrapper classes / model objects
通过为您的类使用包装器对象可以克服这种情况 . 如果是字符串,它可能是一个'简单的可变字符串',如下所示:
在这种情况下,迭代将按预期工作:
支持
List<MString> mstrings
.请注意,如果您拥有模型类,如
User
,并且将在<ui:repeat>
中更改其属性,则类本身将实际上是一个包装器,以便适当地设置属性 .Workaround #2: chained property access
另一种解决方法是直接从
<h:inputText>
标记中访问集合的元素 . 这样,任何此类属性都将通过访问bean,然后集合,然后在所需索引处设置属性来设置 . 过长,但就是这样 . 至于如何提问,<ui:repeat>
提供了一个导出的当前迭代状态变量varStatus
,它将用于访问托管bean中的数组/集合 .在这种情况下,迭代也将按预期工作:
与普通的支持
List<String> strings
.我的解决方案解决方案直接从页面获取值:
保存方法:
getJsParam方法: