JSF 中的动态组件

wax*_*ing 5 java jsf jsf-2

我对在 JSF 中创建动态组件感兴趣。我的意思是一个组件,其外观和行为取决于所传递的变量而有所不同。

让我们举一个实际有效的简单例子。一个复合组件(facelet),根据数据隐藏/显示其自身的不同部分。在本例中,它采用名为“myBean”的属性,您可以想象该属性具有“值”字段和“类型”字段。“type”字段决定应该渲染什么组件。

<?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:jsp="http://java.sun.com/JSP/Page"
      xmlns:h="http://java.sun.com/jsf/html"
      xmlns:ui="http://java.sun.com/jsf/facelets"
      xmlns:f="http://java.sun.com/jsf/core"
      xmlns:c="http://java.sun.com/jsp/jstl/core"
      xmlns:cc="http://java.sun.com/jsf/composite">

    <cc:interface>
        <cc:attribute name="myBean" />
    </cc:interface>

    <cc:implementation>

        <!-- Small input field if type is "shorttext"  -->
        <h:inputText value="#{myBean.value}" rendered="#{myBean.type == 'shorttext'}" />

        <!-- Text area input field if type is "longtext"  -->
        <h:inputTextArea value="#{myBean.value}" rendered="#{myBean.type == 'longtext'}" />

    </cc:implementation>
</html>
Run Code Online (Sandbox Code Playgroud)

虽然这有效,但它很快就会变得笨拙。如果我有二十种不同的类型,我将它们全部指定在同一个文档中,这显然违反了良好的设计。

相反,我希望能够用以下内容替换实现:

    <ui:include value="#{myBean.type}"/>
Run Code Online (Sandbox Code Playgroud)

我理解这是不可能的,因为ui:include在构建组件树时发生,并且组件在渲染阶段进行处理。

但这一定是一个普遍的问题。实现动态组件的最佳方法是什么?我正在使用 JSF 2.0,如果这有什么区别的话。

Bal*_*usC 2

我没有看到更好的方法。XML 不是面向对象的语言。这是通过 XML 可以获得的最好的结果。如果你想以面向对象的方式做到这一点,那么你应该创建一个带有渲染器的真实组件。

顺便说一下,复合组件是在 JSF 2.0 中引入的,因此在 JSF 1.x 中不可用。所以这没有多大区别;)