【发布时间】:2013-02-27 15:15:37
【问题描述】:
我读了很多类似的问题,但我不知道如何解决我的问题:
我编写了一个复合组件,该组件由视图范围的自包含托管 bean 提供支持。
该组件由一个自动完成文本框和一个打开对话框的按钮组成。用户可以通过名称(自动完成)或在树中选择一个节点(对话框)来选择一个项目。
支持 bean 实现了所有需要的东西(数据访问、树逻辑等),并且应该公开选定的项目(作为 POJO)。
现在我有两个问题:
由于树管理的复杂性,
selectedObj属性由在 bean 中执行某些操作的 getter 和 setter 访问:它们不只是访问类字段。现在我将整个 bean 作为属性传递。我怎样才能让 bean 的selectedObj成为我的复合组件的“值”属性?如何在同一个视图中使用我的组件的多个实例?
这是一个组件示例:
<cc:interface>
<cc:attribute name="bean" type="com.yankee.OUTreeBean" required="true"/>
<cc:attribute name="listener" method-signature="void listener()"/>
</cc:interface>
<cc:implementation>
<p:dialog id="#{cc.id}_dialog" widgetVar="_dlg" header="Select OU" modal="true" dynamic="true" >
<p:toolbar>
<!-- some buttons to refresh, expand, collapse etc. -->
</p:toolbar>
<p:tree id="#{cc.id}_tree" value="#{cc.attrs.bean.root}" var="node"
selectionMode="single"
selection="#{cc.attrs.bean.selectedNode}">
<p:ajax event="select" update="@form" listener="#{cc.attrs.listener}" oncomplete="if (!args.validationFailed) _dlg.hide()" />
<p:treeNode>
<h:outputText value="#{node.OU_NAME}" />
</p:treeNode>
</p:tree>
</p:dialog>
<p:autoComplete id="#{cc.id}_inner" value="#{cc.attrs.bean.selectedObj}" completeMethod="#{cc.attrs.bean.completeObj}"
var="obj" itemLabel="#{obj.OU_NAME}" itemValue="#{obj}"
forceSelection="true"
converter="ouConverter"
multiple="false"
minQueryLength="2">
<p:ajax event="itemSelect" listener="#{cc.attrs.listener}" update="@form"/>
</p:autoComplete>
<div style="float: right">
<p:commandButton id="bSearch" icon="ui-icon-search" onclick="_dlg.show()"/>
</div>
</cc:implementation>
COMPONENT 的支持 bean:
@ManagedBean
@ViewScoped
public class OUTreeBean implements Serializable {
private static final long serialVersionUID = 1L;
private List<OU> data; // Data as plain list
protected TreeNode root; // root node of data as a tree
protected TreeNode selectedNode;
@PostConstruct
private void init() throws SQLException {
refreshData();
}
public OU getSelectedObj() {
if (selectedNode == null) {
return null;
}
return ((OU) selectedNode.getData());
}
public void setSelectedObj(OU ou) {
// Find the right tree node and do whatever needed
}
public TreeNode selectedNode getSelectedNode() {
// Blah blah
}
public void setSelectedNode(TreeNode selectedNode) {
// Blah blah
}
public List<OU> completeObj(String namePattern) {
// Autocomplete handler
}
public void refreshData() {
// Blah blah
}
// etc...
}
使用页面摘录:
<ism:selectOUTree id="cbSelectOu" bean="#{myBean.ouFilterBean}" listener="#{myBean.onOUChange}"/>
PAGE 的支持 bean:
@ManagedBean
@ViewScoped
public class MyBean implements Serializable {
private static final long serialVersionUID = 1L;
@ManagedProperty("#{oUTreeBean}")
private OUTreeBean ouFilterBean;
public void onOUChange() throws SQLException {
// Blah blah
}
}
【问题讨论】:
-
它应该与
#{cc.attrs.value}一起使用,如果你有正确的 getter 和 setter,value="#{myBean.ouFilterBean.selectedObj}" -
@Christophe Roussy 好的,谢谢,但目的是从页面的 bean 中“隐藏”组件的支持 bean。
MyBean应该不知道OUTreeBean的存在,因此该组件应该绑定到 MyBean 类型为OU的属性 -
在 myBean 中提供一些 getter 和 setter,这样就可以直接访问对象(通过实现接口来强制执行...)