2.5、栅格布局(GridRow/GridCol)

本文介绍了鸿蒙HarmonyOS中的栅格布局组件GridRow和GridCol,包括栅格系统断点、布局总列数、排列方向、子组件间距以及它们的使用方法和示例。通过栅格布局,可以实现多设备的动态布局,提供统一的定位标注,以及灵活的间距调整,确保布局的一致性和自适应性。
摘要由CSDN通过智能技术生成

概述

栅格布局是一种通用的辅助定位工具,对移动设备的界面设计有较好的借鉴作用。主要优势包括:

  • 提供可循的规律:栅格布局可以为布局提供规律性的结构,解决多尺寸多设备的动态布局问题。通过将页面划分为等宽的列数和行数,可以方便地对页面元素进行定位和排版。

  • 统一的定位标注:栅格布局可以为系统提供一种统一的定位标注,保证不同设备上各个模块的布局一致性。这可以减少设计和开发的复杂度,提高工作效率。

  • 灵活的间距调整方法:栅格布局可以提供一种灵活的间距调整方法,满足特殊场景布局调整的需求。通过调整列与列之间和行与行之间的间距,可以控制整个页面的排版效果。

  • 自动换行和自适应:栅格布局可以完成一对多布局的自动换行和自适应。当页面元素的数量超出了一行或一列的容量时,他们会自动换到下一行或下一列,并且在不同的设备上自适应排版,使得页面布局更加灵活和适应性强。

GridRow 为栅格容器组件,需与栅格子组件 GridCol 在栅格布局场景中联合使用。

我开发的 Demo 展示

在这里插入图片描述

以下代码均经过我 demo 的实战验证,确保代码和效果对应

栅格容器GridRow

栅格系统断点

栅格系统以设备的水平宽度(屏幕密度像素值,单位vp)作为断点依据,定义设备的宽度类型,形成了一套断点规则。开发者可根据需求在不同的断点区间实现不同的页面布局效果。

栅格系统默认断点将设备宽度分为xs、sm、md、lg四类,尺寸范围如下:

断点名称取值范围(vp)设备描述
xs[0, 320)最小宽度类型设备。
sm[320, 520)小宽度类型设备。
md[520, 840)中等宽度类型设备。
lg[840, +∞)大宽度类型设备。

GridRow 栅格组件中,允许开发者使用 breakpoints 自定义修改断点的取值范围,最多支持6个断点,除了默认的四个断点外,还可以启用 xlxxl 两个断点,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备的布局设置。

断点名称设备描述
xs最小宽度类型设备。
sm小宽度类型设备。
md中等宽度类型设备。
lg大宽度类型设备。
xl特大宽度类型设备。
xxl超大宽度类型设备。
  • 针对断点位置,开发者根据实际使用场景,通过一个单调递增数组设置。由于 breakpoints 最多支持六个断点,单调递增数组长度最大为5。
breakpoints: {value: ['100vp', '200vp']}

表示启用 xssmmd 共3个断点,小于100vp为 xs,100vp-200vp为 sm,大于200vp为 md

breakpoints: {value: ['320vp', '520vp', '840vp', '1080vp']}

表示启用xs、sm、md、lg、xl共5个断点,小于320vp为xs,320vp-520vp为sm,520vp-840vp为md,840vp-1080vp为lg,大于1080vp为xl。

  • 栅格系统通过监听窗口或容器的尺寸变化进行断点,通过reference设置断点切换参考物。 考虑到应用可能以非全屏窗口的形式显示,以应用窗口宽度为参照物更为通用。

例如,使用栅格的默认列数12列,通过断点设置将应用宽度分成六个区间,在各区间中,每个栅格子元素占用的列数均不同。

@State bgColors: Color[] = [Color.Red, Color.Orange, Color.Yellow, Color.Green, Color.Pink, Color.Grey, Color.Blue, Color.Brown,Color.Red, Color.Orange, Color.Yellow, Color.Green];
...
GridRow({
  breakpoints: {
    value: ['200vp', '300vp', '400vp', '500vp', '600vp'],
    reference: BreakpointsReference.WindowSize
  }
}) {
   ForEach(this.bgColors, (color, index) => {
     GridCol({
       span: {
         xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
         sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
         md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
         lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
         xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
         xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
       }
     }) {
       Row() {
         Text(`${index}`)
       }.width("100%").height('50vp')
     }.backgroundColor(color)
   })
}                                                                     

我的模拟器是显示3列,就意味着是 md 类型 (中等宽度类型设备)
在这里插入图片描述

布局的总列数

GridRow 中通过 columns 设置栅格布局的总列数。

columns 默认值为 12,即在未设置 columns 时,任何断点下,栅格布局被分成 12 列。

效果图
在这里插入图片描述
对应代码

GridRow() {
  ForEach(this.bgColors, (color, index) => {
    GridCol() {
      Row() {
        Text(`${index+1}`)
      }.width("100%").height('50vp')
    }.backgroundColor(color)
  })
}.width('100%')

columns 为自定义值,栅格布局在任何尺寸设备下都被分为 columns 列。下面分别设置栅格布局列数为 4 和 8,子元素默认占一列,效果如下:

columns 4 的效果图
在这里插入图片描述
对应代码

GridRow({ columns: 4 }) {
    ForEach(this.bgColors, (item, index) => {
      GridCol() {
        Row() {
          Text(`${index + 1}`)
        }.width('100%').height('50')
      }.backgroundColor(item)
    })
  }
  .width('100%').height('100%')
  .onBreakpointChange((breakpoint) => {
    this.currentBp = breakpoint
  })
.height(160)
.border({ color: Color.Blue, width: 2 })
.width('90%')

columns 8 的效果图
在这里插入图片描述

对应代码

GridRow({ columns: 8 }) {
    ForEach(this.bgColors, (item, index) => {
      GridCol() {
        Row() {
          Text(`${index + 1}`)
        }.width('100%').height('50')
      }.backgroundColor(item)
    })
  }
  .width('100%').height('100%')
  .onBreakpointChange((breakpoint) => {
    this.currentBp = breakpoint
  })
.height(160)
.border({ color: Color.Blue, width: 2 })
.width('90%')

排列方向

栅格布局中,可以通过设置GridRow的direction属性来指定栅格子组件在栅格容器中的排列方向。该属性可以设置为GridRowDirection.Row(从左往右排列)或GridRowDirection.RowReverse(从右往左排列),以满足不同的布局需求。通过合理的direction属性设置,可以使得页面布局更加灵活和符合设计要求。

  • 子组件默认从左往右排列
    在这里插入图片描述
    对应代码
GridRow({ 
   direction: GridRowDirection.Row,
   columns: 7
}) {
ForEach(this.bgColors, (item, index) => {
  GridCol() {
    Row() {
      Text(`${index + 1}`)
    }.width('100%').height('50')
  }.backgroundColor(item)
})
}
.width('90%')
.height(160)
.border({ color: Color.Blue, width: 2 })
.margin({ left: 5, top: 5, right: 5 })
  • 子组件从右往左排列。

这里模拟器显示会有 bug,真机不知道有没有这个问题
在这里插入图片描述
对应代码

GridRow({ direction: GridRowDirection.RowReverse})
...

子组件间距

  • GridRow 中通过 gutter 属性设置子元素在水平和垂直方向的间距。

gutter 类型为 number 时,同时设置栅格子组件间水平和垂直方向边距且相等。下例中,设置子组件水平与垂直方向距离相邻元素的间距为 10。
在这里插入图片描述
对应代码

GridRow({ gutter: 10 ,
   columns: 7
}) {
ForEach(this.bgColors, (item, index) => {
  GridCol() {
    Row() {
      Text(`${index + 1}`)
    }.width('100%').height('50')
  }.backgroundColor(item)
})
}
.width('90%')
.height(160)
.border({ color: Color.Blue, width: 2 })
.margin({left:5})

gutter 类型为 GutterOption 时,单独设置栅格子组件水平垂直边距,x 属性为水平方向间距,y 为垂直方向间距。
在这里插入图片描述
对应代码

GridRow({ gutter: { x: 20, y: 50 } ,
   columns: 7
}) {
ForEach(this.bgColors, (item, index) => {
  GridCol() {
    Row() {
      Text(`${index + 1}`)
    }.width('100%').height('50')
  }.backgroundColor(item)
})
}
.width('90%')
.height(160)
.border({ color: Color.Blue, width: 2 })
.margin({left:5})

子组件GridCol

span

子组件占栅格布局的列数,决定了子组件的宽度,默认为1。
当类型为number时,子组件在所有尺寸设备下占用的列数相同。

 GridCol({ span: 2 }) 

在这里插入图片描述

对应代码

GridRow({ columns: 6 }) {
  ForEach(this.bgColors, (color, index) => {
    GridCol({ span: 2 }) {
      Row() {
        Text(`${index}`)
      }.width('100%').height('50vp')
    }
    .backgroundColor(color)
  })
}.margin({left:5, right:5})

当类型为 GridColColumnOption 时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件所占列数设置,各个尺寸下数值可不同。
在这里插入图片描述
对应代码

GridRow({ columns: 8 }) {
  ForEach(this.bgColors, (color, index) => {
    GridCol({ span: { xs: 1, sm: 2, md: 3, lg: 4 } }) {
      Row() {
        Text(`${index}`)
      }.width('100%').height('50vp')
    }
    .backgroundColor(color)
  })
}.margin({left:5, right:5})

offset

栅格子组件相对于前一个子组件的偏移列数,默认为0。

  • 当类型为number时,子组件偏移相同列数。
    在这里插入图片描述
    对应代码
GridRow() {
  ForEach(this.bgColors, (color, index) => {
    GridCol({ offset: 2 }) {
      Row() {
        Text('' + index)
      }.width('100%').height('50vp')
    }
    .backgroundColor(color)
  })
}
.border({ color: Color.Blue, width: 2 })
.margin({left:5, right:5})

order

栅格子组件的序号,决定子组件排列次序。当子组件不设置order或者设置相同的order, 子组件按照代码顺序展示。当子组件设置不同的order时,order较小的组件在前,较大的在后。

当子组件部分设置order,部分不设置order时,未设置order的子组件依次排序靠前,设置了order的子组件按照数值从小到大排列。

  • 当类型为number时,子组件在任何尺寸下排序次序一致。
    在这里插入图片描述
    对应代码
GridRow() {
  GridCol({ order: 4 }) {
    Row() {
      Text('1')
    }.width('100%').height('50vp')
  }.backgroundColor(Color.Red)
  GridCol({ order: 3 }) {
    Row() {
      Text('2')
    }.width('100%').height('50vp')
  }.backgroundColor(Color.Orange)
  GridCol({ order: 2 }) {
    Row() {
      Text('3')
    }.width('100%').height('50vp')
  }.backgroundColor(Color.Yellow)
  GridCol({ order: 1 }) {
    Row() {
      Text('4')
    }.width('100%').height('50vp')
  }.backgroundColor(Color.Green)
}.margin({left:5, right:5})
.border({ color: Color.Blue, width: 2 })

栅格组件的嵌套使用

栅格组件也可以嵌套使用,完成一些复杂的布局。

以下示例中,栅格把整个空间分为12份。第一层GridRow嵌套GridCol,分为中间大区域以及“footer”区域。第二层GridRow嵌套GridCol,分为“left”和“right”区域。子组件空间按照上一层父组件的空间划分,粉色的区域是屏幕空间的12列,绿色和蓝色的区域是父组件GridCol的12列,依次进行空间的划分。

效果图
在这里插入图片描述
对应代码

GridRow() {
  GridCol({ span: { sm: 12 } }) {
    GridRow() {
      GridCol({ span: { sm: 2 } }) {
        Row() {
          Text('left').fontSize(24)
        }
        .justifyContent(FlexAlign.Center)
        .height('90%')
      }.backgroundColor('#ff41dbaa')

      GridCol({ span: { sm: 10 } }) {
        Row() {
          Text('right').fontSize(24)
        }
        .justifyContent(FlexAlign.Center)
        .height('90%')
      }.backgroundColor('#ff4168db')
    }
    .backgroundColor('#19000000')
    .height('100%')
  }

  GridCol({ span: { sm: 12 } }) {
    Row() {
      Text('footer').width('100%').textAlign(TextAlign.Center)
    }.width('100%').height('10%').backgroundColor(Color.Pink)
  }
}.width('100%').height(300)

上一篇 2.4、相对布局(RelativeContainer)
下一篇 2.6、媒体查询(mediaquery)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值