我有两个 jsf 页面。layout.xhtml 和 page.xhtml。布局如下所示:
<ui:composition>
<h:panelGroup id="menu" layout="block">
<h:outputText value="#{menuBean}" />
<h:form>
<ui:repeat var="menuItem" value="#{menuBean.menuItems}">
<button:menuItem label="#{msgs[menuItem.label]}" action="#{menuBean.selectItem(menuItem.label)}" update="#{update}" />
</ui:repeat>
</h:form>
</h:panelGroup>
<ui:repeat var="menuItem" value="#{menuBean.menuItems}">
<h:panelGroup layout="block" rendered="#{menuBean.selectedItemLabel eq menuItem.label}">
<ui:include src="#{menuItem.page}" />
</h:panelGroup>
</ui:repeat>
和这样的页面:
<h:panelGroup binding="#{page}" layout="block">
<ui:decorate template="../template.xhtml">
<ui:param name="menuBean" value="#{pageBean}" />
<ui:param name="update" value=":#{page.clientId}" />
</ui:decorate>
</h:panelGroup>
当我第一次渲染页面时,一切正常(渲染菜单项等)。单击任何按钮后,我得到Target Unreachable, identifier 'menuBean' resolved to null
.
谁能解释发生了什么,为什么menuBean
没有再次分配,如果存在另一种实现这种事情的方法(有一些通用布局页面,在那里传递一些对象并生成页面)?我的 bean 由 Spring 管理。
更新:
我猜这个问题与我的复合按钮有关,如下所示:
<composite:interface name="menuItem">
<composite:attribute name="action" targets="button" />
<composite:attribute name="styleClass" />
<composite:attribute name="label" />
<composite:attribute name="update" />
<composite:attribute name="rendered" />
<composite:insertChildren />
</composite:interface>
<composite:implementation>
<h:commandButton id="button" value="#{cc.attrs.label}" style="width: 150px;" action="#{cc.attrs.action}"
rendered="#{empty cc.attrs.rendered ? true : cc.attrs.rendered}" styleClass="menu-item #{cc.attrs.styleClass}" type="button">
<f:ajax render="#{cc.attrs.update}" />
</h:commandButton>
</composite:implementation>
如果我用标准 h:commandButton 替换标签,一切都会完美。我将对象传递给模板,模板将传递的对象传递给复合对象,但我肯定遗漏了一些东西。