【Harmony】基础组件应用

这个结构和Flutter阿、Compose阿挺像,顺水行舟,都为生命式UI方式,看下组件吧:


image.png
image.png

1. 组件介绍:

@Entry

代表一个界面的入口,跟Activity声明有点像

@Component

代表是一个UI布局的结构体声明,下面跟结构体
结构体内,组件在build方法内,每个组件都必须实现build方法,是用于定义组件的声明式UI描述

2. 基础组件

image.png

基础组件的声明式使用,参数方式传递,参数既可以用直接输入、也可以来源于资源(当然正常编码都用资源啦,涉及到多语言等自动适配啦)

Text

Column() {
  Text(this.message)//外部文字定义使用
    .fontSize('36.00fp')
    .fontWeight(FontWeight.Bold)
  Text("一段文字")//直接字符串
    .fontColor("#ff00ff")
  Text($r("app.string.EntryAbility_label"))//string资源使用
    .fontColor($r('app.color.main_text_color'))//color资源使用
}

Button

Column() {
  Button("button", { type: ButtonType.Normal })//矩形
  Button("button", { type: ButtonType.Capsule })//默认的胶囊型
}

容器:

image.png

Colume:列容器

接口 参数说明
alignItems 水平对齐:HorizontalAlign枚举,默认Center
justifyContent 垂直对齐:FlexAlign枚举,默认Top
Column() {
  Text('Text1')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)

  Text('Text2')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)

  Text('Text3')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)
}
.width('100%')
.height('100%')
.backgroundColor(Color.Green)
.alignItems(HorizontalAlign.End)
.justifyContent(FlexAlign.End) 

Row:行容器

接口 参数说明
alignItems 垂直对齐:VerticalAlign枚举,默认Center
justifyContent 水平对齐:FlexAlign枚举,默认Start
Row() {
  Text('Text1')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)

  Text('Text2')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)

  Text('Text3')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.White)
    .textAlign(TextAlign.Center)
}
.width('100%')
.height('100%')
.backgroundColor(Color.Green)
.alignItems(VerticalAlign.Center)
.justifyContent(FlexAlign.Start)

Stack:栈容器

堆叠型容器

接口 参数说明
alignContent 对齐:Alignment,默认是Center
Stack(/*{ alignContent: Alignment.TopStart }*/) {
  Text('Text1')
    .width(fp2px(50))
    .height(fp2px(50))
    .backgroundColor(Color.Orange)
    .textAlign(TextAlign.Center)

  Text('Text2')
    .width(fp2px(30))
    .height(fp2px(30))
    .backgroundColor(Color.Red)
    .textAlign(TextAlign.Center)

  Text('Text3')
    .width(fp2px(20))
    .height(fp2px(20))
    .backgroundColor(Color.Yellow)
    .textAlign(TextAlign.Center)
}
.width('100%')
.height('100%')
.backgroundColor(Color.Green)

如下图:后面的覆盖在前面的上面


image.png

练习:静态水果排行榜页面

build() {
  Column() {
    /*顶部标题栏*/
    Row() {
      /*左侧*/
      Row() {
        Image($r('app.media.ic_public_back'))
          .width(22)
          .height(22)
          .margin({ right: 18 })

        Text("排行榜")
          .fontSize(20)
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.Start)
      /*右侧*/

      Row() {
        Image($r('app.media.loading'))
          .width(22)
          .height(22)
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.End) //右对齐
    }
    .width('100%')
    .height(47)
    .padding({ left: 26, right: 26 })
    .margin({ top: 18 })


    /*水果排行榜标题*/
    Row() {
      Text("排名")
        .fontSize(14)
        .width('30%')
        .fontColor('#989a9c')
      Text("种类")
        .fontSize(14)
        .width('50%')
        .fontColor('#989a9c')
      Text("得票数")
        .fontSize(14)
        .width('20%')
        .fontColor('#989a9c')
    }
    .width('90%')
    .padding(15)

    /*水果排行内容*/
    Column() {
      List({space: 10}) {
        ListItem() {
          Row() {
            Text("1")
              .width('30%')
            Text("苹果")
              .width('50%')
            Text("12080")
              .width('20%')
          }
        }

        ListItem() {
          Row() {
            Text("2")
              .width('30%')
            Text("葡萄")
              .width('50%')
            Text("10320")
              .width('20%')
          }
        }

        ListItem() {
          Row() {
            Text("3")
              .width('30%')
            Text("西瓜")
              .width('50%')
            Text("9801")
              .width('20%')
          }
        }
      }
      .width('90%')
      .padding(15)
    }
  }
  .backgroundColor("#F1F3F5")
  .width('100%')
  .height('100%')
}

还是有点像模像样的啦: 布局拆解后,按横向纵向排列布局处理,响应式写法确实帅。就是括号嵌套层级有点多,看起来有点难受,看下效果


image.png

练习:动态渲染数据

静态的每个自己添加在实际使用的时候是不切实际的,肯定是根据数据进行遍历添加。试一下foreach

/**
 * 水果排行数组
 */
fruits: Array<Object> = [{
  id: '1',
  name: "苹果",
  vote: "12080"
}, {
  id: '2',
  name: "葡萄",
  vote: "10320"
}, {
  id: '3',
  name: "西瓜",
  vote: "9801"
}, {
  id: '4',
  name: "香蕉",
  vote: "7546"
}, {
  id: '5',
  name: "菠萝",
  vote: "1208"
}]
/**
 * 水果排行榜页面 静态练习
 */
build() {
  Column() {
    /*顶部标题栏*/
    Row() {
      /*左侧*/
      Row() {
        Image($r('app.media.ic_public_back'))
          .width(22)
          .height(22)
          .margin({ right: 18 })

        Text("排行榜")
          .fontSize(20)
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.Start)
      /*右侧*/

      Row() {
        Image($r('app.media.loading'))
          .width(22)
          .height(22)
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.End) //右对齐
    }
    .width('100%')
    .height(47)
    .padding({ left: 26, right: 26 })
    .margin({ top: 18 })


    /*水果排行榜标题*/
    Row() {
      Text("排名")
        .fontSize(14)
        .width('30%')
        .fontColor('#989a9c')
      Text("种类")
        .fontSize(14)
        .width('50%')
        .fontColor('#989a9c')
      Text("得票数")
        .fontSize(14)
        .width('20%')
        .fontColor('#989a9c')
    }
    .width('90%')
    .padding(15)

    /*水果排行内容*/
    Column() {
      List({ space: 10 }) {
        /*采用对象来*/
        ForEach(this.fruits, (item) => {
          ListItem() {
            Row() {
              if (item.id <= 3) {
                Column() {
                  Row() {
                    Text(item.id)
                      .fontSize(14)
                      .fontColor(Color.White)
                  }
                  .width(24)
                  .height(24)
                  .borderRadius(18)
                  .backgroundColor(Color.Blue)
                  .justifyContent(FlexAlign.Center) //水平对齐
                }
                .width('30%')
                .alignItems(HorizontalAlign.Start) //水平对齐
              }
              else {
                Column() {
                  Text(item.id)
                    .fontSize(14)
                    .textAlign(TextAlign.Center)//文字居中,对应上面的Row的水平居中
                    .width(24)
                    .height(24)
                }
                .width('30%')
                .alignItems(HorizontalAlign.Start) //水平对齐
              }
              Text(item.name)
                .width('50%')
              Text(item.vote)
                .width('20%')
            }
          }
        })
      }
      .width('90%')
      .padding(15)
    }
  }
  .backgroundColor("#F1F3F5")
  .width('100%')
  .height('100%')
}

主要关键修改是增加了一个数组Array,然后使用foreach来遍历产生ListItem
效果:


image.png

组件事件修饰器

  • onclick事件
Button("点击")
  .onClick(event => {
    this.count++
    console.log("点击事件触发:" + this.count)
  })
  • @State修饰器
    我们希望对应值修改后能体现在UI上则可以用改状态修饰器修饰对应的值定义
@State count: number = 0

#在count变化后, Text控件自动会刷新
Text("文字" + this.count)
  • @Link 修饰器
    通过@link修饰器可以将子组件的变量与父组件的变量建立关联关系

  • @Builder 修饰器
    可以将组件抽象提取为函数,重复UI利用,在build里调用

3. 模块化

数据与页面分离:

将页面和数据分离开到不同文件,可以方便管理代码
(1)另建ets文件
(2)将类类型、静态数据声明放到独立的ets中
(3)将类型、数据声明加export,让外部可以使用
(4)在需要使用的地方,使用import ··· from 方式来引用
如下:

image.png

布局分离:
一个etx中有多个view时,可以拆分成不同的自定义view,或者说组合多个自定义view到布局:

  1. 新建etx文件,使用 子组件方式,修饰结构体
@Component
export struct TitleComponent{
    //这样实现build方法即可
  build(){
  }
}
  1. 对于子组件中有变量要与父控件保持一致时,使用前面说的@Link,建议关联父控件值时,名称也要与父控件声明一样(不一样也没啥,传参绑定)
@Component
struct TitleComponent{
    //关联父控件中传入变量的值
  @Link isSwitchFruits:boolean
  build(){
    Row() {
      //左侧
      Row() {
        Image($r('app.media.ic_public_back'))
          .width(22)
          .height(22)
          .margin({ right: 18 })

        Text("排行榜")
          .fontSize(20)
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.Start)

      //右侧

      Row() {
        Image($r('app.media.loading'))
          .width(22)
          .height(22)
          .onClick(() => {
            this.isSwitchFruits = !this.isSwitchFruits
          })
      }
      .width('50%')
      .height(100)
      .justifyContent(FlexAlign.End) //右对齐
    }
    .width('100%')
    .height(47)
    .padding({ left: 26, right: 26 })
    .margin({ top: 18 })
  }
}

使用时:

//引用
import {TitleComponent} from '../view/TitleComponent'

//使用传参方式绑定子控件内容,左侧为子控件变量,右侧用取值方式取当前要传的值
TitleComponent({sisSwitchFruits:$isSwitchFruits})
  1. 对于可以重复利用的使用@Builder
/**
 * 水果排行榜页面 静态练习
 */
build() {
  Column() {
    /*顶部标题栏*/
    //使用传参方式绑定子空间内容
    TitleComponent({ sisSwitchFruits: $isSwitchFruits })

    //水果排行榜标题
    ListTitleComponent()

    /*水果排行内容*/
    this.RankList()
  }
  .backgroundColor("#F1F3F5")
  .width('100%')
  .height('100%')
}

/**
 * UI做builder声明方式,UI组件作为函数使用,被build内调用
 */
@Builder RankList() {
  Column() {
    List({ space: 10 }) {
      //采用对象来
      ForEach(this.isSwitchFruits ? this.fruits : this.fruitsData2, (item: Fruit) => {
        ListItem() {
          ItemComponent({
            itemId: item.id,
            itemName: item.name,
            itemVote: item.vote
          })
        }
      })
    }
    .width('90%')
    .padding(15)
  }
}

至此,简单的布局就可以啦:
1.简单的组件使用布局:Column、Row、Stack、ListItem
2.状态改变等:@State 来关注变化、@link 来子控件关联父控件变量、
3.将view独立:独立函数@Builder抽离view到函数,独立文件 @Component 和 import、from

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

推荐阅读更多精彩内容