在一个项目中,我需要从数据库中延迟加载对象,并且对于每个元素,我将放置一个链接以重定向到特定页面。
延迟加载正在运行。当点击第一个元素的链接就可以了,问题出在滚动之后,下一个元素不会调用listner listOi.editer()。
<p:dataScroller value="#{listOi.lazyOi}" var="oi" id="OisChoser" widgetVar="scroller"
chunkSize="5" mode="inline" scrollHeight="531" lazy="true" style="width: 597px;" rows="5" >
<h:panelGroup id="info_OI" class="info_OI" align="center" >
...
<h:commandLink actionListener="#{listOi.editer()}" immediate="true" >
<f:param name="selectedoiId" value="#{oi.id}" />
<span class="crayon" style='cursor: pointer;'></span>
</h:commandLink>
...
</h:panelGroup
</p:dataScroller>
答案 0 :(得分:3)
问题在于PrimeFaces&#39; LazyDataModel
未保留视图中显示的数据的完整模型。它仅跟踪最近加载的项目并丢弃旧项目。这使得JSF无法再访问这些项目。
但是既然你无论如何都要继承那个类(它是抽象的),那么很容易改变这种行为。基本上你想要做的就是跟踪你到目前为止加载的所有数据并返回所要求的数据。至少,您需要覆盖setRowIndex()
,getRowData()
,isRowAvailable()
和load()
。以下是一个适合我的例子。
public class MyLazyModel extends LazyDataModel<SomeType> implements Serializable {
private final List<SomeType> data;
private int rowIndex;
public MyLazyModel() {
super();
data = new ArrayList<>();
}
@Override
public List<SomeType> load(int first, int pageSize, String sortField, SortOrder sortOrder, Map<String, Object> filters) {
List<SomeType> retData;
if (first >= data.size()) {
retData = ... //Get the data from datasource
data.addAll(retData);
return retData;
} else {
return data.subList(first, Math.min(first + pageSize, data.size()));
}
}
@Override
public void setRowIndex(int index) {
if (index >= data.size()) {
index = -1;
}
this.rowIndex = index;
}
@Override
public SomeType getRowData() {
return data.get(rowIndex);
}
@Override
public boolean isRowAvailable() {
if (data == null) {
return false;
}
return rowIndex >= 0 && rowIndex < data.size();
}
}
因为您永远无法确定setWrappedData()
(使用该方法的返回数据在load()
之后调用)仅被调用一次,因此您需要防止向数据模型添加双精度数。在这里,我只通过加载之前从未加载的数据并将数据存储在load()
方法中,完全忽略setWrappedData()
来实现此目的。如果您的模型永远不会失效,它会导致同步问题,但它确实有效。无论如何,您可以通过始终加载数据并用新内容替换旧内容来避免这种情况,但这不是问题的核心。
此外,因为您现在跟踪方法本身中的数据,您需要覆盖LazyDataModel
中依赖LazyDataModel.data
正确的所有方法(或者至少是您的那些子集)正在使用)。
最后注意事项:您当然必须确保返回到JSF页面的模型始终如同here所述。