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