2

我有一个 JSF 复合组件 util_primefaces:inplace_name,它需要一个“管理器”支持 bean,它在编辑实体的“名称”字段时执行持久性更新(使用 p:inplace):

<cc:interface>
  <cc:attribute name="manager" type="com.example.web.AbstractManager" required="false" default="#{blockManager}"/>
  <cc:attribute name="element" type="com.example.entity.Element" required="true"/>
  <cc:attribute name="elid" required="true"/>
  <cc:attribute name="update" required="false" default="@parent"/>
 ..
</cc:interface>

<cc:implementation>
 ..
 <p:inplace id="#{cc.attrs.elid}" editor="true" emptyLabel="UNDEF" >
  <p:ajax 
   event="save" 
   listener="#{cc.attrs.manager.onInplaceNameSaveEvent}"
   process="@this #{cc.attrs.elid}-name"
   update="#{cc.attrs.update}"
  />
 <h:inputText id="#{cc.attrs.elid}-name" value="#{cc.attrs.element.name}"/>
 ..

例如,@ViewScoped @ManagedBean BlockManager 最终扩展了一个 AbstractManager,它有一个监听器方法:

public void onInplaceNameSaveEvent(AjaxBehaviorEvent ae).

[旁白:这里描述了不寻常的“elid”属性的原因,它在这个问题中没有进一步的作用:Primefaces p:inplace:如何更优雅地传播实体合并的 EL 表达式]

当我调用传入显式 #{blockManager} (或 AbstractManager 的其他子类)的复合组件时,它工作正常:

<util_primefaces:inplace_name 
 element="#{tenancy}" 
 elid="tenancy"
 manager="#{blockManager}"
/>

但是,如果我没有传入 #{blockManager},在执行就地编辑和保存时,我会收到一个错误,即方法 onInplaceNameSaveEvent(AjaxBehaviorEvent) 未知:

<util_primefaces:inplace_name 
 element="#{tenancy}" 
 elid="tenancy"
/>

错误是:

WARNING: Method not found: com.example.web.BlockManager@71396a88.onInplaceNameSaveEvent(javax.faces.event.AjaxBehaviorEvent)
javax.el.MethodNotFoundException: Method not found: com.example.web.BlockManager@71396a88.onInplaceNameSaveEvent(javax.faces.event.AjaxBehaviorEvent)
at com.sun.el.util.ReflectionUtil.getMethod(ReflectionUtil.java:155) 

问:为什么在复合组件属性中使用 default="#{blockManager}" 没有正确获取 backing bean?

4

1 回答 1

5

根据标签的文档cc:attribute,值default必须评估为 a java.lang.String

这就是为什么#{blockManager}表达式不能按预期工作的原因,您只能为String属性设置默认值。

要查看会发生什么,您可以测试在 taglib 中注册以下函数(如这里所示):

public static String typeOf(Object o) {
    return o == null ? null : o.getClass().toString();
}

并在这样的复合组件中使用它:

<ui:composition>
    <cc:interface>
        <cc:attribute name="param" type="java.lang.Integer" required="true"
            default="1" />
    </cc:interface>
    <cc:implementation>
        <h:outputText value="#{fnc:typeOf(cc.attrs.param)}" />
    </cc:implementation>
</ui:composition>

现在,如果您在不指定 的情况下使用组件param,如下所示:

<my:comp />

...它将输出class java.lang.String,因为它使用 ValueExpression 的结果作为默认属性,即 String "1"

当您指定参数时:

<my:comp param="1" />

...它输出class java.lang.Integer,因为现在它是强制转换为为cc:attribute.

于 2012-07-22T02:55:15.957 回答