Странное поведение 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>