我有一个非常庞大的数据捕获形式。主要问题是我必须两次点击提交按钮才能提交表单。我第一次按它,它只是刷新,什么都不做。为什么会这样?在我的表单中,有大量的输入文本。
还有一些动态选择菜单,其中项目列表会根据不同菜单中的所选项目而变化。我怀疑问题在于这里存在valueChangeListener,onchange =“submit()”和immediate =“true”,这可能会产生一些我不太了解的副作用。我没有收到任何错误消息。以前有人遇到过这样的问题吗?
<h:commandButton type="submit" value="Add Job Profile" action="{formBean.commitData}"/>
public void commitData() {
QueryRunner run = new QueryRunner(ds);
String insertSql = "sql command here";
try {
int numberOfCommits = run.update(insertSql);
if (numberOfCommits > 0) {
JOptionPane jop = new JOptionPane("Inserted " + numberOfCommits +"record");
JDialog dialog = jop.createDialog("Upload complete");
dialog.setAlwaysOnTop(true);
dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
dialog.setVisible(true);
} else {
JOptionPane jop = new JOptionPane("Upload failed. Please try again." );
JDialog dialog = jop.createDialog("Upload failed");
dialog.setAlwaysOnTop(true);
dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
dialog.setVisible(true);
}
} catch (SQLException sqle) {
sqle.printStackTrace();
}
}
首次点击时我甚至没有收到上传失败的消息。在第二次单击时,上传成功。
编辑:是的,这是动态选择菜单。如果我删除这些菜单,我可以在第一次点击时提交。不过,我该如何解决这个问题?<h:outputLabel value="Family: " />
<h:selectOneMenu id="selectFamily"
value = "#{formBean.selectedFamily}"
valueChangeListener="#{menuBean.changeFamily}"
onchange="submit()"
immediate = "true" >
<f:selectItems id="familyNames" value="#{menuBean.familyNames}" />
</h:selectOneMenu>
<h:outputLabel value="Function: " />
<h:selectOneMenu id="selectFunction"
value="#{formBean.selectedFunction}"
immediate="true" >
<f:selectItems id="functionNames"
value="#{menuBean.functions}" />
</h:selectOneMenu>
// listen for family change and render relevant function
public void changeFamily(ValueChangeEvent event) {
FacesContext context = FacesContext.getCurrentInstance();
String value = (String) event.getNewValue();
switchFunctions(value);
context.renderResponse();
}
答案 0 :(得分:1)
我有类似的问题显示弹出窗口。弹出窗口显示后,单击提交按钮只刷新弹出窗口。没有数据与托管bean同步。但是,再次单击该按钮时,它会提交数据。
我通过更改显示弹出窗口的代码来解决这个问题,来自......
<a4j:commandButton value="Reserve" disabled="#{login.mayReserve == false}" immediate="true"
title="To set an available number to the Reserved state."
oncomplete="#{rich:component('selectCustomer')}.show();" render="selectCustomer" >
<f:setPropertyActionListener target="#{managementController.menuAction}" value="Reserve" />
</a4j:commandButton>
要...
<a4j:commandButton value="Reserve" disabled="#{login.mayReserve == false}" immediate="true"
title="To set an available number to the Reserved state." render="selectCustomer" >
<f:setPropertyActionListener target="#{managementController.menuAction}" value="Reserve" />
<a4j:ajax event="click" oncomplete="#{rich:component('selectCustomer')}.show()" reRender="@all" />
</a4j:commandButton>
我不确定这方面的技术性是什么,但也许它有助于某人前进......
答案 1 :(得分:0)
显然,valueChangeListener
在提交时也被触发了。行context.renderResponse()
将导致完全跳过调用应用程序阶段。当初始菜单值与提交的值不同时,将触发valueChangeListener
。验证这一点。
无论如何,拥有相同的同步下拉菜单与相同形式的其他输入字段的验证相结合实际上是一个皮塔饼。所以简而言之,这里只是一个链接:populate child menus (in JSF)。