我有以下复合组件(仅相关代码):
<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) {
// ...
}