往期鸿蒙全套实战文章必看:(附带鸿蒙全栈学习资料)
双层嵌套list,如何使用LazyForEach起作用
问题场景
在一个外层ListA,分多个ListItem,每个ListItem中都是一个List,在内层List中使用LazyForEach,达到懒加载效果。如果设置外层ListA的每个ListItem 的高度情况下,内层List的LazyForEach会失效,会一次性都会加载出来。如果 不设置外层ListA的每个ListItem 的高度情况下,内层List的 LazyForEach会起作用,但是有部分内层List的数据显示不完全。
解决措施
固定内层List高度。参考如下代码
export class BaseDataSource<T> implements IDataSource {
private readonly listeners: DataChangeListener[] = [];
protected dataset: T[];
constructor(dataset?: T[]) {
this.dataset = dataset ?? [];
}
public resetDataset(dataset: T[]) {
this.dataset = dataset;
this.notifyDataReload();
}
public updateDataAt(index: number, data: T) {
if (index >= 0 && index < this.dataset.length) {
this.dataset[index] = data;
this.notifyDataChange(index);
}
}
public getDataset() {
return this.dataset;
}
public totalCount(): number {
return this.dataset.length;
}
public getData(index: number): T {
return this.dataset[index];
}
/**
* 通知LazyForEach组件需要重载所有子组件
*/
notifyDataReload(): void {
this.listeners.forEach(listener => {
listener.onDataReloaded();
})
}
/**
* 通知LazyForEach组件需要在index对应索引处添加子组件
* @param index
*/
notifyDataAdd(index: number): void {
this.listeners.forEach(listener => {
listener.onDataAdd(index);
})
}
/**
* 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件
* @param index
*/
notifyDataChange(index: number): void {
this.listeners.forEach(listener => {
listener.onDataChange(index);
})
}
/**
* 通知LazyForEach组件需要在index对应索引处删除该子组件
* @param index
*/
notifyDataDelete(index: number): void {
this.listeners.forEach(listener => {
listener.onDataDelete(index);
})
}
/**
* 通知LazyForEach组件将from索引和to索引处的子组件进行交换
* @param from
* @param to
*/
notifyDataMove(from: number, to: number): void {
this.listeners.forEach(listener => {
listener.onDataMove(from, to);
})
}
//----------------------------------------------------------------------------------------------------
// 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听
registerDataChangeListener(listener: DataChangeListener): void {
if (this.listeners.indexOf(listener) < 0) {
this.listeners.push(listener);
}
}
// 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听
unregisterDataChangeListener(listener: DataChangeListener): void {
const pos = this.listeners.indexOf(listener);
if (pos >= 0) {
this.listeners.splice(pos, 1);
}
}
}
class NewPosItemDataSource extends BaseDataSource<string> {
private dataArray: string[] = [];
public totalCount(): number {
return this.dataArray.length;
}
public getData(index: number): string {
return this.dataArray[index];
}
public addData(index: number, data: string): void {
this.dataArray.splice(index, 0, data);
this.notifyDataAdd(index);
}
public pushData(data: string): void {
this.dataArray.push(data);
this.notifyDataAdd(this.dataArray.length - 1);
}
}
const bgColors: ResourceColor[] = [Color.Blue, Color.Gray];
const rowHeight = 60;
@Component
struct header {
title: string = '';
build() {
Column() {
Text(this.title)
.width('100%')
.height(40)
.fontSize(14)
.backgroundColor(Color.Yellow)
.fontColor(Color.Blue)
.textAlign(TextAlign.Center)
}
}
}
@Component
struct ItemCompponet {
title: string = '';
@Prop datas: string[];
generateDataSource() {
let datasource: NewPosItemDataSource = new NewPosItemDataSource();
for (let index = 0; index < this.datas.length; index++) {
const element = this.datas[index];
datasource.pushData(element);
}
return datasource;
}
build() {
Column() {
header({ title: this.title })
List() {
LazyForEach(this.generateDataSource(), (data: string, index) => {
ListItem() {
Text(data)
.width('100%')
.fontSize(14)
.backgroundColor(Color.White)
.fontColor(bgColors[index % bgColors.length])
.textAlign(TextAlign.Center)
}
.height(rowHeight)
}, (data: string, index) => {
console.log(`------- ${data + ' - ' + index.toString()}`);
return data + ' - ' + index.toString();
})
}
.layoutWeight(1)
.scrollBar(BarState.Off)
.cachedCount(10)
.friction(1.25)
.edgeEffect(EdgeEffect.None)
}
}
}
function generateData(pre: string, count: number) {
let datas: string[] = [];
for (let index = 0; index < count; index++) {
const element = pre + '-' + index.toString();
datas.push(element);
}
return datas;
}
@Entry
@Component
struct Index {
@State message: string = 'Hello World';
private scroll: Scroller = new Scroller();
@Builder
private mainListView() {
List({ scroller: this.scroll }) {
ListItem() {
ItemCompponet({ title: 'A', datas: generateData('A', 200) })
}
// .height(40 + 200 * rowHeight) // 添加该高度属性后 内层list的内容区和可视区一样高 导致LazyForEach会失效
ListItem() {
ItemCompponet({ title: 'B', datas: generateData('B', 20) })
}
// .height(40 + 20 * rowHeight) // 添加该高度属性后 内层list的内容区和可视区一样高 导致LazyForEach会失效
}
.divider({ strokeWidth: 10, color: Color.Gray })
.height("100%")
.width("100%")
.scrollBar(BarState.Off)
.edgeEffect(EdgeEffect.None)
}
build() {
Column() {
this.mainListView()
}.width('100%')
.height('100%')
.backgroundColor(Color.White)
}
}