7

多个 ajax 事件的顺序处理支持的 JSF 2.x 功能对我不起作用。我得到了以下场景:

  1. h:inputText(更改)

    <h:inputText id="consumption_input"
       value="#{cc.attrs.consumptionInfo.consumption}">
      <f:ajax
         render="#{cc.attrs.outerRenderString}"
         event="change" listener="#{cc.handleAjaxRequest}" />
    </h:inputText>
    
  2. h:commandButton (动作)

    <h:commandButton
        id="startComparisonButton"
        action="#{rateComparisonBean.startRateComparison()}"
        value="#{bundle.rateResultOverview_startComparison}">
        <!-- This is to avoid mixed requests, ajax and full requests -->
           <f:ajax render="@form"/>
        </h:commandButton>
    

如果自行触发,这两个元素的事件都会得到正确处理。

在一次单击内触发两个事件时会出现问题(在 textInput 中输入一个值,然后单击按钮)。我预计这会导致两个 ajax 请求同步触发(CHANGE-TextField 和 ACTION-commandButton)。

不幸的是,只有一个 Ajax-Request (Change-TextField),第二个事件似乎完全丢失了。

我已经确保 ah:commandButton 的所有先决条件都已满,如此处所指出: commandButton/commandLink/ajax action/listener method not invoked or input value not updated

我很高兴获得有关如何解决此问题的任何提示。

环境:Glassfish 3、Mojarra 2.1.3-FCS

4

2 回答 2

3

JSF AJAX 调用是异步的。发送一个 AJAX 请求(在这种情况下由<h:inputText> onchange事件生成)不会停止 JavaScipt 继续执行,并且在这种情况下会触发提交按钮单击,这反过来又会触发另一个 AJAX 请求。尽管如此,AJAX 请求确实在客户端排队,按照它们发送的确切顺序进行处理,这是由JSF 2.0 规范第 13.3.2 章保证的。

下面是我的测试用例:

风景:

<h:form id="form">
    <h:inputText id="text" value="#{q16363737Bean.text1}">
        <f:ajax render="text2" event="change" listener="#{q16363737Bean.ajaxListenerText}"/>
    </h:inputText>
    <h:commandButton id="button" action="#{q16363737Bean.actionButton}" value="Submit">
        <f:ajax render="text1 text3" listener="#{q16363737Bean.ajaxListenerButton}"/>
    </h:commandButton>
    <br/>
    <h:outputText id="text1" value="Text 1: #{q16363737Bean.text1}."/>
    <h:outputText id="text2" value="Text 2: #{q16363737Bean.text2}."/>
    <h:outputText id="text3" value="Text 3: #{q16363737Bean.text3}."/>
</h:form>

豆子:

@ManagedBean
@ViewScoped
public class Q16363737Bean implements Serializable {

    private String text1 = "I'm text 1";//getter + setter
    private String text2 = "I'm text 2";//getter + setter
    private String text3 = "I'm text 3";//getter + setter

    public void ajaxListenerText(AjaxBehaviorEvent abe) {
        text2 = "I was modified after inputText AJAX call";
    }

    public void ajaxListenerButton(AjaxBehaviorEvent abe) {
        text1 = "I was modified after AJAX listener call of commandButton";
    }

    public void actionButton() {
        text3 = "I was modified after commandButton AJAX call";
    }

}

在检查了一段时间后,我确实发现命令按钮的 AJAX 调用很少被吞下,并且没有完成 UI 更新。似乎某处应该有一些竞争条件。这是一个很好的问题,需要进一步研究。

所以,这可能不是一个答案(虽然我最初认为它会是),而是一个测试用例的命题。尽管我很少遇到这种行为,但这是一个真实的用例,值得充分了解正在发生的事情。

于 2013-05-03T19:30:04.973 回答
0

首先感谢分配快速和复杂的答复。

从头开始的第二个测试指出,ajax 事件已正确排队。

所以我再次检查了上面描述的更复杂的场景。跳过事件接缝的原因与我们的“忙覆盖”有关。把它想象成一堆 JavaScript,它禁用表单元素并在长时间运行的 ajax 请求期间应用覆盖。这是通过应用jsf.ajax.addOnEvent回调来完成的。

无论如何,即使我们禁用了某些控件,在“成功”之后表单处于正确状态并且可以处理第二个事件。

目前,我假设“暂时”禁用的按钮会损害之后应该执行的相关操作事件。

我们目前正在分析这个问题,我会尽快发布最终结果。

于 2013-05-03T21:32:13.447 回答