Tapestry中表单的提交有很多问题,刷新导致表单的重复提交、臭名昭著的dirty form warning、
显示不友好的URL等,这些都可以使用redirect-after-post模式来避免,也就是post表单之后将页面redirect一下,
这样地址栏显示的就是redirect之后的页面,刷新的也将是这个页面,而且redirect之后的页面一般为Page页面,
可以使用friendlyUrl来显示友好的url。
要使用这个模式,不得不提到一个ILink接口,T4中可接受的listener方法中一个方法签名是这样的:
public ILink listenerMethod(Parameter parameter);
这个方法的返回值为ILink,如果T4侦测到是这个返回值的话,将自动redirect到ILink指定的页面。
而且ILink可以为纯页面的Link也可以是带有参数的Link。纯页面Link也就是page service生成的link,
带参数的link为external service生成的link,具体link的生成代码如下:
@InjectObject("engine-service:page")
public abstract IEngineService getPageService();
public ILink getPageLink(String pageName) {
return getPageService().getLink(false, pageName);
}
@InjectObject("engine-service:external")
public abstract IEngineService getExternalService();
public ILink getExternalLink(String pageName, Object[] parameters) {
ExternalServiceParameter esp = new ExternalServiceParameter(pageName,
parameters);
return getExternalService().getLink(false, esp);
}
将上述方法放入到你的项目中自定义的MyBasePage(继承Tapestry的BasePage)中,然后让你的
页面类继承MyBasePage,这样就可以使用上述方法了。
这样在页面提交时可以如下调用
public ILink onDelete(Long id){
//your code to delete object
return getPageLink(getPageName());//删除后重新redirect到本页面
}
当然如果这个页面带参数,例如catgoryId=1,则代码如下:
public ILink onDelete(Long id){
//your code to delete object
return getExternalLink(getPageName(),new Object[]{categoryId});
//删除后重新redirect到本页面,并带一个参数
}
上述方式通过继承来实现方法调用,感觉不是很好,我们可以利用Tapestry的易扩展特性来扩展它,
首先想到的就是注释方式来注入,例如我想实现如下注入
@InjectPageLink(“Home“)
Public abstract ILink getHomeLink();
@InjectExternalLink(“Category“)
Public abstract ILink getCategoryLink(Long categoryId);
这样即去掉了页面的继承,也显得比较直接。
首先是定义注释类
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target( { ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectPageLink {
String value();
}
然后写注释的worker类
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import org.apache.hivemind.ApplicationRuntimeException;
import org.apache.hivemind.Location;
import org.apache.hivemind.service.BodyBuilder;
import org.apache.hivemind.service.MethodSignature;
import org.apache.tapestry.annotations.AnnotationUtils;
import org.apache.tapestry.annotations.MethodAnnotationEnhancementWorker;
import org.apache.tapestry.engine.ILink;
import org.apache.tapestry.enhance.EnhancementOperation;
import org.apache.tapestry.spec.IComponentSpecification;
public class InjectPageLinkAnnotationWorker implements
MethodAnnotationEnhancementWorker {
public void performEnhancement(EnhancementOperation op,
IComponentSpecification spec, Method method, Location location) {
if (!method.getReturnType().equals(ILink.class))
throw new ApplicationRuntimeException(
"InjectPage annotation must return ILink");
InjectPageLink injectPageLink = method.getAnnotation(InjectPageLink.class);
String pageName = injectPageLink.value();
BodyBuilder builder = new BodyBuilder();
builder.begin();
builder.addln("return getPage().getRequestCycle().getInfrastructure().getServiceMap().getService(
org.apache.tapestry.Tapestry.PAGE_SERVICE).getLink(false,\"{0}\");", pageName);
builder.end();
op.addMethod(Modifier.PUBLIC, new MethodSignature(method), builder
.toString(), location);
if (isGetter(method))
op.claimReadonlyProperty(AnnotationUtils.getPropertyName(method));
}
boolean isGetter(Method method) {
return method.getName().startsWith("get")
&& method.getParameterTypes().length == 0;
}
}
External的同样两个类
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target( { ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectExternalLink {
String value();
}
Work类
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import org.apache.hivemind.ApplicationRuntimeException;
import org.apache.hivemind.Location;
import org.apache.hivemind.service.BodyBuilder;
import org.apache.hivemind.service.MethodSignature;
import org.apache.tapestry.Tapestry;
import org.apache.tapestry.annotations.AnnotationUtils;
import org.apache.tapestry.annotations.MethodAnnotationEnhancementWorker;
import org.apache.tapestry.engine.ILink;
import org.apache.tapestry.enhance.EnhancementOperation;
import org.apache.tapestry.spec.IComponentSpecification;
public class InjectExternalLinkAnnotationWorker implements
MethodAnnotationEnhancementWorker {
public void performEnhancement(EnhancementOperation op,
IComponentSpecification spec, Method method, Location location) {
if (!method.getReturnType().equals(ILink.class))
throw new ApplicationRuntimeException(
"injectExternalLink annotation must return ILink");
InjectExternalLink injectExternalLink = method
.getAnnotation(InjectExternalLink.class);
String pageName = injectExternalLink.value();
BodyBuilder builder = new BodyBuilder();
builder.begin();
Class[] parameterTypes = method.getParameterTypes();
int paramCount = Tapestry.size(parameterTypes);
if (paramCount > 0) {
if (parameterTypes[0].isArray()) {
builder.addln(
"return getPage().getRequestCycle().getInfrastructure().getServiceMap().
getService(org.apache.tapestry.Tapestry.EXTERNAL_SERVICE).getLink(false, new org.apache.tapestry.engine.ExternalServiceParameter(\"{0}\",$1));",
pageName);
} else {
builder
.addln(
"java.lang.Object[] params = new java.lang.Object[{0}];",
paramCount);
for (int i = 0; i < paramCount; i++) {
builder.add("params[{0}] = ", i);
if (parameterTypes[i].isPrimitive())
builder.add("($w) ");
// Parameter $0 is this, so $1 is the first real parameter.
builder.addln("${0};", i + 1);
}
builder
.addln(
"return getPage().getRequestCycle().getInfrastructure().getServiceMap().getService(org.apache.tapestry.Tapestry.EXTERNAL_SERVICE).getLink(false, new org.apache.tapestry.engine.ExternalServiceParameter(\"{0}\",params));",
pageName);
}
}
builder.end();
op.addMethod(Modifier.PUBLIC, new MethodSignature(method), builder
.toString(), location);
if (isGetter(method))
op.claimReadonlyProperty(AnnotationUtils.getPropertyName(method));
}
boolean isGetter(Method method) {
// We already know the return type is String
return method.getName().startsWith("get")
&& method.getParameterTypes().length == 0;
}
}
类写好了,然后就是贡献到Tapestry中
<module id="your.model.annotation" version="1.0.0" package="your.package.annotations">
<contribution configuration-id="tapestry.annotation.MethodWorkers">
<worker annotation="InjectPageLink" object="instance:InjectPageLinkAnnotationWorker"/>
<worker annotation="InjectExternalLink" object="instance:InjectExternalLinkAnnotationWorker"/>
<contribution>
<module>
好了将你的module放入项目hivemodule.Xml作为子module。
http://www.iteye.com/post/178572
分享到:
相关推荐
8. **tapestry-hibernate-NOTICE.txt, tapestry-hibernate-core-NOTICE.txt, tapestry-ioc-NOTICE.txt**:与对应的LICENSE文件类似,这些NOTICE文件列出了在Tapestry的这些特定模块中使用到的第三方组件和它们的版权...
在"tapestry-project-4.1.5"这个压缩包中,我们有机会深入探索 Tapestry 4.1.5 版本的特性和应用。 首先,让我们关注"Tapestry-libraries"这个文件夹。它包含了Tapestry 框架运行所需的各种库文件,这些库文件是...
`tapestry-jpa-5.3.7.jar`和`tapestry-hibernate-5.3.7.jar`则与持久层集成有关,它们支持JPA(Java Persistence API)和Hibernate ORM框架,使开发者能够轻松地在Tapestry应用中处理数据库操作。 总的来说,Apache...
11. **Tapestry Hibernate**:`tapestry-hibernate-5.3.8.jar`是Tapestry与Hibernate集成的库,提供了在Tapestry应用中使用Hibernate的相关工具和服务。 通过这个发行包,开发者可以快速搭建Tapestry环境,开发基于...
在本案例中,"tapestry-bin-5.0.18.zip" 是一个与 Tapestry 框架相关的压缩包,这可能是一个用于Web应用开发的工具集或库。Tapestry 是一个开源的Java Web应用程序框架,由Apache软件基金会维护,它允许开发者以组件...
- “tapestry-hibernate”是Tapestry与Hibernate集成的模块,使开发者能够方便地在Tapestry应用中使用Hibernate进行对象关系映射(ORM)操作。 - “tapestry-ioc”是Tapestry的依赖注入(IOC)容器,它简化了服务和...
在"tapestry-ioc-5.0.3-src"这个压缩包中,我们可能找到Tapestry IOC框架的源代码,这对于开发者深入理解其工作原理和定制功能非常有价值。 1. **依赖注入(DI)**:Tapestry IOC的核心概念就是DI,它允许对象在...
这些组件可以通过简单的XML标记(如`<span jwcid="@Insert" ...>`)在页面模板中使用。ognl(Object-Graph Navigation Language)则用于在组件间传递数据和表达式计算。 此外,Tapestry还提供了诸如页面转换、错误...
Tapestry5最新中文教程.doc 作者 Renat Zubairov & Igor Drobiazko译者 沙晓兰 发布于 2008年7月2日 下午9时30分 社区 Java 主题 Web框架 ----------------------------------------- Tapestry5.1实例教程.pdf ...
- **项目名称**:Tapestry-3-Getting Started - **项目目标**:开发一个基于Web的货币转换器应用。 - **技术栈**:Tapestry 3、Eclipse IDE、Apache Tomcat。 #### 二、开发环境搭建 ##### 1. Eclipse IDE安装 - **...
tapestry-spring-5.2.6.jar,java spring框架依赖,可以直接使用
在Tapestry中,大多数的编程都是声明式的,这意味着开发者可以使用XML或注解来定义页面和组件的行为,而不是编写大量的Java代码。这降低了代码的复杂性,提高了可读性和维护性。 4. **页面和组件生命周期**: ...
Tapestry 是一个开源的Java Web应用程序框架,由Apache软件基金会维护。这个“tapestry-4.0.zip”...这个“tapestry-4.0.zip”压缩包中的内容,包含了一系列源码、文档和示例,是深入研究Tapestry 4.0的宝贵资源。
- 使用 `@SessionState` 注解在会话中存储数据。 - 示例代码: `@SessionState private User user;` - **JavaScript 集成**: - 通过 `@JavaScript` 注解引入 JavaScript 脚本。 - 示例代码: `@JavaScript("script...
通过阅读此书,读者能够深入理解Tapestry 5如何将MVC(模型-视图-控制器)模式与组件编程思想结合,实现高度模块化和可重用的Web应用设计。 1. **组件模型**:Tapestry 5的核心是组件模型,它允许开发者创建可复用...
包含: tapestry 的源代码, tapestry集成spring2.0 tapestry快速启动 tapestry upload tapestry hibernate tapestry annotations
这个是开发tapestry3所用的JAR包