視圖存儲(chǔ)在grails-app/views directory目錄下。主要想學(xué)習(xí)如何創(chuàng)建Taglib和如何利用模板技術(shù)。
Grails使用GSP作為表現(xiàn)層,在GSP中g(shù)roovy不只是表明GSP是基于什么技術(shù)的,而且還可以利用Groovy來(lái)創(chuàng)建一些腳本來(lái)在GSP中執(zhí)行。在這點(diǎn)上說(shuō)GSP和JSP很相像。
當(dāng)然內(nèi)嵌的腳本并不利于代碼的重用。Grails中的Taglib和模板給你提供了一個(gè)很好的重用代碼的途徑。
GSP 是Grails的視圖的基礎(chǔ)。List頁(yè)面提供到Show頁(yè)面的連接,Show頁(yè)面允許你導(dǎo)航到Edit頁(yè)面。
MVC的分離策略主要是可以給應(yīng)用程序不同的視圖,Grails通過(guò)不同的插件來(lái)支持不同的表現(xiàn)層技術(shù)。可以通過(guò)命令grails install-plugin 來(lái)查看現(xiàn)在安裝的插件。 或使用grails list-plugins來(lái)獲得當(dāng)前可用的插件。
雖然Grails 并不支持 native和 JSF,但是你還是可以使用他們。一個(gè)Grails程序就是一個(gè)標(biāo)準(zhǔn)的Java EE程序,因此只要你在lib目錄中放置了正確的Jar文件,并在WEB-INF/web.xml文件中進(jìn)行了正確的配置,就可以正常的使用了。Grails實(shí)在一個(gè)標(biāo)準(zhǔn)的servlet容器中開(kāi)發(fā)的,所以Grails程序也支持JSP。
類(lèi)似的增加Ajax框架也同樣的簡(jiǎn)單,拷貝JavaScript庫(kù)到web-app/js目錄下。 Prototype和Scriptaculous是Grails默認(rèn)安裝的。RichUI插件可以很好的從Ajax庫(kù)中選擇正確的UI。
當(dāng)查看插件列表時(shí),可以看到對(duì)富客戶(hù)端,如:Flex,OpenLazlo,GWT和ZK的支持。
在GSP文件中有許多有用的以<g:開(kāi)頭的標(biāo)簽。GSP文件就是Html和Grails標(biāo)簽的混合體。
在控制器中使用def scaffold 的作用是指示Grails動(dòng)態(tài)的在運(yùn)行的時(shí)候產(chǎn)生GSP文件。
輸入命令 grails generate-all Trip會(huì)產(chǎn)生控制器和相關(guān)的GSP文件
當(dāng)訪(fǎng)問(wèn) http://localhost:9090/trip-planner/trip/list時(shí),會(huì)先調(diào)用TripController來(lái)返回Trip的列表,并傳遞給list.gsp來(lái)顯示。
下面介紹一些常用的Grails的標(biāo)簽,包括<g:each>。
<g:each>是個(gè)非常常用的Grails標(biāo)簽。它遍歷列表中的每個(gè)元素,打開(kāi)文件/trip/list.gsp 就可以看到如何使用這個(gè)標(biāo)簽:
<g:each in="${tripList}" status="i" var="trip"> <tr class="${(i % 2) == 0 ? 'even' : 'odd'}"> <td><link action="show" id="${trip.id}">${trip.id?.encodeAsHTML()}</g:link></td> <td>${trip.airline?.encodeAsHTML()}</td> <td>${trip.name?.encodeAsHTML()}</td> <td>${trip.city?.encodeAsHTML()}</td> <td>${trip.startDate?.encodeAsHTML()}</td> <td>${trip.endDate?.encodeAsHTML()}</td> </tr> </g:each> |
The status attribute in the在 <g:each>中,status就是一個(gè)簡(jiǎn)單的計(jì)數(shù)器。Var屬性允許你定義一個(gè)當(dāng)前元素的名字。
另一個(gè)Grails 標(biāo)簽是 <g:link>,它建立一個(gè)HTML的<a href>連接。<g:createLink>標(biāo)簽會(huì)創(chuàng)建一個(gè)真正的URL字符串。在list.gsp上部,你可以看到另一個(gè)和連接相關(guān)的標(biāo)簽<g:createLinkTo>,這個(gè)標(biāo)簽接受一個(gè)dir和 file屬性:
<div class="nav"> <span class="menuButton"><a class="home" href="${createLinkTo(dir:'')}">Home</a></span> <span class="menuButton"><link class="create" action="create">New Trip</g:link></span> </div> |
在list.gsp中還可以看到<g:if>標(biāo)簽:
<h1>Trip List</h1> <if test="${flash.message}"> <div class="message">${flash.message}</div> </g:if> |
<g:paginate>標(biāo)簽顯示的是相關(guān)的分頁(yè)邏輯;<g:sortable>把列的標(biāo)題變成可點(diǎn)擊的,用來(lái)進(jìn)行排序。其他的一些標(biāo)簽,如:<g:form>和 <g:submit>指示顯示相對(duì)應(yīng)的html元素。
自定義標(biāo)簽庫(kù)
雖然標(biāo)準(zhǔn)的Grails標(biāo)簽非常有用,但是最終你還是需要自己的標(biāo)簽。
在Grails中創(chuàng)建標(biāo)簽庫(kù)要比在JSP中創(chuàng)建標(biāo)簽庫(kù)方便。
在Grails中創(chuàng)建標(biāo)簽庫(kù)的第一步是輸入命令grails create-tag-lib Date。這樣就會(huì)創(chuàng)建兩個(gè)文件: grails-app/taglib/DateTagLib.groovy (標(biāo)簽庫(kù))和 grails-app/test/integration/DateTagLibTests.groovy (測(cè)試文件)向文件DateTagLib.groovy中添加以下的代碼:
class DateTagLib { def thisYear = { out << Calendar.getInstance().get(Calendar.YEAR) } } |
這樣就創(chuàng)建了<g:thisYear>標(biāo)簽,年的屬性直接輸出到輸出流中。
測(cè)試標(biāo)簽庫(kù)
在文件DateTagLibTests.groovy里添加如下的測(cè)試代碼:
class DateTagLibTests extends GroovyTestCase { def dateTagLib void setUp(){ dateTagLib = new DateTagLib() } void testThisYear() { String expected = Calendar.getInstance().get(Calendar.YEAR) assertEquals("the years don't match", expected, dateTagLib.thisYear()) } } |
輸入命令grails test-app來(lái)進(jìn)行測(cè)試
命令grails test-app除了運(yùn)行測(cè)試用例外,還會(huì)生成測(cè)試報(bào)告。打開(kāi)文件test/reports/html/index.html可以進(jìn)行查看。
這樣這個(gè)標(biāo)簽就測(cè)試開(kāi)發(fā)完了。
自定義標(biāo)簽的高級(jí)話(huà)題
大部分標(biāo)簽可以有body和屬性。
class DateTagLib { def thisYear = { out << Calendar.getInstance().get(Calendar.YEAR) } def copyright = { attrs, body -> out << "<div id='copyright'>" out << "© ${attrs['startYear']} - ${thisYear()}, ${body()}" out << " All Rights Reserved." out << "</div>" } } |
注意attrs是一個(gè)包含標(biāo)簽屬性的HashMap。可以通過(guò)這個(gè)HashMap來(lái)得到startYear屬性。類(lèi)似的body是作為一個(gè)Closure傳遞進(jìn)來(lái)的。
默認(rèn)的自定義的標(biāo)簽都放在g:命名空間下,如果想修改這個(gè)行為的話(huà),需要在DateTagLib.groovy 中添加static namespace = 'trip'。這樣在GSP中就可以使用以下的方式進(jìn)行調(diào)用:<trip:copyright startYear="2002">FakeCo Inc.</trip:copyright>。
模板
自定義標(biāo)簽庫(kù)是重用代碼的一個(gè)好的辦法,但是大段的代碼的好的重用方式是使用模板。
一個(gè)模板就是可以在多個(gè)GSP文件中共享的一段GSP代碼。在the grails-app/views/trip 目錄下創(chuàng)建的文件,只有Trip的視圖才能訪(fǎng)問(wèn)。只有在grails-app/views 目錄下創(chuàng)建的文件才能全局共享。創(chuàng)建如下的全局模板:
<div id="footer"> <g:copyright startYear='2002'>FakeCo, Inc.</g:copyright> <div id="powered-by"> <img src="${createLinkTo(dir:'images', file:'grails-powered.jpg')}" /> </div> </div> |
下面來(lái)在相應(yīng)的視圖中引入這個(gè)模板:
<html><body> ... <g:render template="/footer" /> </body></html> |
為了自定義 scaffolding,需要使用命令grails install-templates。這個(gè)命令會(huì)向項(xiàng)目添加一個(gè)新的目錄src/templates。在這個(gè)目錄下會(huì)有三個(gè)子目錄artifacts、scaffolding和war。
artifacts 目錄中保存的是生成Controller、DomainClass和TagLib時(shí)需要使用的模板。比如:如果想要所有的控制器都擴(kuò)展一個(gè)相同的抽象類(lèi)。
war 目錄包含一個(gè)web.xml文件。如果需要添加自定義的參數(shù),如:filters 或者 servlets,都可以在這個(gè)文件中進(jìn)行。但你使用grails war命令時(shí),這個(gè)文件就會(huì)被包含在生成的war文件中。
scaffolding 目錄中包含的是生成視圖時(shí),需要指定的信息。打開(kāi) list.gsp 然后添加 <g:render template="/footer" />到文件的底部。
posted @ 2009-12-29 14:21 楊曉晨 閱讀(5725) | 評(píng)論 (1) | 編輯 收藏