使用Apache Tiles3.x构建界面布局
Tiles是一个免费的开源模板Java应用程序的框架。基于复合模式简化的用户界面的构建。对于复杂的网站仍是最简单、最优雅的方式与任何MVC技术一起工作。Struts2对Tiles提供了支持,如今Tiles发展已有13个年头,成为Apache的一个独立项目,我们可以单独使用Tiles来构建用户界面布局。
Tiles项目:http://tiles.apache.org/index.html
Tiles的配置DTD定义:http://tiles.apache.org/framework/tiles-core/dtddoc/index.html
本文主要通过构建一个简单的页面布局来认识Apache Tiles3.x(由于Tiles2.x和Tiles3.x存在较大差异)。
1.准备工作
1.1安装Apache Tiles3.x依赖的Jar
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
< dependency > < groupId >org.apache.tiles</ groupId > < artifactId >tiles-extras</ artifactId > < version >3.0.5</ version > </ dependency > < dependency > < groupId >org.apache.tiles</ groupId > < artifactId >tiles-servlet</ artifactId > < version >3.0.5</ version > </ dependency > < dependency > < groupId >org.apache.tiles</ groupId > < artifactId >tiles-jsp</ artifactId > < version >3.0.5</ version > </ dependency > |
注意:这里使用了Apache3.x完整包依赖。
1.2调试环境
安装jetty-maven-plugin来热部署web应用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
< build > < finalName >tiles</ finalName > < plugins > < plugin > < groupId >org.mortbay.jetty</ groupId > < artifactId >jetty-maven-plugin</ artifactId > < version >7.1.6.v20100715</ version > < configuration > < scanIntervalSeconds >1</ scanIntervalSeconds > < reload >automatic</ reload > < webAppConfig > < contextPath >/tiles</ contextPath > </ webAppConfig > </ configuration > </ plugin > </ plugins > </ build > |
注意:运行mvn jetty:run -Djetty.port=9999 命名,访问http://localhost:9999/tiles 需要额外在Maven的settings.xml文件的插件组中添加插件组标识。
1
2
3
4
5
6
7
|
< pluginGroups > <!-- pluginGroup | Specifies a further group identifier to use for plugin lookup. <pluginGroup>com.your.plugins</pluginGroup> --> < pluginGroup >org.mortbay.jetty</ pluginGroup > </ pluginGroups > |
1.3配置web.xml
在web.xml中添加Tiles监听器
1
2
3
|
< listener > < listener-class >org.apache.tiles.extras.complete.CompleteAutoloadTilesListener</ listener-class > </ listener > |
关于Tiles的监听器可以自定义实现,参见:http://tiles.apache.org/framework/config-reference.html
2.分析界面组成,构建布局文件
假设本案例中的页面构成如图:
分析界面布局,找不通用部分,特殊部分。 在webapp下创建layout文件夹放在布局文件,snippet文件夹放置公共部分。
通过分析,将布局切割为header,body,footer,并且将HTML页面中的meta,script公共部分抽取出来。
/snippet/meta.jsp
1
2
3
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> < meta http-equiv = "Content-Type" content = "text/html; charset=UTF-8" > |
/snippet/script.jsp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> < style > div { width: 480px; height: 80px; background: silver; } #body { background: lime; } </ style > < script type = "text/javascript" > document.writeln("这句话是由JavaScript写入页面的。"); </ script > |
/snippet/header.jsp
1
2
3
4
5
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> < h3 > 这是头部 </ h3 > |
/snippet/footer.jsp
1
2
3
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> < h3 >这是页脚</ h3 > |
/snippet/index_body.jsp
1
2
3
4
5
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> < pre > 这是页面的主体部分 </ pre > |
通过上面的公共部分和主体,构建一个布局文件如下:
/layout/index_layout.jsp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%> <%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles"%> <!DOCTYPE html> < html > < head > < tiles:insertAttribute name = "meta" /> < title >< tiles:insertAttribute name = "title" /></ title > < tiles:insertAttribute name = "script" /> </ head > < body > < div id = "header" > < tiles:insertAttribute name = "header" /> </ div > < div id = "body" > < tiles:insertAttribute name = "body" /> </ div > < div id = "footer" > < tiles:insertAttribute name = "footer" /> </ div > </ body > </ html > |
3.Tiles的复合布局定义
Tiles是通过在xml文件中配置definition进行页面公共部分的重用,页面布局的组合。
/WEB-INF/tiles-defs.xml 定义好公共部分之后,通过配置definition来组合页面布局。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
<? xml version = "1.0" encoding = "UTF-8" ?> <!DOCTYPE tiles-definitions PUBLIC "-//Apache Software Foundation//DTD Tiles Configuration 3.0//EN" "http://tiles.apache.org/dtds/tiles-config_3_0.dtd"> <!-- Definitions for Tiles documentation --> < tiles-definitions > < definition name = "tiles.base.definition" > < put-attribute name = "meta" value = "/snippet/meta.jsp" /> < put-attribute name = "script" value = "/snippet/script.jsp" /> < put-attribute name = "header" value = "/snippet/header.jsp" /> < put-attribute name = "footer" value = "/snippet/footer.jsp" /> </ definition > </ tiles-definitions > |
上面的definition可以说是抽象的,仅仅作为基本的定义抽取了界面中最通用的部分,而且并未指定具体的模版文件(布局文件)。下面通过继承tiles.base.definition来定一个tiles.index.definition其布局模版为/layout/index_layout.jsp。
1
2
3
4
|
< definition name = "tiles.index.definition" extends = "tiles.base.definition" template = "/layout/index_layout.jsp" > < put-attribute name = "body" value = "/snippet/index_body.jsp" /> </ definition > |
上面定义tiles.index.definition,新增了body,其值为/snippet/index_body.jsp页面。
4.使用复合布局
到这里已经将页面的布局进行了分割,组合。现在应用definition来构建一个请求响应页面。
/example/index.jsp
1
2
3
4
5
6
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles"%> < tiles:insertDefinition name = "tiles.index.definition" > < tiles:putAttribute name = "title" value = "这是一个有Apache Tiles构建的页面布局." /> </ tiles:insertDefinition > |
5.启动服务器,访问/example/index.jsp
页面展示效果:
接下来看看页面的源代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
<!DOCTYPE html> < html > < head > < meta http-equiv = "Content-Type" content = "text/html; charset=UTF-8" > < title >这是一个有Apache Tiles构建的页面布局.</ title > < style > div { width: 480px; height: 80px; background: silver; } #body { background: lime; } </ style > < script type = "text/javascript" > document.writeln("这句话是由JavaScript写入页面的。"); </ script > </ head > < body > < div id = "header" > < h3 > 这是头部 </ h3 > </ div > < div id = "body" > < pre > 这是页面的主体部分 </ pre > </ div > < div id = "footer" > < h3 >这是页脚</ h3 > </ div > </ body > </ html > |
该例子中布局index_layout.jsp中body是可变的,title对一个不同的页面有不同的标题设置。在tiles-defx.xml的tiles.index.definition继承了tiles.base.definition,并且添加了其body页面,接着在插入tiles.index.definition的index.jsp页面添加了title。这样做达到的效果是整个站点的header,footer,meta,script抽取到了一个definition,然后通过继承的方式进行扩展,丰富不同的布局的页面组成元素,在具体的响应页面来定义专属该页面的内容。从而达到对页面的布局的控制,公共部分的复用的效果。
6.总结
本文仅仅是一个简单的示例,然而大部分内容被抽取公共部分占去,这样的结果并非意外,对于页面布局的划分,组合,重用才是使用Tiles之前最为繁重和复杂的工作,这些工作能够做的合理,优雅,配置definition自然就轻松多了。
http://aiilive.blog.51cto.com/1925756/1596059
接着上一篇:使用Apache Tiles3.x构建界面布局(一)继续构建页面布局。
上一篇示例中已经对页面的公共部分进行了抽取,创建了布局,并且将使用的布局应用的响应页面了。
1.定义definition来改变公共部分
现在有个新变化,响应页面的footer部分需要发生点变化,这个时候仍然可以通过继承的方式来配置一个definition。
/WEB-INF/tiles-defs.xml 中新增definition
1
2
3
4
|
< definition name = "tiles.override.definition" extends = "tiles.index.definition" > < put-attribute name = "footer" value = "Copy Right <a href='http://aiilive.blog.51cto.com'>野马红尘</a>" ></ put-attribute > </ definition > |
1.1创建响应页面/example/index_override.jsp
1
2
3
4
5
6
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles"%> < tiles:insertDefinition name = "tiles.override.definition" > < tiles:putAttribute name = "title" value = "这是一个有Apache Tiles构建的页面布局." /> </ tiles:insertDefinition > |
1.2访问:/tiles/example/index_override.jsp
注意:footer部分发生了改变,使用了tiles.override.definition的footer属性。
2.响应页面中定义body
/example/home.jsp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles"%> < tiles:insertDefinition name = "tiles.index.definition" > < tiles:putAttribute name = "title" value = "HOME 页面." /> < tiles:putAttribute name = "body" > < h3 >这是HOME页面的body</ h3 > < ul > < li >Apache Tiles</ li > < li >Sitemesh</ li > < li >Smarty</ li > < li >Freemarker</ li > < li >Velocity</ li > </ ul > </ tiles:putAttribute > </ tiles:insertDefinition > |
访问:/tiles/example/home.jsp
3.通过Servlet来访问definition(tiles.index.definition),可编程式!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
package secondriver.tiles.servlet; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.tiles.Attribute; import org.apache.tiles.Definition; import org.apache.tiles.TilesContainer; import org.apache.tiles.access.TilesAccess; import org.apache.tiles.request.ApplicationContext; import org.apache.tiles.request.Request; import org.apache.tiles.request.servlet.ServletRequest; import org.apache.tiles.request.servlet.ServletUtil; public class TilesServlet extends HttpServlet { private static final long serialVersionUID = 2721434552222217027L; @Override protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ApplicationContext applicationContext = ServletUtil .getApplicationContext(getServletContext()); TilesContainer container = TilesAccess.getContainer(applicationContext); Request currentRequest = new ServletRequest(applicationContext, request, response); Definition definition = container.getDefinition( "tiles.index.definition" , currentRequest); definition.putAttribute( "title" , new Attribute( "通过servlet访问" )); definition.putAttribute( "body" , new Attribute( "这是通过Servlet处理后的definition." )); container.render(definition, currentRequest); /** * 不需要处理的时候可以直接使用definition的name * * tiles.index.definition没有定义title属性 * * 而template文件(布局)中用到所以需要在Servlet中进行添加 */ // container.render("tiles.index.definition", currentRequest); } } |
访问:/tiles/tilesservlet
注意:
这里使用了Servlet需要在web.xml中增加Servlet的配置
1
2
3
4
5
6
7
8
|
< servlet > < servlet-name >tilesservlet</ servlet-name > < servlet-class >secondriver.tiles.servlet.TilesServlet</ servlet-class > </ servlet > < servlet-mapping > < servlet-name >tilesservlet</ servlet-name > < url-pattern >/tilesservlet</ url-pattern > </ servlet-mapping > |
在pom.xml中添加servlet-api:
1
2
3
4
5
|
< dependency > < groupId >javax.servlet</ groupId > < artifactId >servlet-api</ artifactId > < version >2.5</ version > </ dependency > |
4.通过标签在响应页面上配置definition,并插入到页面中
1
2
3
4
5
6
7
8
9
10
11
12
13
|
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles"%> < tiles:definition name = "tiles.runtime.definiation" template = "/layout/index_layout.jsp" extends = "tiles.base.definition" > < tiles:putAttribute name = "title" value = "通过标签配置 definition." /> < tiles:putAttribute name = "body" > 通过tiles标签配置的运行时definition </ tiles:putAttribute > </ tiles:definition > < tiles:insertDefinition name = "tiles.runtime.definiation" ></ tiles:insertDefinition > |
访问:/tiles/example/runtime.jsp
本文通过多种方式展示了Tiles的使用和其灵活性。另外Tiles还可以和Freemarker,Velocity,Spring,Struts等多种框架集成使用。
具体可以参考官方文档:http://tiles.apache.org/framework/whats-new.html。
另外更多的示例:http://tiles.apache.org/download.html
http://aiilive.blog.51cto.com/1925756/1596069
使用Apache Tiles3.x构建界面布局的更多相关文章
- 使用 Apache Tiles 3 构建页面布局
参考博客:http://aiilive.blog.51cto.com/1925756/1596059Apache Tiles是一个JavaEE应用的页面布局框架.Tiles框架提供了一种模板机制,可以 ...
- 微信js框架第二篇(创建完整界面布局)
接着昨天的继续谈关于微信新出的这个js框架,今天主要谈一个页面的创建到布局的详细步骤. 一.创建一个完整页面 页面你可以创建在项目的任何节点,只要你在入口文件正确引入创建该页面的路径就可使 ...
- 2013 duilib入门简明教程 -- 界面布局(9)
上一个教程实现的标题栏代码中,并没有看到处理自适应窗口大小的代码,但是窗口大小变化后,按钮的位置会跟着变化,这是因为我们将按钮放到了HorizontalLayout.VerticalLayou ...
- php创建新用户注册界面布局实例
php创建新用户注册界面布局实例 <!DOCTYPE> <html> <head> <title>Load page</title> < ...
- WPF基础知识、界面布局及控件Binding(转)
WPF是和WinForm对应的,而其核心是数据驱动事件,在开发中显示的是UI界面和逻辑关系相分离的一种开放语言.UI界面是在XAML语言环境下开发人员可以进行一些自主设计的前台界面,逻辑关系还是基于c ...
- Unity 编辑器的 界面布局 保存方法
在软件界面的右上角(关闭按钮的下方),点击 layout (界面)的下拉箭头. 弹出选项中的 save layout....(保存界面选项),输入命名,就可以生成这个界面的布局. (软件本身也有 ...
- ios开发UI篇—使用纯代码自定义UItableviewcell实现一个简单的微博界面布局
本文转自 :http://www.cnblogs.com/wendingding/p/3761730.html ios开发UI篇—使用纯代码自定义UItableviewcell实现一个简单的微博界面布 ...
- iOS开发UI篇—使用xib自定义UItableviewcell实现一个简单的团购应用界面布局
iOS开发UI篇—使用xib自定义UItableviewcell实现一个简单的团购应用界面布局 一.项目文件结构和plist文件 二.实现效果 三.代码示例 1.没有使用配套的类,而是直接使用xib文 ...
- UWP开发入门(二十)——键盘弹起时变更界面布局
UWP APP在键盘弹起或隐藏时,并不会自动处理界面布局.有时会出现键盘遮挡了下一个需要填写的文本框,或是下一步按钮的情况.本篇我们以登录界面做例子,用一种巧妙简单的方式在键盘弹起和隐藏时更改界面的布 ...
随机推荐
- 【例题 6-17 UVa 10562】Undraw the Trees
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 模拟+递归 [代码] #include <bits/stdc++.h> using namespace std; con ...
- Android 用SQLite 使用 CursorLoader 中的数据填充列表视图
我做了简单的测试应用程序基于此示例.有一个按钮,插入到数据库和列表视图的数据.都是在 MainActivity 中.在原来的代码是restartLoader() 仅从调用 onResume() ,但它 ...
- wepy小程序实现选项卡
先上效果: 本文是基于前面几篇文章: 使用wepy开发微信小程序商城第一篇:项目初始化 使用wepy开发微信小程序商城第二篇:路由配置和页面结构 使用wepy开发微信小程序商城第三篇:购物车(布局篇) ...
- UVA 11889 - Benefit 可直接枚举
看题传送门 题目大意: 输入两个整数A和C,求最小的整数B,使得lcm(A,B)=C.如果无解,输出NO SOLUTION 思路: A*B=C*gcd(A,B) 所以 B / gcd(A,B) = C ...
- iOS INVALID_USER_SCODE 定位 用户安全码未通过
iOS 高德地图API不能定位及INVALID_USER_SCODE问题,有需要的朋友可以参考下. 一.在使用高德地图的API的时候,没有办法实现定位,在这里说一下在真机测试的时候出现没法定位应该注意 ...
- Java与模式:装饰(Decorator)模式
装饰模式使用被装饰类的一个子类的实例.把client的调用委派到被装饰类,装饰模式的关键在于这样的扩展是全然透明的. 装饰模式在Java种使用也非常广泛,比方我们在又一次定义button.对话框等 ...
- Android系统开发(7)——标准I/O与文件锁
一.常用函数 fopen: FILE *fopen(const char *filename, const char *mode); fread: size_t fread(void *ptz, s ...
- 重新配置vim
重新配置,并非折腾,发个链接吧留着以后用。 都是前辈 vimer程序员的世界 Vim(gvim)配色方案推荐 gvim(vim)使用微软雅黑中文字体 Vim(gvim)编程字体推荐 所需即所获:像 I ...
- [SCSS] Reuse Styles with the SCSS @extend Directive
We can write reusable styles with the SCSS @extend or @mixin directives. Which one is better? It dep ...
- js课程 1-4 js变量的作用域是怎样的
js课程 1-4 js变量的作用域是怎样的 一.总结 一句话总结:只有在函数内部前面带var的变量为局部变量,局部变量只能在函数体内使用. 1.什么情况下会出现NaN类型的错误,举一例? Num ...