一个图书管理系统中的两个对象:Book(书籍),BookType(书籍类型)。Book和BookType之间是多对一关系。
说到主从表的关联关系,自然而然地想起的一种实现方式就是选择框,比如在Book的编辑界面是使用一个类型的下拉选择框,选择一个类型,然后保存。于是就有下例代码:
Seam的解决方案:
其实Seam有另一种解决方案。比如如果你是自动生成代码的方式,在BookEdit.xhtml中就会看到这样的代码:
< action execute ="#{bookHome.wire}" /> : 一最重要的这个。如果没有执行这个方法这段跳转就没有任何效果了。,先来看下这个方法是怎么写的吧:
前面我说过了。这个@In就是拿来做双向注入的。 bookTypeHome是要注入的组件名称。
其他的都是参数了,没啥好解释的。
但为什么在BookTypeList页面点select,怎么就会自动跳转到BookEdit.xhtml呢?奥秘就在这段代码里(BookTypeList.xhtml):
好了,今天到这,困了,上面讲的不明白的欢迎email或qq联系我。。
说到主从表的关联关系,自然而然地想起的一种实现方式就是选择框,比如在Book的编辑界面是使用一个类型的下拉选择框,选择一个类型,然后保存。于是就有下例代码:
<
h:selectOneMenu
id
="type"
value
="#{bookHome.instance.bookType.typeId}"
>
< f:selectItems value ="#{bookTypeList.typeSelectItems}" />
</ h:selectOneMenu >
这段代码的作者(我)原本的想法是让这个下拉框与bookHome.instance里的
bookType.typeId 帮定。比如当前的book类型id是1,修改后将book类型的id改为2,更新到数据库。但是很不幸。这段代码并不能执行预期的行为,或者说它还附加了 其他行为。即Seam已经觉察到了在这个book中的类型的一个属性(主键值)已经改变了。于是,试图更新这个类型。但是JPA的规范中是不允许更改主键 的,这就引起了一个错误。不信可以试下哦,呵呵。(以上描述我已经尽可能说清楚我的想法,但可能还是不怎么清楚,希望大家看不懂的说说哪里看不懂,我好改 正)。我想说的是,不要直接帮定到外键,而是现帮定到一个临时变量,比如在bookHome中多写一个变量:
< f:selectItems value ="#{bookTypeList.typeSelectItems}" />
</ h:selectOneMenu >
public
class
BookHome
extends
EntityHome
<
Book
>
{
private Long typeId;
// ..getter setter
}
然后将下拉框邦定到这个变量上,比如:
private Long typeId;
// ..getter setter
}
<
h:selectOneMenu
id
="type"
value
="#{bookHome.typeId}"
>
< f:selectItems value ="#{bookTypeList.typeSelectItems}" />
</ h:selectOneMenu >
。然后在重载的persist或者update方法中写上:
< f:selectItems value ="#{bookTypeList.typeSelectItems}" />
</ h:selectOneMenu >
BookType newType
=
getEntityManager().find(BookType.
class
, typeId);
instance.setType(newType);
return super.persist();
就完成了。我叫它“转移邦定”,呵呵
instance.setType(newType);
return super.persist();
Seam的解决方案:
其实Seam有另一种解决方案。比如如果你是自动生成代码的方式,在BookEdit.xhtml中就会看到这样的代码:
<
div
class
="association"
id
="bookTypeParent"
>
< h:outputText value ="There is no bookType associated with this book."
rendered ="#{bookHome.instance.bookType == null}" />
< rich:dataTable var ="bookType"
value ="#{bookHome.instance.bookType}"
rendered ="#{bookHome.instance.bookType != null}"
rowClasses ="rvgRowOne,rvgRowTwo"
id ="bookTypeTable" >
< h:column >
< f:facet name ="header" > typeId </ f:facet >
#{bookType.typeId}
</ h:column >
< h:column >
< f:facet name ="header" > bookType typeId </ f:facet >
#{bookType.bookType.typeId}
</ h:column >
< h:column >
< f:facet name ="header" > typeName </ f:facet >
#{bookType.typeName}
</ h:column >
< h:column >
< f:facet name ="header" > action </ f:facet >
< s:link view ="/BookType.xhtml"
id ="viewbookType"
value ="View"
propagation ="none" >
< f:param name ="bookTypeTypeId"
value ="#{bookType.typeId}" />
</ s:link >
</ h:column >
</ rich:dataTable >
< div class ="actionButtons" >
< s:button value ="Select bookType"
view ="/BookTypeList.xhtml" >
< f:param name ="from" value ="BookEdit" />
</ s:button >
</ div >
</ div >
点击“Select bookType”页面就自动跳转到BookTypeList页面,列出所有的类型,每个类型后面都有一个select连接,点击这个连接就可选中这个类 型。然后回到BookEdit.xhtml。很Seam很强大吧,呵呵。其实为何这样能完成一个选择都是在BookEdit.page.xml里配置的。 配置大概如下:
< h:outputText value ="There is no bookType associated with this book."
rendered ="#{bookHome.instance.bookType == null}" />
< rich:dataTable var ="bookType"
value ="#{bookHome.instance.bookType}"
rendered ="#{bookHome.instance.bookType != null}"
rowClasses ="rvgRowOne,rvgRowTwo"
id ="bookTypeTable" >
< h:column >
< f:facet name ="header" > typeId </ f:facet >
#{bookType.typeId}
</ h:column >
< h:column >
< f:facet name ="header" > bookType typeId </ f:facet >
#{bookType.bookType.typeId}
</ h:column >
< h:column >
< f:facet name ="header" > typeName </ f:facet >
#{bookType.typeName}
</ h:column >
< h:column >
< f:facet name ="header" > action </ f:facet >
< s:link view ="/BookType.xhtml"
id ="viewbookType"
value ="View"
propagation ="none" >
< f:param name ="bookTypeTypeId"
value ="#{bookType.typeId}" />
</ s:link >
</ h:column >
</ rich:dataTable >
< div class ="actionButtons" >
< s:button value ="Select bookType"
view ="/BookTypeList.xhtml" >
< f:param name ="from" value ="BookEdit" />
</ s:button >
</ div >
</ div >
<
begin-conversation
join
="true"
/>
< action execute ="#{bookHome.wire}" />
< param name ="bookFrom" />
< param name ="bookBookId" value ="#{bookHome.bookBookId}" />
< param name ="bookTypeFrom" />
< param name ="bookTypeTypeId" value ="#{bookTypeHome.bookTypeTypeId}" />
<
begin-conversation
join
="true"
/>
:开始一个
conversation(我暂称之为“页面流”),如果已存在,就加入。而不重新创建。
< action execute ="#{bookHome.wire}" />
< param name ="bookFrom" />
< param name ="bookBookId" value ="#{bookHome.bookBookId}" />
< param name ="bookTypeFrom" />
< param name ="bookTypeTypeId" value ="#{bookTypeHome.bookTypeTypeId}" />
< action execute ="#{bookHome.wire}" /> : 一最重要的这个。如果没有执行这个方法这段跳转就没有任何效果了。,先来看下这个方法是怎么写的吧:
public
void
wire() {
getInstance(); // 获取instance,放在这里是为了加在instance
BookType bookType = bookTypeHome.getDefinedInstance(); // 获取类型。
if (bookType != null ) { // 如果选择的类型不为null
getInstance().setBookType(bookType); // 设置书籍类型
}
}
那bookTypeHome从哪来的呢?天上掉下的?呵呵,当然不是。就在BookHome的上部分:
getInstance(); // 获取instance,放在这里是为了加在instance
BookType bookType = bookTypeHome.getDefinedInstance(); // 获取类型。
if (bookType != null ) { // 如果选择的类型不为null
getInstance().setBookType(bookType); // 设置书籍类型
}
}
@In(create
=
true
)
BookTypeHome bookTypeHome;
BookTypeHome bookTypeHome;
前面我说过了。这个@In就是拿来做双向注入的。 bookTypeHome是要注入的组件名称。
其他的都是参数了,没啥好解释的。
但为什么在BookTypeList页面点select,怎么就会自动跳转到BookEdit.xhtml呢?奥秘就在这段代码里(BookTypeList.xhtml):
<
s:link
view
="/#{empty from ? 'BookType' : from}.xhtml"
value ="Select"
id ="bookType" >
< f:param name ="bookTypeTypeId"
value ="#{bookType.typeId}" />
</ s:link >
从BookEdit里传来一个from。
value ="Select"
id ="bookType" >
< f:param name ="bookTypeTypeId"
value ="#{bookType.typeId}" />
</ s:link >
<
s:button
value
="Select bookType"
view ="/BookTypeList.xhtml" >
< f:param name ="from" value ="BookEdit" />
</ s:button >
就告诉了BookTypeList,是从BookEdit里来的,点Select的时候就不要去其他地方了。直接回去。
view ="/BookTypeList.xhtml" >
< f:param name ="from" value ="BookEdit" />
</ s:button >
好了,今天到这,困了,上面讲的不明白的欢迎email或qq联系我。。