• 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时不显示,但依然会占用子组件对应的网格。
20
21## 接口
22
23
24WaterFlow(options?: {footer?: CustomBuilder, scroller?: Scroller})
25
26**参数:**
27
28| 参数名     | 参数类型                                        | 必填 | 参数描述                                     |
29| ---------- | ----------------------------------------------- | ------ | -------------------------------------------- |
30| footer |  [CustomBuilder](ts-types.md#custombuilder8) | 否   | 设置WaterFlow尾部组件。  |
31| scroller | [Scroller](ts-container-scroll.md#scroller) | 否   | 可滚动组件的控制器,与可滚动组件绑定。<br/>目前瀑布流仅支持Scroller组件的scrollToIndex接口。 |
32
33
34## 属性
35
36
37除支持[通用属性](ts-universal-attributes-size.md)外,还支持以下属性:
38
39| 名称 | 参数类型 | 描述 |
40| -------- | -------- | -------- |
41| columnsTemplate | string | 设置当前瀑布流组件布局列的数量,不设置时默认1列。<br/>例如, '1fr 1fr 2fr' 是将父组件分3列,将父组件允许的宽分为4等份,第一列占1份,第二列占1份,第三列占2份。并支持[auto-fill](#auto-fill说明)。<br>默认值:'1fr' |
42| rowsTemplate | string | 设置当前瀑布流组件布局行的数量,不设置时默认1行。<br/>例如, '1fr 1fr 2fr'是将父组件分三行,将父组件允许的高分为4等份,第一行占1份,第二行占一份,第三行占2份。并支持[auto-fill](#auto-fill说明)。<br/>默认值:'1fr' |
43| itemConstraintSize | [ConstraintSizeOptions](ts-types.md#constraintsizeoptions) | 设置约束尺寸,子组件布局时,进行尺寸范围限制。               |
44| columnsGap | Length |设置列与列的间距。 <br>默认值:0|
45| rowsGap | Length |设置行与行的间距。<br> 默认值:0|
46| layoutDirection | [FlexDirection](ts-appendix-enums.md#flexdirection) |设置布局的主轴方向。<br/>默认值:FlexDirection.Column|
47
48layoutDirection优先级高于rowsTemplate和columnsTemplate。根据layoutDirection设置情况,分为以下三种设置模式:
49
50- layoutDirection设置纵向布局(FlexDirection.ColumnFlexDirection.ColumnReverse51
52	此时columnsTemplate有效(如果未设置,取默认值)。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件纵向布局,辅轴均分成横向2列。
53
54- layoutDirection设置横向布局(FlexDirection.RowFlexDirection.RowReverse55
56	此时rowsTemplate有效(如果未设置,取默认值)。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件横向布局,辅轴均分成纵向3列。
57
58- layoutDirection未设置布局方向
59
60	布局方向为layoutDirection的默认值:FlexDirection.Column,此时columnsTemplate有效。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件纵向布局,辅轴均分成横向2列。
61
62## 事件
63
64
65除支持[通用事件](ts-universal-events-click.md)外,还支持以下事件:
66
67
68| 名称 | 功能描述 |
69| -------- | -------- |
70| onReachStart(event: () => void) | 瀑布流组件到达起始位置时触发。 |
71| onReachEnd(event: () => void)   | 瀑布流组件到底末尾位置时触发。 |
72
73
74## auto-fill说明
75
76WaterFlow的columnsTemplate、rowsTemplate属性的auto-fill仅支持以下格式:
77
78```css
79repeat(auto-fill, track-size)
80```
81
82其中repeat、auto-fill为关键字。track-size为行高或者列宽,支持的单位包括px、vp、%或有效数字,track-size至少包括一个有效行高或者列宽。
83
84
85## 示例
86
87
88```ts
89// WaterFlowDataSource.ets
90
91// 实现IDataSource接口的对象,用于瀑布流组件加载数据
92export class WaterFlowDataSource implements IDataSource {
93
94  private dataArray: number[] = []
95  private listeners: DataChangeListener[] = []
96
97  constructor() {
98      for (let i = 0; i < 100; i++) {
99          this.dataArray.push(i)
100      }
101  }
102
103  // 获取索引对应的数据
104  public getData(index: number): any {
105      return this.dataArray[index]
106  }
107
108  // 通知控制器数据重新加载
109  notifyDataReload(): void {
110      this.listeners.forEach(listener => {
111          listener.onDataReloaded()
112      })
113  }
114
115  // 通知控制器数据增加
116  notifyDataAdd(index: number): void {
117      this.listeners.forEach(listener => {
118          listener.onDataAdded(index)
119      })
120  }
121
122  // 通知控制器数据变化
123  notifyDataChange(index: number): void {
124      this.listeners.forEach(listener => {
125          listener.onDataChanged(index)
126      })
127  }
128
129  // 通知控制器数据删除
130  notifyDataDelete(index: number): void {
131      this.listeners.forEach(listener => {
132          listener.onDataDeleted(index)
133      })
134  }
135
136  // 通知控制器数据位置变化
137  notifyDataMove(from: number, to: number): void {
138      this.listeners.forEach(listener => {
139          listener.onDataMoved(from, to)
140      })
141  }
142
143  // 获取数据总数
144  public totalCount(): number {
145      return this.dataArray.length
146  }
147
148  // 注册改变数据的控制器
149  registerDataChangeListener(listener: DataChangeListener): void {
150      if (this.listeners.indexOf(listener) < 0) {
151          this.listeners.push(listener)
152      }
153  }
154
155  // 注销改变数据的控制器
156  unregisterDataChangeListener(listener: DataChangeListener): void {
157      const pos = this.listeners.indexOf(listener)
158      if (pos >= 0) {
159          this.listeners.splice(pos, 1)
160      }
161  }
162
163  // 增加数据
164  public Add1stItem(): void {
165      this.dataArray.splice(0, 0, this.dataArray.length)
166      this.notifyDataAdd(0)
167  }
168
169  // 在数据尾部增加一个元素
170  public AddLastItem(): void {
171      this.dataArray.splice(this.dataArray.length, 0, this.dataArray.length)
172      this.notifyDataAdd(this.dataArray.length-1)
173  }
174
175  // 在指定索引位置增加一个元素
176  public AddItem(index: number): void {
177      this.dataArray.splice(index, 0, this.dataArray.length)
178      this.notifyDataAdd(index)
179  }
180
181  // 删除第一个元素
182  public Delete1stItem(): void {
183      this.dataArray.splice(0, 1)
184      this.notifyDataDelete(0)
185  }
186
187  // 删除第二个元素
188  public Delete2ndItem(): void {
189      this.dataArray.splice(1, 1)
190      this.notifyDataDelete(1)
191  }
192
193  // 删除最后一个元素
194  public DeleteLastItem(): void {
195      this.dataArray.splice(-1, 1)
196      this.notifyDataDelete(this.dataArray.length)
197  }
198
199  // 重新加载数据
200  public Reload(): void {
201      this.dataArray.splice(1, 1)
202      this.dataArray.splice(3, 2)
203      this.notifyDataReload()
204  }
205}
206```
207
208```ts
209// WaterflowDemo.ets
210import { WaterFlowDataSource } from './WaterFlowDataSource'
211
212@Entry
213@Component
214struct WaterflowDemo {
215  @State minSize: number = 50
216  @State maxSize: number = 100
217  @State fontSize: number = 24
218  @State colors: number[] = [0xFFC0CB, 0xDA70D6, 0x6B8E23, 0x6A5ACD, 0x00FFFF, 0x00FF7F]
219  scroller: Scroller = new Scroller()
220  datasource: WaterFlowDataSource = new WaterFlowDataSource()
221  private itemWidthArray: number[] = []
222  private itemHeightArray: number[] = []
223
224  // 计算flow item宽/高
225  getSize() {
226    let ret = Math.floor(Math.random() * this.maxSize)
227    return (ret > this.minSize ? ret : this.minSize)
228  }
229
230  // 保存flow item宽/高
231  getItemSizeArray() {
232    for (let i = 0; i < 100; i++) {
233      this.itemWidthArray.push(this.getSize())
234      this.itemHeightArray.push(this.getSize())
235    }
236  }
237
238  aboutToAppear() {
239    this.getItemSizeArray()
240  }
241
242  @Builder 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({ footer: this.itemFoot.bind(this), scroller: this.scroller }) {
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          .width(this.itemWidthArray[item])
268          .height(this.itemHeightArray[item])
269          .backgroundColor(this.colors[item % 5])
270        }, item => item)
271      }
272      .columnsTemplate("1fr 1fr 1fr 1fr")
273      .itemConstraintSize({
274        minWidth: 0,
275        maxWidth: '100%',
276        minHeight: 0,
277        maxHeight: '100%'
278      })
279      .columnsGap(10)
280      .rowsGap(5)
281      .onReachStart(() => {
282        console.info("onReachStart")
283      })
284      .onReachEnd(() => {
285        console.info("onReachEnd")
286      })
287      .backgroundColor(0xFAEEE0)
288      .width('100%')
289      .height('80%')
290      .layoutDirection(FlexDirection.Column)
291    }
292  }
293}
294```
295
296![zh-cn_image_WaterFlow.gif](figures/waterflow.gif)
297