来源:http://www.blogjava.net/natlive/archive/2009/05/21/271890.html
Struts2 的UITag原理:
Struts2 UITag分三部份组成,一部份用于定义Tag的内容与逻辑的UIBean,一部份用于定义JSP Tag,也就是平时我们定义的那种,最后就是Template,它存放在你的theme目录之下,是一个FreeMarker模板文件。
我现在辑写一份MMTag,它主要是用于输出带链接的文字,比如像这样:
<cur:mm message="'I am a boy.'" />
就会输出:
<a href="http://www.blogjava.net/natlive">I am boy.</a>
文件配置:
tangs.tld
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN" "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
<taglib>
<tlib-version>2.2.3</tlib-version>
<jsp-version>1.2</jsp-version>
<short-name>tangs</short-name>
<uri>/tangs</uri>
<display-name>"Tangs Tags"</display-name>
<tag>
<name>button</name>
<tag-class>com.gq.tag.button.ButtonTag</tag-class>
<body-content>empty</body-content>
<attribute>
<name>items</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
<tag>
<name>mm</name>
<tag-class>com.gq.tag.mm.MmTag</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>message</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
</taglib>
MmTag.java
* getBean()返回该Tag中的UIBean。
* populateParams()初始化参数,一般用来初始化UIBean(Component)。
* setXXXX设置属性,和jsp tag是一样的
/**
*创建日期 :2011-11-16
*创建用户 :GongQiang
*变更情况 :
*文档位置 $Archive:test_struct2/com.gq.tag.mm/MmTag.java$
*最后变更 $Author: $
*变更日期 $Date: $
*当前版本 $Revision: $
*
*Copyright (c) 2004 Sino-Japanese Engineering Corp, Inc. All Rights Reserved.
*/
package com.gq.tag.mm;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts2.components.Component;
import org.apache.struts2.views.jsp.ui.AbstractUITag;
import com.opensymphony.xwork2.util.ValueStack;
public class MmTag extends AbstractUITag {
private static final long serialVersionUID = 1L;
private String message;
@Override
public Component getBean(ValueStack stack, HttpServletRequest request, HttpServletResponse response) {
return new MmBean(stack, request, response);
}
@Override
protected void populateParams() {
super.populateParams();
MmBean mm = (MmBean)component;
mm.setMessage(message);
}
public void setMessage(String message) {
this.message = message;
}
}
MmBean.java
* strutsTag注解指明了该UIBean的名字和Tag类的类名。
* getDefaultTemplate()方法用于返回模板的名字,Struts2会自动在后面加入.ftl扩展名以找到特定的模板文件。
* setXXX,设置UIBean的属性,一般Tag中有几个这样的属性,这里就有几个。@StrutsTagAttribute(description="set message", type="String")注解,说明该属性是字符串(也可以是其它),这一步很重要。
* 覆写evaluateExtraParams()方法,在UIBean初始化后会调用这个方法来初始化设定参数,如addParameter方法,会在freemarker里的parameters里加入一个key value。这里要注意findString,还有相关的findxxxx方法,它们是已经封装好了的解释ognl语法的工具,具体是怎么样的,大家可以查看一下UIBean的api doc。
/**
*创建日期 :2011-11-16
*创建用户 :GongQiang
*变更情况 :
*文档位置 $Archive:test_struct2/com.gq.tag.mm/MmBean.java$
*最后变更 $Author: $
*变更日期 $Date: $
*当前版本 $Revision: $
*
*Copyright (c) 2004 Sino-Japanese Engineering Corp, Inc. All Rights Reserved.
*/
package com.gq.tag.mm;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts2.components.UIBean;
import org.apache.struts2.views.annotations.StrutsTag;
import org.apache.struts2.views.annotations.StrutsTagAttribute;
import com.opensymphony.xwork2.util.ValueStack;
@StrutsTag(name="mm", tldTagClass="com.gq.tag.mm.MmTag", description="MM")
public class MmBean extends UIBean {
private String message;
public MmBean(ValueStack stack, HttpServletRequest request, HttpServletResponse response) {
super(stack, request, response);
}
@Override
protected String getDefaultTemplate() {
return "mm";
}
@StrutsTagAttribute(description="set message", type="String")
public void setMessage(String message) {
this.message = message;
}
@Override
protected void evaluateExtraParams() {
super.evaluateExtraParams();
if ( message != null ) {
addParameter("message", findString(message));
System.out.println( "auto_rule_match--->" + findString(message) );
System.out.println( "ongl--->" + getStack().findValue( message ) );
}
}
}
findString( ) 方法并没有想象中的智能,如下说明:
mmTag.jsp 有如下两个标签:
页面直接给定属性<br/>
<tangs:mm message="hahahaha..."/>
<br/>
<br/>
运行时获取值<br/>
<tangs:mm message="name"/>
但是,通过findString( ) 方法,name 并
没能转换成其对应的值(“Foo”),而是
直接输出了。
auto_rule_match--->hahahaha...
ongl--->null
auto_rule_match--->name
ongl--->Foo
findString( ) 会去解析
其值而
非本身的两种格式:${ xxx } 或 %{ xxx }
在源代码目录中建立template/simple目录(这个目录名和你的theme有关),然后在里面建一个mm.ftl文件:
<a href="http://www.yinsha.com">${parameters.message?html}</a>
以上,tangs.tld、MmTag.java、MmBean.java 和 mm.ftl 就完成了<tangs:mm />的定义。
演示:
struts.xml 中添加 action : MmTag
<!-- 演示标签 -->
<package name="tag" extends="struts-default" namespace="/tag">
<action name="Tag" class="com.gq.action.TagTestAction">
<result name="success">/tagTest/tags.jsp</result>
<interceptor-ref name="defaultStack"/>
</action>
<action name="ButtonTag" class="com.gq.tag.button.ButtonActionTest">
<result name="success">/tagTest/buttonTag.jsp</result>
<interceptor-ref name="defaultStack"/>
</action>
<action name="MmTag" class="com.gq.tag.mm.MmActionTest">
<result name="success">/tagTest/mmTag.jsp</result>
<interceptor-ref name="defaultStack"/>
</action>
</package>
/**
*创建日期 :2011-11-16
*创建用户 :GongQiang
*变更情况 :
*文档位置 $Archive:test_struct2/com.gq.tag.mm/MmActionTest.java$
*最后变更 $Author: $
*变更日期 $Date: $
*当前版本 $Revision: $
*
*Copyright (c) 2004 Sino-Japanese Engineering Corp, Inc. All Rights Reserved.
*/
package com.gq.tag.mm;
import com.opensymphony.xwork2.ActionSupport;
public class MmActionTest extends ActionSupport {
private static final long serialVersionUID = 1L;
public String getName(){
return "Foo";
}
@Override
public String execute() throws Exception {
return SUCCESS;
}
}
<%@ page language="java" pageEncoding="UTF-8"%>
<%@taglib prefix="s" uri="/struts-tags" %>
<%@taglib prefix="tangs" uri="/tangs" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<base href="<%=basePath%>">
<title>My JSP 'buttonTag.jsp' starting page</title>
</head>
<body>
页面直接给定属性<br/>
<tangs:mm message="hahahaha..."/>
<br/>
<br/>
运行时获取值<br/>
<tangs:mm message="%{name}"/>
</body>
</html>
图示页面: