4

我有一个页面,我在其中呈现一些 h:panelGroup 面板。这些面板被实现为在启动时在插件注册表中注册的插件。插件 api 的一部分是一个自定义 jsf 组件,我在其中获取扩展点的注册插件,并通过路径包含它们的 facelet 模板:

<c:forEach items="#{pluginRegistry.getPlugins(point)}" var="extension">
    <ui:include src="#{extension.path}" />
</c:forEach>

我包含面板的页面如下所示:

<h:panelGrid id="dashboard" columns="3">
    <cmf:insertPageFragments point="dashboardExtensionPoint" />
</h:panelGrid>

对于每个面板,都有如下 facelet 模板:

<rich:panel id="caseDetailsPanel" header="panel label">
    <!-- panel content -->
</rich:panel>

现在,问题是 pluginsRegistry 返回的列表中的第一个面板使用提供的 id 呈现在页面中,例如 formId:caseDetailsPanel。其余的都生成了像 formId:j_idt223 这样的 id !!!显然,如果我想重新渲染一些面板,我不能这样做。

当环境是带有 JSF 2.1、richfaces 4.2.3.Final 的 jboss AS 7.1 时,就会发生这种情况。当部署在 jboss-eap-6.1 上时,一切看起来都很好,但现在我不能使用这个 jboss 版本。

有关如何解决此问题的任何建议?

4

1 回答 1

5

不能有多个具有相同 ID 的 JSF 组件。每个 JSF 组件都必须有一个唯一的 ID。使用 JSTL 动态创建 JSF 组件时,需要手动分配并确保唯一 ID,否则 JSF 会丢弃提供的 ID 并自动生成唯一 ID。

有几种方法可以实现这一点,具体取决于具体的功能要求和现有代码。

  1. 使用 的迭代索引<c:forEach>

    <c:forEach ... varStatus="loop">
        ...
        <rich:panel id="caseDetailsPanel_#{loop.index}" ...>
    

    这将根据当前的迭代索引生成caseDetailsPanel_0,等。caseDetailsPanel_1

  2. 使用当前迭代项的唯一标识符。根据目前提供的信息尚不清楚是否有任何信息,因此这里只是一个虚构的示例,假设后面的类#{extension}具有id表示技术 DB 标识符的属性。

    <c:forEach ... var="extension">
        ...
        <rich:panel id="caseDetailsPanel_#{extension.id}" ...>
    
  3. 如有必要,将 #1 或 #2 包装在<f:subview>带有唯一标识符的 a 中,这样您就无需修改包含。

    <c:forEach ... varStatus="loop">
        <f:subview id="panel_#{loop.index}">
            <ui:include ... />
    

    <f:subview>围绕它创建一个新的NamingContainer,所以你最终得到formId:panel_0:caseDetailsPanelformId:panel_1:caseDetailsPanel等等。

一个完全不同的选择是使用<ui:repeat>而不是<c:forEach>. <ui:repeat>不在视图构建期间运行,而是在视图渲染期间运行。这样<rich:panel id="caseDetailsPanel">,在组件树中实际上只有一个组件在生成 HTML 期间被多次重用,因此 JSF 将负责生成带有<ui:repeat>索引的正确 ID,如下所示formId:repeatId:0:caseDetailsPanel。但是,这反过来可能会导致麻烦,<ui:include>因为它也在视图构建期间运行,因此无法获得#{extension}

于 2013-05-23T12:52:47.647 回答