将f:属性传递给复合实现中的f:ajax侦听器



我有以下复合组件(仅相关代码):

    <cc:interface>
        <cc:attribute name="message" type="com.virtuafisica.business.problemsmgmt.entity.Message" required="true" />
        <cc:attribute name="notifiedListener" method-signature="void listener(javax.faces.event.AjaxBehaviorEvent)" />
   </cc:interface>
    <cc:implementation> 
        <h:selectBooleanCheckbox value="#{cc.attrs.message.notified}">
            <f:attribute name="msg" value="#{cc.attrs.message}" />
            <c:if test="#{cc.getValueExpression('notifiedListener') != null}">
                <f:ajax event="change" listener="#{cc.attrs.notifiedListener}" />
             </c:if>
        </h:selectBooleanCheckbox>   
    </cc:implementation>

这个组合从Facelet中调用:

<my:message message="#{msg}" notifiedListener="#{problem.changeNotified}"/>

更改事件在后台bean中是这样处理的:

public void changeNotified(AjaxBehaviorEvent event) {
    Message message = (Message) event.getComponent().getAttributes().get("msg");
        if (message == null) {
            logger.log(Level.FINEST, "Null message!");
            return;
        }
        ....
    }

问题是消息是支持bean中的null,尽管event.getComponent()是正确的(是HTMLSelectBooleanCheckbox)。

为什么没有设置属性?

谢谢大家

设置了属性(您可以通过确定getValueExpression("msg")没有返回null来确认它),但是属性值只是计算null。延迟表达式是在您想要获取值的那一刻计算的。因此,在ajax侦听器方法中调用get("msg")的那一刻,#{cc.attrs.message}就会被计算。然而,在这一点上,#{cc}不存在于EL作用域中的任何地方。

解决这个技术问题的方法是手动将其放回EL作用域(然后进行清理!)。

UIComponent component = event.getComponent();
UIComponent cc = UIComponent.getCompositeComponentParent(component);
Map<String, Object> requestScope = FacesContext.getCurrentInstance().getExternalContext().getRequestParameterMap();
try {
    requestScope.put("cc", cc);
    Message message = (Message) component.getAttributes().get("msg");
}
finally {
    requestScope.remove("cc");
}

另一个解决方法是直接从组合中获取它。

UIComponent cc = UIComponent.getCompositeComponentParent(event.getComponent());
Message message = (Message) cc.getAttributes().get("msg");

无论哪种方式,这都是相当笨拙的,因为支持bean不需要知道ajax方法是由组合调用的。简单地说,你遇到了一个设计问题。

考虑创建一个后台组件并从那里委托ajax调用,这样您就可以直接将Message作为方法参数传递。

@FacesComponent("yourCompositeName")
public class YourComposite extends UINamingContainer {
    public void notifiedListener(AjaxBehaviorEvent event) {
        MethodExpression notifiedListener = (MethodExpression) getAttributes().get("notifiedListener");
        Message message = (Message) getAttributes().get("message");
        notifiedListener.invoke(getFacesContext().getELContext(), new Object[] { message });
    }
}

将其注册为<cc:interface componentType>,将方法属性更改为接受Message参数,并让<f:ajax>调用上述支持组件上的侦听器方法,这将反过来委托给方法属性。

<cc:interface componentType="yourCompositeName">
    <cc:attribute name="message" type="com.virtuafisica.business.problemsmgmt.entity.Message" required="true" />
    <cc:attribute name="notifiedListener" method-signature="void listener(com.virtuafisica.business.problemsmgmt.entity.Message)" />
</cc:interface>
<cc:implementation> 
    <h:selectBooleanCheckbox value="#{cc.attrs.message.notified}">
        <c:if test="#{cc.getValueExpression('notifiedListener') != null}">
            <f:ajax listener="#{cc.notifiedListener}" />
        </c:if>
    </h:selectBooleanCheckbox>   
</cc:implementation>

(注意:我删除了event="change",因为这是错误的复选框/单选按钮,默认值是valueChange已经是正确的—对于复选框/单选按钮,其计算值为event="click")

现在您可以在bean中使用这个了。

public void changeNotified(Message message) {
    // ...
}  

相关内容

  • 没有找到相关文章

最新更新