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.Column 或 FlexDirection.ColumnReverse) 60 61 此时columnsTemplate有效(如果未设置,取默认值)。例如columnsTemplate设置为"1fr 1fr"、rowsTemplate设置为"1fr 1fr 1fr"时,瀑布流组件纵向布局,辅轴均分成横向2列。 62 63- layoutDirection设置横向布局(FlexDirection.Row 或 FlexDirection.RowReverse) 64 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 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