• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# WaterFlow
2
3
4瀑布流容器,由“行”和“列”分割的单元格所组成,通过容器自身的排列规则,将不同大小的“项目”自上而下,如瀑布般紧密布局。
5
6
7> **说明:**
8>
9> 该组件从API Version 9 开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。
10
11
12## 子组件
13
14
15包含[FlowItem](ts-container-flowitem.md)子组件。
16
17>  **说明:**
18>
19>  WaterFlow子组件的visibility属性设置为None时不显示,但该子组件周围的columnsGap、rowsGap、margin仍会生效。
20
21## 接口
22
23
24WaterFlow(options?:  WaterFlowOptions)
25
26**参数:**
27
28| 参数名 | 参数类型 | 必填 | 参数描述 |
29| -------- | -------- | -------- | -------- |
30| options |  [WaterFlowOptions](#waterflowoptions对象说明)| 是 | 瀑布流组件参数。 |
31
32## WaterFlowOptions对象说明
33
34| 参数名     | 参数类型                                        | 必填 | 参数描述                                     |
35| ---------- | ----------------------------------------------- | ------ | -------------------------------------------- |
36| footer |  [CustomBuilder](ts-types.md#custombuilder8) | 否   | 设置WaterFlow尾部组件。  |
37| scroller | [Scroller](ts-container-scroll.md#scroller) | 否   | 可滚动组件的控制器,与可滚动组件绑定。<br/>**说明:** <br/>不允许和其他滚动类组件,如:[List](ts-container-list.md)、[Grid](ts-container-grid.md)、[Scroll](ts-container-scroll.md)等绑定同一个滚动控制对象。 |
38
39
40## 属性
41
42
43除支持[通用属性](ts-universal-attributes-size.md)外,还支持以下属性:
44
45| 名称 | 参数类型 | 描述 |
46| -------- | -------- | -------- |
47| columnsTemplate | string | 设置当前瀑布流组件布局列的数量,不设置时默认1列。<br/>例如, '1fr 1fr 2fr' 是将父组件分3列,将父组件允许的宽分为4等份,第一列占1份,第二列占1份,第三列占2份。<br>可使用columnsTemplate('repeat(auto-fill,track-size)')根据给定的列宽track-size自动计算列数,其中repeat、auto-fill为关键字,track-size为可设置的宽度,支持的单位包括px、vp、%或有效数字,默认单位为vp,使用方法参见示例2。<br>默认值:'1fr' |
48| rowsTemplate | string | 设置当前瀑布流组件布局行的数量,不设置时默认1行。<br/>例如, '1fr 1fr 2fr'是将父组件分三行,将父组件允许的高分为4等份,第一行占1份,第二行占一份,第三行占2份。<br>可使用rowsTemplate('repeat(auto-fill,track-size)')根据给定的行高track-size自动计算行数,其中repeat、auto-fill为关键字,track-size为可设置的高度,支持的单位包括px、vp、%或有效数字,默认单位为vp。<br/>默认值:'1fr' |
49| itemConstraintSize | [ConstraintSizeOptions](ts-types.md#constraintsizeoptions) | 设置约束尺寸,子组件布局时,进行尺寸范围限制。               |
50| columnsGap | [Length](ts-types.md#length) |设置列与列的间距。 <br>默认值:0|
51| rowsGap | [Length](ts-types.md#length) |设置行与行的间距。<br> 默认值:0|
52| layoutDirection | [FlexDirection](ts-appendix-enums.md#flexdirection) |设置布局的主轴方向。<br/>默认值:FlexDirection.Column|
53| enableScrollInteraction<sup>10+</sup>  |  boolean  |   设置是否支持滚动手势,当设置为false时,无法通过手指或者鼠标滚动,但不影响控制器的滚动接口。<br/>默认值:true      |
54| nestedScroll<sup>10+</sup>                 | [NestedScrollOptions](ts-container-scroll.md#nestedscrolloptions10对象说明)         | 嵌套滚动选项。设置向前向后两个方向上的嵌套滚动模式,实现与父组件的滚动联动。 |
55| friction<sup>10+</sup> | number \| [Resource](ts-types.md#resource)    | 设置摩擦系数,手动划动滚动区域时生效,只对惯性滚动过程有影响,对惯性滚动过程中的链式效果有间接影响。<br/>默认值:非可穿戴设备为0.6,可穿戴设备为0.9<br/>**说明:** <br/>设置为小于等于0的值时,按默认值处理 |
56
57layoutDirection优先级高于rowsTemplate和columnsTemplate。根据layoutDirection设置情况,分为以下三种设置模式:
58
59- layoutDirection设置纵向布局(FlexDirection.ColumnFlexDirection.ColumnReverse60
61	此时columnsTemplate有效(如果未设置,取默认值)。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件纵向布局,辅轴均分成横向2列。
62
63- layoutDirection设置横向布局(FlexDirection.RowFlexDirection.RowReverse64
65	此时rowsTemplate有效(如果未设置,取默认值)。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件横向布局,辅轴均分成纵向3列。
66
67- layoutDirection未设置布局方向
68
69	布局方向为layoutDirection的默认值:FlexDirection.Column,此时columnsTemplate有效。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件纵向布局,辅轴均分成横向2列。
70
71## 事件
72
73
74除支持[通用事件](ts-universal-events-click.md)外,还支持以下事件:
75
76
77| 名称 | 功能描述 |
78| -------- | -------- |
79| onReachStart(event: () => void) | 瀑布流组件到达起始位置时触发。 |
80| onReachEnd(event: () => void)   | 瀑布流组件到底末尾位置时触发。 |
81| onScrollFrameBegin<sup>10+</sup>(event: (offset: number, state: [ScrollState](ts-container-list.md#scrollstate枚举说明)) => { offsetRemain: number }) | 瀑布流开始滑动时触发,事件参数传入即将发生的滑动量,事件处理函数中可根据应用场景计算实际需要的滑动量并作为事件处理函数的返回值返回,瀑布流将按照返回值的实际滑动量进行滑动。<br/>\- offset:即将发生的滑动量,单位vp。<br/>\- state:当前滑动状态。<br/>- offsetRemain:实际滑动量,单位vp。<br/>触发该事件的条件:手指拖动WaterFlow、WaterFlow惯性划动时每帧开始时触发;WaterFlow超出边缘回弹、使用滚动控制器和拖动滚动条的滚动不会触发。|
82
83
84## 示例
85
86### 示例1
87WaterFlow的基本使用。
88```ts
89// WaterFlowDataSource.ets
90
91// 实现IDataSource接口的对象,用于瀑布流组件加载数据
92export class WaterFlowDataSource implements IDataSource {
93  private dataArray: number[] = []
94  private listeners: DataChangeListener[] = []
95
96  constructor() {
97    for (let i = 0; i < 100; i++) {
98      this.dataArray.push(i)
99    }
100  }
101
102  // 获取索引对应的数据
103  public getData(index: number): number {
104    return this.dataArray[index]
105  }
106
107  // 通知控制器数据重新加载
108  notifyDataReload(): void {
109    this.listeners.forEach(listener => {
110      listener.onDataReloaded()
111    })
112  }
113
114  // 通知控制器数据增加
115  notifyDataAdd(index: number): void {
116    this.listeners.forEach(listener => {
117      listener.onDataAdd(index)
118    })
119  }
120
121  // 通知控制器数据变化
122  notifyDataChange(index: number): void {
123    this.listeners.forEach(listener => {
124      listener.onDataChange(index)
125    })
126  }
127
128  // 通知控制器数据删除
129  notifyDataDelete(index: number): void {
130    this.listeners.forEach(listener => {
131      listener.onDataDelete(index)
132    })
133  }
134
135  // 通知控制器数据位置变化
136  notifyDataMove(from: number, to: number): void {
137    this.listeners.forEach(listener => {
138      listener.onDataMove(from, to)
139    })
140  }
141
142  // 获取数据总数
143  public totalCount(): number {
144    return this.dataArray.length
145  }
146
147  // 注册改变数据的控制器
148  registerDataChangeListener(listener: DataChangeListener): void {
149    if (this.listeners.indexOf(listener) < 0) {
150      this.listeners.push(listener)
151    }
152  }
153
154  // 注销改变数据的控制器
155  unregisterDataChangeListener(listener: DataChangeListener): void {
156    const pos = this.listeners.indexOf(listener)
157    if (pos >= 0) {
158      this.listeners.splice(pos, 1)
159    }
160  }
161
162  // 增加数据
163  public add1stItem(): void {
164    this.dataArray.splice(0, 0, this.dataArray.length)
165    this.notifyDataAdd(0)
166  }
167
168  // 在数据尾部增加一个元素
169  public addLastItem(): void {
170    this.dataArray.splice(this.dataArray.length, 0, this.dataArray.length)
171    this.notifyDataAdd(this.dataArray.length - 1)
172  }
173
174  // 在指定索引位置增加一个元素
175  public addItem(index: number): void {
176    this.dataArray.splice(index, 0, this.dataArray.length)
177    this.notifyDataAdd(index)
178  }
179
180  // 删除第一个元素
181  public delete1stItem(): void {
182    this.dataArray.splice(0, 1)
183    this.notifyDataDelete(0)
184  }
185
186  // 删除第二个元素
187  public delete2ndItem(): void {
188    this.dataArray.splice(1, 1)
189    this.notifyDataDelete(1)
190  }
191
192  // 删除最后一个元素
193  public deleteLastItem(): void {
194    this.dataArray.splice(-1, 1)
195    this.notifyDataDelete(this.dataArray.length)
196  }
197
198  // 重新加载数据
199  public reload(): void {
200    this.dataArray.splice(1, 1)
201    this.dataArray.splice(3, 2)
202    this.notifyDataReload()
203  }
204}
205```
206
207```ts
208// Index.ets
209import { WaterFlowDataSource } from './WaterFlowDataSource'
210
211@Entry
212@Component
213struct WaterFlowDemo {
214  @State minSize: number = 80
215  @State maxSize: number = 180
216  @State fontSize: number = 24
217  @State colors: number[] = [0xFFC0CB, 0xDA70D6, 0x6B8E23, 0x6A5ACD, 0x00FFFF, 0x00FF7F]
218  scroller: Scroller = new Scroller()
219  dataSource: WaterFlowDataSource = new WaterFlowDataSource()
220  private itemWidthArray: number[] = []
221  private itemHeightArray: number[] = []
222
223  // 计算FlowItem宽/高
224  getSize() {
225    let ret = Math.floor(Math.random() * this.maxSize)
226    return (ret > this.minSize ? ret : this.minSize)
227  }
228
229  // 设置FlowItem的宽/高数组
230  setItemSizeArray() {
231    for (let i = 0; i < 100; i++) {
232      this.itemWidthArray.push(this.getSize())
233      this.itemHeightArray.push(this.getSize())
234    }
235  }
236
237  aboutToAppear() {
238    this.setItemSizeArray()
239  }
240
241  @Builder
242  itemFoot() {
243    Column() {
244      Text(`Footer`)
245        .fontSize(10)
246        .backgroundColor(Color.Red)
247        .width(50)
248        .height(50)
249        .align(Alignment.Center)
250        .margin({ top: 2 })
251    }
252  }
253
254  build() {
255    Column({ space: 2 }) {
256      WaterFlow() {
257        LazyForEach(this.dataSource, (item: number) => {
258          FlowItem() {
259            Column() {
260              Text("N" + item).fontSize(12).height('16')
261              Image('res/waterFlowTest(' + item % 5 + ').jpg')
262                .objectFit(ImageFit.Fill)
263                .width('100%')
264                .layoutWeight(1)
265            }
266          }
267          .onAppear(() => {
268            // 即将触底时提前增加数据
269            if (item + 20 == this.dataSource.totalCount()) {
270              for (let i = 0; i < 100; i++) {
271                this.dataSource.addLastItem()
272              }
273            }
274          })
275          .width('100%')
276          .height(this.itemHeightArray[item % 100])
277          .backgroundColor(this.colors[item % 5])
278        }, (item: string) => item)
279      }
280      .columnsTemplate("1fr 1fr")
281      .columnsGap(10)
282      .rowsGap(5)
283      .backgroundColor(0xFAEEE0)
284      .width('100%')
285      .height('100%')
286    }
287  }
288}
289```
290
291![zh-cn_image_WaterFlow.gif](figures/waterflow-perf-demo.gif)
292
293### 示例2
294auto-fill的使用。
295```ts
296//index.ets
297import { WaterFlowDataSource } from './WaterFlowDataSource'
298
299@Entry
300@Component
301struct WaterFlowDemo {
302  @State minSize: number = 80
303  @State maxSize: number = 180
304  @State colors: number[] = [0xFFC0CB, 0xDA70D6, 0x6B8E23, 0x6A5ACD, 0x00FFFF, 0x00FF7F]
305  dataSource: WaterFlowDataSource = new WaterFlowDataSource()
306  private itemWidthArray: number[] = []
307  private itemHeightArray: number[] = []
308
309  // 计算FlowItem宽/高
310  getSize() {
311    let ret = Math.floor(Math.random() * this.maxSize)
312    return (ret > this.minSize ? ret : this.minSize)
313  }
314
315  // 设置FlowItem宽/高数组
316  setItemSizeArray() {
317    for (let i = 0; i < 100; i++) {
318      this.itemWidthArray.push(this.getSize())
319      this.itemHeightArray.push(this.getSize())
320    }
321  }
322
323  aboutToAppear() {
324    this.setItemSizeArray()
325  }
326
327  build() {
328    Column({ space: 2 }) {
329      WaterFlow() {
330        LazyForEach(this.dataSource, (item: number) => {
331          FlowItem() {
332            Column() {
333              Text("N" + item).fontSize(12).height('16')
334            }
335          }
336          .width('100%')
337          .height(this.itemHeightArray[item % 100])
338          .backgroundColor(this.colors[item % 5])
339        }, (item: string) => item)
340      }
341      .columnsTemplate('repeat(auto-fill,80)')
342      .columnsGap(10)
343      .rowsGap(5)
344      .padding({left:5})
345      .backgroundColor(0xFAEEE0)
346      .width('100%')
347      .height('100%')
348    }
349  }
350}
351```
352
353![waterflow_auto-fill.png](figures/waterflow_auto-fill.png)