在我的应用程序中,我有三个下拉菜单(p:selectOneMenu
),比如A,B,C。其中两个是依赖的,比如B和C。通过更改 B 的值,我将值动态加载到 C。还有一个文本框。当从这三个下拉列表触发更改时事件时,文本框的值由 ajax 生成。
这是 xhtml:
<p:selectOneMenu id="customerMenu" value="#{adminController.activityDTO.customerId}" required="true" label="Customer Name" style="width: 200px">
<f:selectItem itemLabel="Select One" itemValue="" />
<f:selectItems value="#{adminController.customers}" var="customer" itemLabel="#{customer.customerName}" itemValue="#{customer.customerId}" />
<p:ajax listener="#{adminController.generateActivityName}" update="activityId" />
</p:selectOneMenu>
<p:selectOneMenu id="activityTypeMenu" value="#{adminController.activityDTO.activityParentType}" required="true" label="Activity Type"
style="width: 200px">
<f:selectItem itemLabel="Select One" itemValue="" />
<f:selectItems value="#{adminController.activityTypes}" var="activityType" itemLabel="#{activityType.parent}" itemValue="#{activityType.parent}" />
<p:ajax listener="#{adminController.updateDependentActivity}" update="activitySubType" />
</p:selectOneMenu>
<p:selectOneMenu id="activitySubTypeMenu" value="#{adminController.activityDTO.activitySubType}" required="true" label="Activity Sub Type"
style="width: 200px">
<f:selectItem itemLabel="Select One" itemValue="" />
<f:selectItems value="#{adminController.activitySubTypes}" var="activityType" itemLabel="#{activityType.name}" itemValue="#{activityType.id}" />
<p:ajax listener="#{adminController.generateActivityId}" update="activityId" />
</p:selectOneMenu>
<p:inputText id="activityId" autocomplete="off" readonly="true" value="#{adminController.activityDTO.activityId}"
label="#{adbBundle['admin.addActivityPanel.addActivityTable.activityId']}" required="true" />
activityTypeMenu
和activitySubTypeMenu
取决于我填充activitySubTypeMenu
activityTypeMenu
的选定值。
现在我面临的问题是:
- 假设我在
activityTypeMenu
中选择了"外部"和"内部",默认为"选择一个"。如果我从activityTypeMenu
中选择"外部",activitySubTypeMenu
将具有"项目"和"服务"。但是,如果我选择默认的"选择一个",则activitySubTypeMenu
仍然保留先前动态填充的值。这是因为required="true"
属性拒绝触发我从中加载动态值的后端方法。 - 我尝试将
<f:selectItem itemLabel="Select One" itemValue="" />
的itemValue
设置为#{null}
,然后在选择"选择一个"选项时触发后端方法,我可以将空列表设置为activitySubTypes
这样activitySubTypeMenu
为空。但在这种情况下,required="true"
变得毫无意义。我的意思是,我也有保存按钮,单击该按钮时没有从activityTypeMenu
中选择任何选项(即选择"选择一个"),并且activitySubTypeMenu
不会抛出ValidatorException
并且组件不会通过 Primefaces 的错误 css 类进行样式设置样式。 - 此外,如果我没有将
<f:selectItem itemLabel="Select One" itemValue="" />
itemValue
设置为#{null}
,那么将所选值更改为默认选项("选择一个")不会清除activityId
p:inputText
。如果我使用#{null}
那么我可以触发后端方法,从中我可以将文本框的值设置为空。
我如何解决这个问题并获得理想的结果。我想要的是:
- 如果该选项设置为"选择一个",则从属菜单将为空,文本框的菜单将为空。
- 我想使用
required="true"
属性。
您可以在 required
属性中使用 EL。只有当按下特定的提交按钮或提交特定的组件值时(因此,通过请求参数映射#{param}
中存在的客户端 ID),才能让所需的表达式计算true
。
下面的启动示例应该可以满足您的需求。
<p:selectOneMenu binding="#{menu1}" ... required="#{not empty param[submit.clientId]}">
...
</p:selectOneMenu>
<p:selectOneMenu ... required="#{not empty param[menu1.clientId]}">
...
</p:selectOneMenu>
<p:commandButton binding="#{submit}" ... />
这样,仅当按下表单的主提交按钮时才需要第一个菜单(因此在触发事件侦听器时不需要),并且仅当第一个菜单具有值时才需要第二个菜单。