2010-08-17 82 views
4

我有兴趣在JSF中创建动态组件。我的意思是一个组件的外观和行为依赖于它传递的变量。JSF中的动态组件

让我们来看一个实际工作的简单例子。根据数据隐藏/显示其自身不同部分的复合组件(facelet)。在这种情况下,它需要一个名为“myBean”的属性,您可以想象它具有“值”字段和“类型”字段。 “类型”字段确定应该呈现哪个组件。

<?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> 

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

相反,我希望能够与像更换执行:

<ui:include value="#{myBean.type}"/> 

我的理解是不可能的,因为建筑组件树时出现ui:include,而该组件在处理渲染阶段。

然而,这一定是一个常见问题。什么是实现动态组件的最佳方法?我正在使用JSF 2.0,如果这有所作为。

回答

2

我没有看到更好的方法。 XML不是OO语言。这是您用XML获得的最好结果。如果你想以OO的方式来做,那么你应该用渲染器创建一个真实的组件。

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

+0

你知道,我自己也得出了同样的结论。看起来,在动态修改组件树的过程中,这种方法甚至可能会带来好处:即使组件树位于ui:repeat(fly-weight模式)中,组件树也只会是一次。 +1,但现在我将把问题留给公开。 – waxwing 2010-08-17 12:05:09