問題

私はユーザーが好きなようにアイテムを固定したり削除したりできる動的ダッシュボードを使って作業しています。今私はバッキングBeanからビューに既存の複合コンポーネントを追加したいという問題があります。私はインターネットからこれを行う正しい方法を見つけようとしましたが、これまでのところ成功しませんでした。追加したい単純な複合コンポーネント

 <?xml version='1.0' encoding='UTF-8' ?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
      xmlns:cc="http://java.sun.com/jsf/composite"
      xmlns:ui="http://java.sun.com/jsf/facelets"
      xmlns:h="http://java.sun.com/jsf/html"
      xmlns:f="http://java.sun.com/jsf/core"
      xmlns:p="http://primefaces.org/ui"
      xmlns:composite="http://java.sun.com/jsf/composite">
    <!-- INTERFACE -->
    <cc:interface>

    </cc:interface>

    <!-- IMPLEMENTATION -->
    <cc:implementation>
        <h:outputText value="TEST"/>
    </cc:implementation>
</html>
 

複合コンポーネントを返すコードは次のとおりです。

 public static UIComponent getCompositeComponent(String xhtml, String namespace) {
    FacesContext fc = FacesContext.getCurrentInstance();
    Application app = fc.getApplication();
    Resource componentResource = app.getResourceHandler().createResource(xhtml, namespace);

    UIPanel facet = (UIPanel) app.createComponent(UIPanel.COMPONENT_TYPE);
    facet.setRendererType("javax.faces.Group");
    UIComponent composite = app.createComponent(fc, componentResource);
    composite.getFacets().put(UIComponent.COMPOSITE_FACET_NAME, facet);

    return composite;
}
 

そして、ここで私は関数を使用しています:

 Column column = new Column();
UIComponent test = HtmlUtil.getCompositeComponent("test.xhtml", "comp");
column.getChildren().add(test);
 

しかし、列の中には何もレンダリングされません。どのようにこれを行うことができますか?私は私のユースケースに合っていないので、 rendered = "#{bean.isThisRendered}" の方法で行きたくありません。

  ベストアンサー

このコードは不完全です。複合コンポーネントの実装に複合コンポーネントリソースを含めるには、 FaceletContext#includeFacelet() を使用する必要があります。ジョブを実行するユーティリティメソッドを次に示します。 #{cc}をELスコープで作成する必要があるコンテキストであるため、親を手に入れることが重要です。したがって、このユーティリティメソッドは、指定された親の子としてコンポジットを直ちに追加します。さらに、コンポーネントに固定IDを与えることが重要

 public static void includeCompositeComponent(UIComponent parent, String libraryName, String resourceName, String id) {
    // Prepare.
    FacesContext context = FacesContext.getCurrentInstance();
    Application application = context.getApplication();
    FaceletContext faceletContext = (FaceletContext) context.getAttributes().get(FaceletContext.FACELET_CONTEXT_KEY);

    // This basically creates <ui:component> based on <composite:interface>.
    Resource resource = application.getResourceHandler().createResource(resourceName, libraryName);
    UIComponent composite = application.createComponent(context, resource);
    composite.setId(id); // Mandatory for the case composite is part of UIForm! Otherwise JSF can't find inputs.

    // This basically creates <composite:implementation>.
    UIComponent implementation = application.createComponent(UIPanel.COMPONENT_TYPE);
    implementation.setRendererType("javax.faces.Group");
    composite.getFacets().put(UIComponent.COMPOSITE_FACET_NAME, implementation);

    // Now include the composite component file in the given parent.
    parent.getChildren().add(composite);
    parent.pushComponentToEL(context, composite); // This makes #{cc} available.
    try {
        faceletContext.includeFacelet(implementation, resource.getURL());
    } catch (IOException e) {
        throw new FacesException(e);
    } finally {
        parent.popComponentFromEL(context);
    }
}
 

したがって、特定の例では、次のように使用します。

 includeCompositeComponent(column, "comp", "test.xhtml", "someUniqueId");
 

  同じタグがついた質問を見る

dynamicjsf-2faceletscomposite-component