以下是<f:viewAction>
的简单用例。
<f:metadata>
<f:viewParam name="id" value="#{testManagedBean.id}" maxlength="20"/>
<f:viewAction action="#{testManagedBean.viewAction}"/>
</f:metadata>
涉及托管bean。
@ManagedBean
@ViewScoped
public final class TestManagedBean implements Serializable {
private static final long serialVersionUID = 1L;
private Long id; //Getter and setter.
public void viewAction() {
System.out.println("viewAction() called : " + id);
}
}
参数id
通过URL传递。如果通过相关网址传递了非xxx
这样的非数字值,并且未调用与viewAction()
的侦听器关联的<f:viewAction>
方法,则会出现转换错误。
在这种情况下,id
的值为null
。我希望重定向到另一个页面,当id
无法转换为所需的目标类型时(如本例所示)或id
未根据指定的验证标准进行验证,以避免可能发生的潜在异常只要在相应的托管bean中尝试访问这些参数,就会抛出PrimeFaces的LazyDataModel#load()
方法或相关托管bean中的其他位置。为此,应调用viewAction()
方法。
如何处理?我应该使用
<f:event type="preRenderView">
与<f:viewAction>
?
答案 0 :(得分:8)
这是specified行为。当PROCESS_VALIDATIONS
阶段以validation failure结尾时,UPDATE_MODEL_VALUES
和INVOKE_APPLICATION
阶段都会被跳过。与<h:form>
的“常规”形式完全相同。将<f:viewParam>
视为<h:inputText>
,将<f:viewAction>
视为<h:commandButton>
,它将变得更加清晰。
根据您的特定要求,在转换/验证失败时执行重定向,至少有3个解决方案:
如您所知,添加<f:event listener>
。我宁愿挂钩postValidate
事件而不是更好的自我可记录性。
<f:metadata>
<f:viewParam name="id" value="#{bean.id}" maxlength="20" />
<f:event type="postValidate" listener="#{bean.redirectIfNecessary}" />
<f:viewAction action="#{bean.viewAction}" />
</f:metadata>
public void redirectIfNecessary() throws IOException {
FacesContext context = FacesContext.getCurrentInstance();
if (!context.isPostback() && context.isValidationFailed()) {
context.getExternalContext().redirect("some.xhtml");
}
}
对FacesContext#isPostback()
的检查可防止在同一视图(如果有)中“常规”表单的验证失败时执行重定向。
扩展内置LongConverter
,您可以在getAsObject()
中执行重定向(验证器不可用,因为Long
的默认转换器在非数字输入上已经失败;如果转换器失败,验证器永远不会被解雇)。然而,这是一种糟糕的设计(紧耦合)。
<f:metadata>
<f:viewParam name="id" value="#{bean.id}" converter="idConverter" />
<f:viewAction action="#{bean.viewAction}" />
</f:metadata>
@FacesConverter("idConverter")
public class IdConverter extends LongConverter {
@Override
public Object getAsObject(FacesContext context, UIComponent component, String value) {
if (value == null || !value.matches("[0-9]{1,20}")) {
try {
context.getExternalContext().redirect("some.xhtml");
return null;
}
catch (IOException e) {
throw new FacesException(e);
}
}
else {
return super.getAsObject(context, component, value);
}
}
}
如果需要,您可以在<f:attribute>
内使用<f:viewParam>
将“参数”传递给转换器。
<f:viewParam name="id" value="#{bean.id}" converter="idConverter">
<f:attribute name="redirect" value="some.xhtml" />
</f:viewParam>
String redirect = (String) component.getAttributes().get("redirect");
context.getExternalContext().redirect(redirect);
创建一个自定义标记处理程序,它与<f:event listener>
基本相同,但不需要额外的辅助bean方法。
<html ... xmlns:my="http://example.com/ui">
<f:metadata>
<f:viewParam name="id" value="#{bean.id}" maxlength="20" />
<my:viewParamValidationFailed redirect="some.xhtml" />
<f:viewAction action="#{bean.viewAction}" />
</f:metadata>
com.example.taghandler.ViewParamValidationFailed
public class ViewParamValidationFailed extends TagHandler implements ComponentSystemEventListener {
private String redirect;
public ViewParamValidationFailed(TagConfig config) {
super(config);
redirect = getRequiredAttribute("redirect").getValue();
}
@Override
public void apply(FaceletContext context, UIComponent parent) throws IOException {
if (parent instanceof UIViewRoot && !context.getFacesContext().isPostback()) {
((UIViewRoot) parent).subscribeToEvent(PostValidateEvent.class, this);
}
}
@Override
public void processEvent(ComponentSystemEvent event) throws AbortProcessingException {
FacesContext context = FacesContext.getCurrentInstance();
if (context.isValidationFailed()) {
try {
context.getExternalContext().redirect(redirect);
}
catch (IOException e) {
throw new AbortProcessingException(e);
}
}
}
}
/WEB-INF/my.taglib.xml
<?xml version="1.0" encoding="UTF-8"?>
<facelet-taglib
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-facelettaglibrary_2_0.xsd"
version="2.0"
>
<namespace>http://example.com/ui</namespace>
<tag>
<tag-name>viewParamValidationFailed</tag-name>
<handler-class>com.example.taghandler.ViewParamValidationFailed</handler-class>
</tag>
</facelet-taglib>
/WEB-INF/web.xml
<context-param>
<param-name>javax.faces.FACELETS_LIBRARIES</param-name>
<param-value>/WEB-INF/my.taglib.xml</param-value>
</context-param>
是的,这是一些代码,但它最终是干净且可重复使用的<my:viewParamValidationFailed>
代码,实际上非常适合新的OmniFaces功能。
答案 1 :(得分:1)
为什么不自行验证id
?
@ManagedBean
@ViewScoped
public final class TestManagedBean implements Serializable
{
private String id; //Getter and setter.
private Long validId; //Getter and setter.
public void viewAction() {
try {
validId = Long.parseLong(id);
} catch (NumberFormatException ex) {
FacesContext facesContext = FacesContext.getCurrentInstance();
String outcome = "redirect.xhtml";
facesContext.getApplication().getNavigationHandler().handleNavigation(facesContext, null, outcome);
}
}
}