1# 栅格布局 (GridRow/GridCol) 2 3 4## 概述 5 6栅格布局是一种通用的辅助定位工具,对移动设备的界面设计有较好的借鉴作用。主要优势包括: 7 81. 提供可循的规律:栅格布局可以为布局提供规律性的结构,解决多尺寸多设备的动态布局问题。通过将页面划分为等宽的列数和行数,可以方便地对页面元素进行定位和排版。 9 102. 统一的定位标注:栅格布局可以为系统提供一种统一的定位标注,保证不同设备上各个模块的布局一致性。这可以减少设计和开发的复杂度,提高工作效率。 11 123. 灵活的间距调整方法:栅格布局可以提供一种灵活的间距调整方法,满足特殊场景布局调整的需求。通过调整列与列之间和行与行之间的间距,可以控制整个页面的排版效果。 13 144. 自动换行和自适应:栅格布局可以完成一对多布局的自动换行和自适应。当页面元素的数量超出了一行或一列的容量时,他们会自动换到下一行或下一列,并且在不同的设备上自适应排版,使得页面布局更加灵活和适应性强。 15 16[GridRow](../reference/apis-arkui/arkui-ts/ts-container-gridrow.md)为栅格容器组件,需与栅格子组件[GridCol](../reference/apis-arkui/arkui-ts/ts-container-gridcol.md)在栅格布局场景中联合使用。 17 18 19## 栅格容器GridRow 20 21 22### 栅格系统断点 23 24栅格系统以设备的水平宽度([屏幕密度像素值](../reference/apis-arkui/arkui-ts/ts-pixel-units.md),单位vp)作为断点依据,定义设备的宽度类型,形成了一套断点规则。开发者可根据需求在不同的断点区间实现不同的页面布局效果。 25 26栅格系统默认断点将设备宽度分为xs、sm、md、lg四类,尺寸范围如下: 27 28| 断点名称 | 取值范围(vp) | 设备描述 | 29| ---- | --------------- | --------- | 30| xs | [0, 320) | 最小宽度类型设备。 | 31| sm | [320, 520) | 小宽度类型设备。 | 32| md | [520, 840) | 中等宽度类型设备。 | 33| lg | [840, +∞) | 大宽度类型设备。 | 34 35在GridRow栅格组件中,允许开发者使用breakpoints自定义修改断点的取值范围,最多支持6个断点,除了默认的四个断点外,还可以启用xl,xxl两个断点,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备的布局设置。 36 37| 断点名称 | 设备描述 | 38| ---- | --------- | 39| xs | 最小宽度类型设备。 | 40| sm | 小宽度类型设备。 | 41| md | 中等宽度类型设备。 | 42| lg | 大宽度类型设备。 | 43| xl | 特大宽度类型设备。 | 44| xxl | 超大宽度类型设备。 | 45 46- 针对断点位置,开发者根据实际使用场景,通过一个单调递增数组设置。由于breakpoints最多支持六个断点,单调递增数组长度最大为5。 47 48 49 ```ts 50 breakpoints: {value: ['100vp', '200vp']} 51 ``` 52 53 表示启用xs、sm、md共3个断点,小于100vp为xs,100vp-200vp为sm,大于200vp为md。 54 55 56 ```ts 57 breakpoints: {value: ['320vp', '520vp', '840vp', '1080vp']} 58 ``` 59 60 表示启用xs、sm、md、lg、xl共5个断点,小于320vp为xs,320vp-520vp为sm,520vp-840vp为md,840vp-1080vp为lg,大于1080vp为xl。 61 62- 栅格系统通过监听窗口或容器的尺寸变化进行断点,通过reference设置断点切换参考物。 考虑到应用可能以非全屏窗口的形式显示,以应用窗口宽度为参照物更为通用。 63 64例如,使用栅格的默认列数12列,通过断点设置将应用宽度分成六个区间,在各区间中,每个栅格子元素占用的列数均不同。 65 66 67```ts 68@State bgColors: ResourceColor[] = 69 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 70 'rgb(255,192,0)', 'rgb(170,10,33)']; 71... 72GridRow({ 73 breakpoints: { 74 value: ['200vp', '300vp', '400vp', '500vp', '600vp'], 75 reference: BreakpointsReference.WindowSize 76 } 77}) { 78 ForEach(this.bgColors, (color:ResourceColor, index?:number|undefined) => { 79 GridCol({ 80 span: { 81 xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。 82 sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。 83 md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。 84 lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。 85 xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。 86 xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。 87 } 88 }) { 89 Row() { 90 Text(`${index}`) 91 }.width("100%").height('50vp') 92 }.backgroundColor(color) 93 }) 94} 95``` 96 97 98 99 100### 布局的总列数 101 102GridRow中通过columns设置栅格布局的总列数。 103 104- columns默认值为12,即在未设置columns时,任何断点下,栅格布局被分成12列。 105 106 107 ```ts 108 @State bgColors: ResourceColor[] = 109 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 110 'rgb(255,192,0)', 'rgb(170,10,33)', 'rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)']; 111 ... 112 GridRow() { 113 ForEach(this.bgColors, (item:ResourceColor, index?:number|undefined) => { 114 GridCol() { 115 Row() { 116 Text(`${index}`) 117 }.width('100%').height('50') 118 }.backgroundColor(item) 119 }) 120 } 121 ``` 122 123  124 125- 当columns为自定义值,栅格布局在任何尺寸设备下都被分为columns列。下面分别设置栅格布局列数为4和8,子元素默认占一列,效果如下: 126 127 ```ts 128 class CurrTmp{ 129 currentBp: string = 'unknown'; 130 set(val:string){ 131 this.currentBp = val 132 } 133 } 134 let BorderWH:Record<string,Color|number> = { 'color': Color.Blue, 'width': 2 } 135 @State bgColors: ResourceColor[] = 136 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 137 'rgb(255,192,0)', 'rgb(170,10,33)']; 138 @State currentBp: string = 'unknown'; 139 ... 140 Row() { 141 GridRow({ columns: 4 }) { 142 ForEach(this.bgColors, (item: ResourceColor, index?:number|undefined) => { 143 GridCol() { 144 Row() { 145 Text(`${index}`) 146 }.width('100%').height('50') 147 }.backgroundColor(item) 148 }) 149 } 150 .width('100%').height('100%') 151 .onBreakpointChange((breakpoint:string) => { 152 let CurrSet:CurrTmp = new CurrTmp() 153 CurrSet.set(breakpoint) 154 }) 155 } 156 .height(160) 157 .border(BorderWH) 158 .width('90%') 159 160 Row() { 161 GridRow({ columns: 8 }) { 162 ForEach(this.bgColors, (item: ResourceColor, index?:number|undefined) => { 163 GridCol() { 164 Row() { 165 Text(`${index}`) 166 }.width('100%').height('50') 167 }.backgroundColor(item) 168 }) 169 } 170 .width('100%').height('100%') 171 .onBreakpointChange((breakpoint:string) => { 172 let CurrSet:CurrTmp = new CurrTmp() 173 CurrSet.set(breakpoint) 174 }) 175 } 176 .height(160) 177 .border(BorderWH) 178 .width('90%') 179 ``` 180 181  182 183- 当columns类型为GridRowColumnOption时,支持下面六种不同尺寸(xs, sm, md, lg, xl, xxl)设备的总列数设置,各个尺寸下数值可不同。 184 185 ```ts 186 @State bgColors: ResourceColor[] = 187 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 188 'rgb(255,192,0)', 'rgb(170,10,33)']; 189 GridRow({ columns: { sm: 4, md: 8 }, breakpoints: { value: ['200vp', '300vp', '400vp', '500vp', '600vp'] } }) { 190 ForEach(this.bgColors, (item: ResourceColor, index?:number|undefined) => { 191 GridCol() { 192 Row() { 193 Text(`${index}`) 194 }.width('100%').height('50') 195 }.backgroundColor(item) 196 }) 197 } 198 ``` 199 200  201 202若只设置sm, md的栅格总列数,则较小的尺寸使用默认columns值12,较大的尺寸使用前一个尺寸的columns。这里只设置sm:4, md:8,则较小尺寸的xs:12,较大尺寸的参照md的设置,lg:8, xl:8, xxl:8 203 204 205### 排列方向 206 207栅格布局中,可以通过设置GridRow的direction属性来指定栅格子组件在栅格容器中的排列方向。该属性可以设置为GridRowDirection.Row(从左往右排列)或GridRowDirection.RowReverse(从右往左排列),以满足不同的布局需求。通过合理的direction属性设置,可以使得页面布局更加灵活和符合设计要求。 208 209- 子组件默认从左往右排列。 210 211 212 ```ts 213 GridRow({ direction: GridRowDirection.Row }){} 214 ``` 215 216  217 218- 子组件从右往左排列。 219 220 221 ```ts 222 GridRow({ direction: GridRowDirection.RowReverse }){} 223 ``` 224 225  226 227 228### 子组件间距 229 230GridRow中通过gutter属性设置子元素在水平和垂直方向的间距。 231 232- 当gutter类型为number时,同时设置栅格子组件间水平和垂直方向边距且相等。下例中,设置子组件水平与垂直方向距离相邻元素的间距为10。 233 234 235 ```ts 236 GridRow({ gutter: 10 }){} 237 ``` 238 239  240 241- 当gutter类型为GutterOption时,单独设置栅格子组件水平垂直边距,x属性为水平方向间距,y为垂直方向间距。 242 243 244 ```ts 245 GridRow({ gutter: { x: 20, y: 50 } }){} 246 ``` 247 248  249 250 251## 子组件GridCol 252 253GridCol组件作为GridRow组件的子组件,通过给GridCol传参或者设置属性两种方式,设置span(占用列数),offset(偏移列数),order(元素序号)的值。 254 255- 设置span。 256 257 258 ```ts 259 let Gspan:Record<string,number> = { 'xs': 1, 'sm': 2, 'md': 3, 'lg': 4 } 260 GridCol({ span: 2 }){} 261 GridCol({ span: { xs: 1, sm: 2, md: 3, lg: 4 } }){} 262 GridCol(){}.span(2) 263 GridCol(){}.span(Gspan) 264 ``` 265 266- 设置offset。 267 268 269 ```ts 270 let Goffset:Record<string,number> = { 'xs': 1, 'sm': 2, 'md': 3, 'lg': 4 } 271 GridCol({ offset: 2 }){} 272 GridCol({ offset: { xs: 2, sm: 2, md: 2, lg: 2 } }){} 273 GridCol(){}.offset(Goffset) 274 ``` 275 276- 设置order。 277 278 279 ```ts 280 let Gorder:Record<string,number> = { 'xs': 1, 'sm': 2, 'md': 3, 'lg': 4 } 281 GridCol({ order: 2 }){} 282 GridCol({ order: { xs: 1, sm: 2, md: 3, lg: 4 } }){} 283 GridCol(){}.order(2) 284 GridCol(){}.order(Gorder) 285 ``` 286 287 288### span 289 290子组件占栅格布局的列数,决定了子组件的宽度,默认为1。 291 292- 当类型为number时,子组件在所有尺寸设备下占用的列数相同。 293 294 295 ```ts 296 @State bgColors: ResourceColor[] = 297 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 298 'rgb(255,192,0)', 'rgb(170,10,33)']; 299 ... 300 GridRow({ columns: 8 }) { 301 ForEach(this.bgColors, (color:ResourceColor, index?:number|undefined) => { 302 GridCol({ span: 2 }) { 303 Row() { 304 Text(`${index}`) 305 }.width('100%').height('50vp') 306 } 307 .backgroundColor(color) 308 }) 309 } 310 ``` 311 312  313 314- 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件所占列数设置,各个尺寸下数值可不同。 315 316 317 ```ts 318 @State bgColors: ResourceColor[] = 319 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 320 'rgb(255,192,0)', 'rgb(170,10,33)']; 321 ... 322 GridRow({ columns: 8 }) { 323 ForEach(this.bgColors, (color:ResourceColor, index?:number|undefined) => { 324 GridCol({ span: { xs: 1, sm: 2, md: 3, lg: 4 } }) { 325 Row() { 326 Text(`${index}`) 327 }.width('100%').height('50vp') 328 } 329 .backgroundColor(color) 330 }) 331 } 332 ``` 333 334  335 336 337### offset 338 339栅格子组件相对于前一个子组件的偏移列数,默认为0。 340 341- 当类型为number时,子组件偏移相同列数。 342 343 344 ```ts 345 @State bgColors: ResourceColor[] = 346 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 347 'rgb(255,192,0)', 'rgb(170,10,33)']; 348 ... 349 GridRow() { 350 ForEach(this.bgColors, (color:ResourceColor, index?:number|undefined) => { 351 GridCol({ offset: 2 }) { 352 Row() { 353 Text('' + index) 354 }.width('100%').height('50vp') 355 } 356 .backgroundColor(color) 357 }) 358 } 359 ``` 360 361  362 363 栅格默认分成12列,每一个子组件默认占1列,偏移2列,每个子组件及间距共占3列,一行放四个子组件。 364 365- 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件所占列数设置,各个尺寸下数值可不同。 366 367 368 ```ts 369 @State bgColors: ResourceColor[] = 370 ['rgb(213,213,213)', 'rgb(150,150,150)', 'rgb(0,74,175)', 'rgb(39,135,217)', 'rgb(61,157,180)', 'rgb(23,169,141)', 371 'rgb(255,192,0)', 'rgb(170,10,33)']; 372 ... 373 374 GridRow() { 375 ForEach(this.bgColors, (color:ResourceColor, index?:number|undefined) => { 376 GridCol({ offset: { xs: 1, sm: 2, md: 3, lg: 4 } }) { 377 Row() { 378 Text('' + index) 379 }.width('100%').height('50vp') 380 } 381 .backgroundColor(color) 382 }) 383 } 384 ``` 385 386  387 388 389### order 390 391栅格子组件的序号,决定子组件排列次序。当子组件不设置order或者设置相同的order, 子组件按照代码顺序展示。当子组件设置不同的order时,order较小的组件在前,较大的在后。 392 393当子组件部分设置order,部分不设置order时,未设置order的子组件依次排序靠前,设置了order的子组件按照数值从小到大排列。 394 395- 当类型为number时,子组件在任何尺寸下排序次序一致。 396 397 398 ```ts 399 GridRow() { 400 GridCol({ order: 4 }) { 401 Row() { 402 Text('1') 403 }.width('100%').height('50vp') 404 }.backgroundColor('rgb(213,213,213)') 405 GridCol({ order: 3 }) { 406 Row() { 407 Text('2') 408 }.width('100%').height('50vp') 409 }.backgroundColor('rgb(150,150,150)') 410 GridCol({ order: 2 }) { 411 Row() { 412 Text('3') 413 }.width('100%').height('50vp') 414 }.backgroundColor('rgb(0,74,175)') 415 GridCol({ order: 1 }) { 416 Row() { 417 Text('4') 418 }.width('100%').height('50vp') 419 }.backgroundColor('rgb(39,135,217)') 420 } 421 ``` 422 423  424 425- 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件排序次序设置。在xs设备中,子组件排列顺序为1234;sm为2341,md为3412,lg为2431。 426 427 428 ```ts 429 GridRow() { 430 GridCol({ order: { xs:1, sm:5, md:3, lg:7}}) { 431 Row() { 432 Text('1') 433 }.width('100%').height('50vp') 434 }.backgroundColor(Color.Red) 435 GridCol({ order: { xs:2, sm:2, md:6, lg:1} }) { 436 Row() { 437 Text('2') 438 }.width('100%').height('50vp') 439 }.backgroundColor(Color.Orange) 440 GridCol({ order: { xs:3, sm:3, md:1, lg:6} }) { 441 Row() { 442 Text('3') 443 }.width('100%').height('50vp') 444 }.backgroundColor(Color.Yellow) 445 GridCol({ order: { xs:4, sm:4, md:2, lg:5} }) { 446 Row() { 447 Text('4') 448 }.width('100%').height('50vp') 449 }.backgroundColor(Color.Green) 450 } 451 ``` 452 453  454 455 456## 栅格组件的嵌套使用 457 458栅格组件也可以嵌套使用,完成一些复杂的布局。 459 460以下示例中,栅格把整个空间分为12份。第一层GridRow嵌套GridCol,分为中间大区域以及“footer”区域。第二层GridRow嵌套GridCol,分为“left”和“right”区域。子组件空间按照上一层父组件的空间划分,粉色的区域是屏幕空间的12列,绿色和蓝色的区域是父组件GridCol的12列,依次进行空间的划分。 461 462```ts 463@Entry 464@Component 465struct GridRowExample { 466 build() { 467 GridRow() { 468 GridCol({ span: { sm: 12 } }) { 469 GridRow() { 470 GridCol({ span: { sm: 2 } }) { 471 Row() { 472 Text('left').fontSize(24) 473 } 474 .justifyContent(FlexAlign.Center) 475 .height('90%') 476 }.backgroundColor('#ff41dbaa') 477 478 GridCol({ span: { sm: 10 } }) { 479 Row() { 480 Text('right').fontSize(24) 481 } 482 .justifyContent(FlexAlign.Center) 483 .height('90%') 484 }.backgroundColor('#ff4168db') 485 } 486 .backgroundColor('#19000000') 487 } 488 489 GridCol({ span: { sm: 12 } }) { 490 Row() { 491 Text('footer').width('100%').textAlign(TextAlign.Center) 492 }.width('100%').height('10%').backgroundColor(Color.Pink) 493 } 494 }.width('100%').height(300) 495 } 496} 497``` 498 499 500 501 502 503综上所述,栅格组件提供了丰富的自定义能力,功能异常灵活和强大。只需要明确栅格在不同断点下的Columns、Margin、Gutter及span等参数,即可确定最终布局,无需关心具体的设备类型及设备状态(如横竖屏)等。 504