Странное поведение JSF p: dataTable (первая запись неправильная в P: columns)

у меня на второй странице p:dataTable. Левый в основном используется для выбора

<p:dataTable id="leftTable" var="item" value="#{bean.items}"
             selection="#{bean.item}" selectionMode="single">
  <p:ajax event="rowSelect" update=":rightTable" listener="#{bean.select}"/>  
  <p:column>
    <h:outputText value="#{item.value}" />  
  </p:column>
</p:dataTable>

это довольно просто и работает нормально. The rightTable является более сложным, это упрощенный пример:

<p:dataTable id="rightTable" var="row" value="#{bean.rows}">
  <p:columns var="col" value="#{bean.cols}">  
    <h:outputText value="#{bean.map[row.id][col.id]}"/>
  </p:columns>
</p:dataTable>

Ну, это тоже отлично работает. Теперь я реализовал composite component заменить leftTable. Этот компонент также имеет selection и select атрибуты и вообще это тоже работает. Тот же метод, что и от p:dataTable is называют и правильно элемент установлен.

но, и это действительно неприятная вещь: если я использую свой пользовательский компонент, всегда значение в первой ячейке (столбец 0, строка 0) составляет null. Значения берутся из Map<Long,Map<Long,String>> и я проверил, что значение конкретных row.id,col.id устанавливается после вызова метода.

я совершенно невежественный пень с этой проблемой и ожидаю, что на это действительно трудно ответить вопрос и действительно ценю, если кто-то может даже помочь мне debug этот вопрос более подробно.

Upate 1: По запросу я проверил #{row.id},#{col.id}:

95,626 | 95,528
96,527 | 96,528
97,527 | 97,528

в первой ячейке col.id неправильно. Он должен быть!--14--> но на самом деле 626 (это значение предыдущего запроса). Почему это происходит? Как я могу получить правильное значение?

обновление 2: это мое компонент:

<composite:interface componentType="my.MenuDmClick">
  <composite:attribute name="actionListener" required="true"
        method-signature="void listener(javax.faces.event.AjaxBehaviorEvent)"/>
  <composite:attribute name="selection"/>
  <composite:attribute name="update/>
  <composite:attribute name="dm"/>
</composite:interface>
<composite:implementation>
  <ui:repeat var="item" value="#{cc.attrs.dm.wrappedData}">
  <li>
    <p:commandLink actionListener="#{cc.actionListener(item)}"
                   update="#{cc.attrs.update}">
      <h:outputText value="#{item.name}"/>
    </p:commandLink>
  </li>
  </ui:repeat>
</composite:implementation>

и это бэк-бин:

@FacesComponent(value="my.MenuDmClick")
public class MenuDmClick extends UINamingContainer
{
  public void actionListener(Object ejb)
  {
    FacesContext context = FacesContext.getCurrentInstance();
    ValueExpression mSelection = this.getValueExpression("selection");

    if(mSelection!=null){mSelection.setValue(context.getELContext(), ejb);}
    else{throw new PropertyNotFoundException("'selection' must be a ValueExpression!");}

    MethodExpression ajaxEventListener =
         (MethodExpression) getAttributes().get("actionListener");
    ajaxEventListener.invoke(context.getELContext(), new Object[] {});
  }
}

1 ответов


после попытки обычных вещей, таких как

  • ui: повтор и c: forEach
  • immediate=true / false

я наткнулся на process=@this ... и это работает как шарм!

<c:forEach var="item" items="#{cc.attrs.dm.wrappedData}">
  <li>
    <p:commandLink process="@this"
        actionListener="#{cc.actionListener(item)}" update="#{cc.attrs.update}">
      <h:outputText value="#{item.round.name}"/>
    </p:commandLink>
  </li>
</c:forEach>