17、鸿蒙/布局/栅格布局 (GridRow/GridCol)

概述

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

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

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

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

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

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

栅格容器GridRow

栅格系统断点

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

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


GridRow.png

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

GridRow.png
  • 针对断点位置,开发者根据实际使用场景,通过一个单调递增数组设置。由于breakpoints最多支持六个断点,单调递增数组长度最大为5。
breakpoints: {value: ['100vp', '200vp']}

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

breakpoints: {value: ['320vp', '520vp', '840vp', '1080vp']}
  • 栅格系统通过监听窗口或容器的尺寸变化进行断点,通过reference设置断点切换参考物。 考虑到应用可能以非全屏窗口的形式显示,以应用窗口宽度为参照物更为通用。
    例如,使用栅格的默认列数12列,通过断点设置将应用宽度分成六个区间,在各区间中,每个栅格子元素占用的列数均不同。
GridRow({
      breakpoints: {
        value: ['200vp', '300vp', '400vp', '500vp', '600vp'],
        reference: BreakpointsReference.WindowSize
      }
    }){
      GridCol({
        span: {
          xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
          sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
          md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
          lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
          xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
          xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
        }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
      GridCol({
          span: {
            xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
            sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
            md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
            lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
            xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
            xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
          }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
      GridCol({
          span: {
            xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
            sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
            md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
            lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
            xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
            xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
          }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
      GridCol({
        span: {
          xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
          sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
          md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
          lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
          xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
          xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
        }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
      GridCol({
        span: {
          xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
          sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
          md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
          lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
          xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
          xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
        }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
      GridCol({
        span: {
          xs: 2, // 在最小宽度类型设备上,栅格子组件占据的栅格容器2列。
          sm: 3, // 在小宽度类型设备上,栅格子组件占据的栅格容器3列。
          md: 4, // 在中等宽度类型设备上,栅格子组件占据的栅格容器4列。
          lg: 6, // 在大宽度类型设备上,栅格子组件占据的栅格容器6列。
          xl: 8, // 在特大宽度类型设备上,栅格子组件占据的栅格容器8列。
          xxl: 12 // 在超大宽度类型设备上,栅格子组件占据的栅格容器12列。
        }
      }){
        Row(){
          Text('新闻')
        }.backgroundColor(Color.Red).width(100).height(100)
      }
}

布局的总列数

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

  • columns默认值为12,即在未设置columns时,任何断点下,栅格布局被分成12列。
@State bgColor: Color[] = [
    Color.Red,
    Color.Yellow,
    Color.Gray,
    Color.Green,
    Color.Black,
    Color.Blue,
    Color.Pink,
    Color.Orange,
    Color.Pink,
    Color.Brown,
    Color.Gray,
    Color.Black,
];
...
GridRow(){
      ForEach(this.bgColor, (item: Color, index: number)=>{
        GridCol(){
          Row(){
            Text(index.toString()).textAlign(TextAlign.Center)
          }.width(100).height(100).backgroundColor(item)
        }
      })
}
  • 当columns为自定义值,栅格布局在任何尺寸设备下都被分为columns列。下面分别设置栅格布局列数为4,子元素默认占一列,效果如下:
GridRow({ columns: 4 }){
      ForEach(this.bgColor, (item: Color, index: number)=>{
        GridCol(){
          Row(){
            Text(index.toString()).textAlign(TextAlign.Center)
          }.width(100).height(100).backgroundColor(item)
        }
      })
}
  • 当columns类型为GridRowColumnOption时,支持下面六种不同尺寸(xs, sm, md, lg, xl, xxl)设备的总列数设置,各个尺寸下数值可不同。
 GridRow({ columns: { sm: 2, md: 4, lg: 8 }, breakpoints: { value: ['200vp', '300vp', '400vp', '500vp', '600vp'] } }){
        ForEach(this.bgColor, (item: Color, index: number)=>{
          GridCol(){
            Row(){
              Text(index.toString()).textAlign(TextAlign.Center)
                .width('100%')
            }.width('100%').height(100).backgroundColor(item)
          }
        })
}

若只设置sm, md的栅格总列数,则较小的尺寸使用默认columns值12,较大的尺寸使用前一个尺寸的columns。这里只设置sm: 2, md: 4, lg: 8,则较小尺寸的xs:12,较大尺寸的参照md的设置,lg:8, xl:8, xxl:8

排列方向

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

  • 子组件默认从左往右排列。
GridRow({ direction: GridRowDirection.Row }){}
  • 子组件从右往左排列。
GridRow({ direction: GridRowDirection.RowReverse }){}

子组件间距

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

  • 当gutter类型为number时,同时设置栅格子组件间水平和垂直方向边距且相等。下例中,设置子组件水平与垂直方向距离相邻元素的间距为10。
 GridRow({ gutter: 10 }){}
  • 当gutter类型为GutterOption时,单独设置栅格子组件水平垂直边距,x属性为水平方向间距,y为垂直方向间距。
GridRow({ gutter: { x: 20, y: 50 } }){}

子组件GridCol

GridCol组件作为GridRow组件的子组件,通过给GridCol传参或者设置属性两种方式,设置span(占用列数),offset(偏移列数),order(元素序号)的值。

  • 设置span。
GridCol({ span: 2 }){}
GridCol({ span: { xs: 1, sm: 2, md: 3, lg: 4 } }){}
GridCol(){}.span(2)
  • 设置offset。
GridCol({ offset: 2 }){}
GridCol({ offset: { xs: 2, sm: 2, md: 2, lg: 2 } }){}
  • 设置order。
GridCol({ order: 2 }){}
GridCol({ order: { xs: 1, sm: 2, md: 3, lg: 4 } }){}
GridCol(){}.order(2)

span

子组件占栅格布局的列数,决定了子组件的宽度,默认为1。

  • 当类型为number时,子组件在所有尺寸设备下占用的列数相同。
// 一行2个
GridCol({ span: 2 })
  • 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件所占列数设置,各个尺寸下数值可不同。
 GridCol({
            span: { xs: 1, sm: 2, md: 3, lg: 4 }
})

offset

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

  • 当类型为number时,子组件偏移相同列数
 GridCol({
  offset: 1
})
  • 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件所占列数设置,各个尺寸下数值可不同。
GridCol({ offset: { xs: 1, sm: 2, md: 3, lg: 4 } })

order

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

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

  • 当类型为number时,子组件在任何尺寸下排序次序一致。
GridRow({
      breakpoints: {
        value: ['200vp', '300vp', '400vp', '500vp', '600vp'],
        reference: BreakpointsReference.WindowSize
      }
    }){
      GridCol({
        order: 3
      }){
        Row(){
          Text('新闻1')
        }.backgroundColor(Color.Red).width('100%').height(100)
      }
      GridCol({
        order: 2
      }){
        Row(){
          Text('新闻2')
        }.backgroundColor(Color.Red).width('100%').height(100)
      }
      GridCol({

      }){
        Row(){
          Text('新闻3')
        }.backgroundColor(Color.Red).width('100%').height(100)
      }

}
  • 当类型为GridColColumnOption时,支持六种不同尺寸(xs, sm, md, lg, xl, xxl)设备中子组件排序次序设置。在xs设备中,子组件排列顺序为1234;sm为2341,md为3412,lg为2431。
GridCol({ order: { xs:1, sm:5, md:3, lg:7}})

综上所述,栅格组件提供了丰富的自定义能力,功能异常灵活和强大。只需要明确栅格在不同断点下的Columns、Margin、Gutter及span等参数,即可确定最终布局,无需关心具体的设备类型及设备状态(如横竖屏)等。

©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,284评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,115评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,614评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,671评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,699评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,562评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,309评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,223评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,668评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,859评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,981评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,705评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,310评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,904评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,023评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,146评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,933评论 2 355

推荐阅读更多精彩内容