HarmonyOS鸿蒙开发学习:鸿蒙基础-基础环境-ArkTS-组件-样式

news2025/1/10 11:37:33

鸿蒙基础-基础环境-ArkTS-组件-样式

DevEcoStudio编辑器下载链接
链接: https://pan.baidu.com/s/18C9i35YPh4GsHpbSif8KQw?pwd=d6e9 提取码: d6e9

安装教程

下载对应的版本

  • windows
  • mac英特尔
  • mac Arm

API12 的编辑器
API12的模拟器(mac英特尔安装API11)

  • Mac Arm芯片版本安装

解压编辑器版本
image.png

image.png

将左侧内容拖入

image.png

点击打开

  • 打开项目的配置项

image.png
image.png

新建一个路径 HarmonyOS-SDK
将资源下载包中的sdk的内容进行解压,放入到刚刚新建的目录
image.png
将设置中的安装路径改变成新建的目录,最终点击确认,sdk就安装好了

image.png

打开原有项目发现
image.png

一个配置文件的依赖,线上没有提供云下载地址,只能采用离线的版本 。hvigor

下载资源包中,还有提供一个dependencies, 这就是我们需要替换的版本
image.png
image.png
image.png
image.png

安装模拟器

  • 需要实名登录

image.png

  • 安装插件

image.png
image.png

internal-plugin-SNAPSHOT.zip
如果大家的mac安装不上 这个插件,可以采用文档中提供的,下载之后安装即可

image.png

下载资源中的模拟器,拷贝到sdk的目录
注意: 模拟器的名字是小写的
image.png

  • 创建镜像目录

注意
** 如果用的是API12的模拟器,需要把镜像放入到HarmonyOS-NEXT-DB1 目录下。**
** 如果用的是API11的模拟器,需要把镜像放入到HarmonyOS-NEXT-DP2 目录下。**

system-image
HarmonyOS-NEXT-DB1
phone_arm/phone_x86
HarmonyOS-NEXT-DP2
phone_arm/phone_x86

image.png

Windows安装模拟器

  • DevEcoStudio快捷键

起步

起步-鸿蒙简介

  • HarmonyOS 是新一代的智能终端操作系统,为不同设备的智能化、互联与协同提供了统一的语言。带来简洁,流畅,连续,安全可靠的全场景交互体验。

历程:

时间事件
2019HarmonyOS 1.0,华为在东莞举行华为开发者大会,正式发布操作系统鸿蒙 OS,主要用于物联网
2020HarmonyOS 2.0,基于开源项目 OpenHarmony 开发的面向多种全场景智能设备的商用版本
2021HarmonyOS 3.0,先后优化游戏流畅度、地图三维体验、系统安全,另外系统的稳定性也得到了增强
2023.2HarmonyOS 3.1,系统纯净能力进一步提升,对后台弹窗、 隐藏应用、后台跳转等情况
2023.7华为 Mate 50 系列手机获推 HarmonyOS 4.0
2024HarmonyOS Next 开发者预览版发布,将不在兼容安卓应用

起步-DevEco Studio

image.png

安装 DevEco Studio 编辑器

DevEcoStudio编辑器下载链接
链接: https://pan.baidu.com/s/1TyrmbTkrOEsTB8HcaMR4og?pwd=fjjw 提取码: fjjw

  1. 安装:DevEco Studio → 一路 Next(只演示windows)

image.png
image.png
image.png
image.png
image.png
image.png

  1. 运行: 选择not import System Img

这里最好别选在C盘

  • 下一步

image.png
image.png

  • 下载sdk

image.png

  • 安装完成

image.png

  • 安装完成

image.png

  • 创建一个新项目

image.png

  • 填写信息

image.png
等待创建完成

  • 安装中文插件(windows)

image.png

  • Mac的选择

image.png

  • 选择Plugins

image.png

  • 点OK重启

image.png

  • 看到效果

image.png

起步-认识工作区

image.png

  • 通过左侧目录找到对应的应用文件,在编辑区进行代码编写,在右侧看预览效果
  • 连按两下shift,可以快速寻找文件

起步-如何排错

image.png

写代码时,我们会经常遇到这种情况,右侧出现不能够开启预览器的提示,让我们打开预览器日志看错误

  • 解决该问题的思路 1. 按照编辑器提示的,打开预览器日志

image.png

  • 如果是明确的语法错误或者api错误,编辑器会指出我们代码的行数,我们可以点击提示的代码行,直接跳到对应位置,直接检查代码的问题

image.png

  • 跳到对应的位置

image.png

解决思路2: 如果当前文件不多,可以点开你创建的所有文件,查看文件中是否有报错的地方,文件报错,在右侧以及代码区会有明显的报错提示
image.png

解决思路3: 通过统一构建,暴露哪些文件及代码无法编译通过
image.png

image.png

如何刷新看效果

预览器是有热更新的

  • 原则上写完右侧自动更新
  • 如果不自动刷新- 语法错了,编辑器卡住了

image.png

  • 直接点击刷新按钮

image.png

  • 刷新只能针对 带有@Entry和@Preview的文件,否则无法看到效果

起步-审查和多设备预览

效果预览方法:
info

  • Preview(预览器)
  • 本地模拟器(只有Mac(ARM)芯片)
  • 远程模拟器
  • 云手机(需要审核及申请,暂无消息)
  • 本地真机(Meta60-Meta60Pro X5或者是装载OpenHarmony的工程机(价格较低,但是依然存在一些问题))

danger
只有装了Next预览版系统的手机才可以进行真机调试和预览,目前Next预览器系统装机量有限,需要申请和审核,小道消息Q2季度会进一步扩大开放名额

  • 预览器的多设备预览

image.png
danger
2in1的意思是 平板电脑二合一的状态

未命名.gif

  • 审查元素

image.png

image.png

  • 真机预览调试

image.png
info

  • 刷了Next预览版本的手机插上数据线,连接到电脑之后(注意:这里需要开启手机的usb调试,设置中搜索usb调试,打开,并且处理开发者模式,打开手机设置-关于手机,连续点击手机系统的版本号,直到出现您已处于开发者模式的提示未知)

  • 运行
    info
    点击绿色三角- 运行到真机

  • debugger模式
    info
    点击小虫子-断点调试到真机(4.0版本中真机不支持断点调试-next真机支持断点调试)

起步-工程目录结构

image.png

info
我们详解下目录结构

  • AppScope > app.json5:应用的全局配置信息。

  • **entry:**应用/服务模块,编译构建生成一个HAP。

    • src > main > ets:用于存放ArkTS源码。
    • src > main > ets > entryability:应用/服务的入口。
    • src > main > ets > pages:应用/服务包含的页面。
    • **src > main > resources:**用于存放应用/服务模块所用到的资源文件,如图形、多媒体、字符串、布局文件等。
    • src > main > module.json5:Stage模型模块配置文件,主要包含HAP的配置信息、应用在具体设备上的配置信息以及应用的全局配置信息。
  • **entry > build-profile.json5:**当前的模块信息、编译信息配置项,包括buildOption、targets配置等。

  • entry > hvigorfile.ts:模块级编译构建任务脚本。

  • entry >oh-package.json5:配置三方包声明文件的入口及包名。

  • oh_modules:用于存放三方库依赖信息,包含应用/服务所依赖的第三方库文件。关于原npm工程适配ohpm操作,请参考历史工程适配OHPM包管理。

  • **build-profile.json5:**应用级配置信息,包括签名、产品配置等。

  • **hvigorfile.ts:**应用级编译构建任务脚本。

  • 资源目录结构

image.png

什么是Stage模型

info
通过上层的目录结构我们要分析出如下关系
State
-Module(模块-对应Hap包)
-ability(应用服务入口)
-pages(页面)
-component(自定义组件)
-resources(资源)

image.png

应用模型Stage&FA

鸿蒙的战略 兼容安卓-把java + 前端拉入到开发阵容中-FA模型

官方介绍

  • 应用模型是HarmonyOS为开发者提供的应用程序所需能力的抽象提炼,它提供了应用程序必备的组件和运行机制。有了应用模型,开发者可以基于一套统一的模型进行应用开发,使应用开发更简单、高效。

换言之- 应用模型是鸿蒙开发一切的基础,因为只有基于该应用模型我们才可以开发对应的应用和业务。

应用模型包含几个要素**应用组件-**应用进程-应用线程-应用任务管理-应用配置文件

提问:应用模型是只有一个吗?
回答:鸿蒙前后推出了两种应用模型- FA(Feature Ability),Stage,目前FA已经不再主推

image.png

  • HarmonyOS Next也将Stage模型作为主推模型,所以我们本次训练营将学习Stage模型相关的应用开发能力。

下面是官方的Stage模型概念图
image.png

总结:应用模型是开发鸿蒙应用的基础底座,但是鸿蒙先后推出了FA和Stage,鸿蒙4.0和鸿蒙Next都将Stage作为主推方向,所以我们主要基于Stage模型来学习和开发我们目前的应用

什么是UIAbility-(界面能力组件)

从上一个小节中,我们发现Stage模型提到了UIAbility组件包含UI界面绘制,主要和用户交互。

  • UIAbility组件是一种包含UI界面的应用组件,主要用于和用户交互。

官网介绍-UIAbility是系统调度的基本单元,可以给应用提供绘制界面的窗口。
info
UIAbility的设计理念:

  1. 原生支持应用组件级的跨端迁移和多端协同。
  2. 支持多设备和多窗口形态。

UIAbility组件是系统调度的基本单元,为应用提供绘制界面的窗口。一个应用可以包含一个或多个UIAbility组件。例如,在支付应用中,可以将入口功能和收付款功能分别配置为独立的UIAbility。

每一个UIAbility组件实例都会在最近任务列表中显示一个对应的任务。

image.png

对于开发者而言,可以根据具体场景选择单个还是多个UIAbility,划分建议如下:

  • 如果开发者希望在任务视图中看到一个任务,则建议使用一个UIAbility,多个页面的方式。

  • 如果开发者希望在任务视图中看到多个任务,或者需要同时开启多个窗口,则建议使用多个UIAbility开发不同的模块功能。

  • 场景- 支付/小程序/鉴权

为使应用能够正常使用UIAbility,需要在module.json5配置文件的abilities标签中声明UIAbility的名称、入口、标签等相关信息。

{
  "module": {
    ...
    "abilities": [
      {
        "name": "EntryAbility", // UIAbility组件的名称
        "srcEntry": "./ets/entryability/EntryAbility.ets", // UIAbility组件的代码路径
        "description": "$string:EntryAbility_desc", // UIAbility组件的描述信息
        "icon": "$media:icon", // UIAbility组件的图标
        "label": "$string:EntryAbility_label", // UIAbility组件的标签
        "startWindowIcon": "$media:icon", // UIAbility组件启动页面图标资源文件的索引
        "startWindowBackground": "$color:start_window_background", // UIAbility组件启动页面背景颜色资源文件的索引
        ...
      }
    ]
  }
}

warning
上述文件不用我们手动填写,我们新建ability的时候,会自动填入

组件基础

组件-什么是ArkTS

info
ArkTS提供了语言运行时相关能力
ArkTS是HarmonyOS优选的主力应用开发语言。ArkTS围绕应用开发在TypeScript(简称TS)生态基础上做了进一步扩展,保持了TS的基本风格,同时通过规范定义强化开发期静态检查和分析,提升程序执行稳定性和性能。

特别注意:尤其是学过鸿蒙4.0的,Next版本极大增强了TS语言中动态类型的限制,几乎不再支持动态类型

从API version 10开始,ArkTS进一步通过规范强化静态检查和分析,对比标准TS的差异可以参考从TypeScript到ArkTS的适配规则:

  • 强制使用静态类型:静态类型是ArkTS最重要的特性之一。如果使用静态类型,那么程序中变量的类型就是确定的。同时,由于所有类型在程序实际运行前都是已知的,编译器可以验证代码的正确性,从而减少运行时的类型检查,有助于性能提升。
  • 禁止在运行时改变对象布局:为实现最大性能,ArkTS要求在程序执行期间不能更改对象布局。
  • 限制运算符语义:为获得更好的性能并鼓励开发者编写更清晰的代码,ArkTS限制了一些运算符的语义。比如,一元加法运算符只能作用于数字,不能用于其他类型的变量。
  • 不支持Structural typing:对Structural typing的支持需要在语言、编译器和运行时进行大量的考虑和仔细的实现,当前ArkTS不支持该特性。根据实际场景的需求和反馈,我们后续会重新考虑。
  • 由于文档权限的限制,这里有一份openHarmony的 上的Next版本的ts类型的迁移说明 地址 (内容基本一致)

openHarmony 和HarmonyOS的关系和区别
OpenHarmony是指鸿蒙(HarmonyOS)的开源版本,由华为官方开源,遵循Apache 2.0许可证。OpenHarmony包含了HarmonyOS的核心代码库,但并不包括商业版HarmonyOS中的一些特有功能和技术。
HarmonyOS是华为自主研发的分布式操作系统,旨在打造一个全场景、全终端的智能生态系统。它支持多种设备类型,包括手机、平板、智能手表、智能音箱、电视、汽车、家居等,可以实现设备之间的协同工作和资源共享。
OpenHarmony和HarmonyOS的关系就像是Java和Android的关系一样。可以将OpenHarmony看作是HarmonyOS的底层技术和代码库的一部分,而HarmonyOS则是在此基础上进行了更多的开发和扩展,添加了更多的功能和应用。相比之下,OpenHarmony更加开放和自由,可以被开发者用于构建各种类型的设备和应用,而HarmonyOS则更加注重整合、统一和优化整个生态系统。

image.png

  • ArkTS的特性-扩展能力
  1. **基本语法 **
    • 定义声明式UI、自定义组件、动态扩展UI元素;
    • 提供ArkUI系统组件,提供组件事件、方法、属性;
    • 共同构成 UI 开发主体
  2. **状态管理 **
    • 组件状态、组件数据共享、应用数据共享、设备共享;
  3. **渲染控制 **
    • 条件渲染、循环渲染、数据懒加载;

ArkTS以声明方式组合和扩展组件来描述应用程序的UI,同时还提供了基本的属性、事件和子组件配置方法,帮助开发者实现应用交互逻辑。

  • 命令式UI- document.createElement(“div”)-
  • 声明式UI

image.png

下图是关于ArtTS的一个整体的应用架构(官网)
image.png

info
总结:

  • AktTS提供原有前端范畴内的一切TypeScript和JavaScript的类型及方法支持
  • Next版本加了很多限制-要看迁移指南
  • ArkTS采用声明式UI的方法来绘制页面,设置属性,绑定事件

ArkTS重点迁移说明

  • 对象字面量不能用于类型声明

image.png

  • 不支持in操作符

image.png

  • 不支持解构赋值

image.png

  • 不支持通过索引访问字段

image.png
image.png

有一种情况下,可以通过索引访问字段,即传入的参数类型为object, 但是传入时必须用class的类型传入,如下

@Entry
@Component
struct Index {
  @State message: string = 'Hello World1';
  test (obj: object) {
   console.log(Object.keys(obj).map(item => `${item}=${obj[item]}`).toString())
  }
  aboutToAppear(): void {
    let o = new TestObj()
    this.test(o)
  }
  build() {
    Row() {
      Column() {
        Text(this.message)
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
      }
      .width('100%')
    }
    .height('100%')
  }
}
class TestObj {
  a: number = 0
  b: string = "123"
}

以上场景一般用作请求封装时,无法确定侦测传入参数的类型时使用

  • 部分支持延展运算符

image.png

  • 不支持bind/call/apply改变this指向

this始终指向调用者。谁调用这个方法,this指向谁

image.png

基础-组件结构

info
接下来,我们来解析我们的UI的结构

image.png

ArkTS通过装饰器 @Component@Entry 装饰 struct 关键字声明的数据结构,构成一个自定义组件。
自定义组件中提供了一个 build 函数,开发者需在该函数内以链式调用的方式进行基本的 UI 描述,UI 描述的方法请参考 UI 描述规范。

  • struct-自定义组件基于struct实现

要想实现一段UI的描述,必须使用struct关键字来声明- 注意不能有继承关系-组件名不能系统组件名重名

语法: struct 组件名 {}

@Component
struct Index {
  
}
@CustomDialog 
struct Index2 {
  
}

info
struct关键字声明的UI描述-必须被@Component或者@CustomDialog修饰

  • Component修饰符

Component装饰器只能修饰struct关键字声明的结构,被修饰后的struct具备组件的描述(渲染)能力

  • build函数

用于定义组件的UI描述,一个struct结构必须实现build函数

@Component
struct MyComponent {
  build() {
  }
}

info
注意:
1.build函数是组件(Component)必须提供以及实现的一个函数,build函数可以没有内容,如果有的话,必须有且只有一个容器组件(可以放置子组件的组件)- 只有entry里面有限制- component里面没有限制
2.Component的组件build函数 可以放没有子组件的组件
image.png

  • 常见容器组件- Flex-Column-Row-List-Grid-Panel

  • entry修饰符

entry将自定义组件定义为UI页面的入口,也就是我们原来前端常说的一个页面,最多可以使用entry装饰一个自定义组件(在一个ets文件中)-如下面的代码就是不被允许的

@Entry
@Component
struct Index {

  build() {
  
  }
}
@Entry
@Component
struct Index2 {
  build() {

  }
}

info
entry修饰的组件,最终会被注册,具体文件位置-main/resources/base/profile/main_pages.json

  1. 自动注册-新建组件时,采用新建Page的方式
  2. 手动注册-新建一个ets文件,自己在main_pages.json中手动添加路径

注意:
如果你手动删除了某一个带entry的组件,你需要手动去main_page中去删除该路径,否则编译会报错

  • 组件复用

在很多情况下,由于业务的复杂度,我们经常会将一个大的业务拆成若干个组件,进行组装,这里我们非常灵活的复用组件,比如

image.png

info

  • 我们可以把上图抽象成三个组件- Header- Main- Footer

代码

import { MeiTuanFooter } from '../views/MeiTuan/MeiTuanFooter';
import { MeiTuanHeader } from '../views/MeiTuan/MeiTuanHeader';
import { MeiTuanMain } from '../views/MeiTuan/MeiTuanMain';

@Entry
@Component
struct MeiTuan {
  @State message: string = 'Hello World';

  build() {
    Column() {
      MeiTuanHeader()  // 60
      MeiTuanMain()
        .layoutWeight(1)
        .backgroundColor(Color.Green)
      MeiTuanFooter()  // 60
    }.height('100%')
  }
}

@Preview
@Component
struct MeiTuanHeader {
  build() {
    Row () {
      Text("美团头部")
        .width('100%')
        .textAlign(TextAlign.Center)
    }
    .width('100%')
    .height(60)
    .backgroundColor(Color.Pink)
  }
}
export { MeiTuanHeader }
@Component
struct MeiTuanMain {
  build() {
    Row() {
      Text("美团中部")
        .fontColor(Color.Red)
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
  }
}

export { MeiTuanMain }

@Component
struct MeiTuanFooter {
  build() {
    Row () {
      Text("美团底部")
        .fontColor(Color.White)
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height(60)
    .backgroundColor(Color.Blue)
  }
}
export  { MeiTuanFooter  }

image.png

info
总结:

  • 一个UI描述必须使用struct来声明,不能继承
  • struct必须被Component或者CustomDialog修饰
  • struct必须实现build方法,build方法可以没有元素,但是有的话有且只有一个可容纳子组件的容器组件(entry修饰的组件)
  • entry修饰符表示该组件是页面级组件,一个文件中只允许修饰一个struct组件
  • 采用分拆组件的形式可以有效解解耦我们的业务

基础-系统组件(ArkUI)

ArkUI: Ability Kit在UIAbility组件可以使用ArkUI提供的组件、事件、动效、状态管理等能力。
这里所指的Kit其实是Next版本中,鸿蒙将各个能力集进行了统一的分类

ArkUI将组件大概分为这么几类

  • 基础组件
  • 容器组件
  • 媒体组件(只有一个Video)
  • 绘制组件
  • 画布组件
  • 高级组件 额外新增
  • 安全组件 额外新增

大家关心的地图功能在Next版本中以API形式呈现
image.png

基本组件使用

  • Text 文本组件-(Span子组件 ImageSpan组件)
  • Column 列组件,纵向排列,Flex布局主轴是Y (任意子组件)
  • Row 行组件,横向排列,Flex布局主轴是X (任意子组件)
  • Flex 以弹性方式布局子组件的容器组件。(存在二次布局,官方推荐有性能要求,使用Column和Row代替) (任意子组件)
  • Button 按钮组件 (单子组件)
  • TextInput 输入框组件 (无子组件)
  • Image (无子组件)
  • Button (单个子组件)
  • List (仅支持ListItem子组件)
  • Scroll (仅支持单个子组件)
  • Stack(堆叠容器,子组件按照顺序依次入栈,后一个子组件覆盖前一个子组件)
  • Grid(网格容器,由“行”和“列”分割的单元格所组成,通过指定“项目”所在的单元格做出各种各样的布局。仅支持GridItem组件)
  • GridRow(栅格容器组件,仅可以和栅格子组件(GridCol)在栅格布局场景中使用。)
  • GirdCol(栅格子组件,必须作为栅格容器组件(GridRow)的子组件使用。)

组件使用
info

  • 使用组件采用 **组件名() **的语法
  • 有构造参数采用 **组件名(参数)**的语法
  • 组件里放置子组件采用 **组件名() { 子组件的语法 } **的链式语法
  • 组件设置属性采用 **组件名().width().height() **的语法
  • 组件又有属性又有子组件采用 **组件名(){ … 子组件 }.width().height() **的语法

基础布局

横向布局-采用Row
百分比说明: 鸿蒙的里面的百分比指的是相对当前父容器,并不是当前的手机屏幕
在写宽度高度时,直接写数字默认单位为vp虚拟像素,屏幕会进行适配。

image.png

Row组件默认情况下,子组件内容会垂直方向居中-** 内容超出不会换行**

@Entry
@Component
struct ComponentCase {
  @State message: string = 'Hello World'

  build() {
    Column() {
      // css 支持调整布局
      Row({ space: 15 }) {
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Pink)
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Red)
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Blue)
      }
      .width('100%')
      // .justifyContent(FlexAlign.Start)

      .justifyContent(FlexAlign.Center)

      // .justifyContent(FlexAlign.End)
      // .justifyContent(FlexAlign.SpaceBetween)
      // .justifyContent(FlexAlign.SpaceAround)
      // .justifyContent(FlexAlign.SpaceEvenly)
    }
    .width('100%')
    .height('100%')
  }
}

纵向布局

image.png

@Entry
@Component
struct ComponentCase {
  @State message: string = 'Hello World'

  build() {
    Column({ space: 10 }) {
      // css 支持调整布局
      Row({ space: 15 }) {
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Pink)
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Red)
        Column()
          .width(100)
          .height(200)
          .backgroundColor(Color.Blue)
      }
      .width('100%')
      // .justifyContent(FlexAlign.Start)
      .justifyContent(FlexAlign.Center)

      Column({ space: 15 }) {
        Column()
          .width(200)
          .height(100)
          .backgroundColor(Color.Pink)
        Column()
          .width(200)
          .height(100)
          .backgroundColor(Color.Red)
        Column()
          .width(200)
          .height(100)
          .backgroundColor(Color.Blue)
      }
      .width('100%')
      .justifyContent(FlexAlign.Center)

    }
    .width('100%')
    .height('100%')
  }
}

自定义组件应用

@Entry
@Component
struct Layout {

  build() {
    Column() {
      RowCase()
      ColumnCase()
    }.height('100%').backgroundColor(Color.Grey)
  }
}

@Component
struct RowCase {
  build() {
    Row() {
      Column().height(150).width(100).backgroundColor(Color.Pink)
      Column().height(150).width(100).backgroundColor(Color.Red)
      Column().height(150).width(100).backgroundColor(Color.Blue)
    }
    .width('100%')
    .alignItems(VerticalAlign.Top)
    .justifyContent(FlexAlign.SpaceAround)
  }
}

@Component
struct ColumnCase {
  build() {
    Column() {
      Column().height(100).width(150).backgroundColor(Color.Pink)
      Column().height(100).width(150).backgroundColor(Color.Red)
      Column().height(100).width(150).backgroundColor(Color.Blue)
    }.height(400).width('100%').justifyContent(FlexAlign.SpaceEvenly)
  }
}
  • 百度的小案例

image.png

@Entry
@Component
struct Baidu {

  build() {
    Column({ space: 20 }) {
      Image("https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png")
        .width(160)
      Row() {
        TextInput()
          .borderRadius({
            topLeft: 6,
            bottomLeft: 6
          })
          .height(40)
          .layoutWeight(1)
          .backgroundColor(Color.White)
          .border({
             color: "#c4c7ce",
             width: 2
          })
        Button("百度一下")
          .type(ButtonType.Normal)
          .backgroundColor("#516aee")
          .padding({
            left: 10,
            right: 10,
            top: 6,
            bottom: 6
          })
          .translate({
            x: -2
          })
          .borderRadius({
            topRight: 6,
            bottomRight: 6
          })
      }
      .padding({
        left: 10,
        right: 10
      })
      .width('100%')
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
  }
}

warning
Row 和Column的布局方式成为线性布局- 不是横向排列就是纵向排列

  • 线性布局中永远不会产生换行
  • 均不支持出现滚动条
  • 横向排列的垂直居中,总行排列的水平居中
  • 主轴-排列方向的轴
  • 侧轴-排列方向垂直的轴

堆叠布局

image.png
info
只要在Stack内部-后者永远会覆盖前者

image.png

@Entry
@Component
struct Baidu {
  build() {
    Column({ space: 12 }) {
      //   图片
      Row() {
        Stack({alignContent:Alignment.TopEnd}) {
          Image('https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png')
            .width(160)
          Text('鸿蒙版')
            .fontSize(12)
            .offset({
              x:-20,
              y:30
            })
        }
      }
      .width('100%')
      .justifyContent(FlexAlign.Center)

      //   输入框+按钮
      Row() {
        TextInput()
          .layoutWeight(1)
          .border({
            width: 2,
            color: '#ccc'
          })
          .height(40)
          .borderRadius({
            topLeft: 4,
            bottomLeft: 4
          })
          .backgroundColor('#fff')
        Button('百度一下')
          .type(ButtonType.Normal)
          .translate({
            x: -2
          })
          .borderRadius({
            topRight: 4,
            bottomRight: 4
          })
      }
    }
    .width('100%')
    .height('100%')
    .padding(16)
    .justifyContent(FlexAlign.Center)
  }
}

info
Stack的参数 可以设置子组件的排列方式-alignContent

  • Top(顶部)
  • TopStart(左上角)
  • TopEnd(右上角)
  • Start(左侧)
  • End(右侧)
  • Center(中间)
  • Bottom(底部)
  • BottomStart(左下角)
  • BottomEnd(右下角)

image.png

@Entry
@Component
struct FontJump {
  build() {
    Row() {
      Stack() {
        Text('抖音')
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          .fontColor('#ff2d83b3')
          .translate({
            x:-2,
            y:2
          })
          .zIndex(1)
        Text('抖音')
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          .fontColor('#ffe31fa9')
          .translate({
            x:2,
            y:-2
          })
          .zIndex(2)

        Text('抖音')
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          .fontColor('#ff030000')
          .translate({
            x:0,
            y:0
          })
          .zIndex(3)

      }
      .width('100%')
    }
    .height('100%')
  }
}

弹性布局

image.png

@Entry
@Component
struct ComponentCase {
  @State message: string = 'Hello World'
  build() {
    Scroll() {
      Row() {
        Column() {
          Flex({ direction: FlexDirection.Row, justifyContent: FlexAlign.SpaceAround }) {
            Column()
              .width(100)
              .height(200)
              .backgroundColor(Color.Pink)
            Column()
              .width(100)
              .height(200)
              .backgroundColor(Color.Red)
            Column()
              .width(100)
              .height(200)
              .backgroundColor(Color.Blue)
          }
          .margin({
            top: 200
          })

        }
        .width('100%')
      }
    }

  }
}

warning
Flex布局设置属性设置方向 是通过参数的,并非通过属性

  • 属性?

image.png

  • 参数?

image.png

网格布局

Grid布局

  • 想要控制屏幕的分栏 分几列, 怎么分 特别像前端的栅格布局
  • Row组件默认情况下,里面的元素的纵向是居中的
  • Column组件默认横向是居中的

image.png

info
Grid组件下只能放置GridItem组件

Grid可以设置columnsTemplate和rowsTemplate
columnsTemplate是设置横向的分配,如果设置 1fr 1fr 表示,等分为两份, 如果设置1fr 2fr表示左边一份,右边两份, 在设置columnsTemplate不设置rowsTemplate的情况下,如果内容超出容器区域,会自动出现滚动条 columnsGap设置列和列之间的间距,rowsGap设置行和行之间的间距

@Entry
@Component
struct GridCase {
  build() {
    Grid() {
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
      GridItemCase()
    }
    .width("100%")
    .height("100%")
    .columnsTemplate("1fr 1fr")
    .columnsGap(10)
    .rowsGap(10)
    .padding(10)

  }
}

@Component
struct GridItemCase {
  build() {
    GridItem() {
      Row() {
        Column() {
          Text("内容")
        }
        .width('100%')
      }
      .height(200)
      .borderRadius(4)
      .backgroundColor(Color.Pink)
    }

  }
}

滚动条说明

在基本的布局组件 Column/Row/Flex/Stack中不论内容超出与否,皆不会出现滚动条

  • 出现滚动条的组件
  • Grid
  • List(列表)
  • Scroll(滚动条)
  • Swiper(轮播)
  • WaterFlow(瀑布流)

出现滚动条的前提条件是- 上述组件中的子组件的内容超出了父容器组件的宽度或者高度

  • 使用最基本的Scroll组件出现一个滚动条

image.png

  • 先实现基本的布局
@Entry
@Component
struct ScrollCase {
  build() {
    Column() {
      Row()
        .width('100%')
        .height(50)
        .backgroundColor(Color.Red)

      Column() {

      }
      .width('100%')
      .layoutWeight(1)
      .backgroundColor(Color.Orange)

      Row()
        .width('100%')
        .height(50)
        .backgroundColor(Color.Blue)
    }
    .justifyContent(FlexAlign.SpaceBetween)
    .width('100%')
    .height('100%')
  }
}

image.png

  • 实现区域滚动
@Entry
@Component
struct ScrollCase {
  build() {
    Column() {
      Row()
        .width('100%')
        .height(50)
        .backgroundColor(Color.Red)
      Scroll(){
        Column() {
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
          ScrollItem()
        }
        .width('100%')
        .backgroundColor(Color.Orange)
      }
      .layoutWeight(1)
      Row()
        .width('100%')
        .height(50)
        .backgroundColor(Color.Blue)
    }
    .justifyContent(FlexAlign.SpaceBetween)
    .width('100%')
    .height('100%')
  }
}
@Component
struct ScrollItem {
  build() {
    Row() {
      Text("滚动区域内容")
    }
    .width('100%')
    .height(80)
    .backgroundColor(Color.Pink)
    .borderRadius(8)
    .margin({
      top: 20,
      bottom: 10
    })
    .justifyContent(FlexAlign.Center)
  }
}

image.png

  • 如何控制滚动
    info
    Scroll的滚动一般由用户的手指触发

  • 我们也可以使用一个对象来控制滚动条 scroller

@Entry
@Component
struct ScrollCase02 {
  scroller: Scroller = new Scroller()
  build() {
    Row() {
      Column() {
        // 有且只有一个组件
        Scroll(this.scroller) {
          Row({ space: 20 }) {
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
          }
        }.
        height(200)
        .scrollable(ScrollDirection.Horizontal)
        .width('100%')
        .backgroundColor(Color.Orange)
        Row() {
          Button("滚到左侧")
            .onClick(() => {
              this.scroller.scrollEdge(Edge.Start)
            })
          Button("滚到右侧")
            .onClick(() => {
              this.scroller.scrollEdge(Edge.End)

            })
        }
      }
    }
    .width('100%')
    .height('100%')
    .backgroundColor(Color.Pink)
  }
}

@Component
struct Actor {
  build() {
    Row() {
      Text("热辣滚烫-贾玲")
        .fontColor(Color.White)
    }
    .backgroundColor(Color.Red)
    .justifyContent(FlexAlign.Center)
    .width(100)
    .height(180)
  }
}

image.png

  • 横向滚动
    info

  • 只需要将Scroll的滚动方向调节成横向即可

@Entry
@Component
struct ScrollCase02 {
  @State message: string = 'Hello World';
  scroller: Scroller = new Scroller()
  build() {
    Row() {
      Column() {
        // 有且只有一个组件
        Scroll(this.scroller) {
          Row({ space: 20 }) {
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
            Actor()
          }
        }.
        height(200)
        .scrollable(ScrollDirection.Horizontal)
        .width('100%')
        .backgroundColor(Color.Orange)
        Row() {
          Button("滚到左侧")
            .onClick(() => {
              this.scroller.scrollEdge(Edge.Start)
            })
          Button("滚到右侧")
            .onClick(() => {
              this.scroller.scrollEdge(Edge.End)

            })
        }
      }
    }
    .width('100%')
    .height('100%')
    .backgroundColor(Color.Pink)
  }
}

@Component
struct Actor {
  build() {
    Row() {
      Text("热辣滚烫-贾玲")
        .fontColor(Color.White)
    }
    .backgroundColor(Color.Red)
    .justifyContent(FlexAlign.Center)
    .width(100)
    .height(180)
  }
}

image.png
在arkUI中,我们的内容如果超过了屏幕显示,则不会显示滚动条,需要使用Scroll来包裹
需要注意的是: 该组件滚动的前提是1.设置或使用了默认的滚动方向,2.子组件大与容器Scroll大小,否则不能滚动

组件-事件监听

监听原生组件的事件和设置属性的方式是一样的都是链式调用,值得注意的是,我们注册事件必须使用箭头函数的写法,Next版本禁止使用匿名函数的形式来给组件注册事件

  • 匿名函数 function () {} (ES5函数声明方式的一种,在鸿蒙中属于弃用语法
  • 组件外部函数 function () {} (组件内可以使用组件外的声明的函数,但需要注意this指向
  • 尝试给一个TextInput注册一个值改变事件提交事件,给登录按钮注册点击事件

image.png
image.pngimage.png
danger

  • promAction.showToast()轻量级提示,需要引入一个包才可以使用的,自动消失
  • promAction.showDialog()弹层级提示,需要引入一个包才可以使用的,点击消失
  • AlertDialog.show() 弹层级提示,不需要引入包使用的,点击消失
import { promptAction } from '@kit.ArkUI';
@Entry
@Component
struct Event {

  build() {
    Row() {
      Column({ space: 15 }) {
        Row() {
          TextInput({ placeholder: '请输入用户名' })
            .backgroundColor('#f4f5f6')
            .width('100%')
            .onChange((value) => {
              promptAction.showToast({ message: value })
            })
            .onSubmit(() => {
              AlertDialog.show({
                message:'登录失败'
              })
            })
        }.padding({
          left: 20,
          right: 20
        })

        Row() {
          Button("登录")
            .width('100%')
            .onClick(() => {
              promptAction.showDialog({
                message: '登录成功'
              })
            })

        }.padding({
          left: 20,
          right: 20
        })

      }
      .width('100%')
    }
    .height('100%')
  }
}

info
请注意:在注册事件中的逻辑必须使用箭头函数 () => {}

  1. 因为function中this指向为undefind
  2. 箭头函数中的this指向当前struct实例,可以方便的调用方法和获取属性

info
当我们事件处理逻辑比较复杂,写在UI描述中无法抽提的时候,我们可以在struct结构体中定义

struct Event {
  login () {
    AlertDialog.show({
      message: '登录成功'
    })
  }

  ...
  buile(){
    Button("登录")
            .width('100%')
            .onClick(() => {
              this.login()
            })
  }
   
}
  • 为什么说一定要用箭头函数
import { promptAction } from '@kit.ArkUI';
@Entry
@Component
struct FunctionCase {
  @State message: string = 'Hello World';
  showThis(){
    promptAction.showToast({
      message:this.message
    })
  }
  build() {
    Row() {
      Column() {
        Text(this.message)
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          //报错
          //.onClick(this.showThis)
          //推荐
          .onClick(()=>{
             this.showThis()
          })
      }
      .width('100%')
    }
    .height('100%')
  }
}

info
关于箭头函数和this的说明:
1.推荐使用箭头函数
2.call和apply是错误级别不支持,bind是警告,能用也别用!
3.this是当前上下文,一般指函数调用者

  • 尝试获取组件的大小-组件区域变化事件onAreaChange
    info
    如何获取某个组件的大小呢?比如获取下面百度图片的大小

image.png

import { promptAction } from '@kit.ArkUI';

@Entry
@Component
struct EventCase {
  // 登录方法
  login() {
    promptAction.showToast({ message: '登录成功' })
  }
  loginToButton(){
    promptAction.showToast({ message: '登录失败' })
  }
  build() {
    Row() {
      Column({ space: 20 }) {
        Image('https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png')
          .width(160)
          .onAreaChange((oldValue:Area,newValue:Area)=>{
            AlertDialog.show({
              message:`oldValue:${JSON.stringify(oldValue)}
                    newValue:${JSON.stringify(newValue)}`
            })
          })
        TextInput({ placeholder: '请输入用户名', text:'' })
          .height(40)
          .onChange((value) => {
            // 监听系统组件的事件的话  回调函数的参数都是有具体类型  一般的话不用给类型
            promptAction.showToast({ message: value })
          })
        TextInput({ placeholder: '请输入密码', text: '' })
          .height(40)
          .onChange((value) => {
            // 监听系统组件的事件的话  回调函数的参数都是有具体类型  一般的话不用给类型
            promptAction.showToast({ message: value })
          })
          .type(InputType.Password)
          .onSubmit(() => {
            this.login()
          })
        Button("登录")
          .width('100%')
          .onClick(this.loginToButton)
      }
      .padding({
        left: 20,
        right: 20
      })
      .width('100%')
    }
    .height('100%')
  }
}

基础-组件状态

info
当我们需要在组件中记录一些状态时,变量应该显示的在struct中声明,并注明类型
比如-登录账户和密码

  username: string = "admin"
  password: string = "123456"
  • 实现一个简单的登录页面

image.png

  • @State修饰符的作用

如果没有@State修饰符,改变状态后,页面不会进行状态更新

info
1.State修饰的类型:Object、class、string、number、boolean、enum类型,以及这些类型的数组。
2.类型必须被指定,嵌套类型的场景请参考观察变化。
3.不支持any,不支持简单类型和复杂类型的联合类型,不允许使用undefined和null。

加上该修饰符后,你惊奇的发现按钮随着数据的变化在变化,因为我们在值改变的时候赋值,造成了build的重新执行,来保证我们状态的变化。
可以理解成没有@State修饰符,数据只会作用页面一次!!!

interface myObjType {
  age:number
}
interface moreStepType {
  info:myObjType
}
@Entry
@Component
struct ThisCase {
  noStateMessage: string = '6666666666666666666'
  @State
  message: string = '999999999999999999999'
  @State
  myObj:myObjType = { age: 18 };
  noStateObj:myObjType = { age: 19 };
  @State
  moreStepObj:moreStepType = {
    info:{
      age:20
    }
  }
  innerFunction: () => void = () => {
    this.myObj = { age: 19 }
  }

  build() {
    Row() {
      Column() {
        Text(JSON.stringify(this))
          .fontWeight(FontWeight.Bold)
            .onClick(() => {
              this.innerFunction()
            })
      }
      .width('100%')
      .padding(20)
    }
    .height('100%')
  }
}

image.png
info
改变状态:引用数据类型只能检测到自身和第一层变化

  • 添加一个登录验证的需求 :账号admin密码123456时登录提示登录成功,否则提示用户名或者密码错误
import { promptAction } from '@kit.ArkUI';

@Entry
@Component
struct EventCase {
  @State message: string = 'Hello World';
  @State
  username: string = ""
  @State
  password: string = ""
  // 登录方法
  login() {
    if(this.username.trim()  === 'admin' && this.password === '123456') {
      // promptAction.showToast({ message: '登录成功' })
      AlertDialog.show({ message: '登录成功' })
      return
    }
    promptAction.showToast({ message: '用户名或者密码错误' })
  }

  build() {
    Row() {
      Column({ space: 20 }) {
        TextInput({ placeholder: '请输入用户名', text: this.username })
          .height(40)
          .onChange((value) => {
            // 监听系统组件的事件的话  回调函数的参数都是有具体类型  一般的话不用给类型
            console.log(value)
          })
        TextInput({ placeholder: '请输入密码', text: this.password })
          .height(40)
          .onChange((value) => {
            // 监听系统组件的事件的话  回调函数的参数都是有具体类型  一般的话不用给类型
            console.log(value)
          })
          .type(InputType.Password)
          .onSubmit(() => {
            this.login()
          })
        Button("登录")
          .width('100%')
          .onClick(() => {
            this.login()
          })


      }
      .padding({
        left: 20,
        right: 20
      })
      .width('100%')
    }
    .height('100%')
  }
}
  • 再加一个小需求,当用户名和密码为空时,不让用户点按钮
    danger
    可用enable设置按钮是否启用
getBtnEnable () {
    return  !!this.username && !!this.password
  }

 Row() {
           Button("登录")
          .width('100%')
          .onClick(() => {
            this.login()
          })
          // 隐士转化 ""
          .enabled(this.getBtnEnable())

接下来,我们来完成一个嵌套数据修改的案例,来观察State修饰符的特性

image.png

  • 声明一个关于人的接口
interface IAddress {
  province: string
  city: string
  area: string
}
interface IPerson {
  name: string
  age: number
  sex: "男" | "女"
  address: IAddress
}

可能比较疑惑,为什么这里还需要声明两个接口,因为Next版本不支持对象字面量类型声明,所以这里需要,用interface代替

  • 类型怎么赋值呢?
  • 使用npm全局安装插件
$ npm i -g interface2class   # 特别注意 $符号不用输入,$符号表示它是一个命令行命令

danger
npm config get registry

  • 设置淘宝镜像
$ npm config get registry
$ npm config set registry https://registry.npmmirror.com/
  • 脚本禁用-只针对windows
    info
    解决方法
  1. 以管理员身份运行PowerShell
  2. 执行:get-ExecutionPolicy,如果显示Restricted,表示状态是禁止的
  3. 执行: set-ExecutionPolicy RemoteSigned
  4. 选择Y

info
脚本策略-操作系统允许npm包的命令可执行

image.png

  • 测试是否安装成功
$ i2c -V

image.png

  • 执行带有interface的文件

image.png

  • 自动生成实现类

image.png

  • 声明一个State类型
 @State
  p: IPersonModel = new IPersonModel({
    name: '小张',
    age: 20,
    sex: "女",
    address: {
      province: '广东',
      city: '广州',
      area: '天河'
    }
  })
  • 使用UI组件实现双向绑定
@Entry
@Component
struct StateCase {
  @State
  p: IPersonModel = new IPersonModel({
    name: '小张',
    age: 20,
    sex: "女",
    address: {
      province: '广东',
      city: '广州',
      area: '天河'
    }
  })
  build() {
    Column({ space: 10 }) {
      Text(JSON.stringify(this.p))
      Row({ space: 15 }) {
        Text("姓名:")
        TextInput({ text: this.p.name }).layoutWeight(1)
          .onChange(value => {
            this.p.name = value
          })
      }.padding(10)
      Row({ space: 15 }) {
        Text("年龄:")
        TextInput({ text: this.p.age.toString() }).layoutWeight(1)
          .onChange(value => {
             this.p.age = parseInt(value)
          })
      }.padding(10)
      Row({ space: 15 }) {
        Text("性别:")
        Select([{ value: '男' }, { value: '女' }])
          .layoutWeight(1)
          .value(this.p.sex)
          .onSelect((index: number, value: string) => {
            this.p.sex = value as "男" | "女"
          })
      }.padding(10)
      Row({ space: 15 }) {
        Text("地址:")
        TextInput({ text: this.p.address.province }).layoutWeight(1)
          .onChange((value) => {
            this.p.address.province = value
          })
        TextInput({ text: this.p.address.city }).layoutWeight(1)
          .onChange((value) => {
            this.p.address.city = value
          })
        TextInput({ text: this.p.address.area }).layoutWeight(1)
          .onChange((value) => {
            this.p.address.area = value
          })
      }.padding(10)

    }
  }
}
interface IAddress {
  province: string
  city: string
  area: string
}
interface IPerson {
  name: string
  age: number
  sex: "男" | "女"
  address: IAddress
}
export class IAddressModel implements IAddress {
  province: string = ''
  city: string = ''
  area: string = ''

  constructor(model: IAddress) {
    this.province = model.province
    this.city = model.city
    this.area = model.area
  }
}
export class IPersonModel implements IPerson {
  name: string = ""
  age: number = 0
  sex: "男" | "女" = "男"
  address: IAddress = new IAddressModel({} as IAddress)

  constructor(model: IPerson) {
    this.name = model.name
    this.age = model.age
    this.sex = model.sex
    this.address = model.address
  }
}

  • 更新数据-第一层数据
  // 放置按钮
      Row({ space: 10 }) {
        Button("修改姓名")
          .onClick(() => {
             this.p.username = "老张"
          })
        Button("修改年龄")
          .onClick(() => {
            this.p.age++
          })
        Button("修改性别")
          .onClick(() => {
            this.p.sex = this.p.sex === "男" ? "女" : "男"
          })
      }
      .padding({
        left: 10,
        right: 10
      })
      .width("100%")

image.png

  • 更新第二层数据

danger
所有的鸿蒙的更新机制-关于对象层的, 所有的监听机制都只能监听到Object.keys(对象)中的可见属性,但凡超过一层,是监测不到变化的。

  Row({ space: 10 }) {
        Button("修改地址")
          .onClick(() => {
            // this.p.address.province = "北京"
            // this.p.address.city = "北京"
            // this.p.address.area = "顺义"
            // this.p.age++
            // this.p.address = new IAddressModel({
            //   province: this.p.address.province,
            //   city: this.p.address.city,
            //   area:"海珠区"
            // })
            this.p.address.area = "海珠区" // 虽然不会造成更新 但是数据会变化
            this.p.address = new IAddressModel(this.p.address)
          })

      }
      .padding({
        left: 10,
        right: 10
      })
      .width("100%")

发现什么问题没有?
我们发现第一层数据 比如 name/age/sex都是可以双向同步的,但是第二层的数据无法进行响应式更新

  • 上述代码中,我们也没有使用 ,因为在目前的编辑器中使用 ,因为在目前的编辑器中使用 ,因为在目前的编辑器中使用绑定嵌套的数据编辑器会报错,虽然效果对,但是报错总归不太好,所以我们手动进行了数据同步

在鸿蒙开发中,我们应该明确,所有的响应式更新都只能是系统可观测的响应式更新,那什么是可观测的呢?
就是只有对象或者数组第一层的数据发生了变化,才可以被观测到,所以基于这种特性,我们需要来处理下

image.png

下方列出系统可观测的数据和行为变化

image.png
image.png
image.png
image.png

组件-双向绑定

  • 双向绑定
    info
    数据-视图

  • 数据驱动视图

  • 视图中的内容发生变化-数据也会同步修改

  • MVVM-M-V-VM

  • Model数据模型

  • V-View视图

  • VM- ViewModel 桥梁

在鸿蒙Next版本中,推出了一系列双向绑定的组件

  • 双向绑定- 数据和视图双向同步-数据变化-视图更新,视图更新,-数据更新

  • 鸿蒙Next版本所支持的所有双向绑定组件

image.png
image.png

  • 双向绑定语法

$$语法:内置组件双向同步

  • 当前$$支持基础类型变量,以及@State、@Link和@Prop装饰的变量。

如果绑定TextInput组件,如 TextInput({ text: $$this.xxx })

danger
值得注意的是:上述组件中有的双向组件是属性,有的双向绑定是参数

  • 参数是在组件({ text: $$this.xx })

  • 属性是在组件().text($$this.xxx)

  • 不支持嵌套数据的双向绑定如 组件({ text: $$this.xx.xx })

  • 测试几个正常的双向绑定

image.png

@Entry
@Component
struct MvvmCase {
  @State
  isChecked: boolean = false
  @State
  myDate: Date = new Date('2024-05-01')
  @State
  myRatio: boolean = false
  @State
  mySearch: string = ''
  @State
  myToggle: boolean = false
  @State
  mySelect: string = '请选择'
  build() {
    Column({ space: 12 }) {
      Column() {
        Text('Checkbox双向绑定')
        Checkbox().select($$this.isChecked)
        Text('' + this.isChecked)
      }

      MyDivider()
      Column() {
        Text('DatePicker双向绑定')
        DatePicker({
          selected: $$this.myDate
        })
        Text('' + this.myDate)
      }

      MyDivider()
      Column() {
        Text('Radio双向绑定')
        Radio({ value: 'radio', group: 'radioSelect' }).checked($$this.myRatio)
        Text('' + this.myRatio)
      }

      MyDivider()
      Column() {
        Text('Search双向绑定')
        Search({value:$$this.mySearch})
        Text('' + this.mySearch)
      }
      MyDivider()
      Column() {
        Text('Toggle双向绑定')
        Toggle({ type: ToggleType.Switch,isOn:$$this.myToggle})
        Text('' + this.myToggle)
      }
      MyDivider()
      Column() {
        Text('Select双向绑定')
        Select([{ value: 'aaa' },
          { value: 'bbb'},
          { value: 'ccc'},
          { value: 'ddd'}])
          .value($$this.mySelect)
        Text('' + this.mySelect)
      }
    }
    .width("100%")
    .height("100%")
  }
}

@Component
struct MyDivider {
  build() {
    Divider().height(5).backgroundColor(Color.Pink)
  }
}

样式

样式-语法(链式&枚举)

ArkTS以声明方式组合和扩展组件来描述应用程序的UI;
同时还提供了基本的属性、事件和子组件配置方法,帮助开发者实现应用交互逻辑。

1)样式属性:通用属性 和 组件属性

  • 属性方法以 . 链式调用的方式配置系统组件的样式和其他属性
@Entry
@Component
struct Index {
  build() {
    Text('演示')
      .backgroundColor('red')
      .fontSize(50)
      .width('100%')
      .height(100)
  }
}

2)枚举值

  • 对于系统组件,ArkUI还为其属性预定义了一些枚举类型。文档链接
@Entry
@Component
struct Index {
  build() {
    Text('演示')
      .fontSize(50)
      .width('100%')
      .height(100)
      .backgroundColor(Color.Blue)
      .textAlign(TextAlign.Center)
      .fontColor(Color.White)
  }
}

info

  • 样式相关属性通过链式函数的方式进行设置
  • 如果类型是枚举的,通过枚举传入对应的值

注意: 有的属性强烈建议使用枚举(大部分枚举值都是数字,但是数字无法体现代码含义)
有的组件如fontColor可以使用系统自带颜色枚举,也可以使用色值

样式-单位px/vp/fp/lpx

官方定义

使用虚拟像素,使元素在不同密度的设备上具有一致的视觉体量。

1) vp 是什么?virtual pixel

  • 屏幕密度相关像素,根据屏幕像素密度转换为屏幕物理像素,当数值不带单位时,默认单位 vp;在实际宽度为1440物理像素的屏幕上,1vp 约等于 3px(物理像素)

image.png

image.png

  • 系统还提供了对应的像素转化方法

image.png
info
在样式中,我们如果写px,那么px直接表示的是物理像素,也就是分辨率,那么我们的手机分辨率密度各有不同,无法针对这种密度写一个固定值,所以vp会自动根据手机密度去进行适配,所以vp它提供了一种灵活的方式来适应不同屏幕密度的显示效果。
设计图按照1080设计- 换算成360写vp就可以了

  • 上图的意思是,使用这个单位在不同屏幕物理分辨率的实际尺寸一致(A设备1英寸,B设备1英寸)。

2)在不同屏幕物理分辨率下,要想实现等比例适配, 可以吗?
如下图:
GIF 2024-5-10 12-14-22.gif
info
设置lpx基准值 - resources/base/profile/main-pages.json
添加window属性,设置desigWidth,不设置也可以使用lpx,默认720

image.png

@Entry
@Component
struct PXCase {
  build() {
    Row() {
      Column() {
        Text('375lpx')
        .width('375lpx')
        .height('72lpx')
          .textAlign(TextAlign.Center)
        .backgroundColor(Color.Red)
        Divider().strokeWidth(2)
        Row(){
          Text('72lpx')
        }
        .width('72lpx')
        .height('25lpx')
        .backgroundColor(Color.Brown)
      }
      .width('100%')
    }
    .height('100%')
  }
}

danger
伸缩布局的方案

  • 设定基准值,使用lpx,类似于前端的rem
  • 监听元素的变化-可以拿到宽高-重新计算
  • layoutWeight(number)- 剩余资源再分配

伸缩 layoutWeight(flex: number) 占剩余空间多少份,可以理解成CSS的 flex: 1
如图-手机端
image.png
-pad
image.png
我们可以使用layoutWeight属性,让右侧内容去占满剩余宽度

build() {
    Row() {
      Text("左侧内容")
      Text("右侧内容")
        .textAlign(TextAlign.End)
        .width('80%')
        .height(60)
        .backgroundColor('red')
        .layoutWeight(1)
    }.width('100%')
    .height('100%')

  }

image.png

@Entry
@Component
struct LayoutCase {
  @State message: string = 'Hello World';

  build() {
    Column() {
      Row() {

      }
      .width('100%')
      .height(50)
      .backgroundColor(Color.Blue)

      Column() {

      }
      .width('100%')
      .backgroundColor(Color.Green)
      .layoutWeight(1)  // Grid中的columnsTemplate 1fr 1fr
      Column() {

      }
      .width('100%')
      .backgroundColor(Color.Orange)
      .layoutWeight(1)

      Row() {

      }
      .width('100%')
      .height(50)
      .backgroundColor(Color.Red)
    }
    .height("100%")
    .width("100%")
    .justifyContent(FlexAlign.SpaceBetween)
  }
}

info
瓜分当前组件的剩余空间
Row() {
Text().width(10)
Text().width(10)
Text().layoutWeight(5) //表示这个元素占据剩余所有的空间 (100 - 10 - 10)* (5/(5 + 1))
Text().layoutWeight(1) //表示这个元素占据剩余所有的空间 100 - 10 - 10
}.width(100)

  • 内容等比例缩放-可以使用aspectRatio属性设置宽高比

image.png
设置元素宽高比 aspectRatio(ratio: number)

如我们如果希望一个元素始终占整个屏幕宽度的50%,且为一个正方形

image.png

 Column()
        .width('50%')
        .height('50%')
        .backgroundColor('blue')
        .aspectRatio(1)
@Entry
@Component
struct Index {
  build() {
    Text('left')
      .width('50%')
        // 宽高比例
      .aspectRatio(1)
      .backgroundColor('red')
  }
}

info

  • vp 是鸿蒙默认单位,和屏幕像素有关,最终表现视觉大小在任何设备一致(屏幕越大看的越多)
  • 鸿蒙一般以伸缩 layoutWeight、网格、栅格进行布局适配,如要等比例缩放可以设置高宽比 aspectRatio(屏幕越大内容越大)

Image和资源Resource

info
项目开发离不开图片-图片在页面中必须使用Image/ImageSpan/动画图片组件
Image为图片组件,常用于在应用中显示图片。Image支持加载string、PixelMap和Resource类型的数据源,支持png、jpg、bmp、svg和gif类型的图片格式。

  • 1.使用本地图片-拖一张图片放置到ets目录下-比如assets文件下
    info
    命名不要出现中文、空格、括号等特殊符号!

image.png

Image('/assets/a.png')
          .width(100)
          .height(100)

image.png

  • 2.使用Resource下的图片-media

image.png

//不需要图片后缀
Image($r('app.media.a'))
          .width(100)
          .height(100)

  • 3.使用Resource下的图片-rawfile

image.png

 Image($rawfile('a.png'))
          .width(100)
          .height(100)

  • 4.使用网络图片
        Image("https://foruda.gitee.com/avatar/1705232317138324256/1759638_itcast_panpu_1705232317.png")
          .width(100)
          .height(100)


info
尤其注意: 使用网络图片时,在preview中时,可以预览,但是在模拟器和真实项目中,必须申请网络权限

"requestPermissions": [{
  "name":"ohos.permission.INTERNET"
}],
  • 5.使用字体图标

阿里巴巴矢量图标库 iconfont
华为官方图标下载 链接
info
搜索下载需要的图标,下载类型为SVG,使用fillColor()可以进行图标颜色修改
部分图标不能修改,需要手动添加fill属性才能生效

image.pngimage.png

        Image($r('app.media.ic_like'))
          .width(100)
          .height(100)
          .fillColor(Color.Red)

image.png

  • 6.使用系统内置图标
      //有的图标必须给大小
      Image($r('sys.media.ohos_ic_public_sound'))
          .width(100)
          .height(100)
          .fillColor(Color.Red)

image.png

image.png

@Entry
@Component
struct ImageCase {
  build() {
    Row() {
      Column({space:10}) {
        // 本地自建目录
        Image('/assets/a.png')
          .width(100)
          .height(100)
        // 推荐目录
        Image($r('app.media.a'))
          .width(100)
          .height(100)
        // 源文件目录
        Image($rawfile('a.png'))
          .width(100)
          .height(100)
        // 网络图片
        Image("https://foruda.gitee.com/avatar/1705232317138324256/1759638_itcast_panpu_1705232317.png")
          .width(100)
          .height(100)
        // 字体图标用法
        Image($r('app.media.ic_like'))
          .width(100)
          .height(100)
          .fillColor(Color.Red)
        // 系统内置图标用法
        Image($r('sys.media.ohos_ic_public_sound'))
          .width(100)
          .height(100)
          .fillColor(Color.Red)
      }
      .width('100%')
    }
    .height('100%')
  }
}

info
我们已经知道resources/base下media和profile的作用了,那么element是干嘛的呢?

  • 自定义资源颜色-在color.json中定义一个颜色

image.png

{
  "color": [
    {
      "name": "start_window_background",
      "value": "#FFFFFF"
    },
    {
      "name": "my_first_color",
      "value": "#FF00FF0F"
    }
  ]
}

在页面使用这个颜色

 Text(this.message)
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          .fontColor($r('app.color.my_first_color'))
  • 自定义资源文字-在string.json中定义一个文字

image.png

{
  "string": [
    {
      "name": "module_desc",
      "value": "module description"
    },
    {
      "name": "EntryAbility_desc",
      "value": "description"
    },
    {
      "name": "EntryAbility_label",
      "value": "label"
    },
    {
      "name": "my_first_string",
      "value": "老潘"
    }
  ]
}

在页面上使用这个文字

 Text($r('app.string.my_first_string'))
          .fontSize(50)
          .fontWeight(FontWeight.Bold)
          .fontColor($r('app.color.my_first_color'))

image.png

  • 国际化处理
    danger
    需要注意的是,手动添加虽然可以使用,但是会引起警告
    推荐编辑添加,除了添加base/string.json,对en_US和zh_CN也进行添加

image.png
image.png
添加后运行到模拟器,改变系统语言会发现,呈现文字也会跟随改变
GIF 2024-5-10 17-29-23.gif

info
写一个知乎的评论练习练习排版布局和图片样式等

image.png

@Entry
@Component
struct ZHCase {
  build() {
    Column() {
      // 1.标题
      Row() {
        Row() {
          Image($r('sys.media.ohos_ic_public_arrow_left'))
            .width(24)
            .height(24)
        }
        .width(30)
        .height(30)
        .backgroundColor('#f4f4f4')
        .borderRadius(15)
        .justifyContent(FlexAlign.Center)

        Text('评论回复')
          .layoutWeight(1)
          .textAlign(TextAlign.Center)
          .margin({
            right: 30
          })
      }
      .width('100%')
      .padding(16)
      .border({
        width: {
          bottom: 1
        },
        color: '#f4f5f6'
      })

      // 2.评论
      Row({space:16}){
        Image('https://foruda.gitee.com/avatar/1705232317138324256/1759638_itcast_panpu_1705232317.png')
          .width(60)
          .borderRadius(30)
        Column({space:16}){
          Text('潘神')
            .fontSize(20)
            .fontWeight(FontWeight.Bold)
          Text('写布局的口诀是从上到下,从左到右,从大到小,从外往里!🔥')
            .width('100%')
          Row(){
            Text('10-21·IP:北京')
              .fontSize(12)
              .fontColor('#ccc')
            Row({space:4}){
              Image($r('app.media.ic_like'))
                .width(16)
                .fillColor('#ccc')
              Text('100')
                .fontSize(12)
                .fontColor('#ccc')
            }
          }
          .width('100%')
          .justifyContent(FlexAlign.SpaceBetween)
        }
        .layoutWeight(1)
        .alignItems(HorizontalAlign.Start)

      }
      .padding(16)
      .alignItems(VerticalAlign.Top)
    }
    .width('100%')
    .justifyContent(FlexAlign.Start)
  }
}

样式-@Styles 复用

注意: Styles和Extend均只支持在当前文件下的全局或者组件内部定义,如果你想要在其他文件导出一个公共样式,导出公共使用,ArtTS是不支持的,这种方式还是需要考虑组件复用。

在开发过程中会出现大量代码在进行重复样式设置,@Styles 可以帮我们进行样式复用

info
通用属性 通用事件
在Styles修饰的函数中能够点出来就是通用属性和事件-Text的字体颜色-字体大小不属于通用属性
Styles修饰的函数不允许传参数

  • 当前 @Styles 仅支持 通用属性 和 通用事件。
    info
    全局Styles不支持箭头函数语法

warning
注意: 全局Styles扩展符只能和使用它的组件位于同一个文件,不允许导入导出,导入导出也使用不了

  • 支持 全局 定义和 组件内 定义,同时存在组件内覆盖全局生效。
import { promptAction } from '@kit.ArkUI'

@Styles
function payStyle () {
  .width('100%')
  .height(50)
  .borderRadius(4)
  .backgroundColor("#00c168")
  .onClick(() => {
    promptAction.showToast({ message: '微信支付成功' })
  })
}

@Entry
@Component
struct StyleCase {
  @Styles
  payStyle() {
    .width('100%')
    .height(50)
    .borderRadius(4)
    .backgroundColor("#ff1256e0")
    .onClick(() => {
      promptAction.showToast({ message: '支付宝支付成功' })
    })
 }


  build() {
    Column({ space: 20 }) {
      Row() {
        Button("微信支付", { type: ButtonType.Normal })
          .payStyle()
          .fontColor(Color.White)
      }
      .padding(10)
      Row() {
        Button("微信支付", { type: ButtonType.Normal })
          .payStyle()
          .fontColor(Color.White)
      }
      .padding(10)
      Row() {
        Button("微信支付", { type: ButtonType.Normal })
          .payStyle()
          .fontColor(Color.White)
      }
      .padding(10)

    }
  }
}

image.png

样式-@Extend 复用

info
假设我们就想针对 Text进行字体和样式的复用,此时可以使用Extend来修饰一个全局的方法

  • 使用 @Extend 装饰器修饰的函数只能是 全局
  • 函数可以进行 传参,如果参数是状态变量,状态更新后会刷新UI
  • 且参数可以是一个函数,实现复用事件且可处理不同逻辑

warning
注意: Extend扩展符只能和使用它的组件位于同一个文件,不允许导入导出,导入导出也使用不了

image.png

import { promptAction } from '@kit.ArkUI'

@Entry
@Component
struct ExtendCase {

  build() {
     Column({ space: 20 }) {
       Button("微信支付")
         .payButton("alipay")
       Button("微信支付")
         .payButton("wechat")
       Button("微信支付")
         .payButton("alipay")
       Button("微信支付")
         .payButton("wechat")
           Button("微信支付")
         .payButton("alipay")
           Button("微信支付")
         .payButton("wechat")
           Button("微信支付")
         .payButton("alipay")

     }
     .padding(20)
    .width('100%')
  }
}

// 不允许导出
@Extend(Button)
function  payButton (type: "alipay" | "wechat") {
  .type(ButtonType.Normal)
  .fontColor(Color.White)
  .width('100%')
  .height(50)
  .borderRadius(4)
  .backgroundColor(type === "wechat" ? "#00c168" : "#ff1256e0")
  .onClick(() => {
    if(type === "alipay") {
      promptAction.showToast({ message: '支付宝支付成功' })
    }else {
      promptAction.showToast({ message: '微信支付成功' })
    }

  })
}

多态样式stateStyles

@Styles和@Extend仅仅应用于静态页面的样式复用,stateStyles可以依据组件的内部状态的不同,快速设置不同样式。这就是我们本章要介绍的内容stateStyles(又称为:多态样式)。
ArkUI 提供以下五种状态:

  • focused:获焦态。
  • normal:正常态。
  • pressed:按压态。
  • disabled:不可用态。
  • selected: 选中态

warning
假设我们想做一个微信中点击的选中状态, 如图
image.png

该图在点击时会有变色,抬起时消失,此时就可以利用多态样式进行设置

  • 实现一个基本的Row样式
@Entry
@Component
struct StateStylesCase {
  build() {
    Column({ space: 20 }) {
       Row() {
         Text("你今天想我了吗")
       }
       .padding(20)
       .height(80)
       .border({
         color: '#f3f4f5',
         width: 3
       })
       .borderRadius(4)
       // 多态样式
       .stateStyles({
         // 正常态
         normal: {
           .backgroundColor(Color.White)
         },
         pressed: {
           .backgroundColor("#eee")
         }
       })
      .width('100%')
    }
    .padding(20)
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
  }
}

image.png

info
按钮时,出现变色,需要同时设置pressed和normal两个属性, 如果只设置pressed,松手之后样式无法恢复

info
注意- 多态样式只能设置通用属性

  • 禁用状态样式
    warning
    鸿蒙所有组件都会有一个enable属性,enable为true时表示该组件可用,为false时,表示该组件禁用,禁用态就表示组件禁用时的样式

  • 设置Row禁用时的状态背景为灰背景

image.png

@Entry
@Component
struct StateStylesCase {
  @State
  btnEnable: boolean = true
  build() {
    Column({ space: 20 }) {
       Row() {
         Text("你今天想我了吗")
       }
       .padding(20)
       .height(80)
       .border({
         color: '#f3f4f5',
         width: 3
       })
       .borderRadius(4)
       // 多态样式
       .stateStyles({
         // 正常态
         normal: {
           .backgroundColor(Color.White)
         },
         pressed: {
           .backgroundColor("#eee")
         },
         disabled: {
           .backgroundColor("#999")
         }
       })
       .enabled(this.btnEnable)
      .width('100%')
      Button("禁用/解禁")
        .onClick(() => {
          this.btnEnable = !this.btnEnable
        })
    }
    .padding(20)
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
  }
}
  • 获焦状态
    info
    假设我们有个账号和密码输入框,需要在聚焦态时设置选中样式,同样可以使用多态样式来设置
    但是在预览器和真机中,有个非常诡异的设计,只能通过tab键切换才可以使得输入框进入聚焦态
    如图
    image.png
 TextInput({
          placeholder: '请输入账号'
        })
          .stateStyles({
            focused: {
              .border({
                color: Color.Red,
                width: 1
              })
            },
            normal: {
              .border({
                width: 0
              })
            }
          })
        TextInput({
          placeholder: '请输入密码',
        })
          .stateStyles({
            focused: {
              .border({
                color: Color.Red,
                width: 1
              })
            },
            normal: {
              .border({
                width: 0
              })
            }
          })
          .type(InputType.Password)
          .showPasswordIcon(true)
  • 全部代码
@Entry
@Component
struct StateStylesCase {
 @State
 rowEnable: boolean = true
  build() {
    Row() {
      Column({ space: 20 }) {
        TextInput({
          placeholder: '请输入账号'
        })
          .stateStyles({
            focused: {
              .border({
                color: Color.Red,
                width: 1
              })
            },
            normal: {
              .border({
                width: 0
              })
            }
          })
        TextInput({
          placeholder: '请输入密码',
        })
          .stateStyles({
            focused: {
              .border({
                color: Color.Red,
                width: 1
              })
            },
            normal: {
              .border({
                width: 0
              })
            }
          })
          .type(InputType.Password)
          .showPasswordIcon(true)


        Row() {
          Text("你干什么吃的")
        }
        .height(60)
        .width('100%')
        .border({
          color: '#ccc',
          width: 1
        }).stateStyles({
          pressed: {
            .backgroundColor(Color.Gray)
          },
          normal: {
            .backgroundColor(Color.White)
          },
          disabled: {
            .backgroundColor("#f3f4f5")
          }
        })
        .enabled(this.rowEnable)
        Button("禁用")
          .onClick(() => {
            this.rowEnable = !this.rowEnable
          })
      }


    }
    .height('100%')
  }
}


info

  • 使用比较多的应该是 normal pressed 结合下的按压效果
  • enabled(true|false) 开启|禁用

界面渲染

渲染-条件渲染

info
在ArkTS中 我们要根据某个状态来控制元素或者组件的显示隐藏 可以采用条件渲染

  • if/else(创建销毁元素)

  • 元素高宽-透明度-位置控制 (属性控制)

  • visibility属性控制

  • 使用if/else

通过一个switch开关来控制图片的显示隐藏

GIF 2024-5-11 15-43-49.gif

@Entry
@Component
struct ConditionCase {
  @State isShow: boolean = false;

  build() {
    Row() {
      Column() {
        Toggle({
          type: ToggleType.Switch,
          isOn: $$this.isShow
        })
        // 1.控制是否渲染
        // if(this.isShow){
        //   Image($r('app.media.b')).width(100)
        // }else{
        // Image($r('app.media.b')).width(100)
        // }
        // 2.控制是否展示
        Image($r('app.media.b')).width(100)// 展示
          // .visibility(Visibility.Visible)
          // 隐藏(不显示也会占位置,不影响其他元素排列的话优先使用这个,性能高)
          // .visibility(this.isShow?Visibility.Visible:Visibility.Hidden)
          // 隐藏(不显示就不占位置,和if else一样)
          .visibility(this.isShow ? Visibility.Visible : Visibility.None)
      }
      .width('100%')
    }
    .height('100%')
  }
}
  • 多种条件控制

GIF 2024-5-11 16-28-40.gif
info
分析:
1.页面排版布局样式实现
2.下拉框的双向绑定
3.条件渲染

@Entry
@Component
struct ConditionCase02 {
  @State myVip: number = 0;
  @State optionValue:string = '暂不开通'

  build() {
    Row() {
      Column({space:20}) {
        Row(){
          Text('开通会员:')
          Select([{value:'暂不开通'},{value:'VIP'},{value:'SVIP'}])
            .width('50%')
            .selected($$this.myVip)
            .value($$this.optionValue)
        }
        Row({ space: 20 }) {
          Image($r('app.media.b')).width(30).borderRadius(30)
          Text('西北吴彦祖')
          if (this.myVip === 0) {
            Text('VIP')
              .VIPStyle(this.myVip)
              .backgroundColor('#ccc')
          } else if (this.myVip === 1) {
            Text('VIP')
              .VIPStyle(this.myVip)
              .backgroundColor('#ffffb803')
          } else if (this.myVip === 2) {
            Text('SVIP')
              .VIPStyle(this.myVip)
              .backgroundColor('#ffb00909')
          }

        }.width('100%')
        .justifyContent(FlexAlign.Center)
      }
      .width('100%')
      .padding(20)
    }
    .height('100%')
  }
}
@Extend(Text)
function VIPStyle(type: number) {
  .padding({
    left: 12,
    right: 12,
    bottom: 4,
    top: 4
  })
  .fontColor('#fff')
  .borderRadius(20)
  .fontSize(12)
}
  • 案例-实现加载数据的loading效果

warning
image.png

  • 封装loading组件
@Preview // 表示该组件可预览 Preview只能看效果 没有交互
@Component
export struct HmLoading {
  @State // 响应式驱动视图
  value: number = 0
  timer: number = -1  // 显示声明
  aboutToAppear(): void {
   this.timer = setInterval(() => {
      if(this.value === 100) {
        this.value = 0
      }
      this.value++
    }, 10)
  }
  aboutToDisappear(): void {
    clearInterval(this.timer)
  }

  build() {
     Progress({
       total: 100,
       value: this.value,
       type: ProgressType.Ring
     })
  }
}

// 默认导出
export default HmLoading  // 一个文件只能有一个默认导出
// 按需导出
// export { HmLoading, HmLoading1, HmLoading3 }


  • 在LoadingCase中使用
import HmLoading from './Components/HmLoading';
import { promptAction } from '@kit.ArkUI';

@Entry
@Component
struct LoadingCase {
  @State showLoading: boolean = false;

  // 页面刚进入 应该去获取数据  等待的过程中 显示进度 数据获取完 进度消失
  // 钩子函数
  aboutToAppear(): void {
     // 请求数据
    // setTimeout/setInterval
    this.showLoading = true
    setTimeout(() => {
      this.showLoading = false
    },10000)
  }
  build() {
    Row() {
      Column() {
        if(this.showLoading) {
          HmLoading()
        }
      }
      .width('100%')
    }
    .height('100%')
  }
}

// 封装一个Loading组件

渲染-循环渲染

  • ForEach-最常用的
  • LazyForEach-懒加载渲染(复杂-后面讲)
    info
    循环渲染使用 ForEach方法来进行

ForEach 接口基于数组类型数据来进行循环渲染,需要与容器组件配合使用。

ForEach(
  // 数据源
  arr: Array,
  // 组件生成函数
  itemGenerator: (item: 单项, index?: number) => void,
  // 键值生成函数
  keyGenerator?: (item: 单项, index?: number): string => string
)
  • 定义数据类型
interface PayRecord {
  OrderName:string
  OrderDate:Date
  OrderAmount:number
}
@State PayRecordList: PayRecord[] = [
  {
    OrderName: '给老婆买口红',
    OrderDate: new Date('2024/05/11'),
    OrderAmount: 399.00
  },
  {
    OrderName: '给老婆买花',
    OrderDate: new Date('2024/05/11'),
    OrderAmount: 99.00
  },
  {
    OrderName: '给自己买手机',
    OrderDate: new Date('2024/05/11'),
    OrderAmount: 9999.00
  }
]
  • 在页面中生成数据,用ForEach循环
interface PayRecord {
  OrderName:string
  OrderDate:Date
  OrderAmount:number
}
@Entry
@Component
struct ForEachCase {
  @State showDialog: boolean = false
  @State PayRecordList: PayRecord[] = [
    {
      OrderName: '给老婆买口红',
      OrderDate: new Date('2024/05/11'),
      OrderAmount: 399.00
    },
    {
      OrderName: '给老婆买花',
      OrderDate: new Date('2024/05/11'),
      OrderAmount: 99.00
    },
    {
      OrderName: '给自己买手机',
      OrderDate: new Date('2024/05/11'),
      OrderAmount: 9999.00
    }
  ]

  build() {
    Stack() {
      Column() {
        // 标题
        Row() {
          Text('支付记录')
            .layoutWeight(1)
            .textAlign(TextAlign.Center)
            .margin({
              left: 30
            })
        }
        .width('100%')
        .padding(16)
        .border({
          width: {
            bottom: 1
          },
          color: '#f4f5f6'
        })

        // 列表
        Column() {
          // 要循环的结构体
          // Column({ space: 20 }) {
          //   Text('给老婆买了一朵花')
          //     .fontWeight(FontWeight.Bold)
          //     .width('100%')
          //   Row() {
          //     Text('¥43.00')
          //     Text('2024/5/11')
          //   }
          //   .width('100%')
          //   .justifyContent(FlexAlign.SpaceBetween)
          // }
          // .width('100%')
          // .padding(20)
          // 使用循环组件,依次生成结构体
          ForEach(this.PayRecordList, (item: PayRecord) => {
            Column({ space: 20 }) {
              Text(item.OrderName)
                .fontWeight(FontWeight.Bold)
                .width('100%')
              Row() {
                Text('¥' + item.OrderAmount.toFixed(2))
                  .fontColor(Color.Red)
                Text(item.OrderDate.toLocaleDateString())
              }
              .width('100%')
              .justifyContent(FlexAlign.SpaceBetween)
            }
            .width('100%')
            .padding(20)
          })

        }
        .justifyContent(FlexAlign.Start)
        .width('100%')
        .layoutWeight(1)

      }
      .width('100%')
      .height('100%')
    }
  }
}

image.png

image.png

  • 新建一个list数据,进行循环
    warning
    使用下面的接口
interface GoodItem {
  goods_name: string
  goods_price: number
  goods_img: string
  goods_count: number
  id: number
}
  • 拷贝图片到assets

图片.zip
image.png

  • 声明数据
@State list: GoodItem[] = [
    {
      "id": 1,
      "goods_name": "班俏BANQIAO超火ins潮卫衣女士2020秋季新款韩版宽松慵懒风薄款外套带帽上衣",
      "goods_img": "assets/1.webp",
      "goods_price": 108,
      "goods_count": 1,
    },
    {
      "id": 2,
      "goods_name": "嘉叶希连帽卫衣女春秋薄款2020新款宽松bf韩版字母印花中长款外套ins潮",
      "goods_img": "assets/2.webp",
      "goods_price": 129,
      "goods_count": 1,
    },
    {
      "id": 3,
      "goods_name": "思蜜怡2020休闲运动套装女春秋季新款时尚大码宽松长袖卫衣两件套",
      "goods_img": "assets/3.webp",
      "goods_price": 198,
      "goods_count": 1,
    },
    {
      "id": 4,
      "goods_name": "思蜜怡卫衣女加绒加厚2020秋冬装新款韩版宽松上衣连帽中长款外套",
      "goods_img": "assets/4.webp",
      "goods_price": 99,
      "goods_count": 1,
    },
    {
      "id": 5,
      "goods_name": "幂凝早秋季卫衣女春秋装韩版宽松中长款假两件上衣薄款ins盐系外套潮",
      "goods_img": "assets/5.webp",
      "goods_price": 156,
      "goods_count": 1,
    },
    {
      "id": 6,
      "goods_name": "ME&CITY女装冬季新款针织抽绳休闲连帽卫衣女",
      "goods_img": "assets/6.webp",
      "goods_price": 142.8,
      "goods_count": 1,
    },
    {
      "id": 7,
      "goods_name": "幂凝假两件女士卫衣秋冬女装2020年新款韩版宽松春秋季薄款ins潮外套",
      "goods_img": "assets/7.webp",
      "goods_price": 219,
      "goods_count": 2,
    },
    {
      "id": 8,
      "goods_name": "依魅人2020休闲运动衣套装女秋季新款秋季韩版宽松卫衣 时尚两件套",
      "goods_img": "assets/8.webp",
      "goods_price": 178,
      "goods_count": 1,
    },
    {
      "id": 9,
      "goods_name": "芷臻(zhizhen)加厚卫衣2020春秋季女长袖韩版宽松短款加绒春秋装连帽开衫外套冬",
      "goods_img": "assets/9.webp",
      "goods_price": 128,
      "goods_count": 1,
    },
    {
      "id": 10,
      "goods_name": "Semir森马卫衣女冬装2019新款可爱甜美大撞色小清新连帽薄绒女士套头衫",
      "goods_img": "assets/10.webp",
      "goods_price": 153,
      "goods_count": 1,
    }
  ]

使用ForEach遍历

import { GoodItem } from './models'

@Entry
@Component
struct ForEachGoodCase {
  @State list: GoodItem[] = [
    {
      "id": 1,
      "goods_name": "班俏BANQIAO超火ins潮卫衣女士2020秋季新款韩版宽松慵懒风薄款外套带帽上衣",
      "goods_img": "assets/1.webp",
      "goods_price": 108,
      "goods_count": 1,
    },
    {
      "id": 2,
      "goods_name": "嘉叶希连帽卫衣女春秋薄款2020新款宽松bf韩版字母印花中长款外套ins潮",
      "goods_img": "assets/2.webp",
      "goods_price": 129,
      "goods_count": 1,
    },
    {
      "id": 3,
      "goods_name": "思蜜怡2020休闲运动套装女春秋季新款时尚大码宽松长袖卫衣两件套",
      "goods_img": "assets/3.webp",
      "goods_price": 198,
      "goods_count": 1,
    },
    {
      "id": 4,
      "goods_name": "思蜜怡卫衣女加绒加厚2020秋冬装新款韩版宽松上衣连帽中长款外套",
      "goods_img": "assets/4.webp",
      "goods_price": 99,
      "goods_count": 1,
    },
    {
      "id": 5,
      "goods_name": "幂凝早秋季卫衣女春秋装韩版宽松中长款假两件上衣薄款ins盐系外套潮",
      "goods_img": "assets/5.webp",
      "goods_price": 156,
      "goods_count": 1,
    },
    {
      "id": 6,
      "goods_name": "ME&CITY女装冬季新款针织抽绳休闲连帽卫衣女",
      "goods_img": "assets/6.webp",
      "goods_price": 142.8,
      "goods_count": 1,
    },
    {
      "id": 7,
      "goods_name": "幂凝假两件女士卫衣秋冬女装2020年新款韩版宽松春秋季薄款ins潮外套",
      "goods_img": "assets/7.webp",
      "goods_price": 219,
      "goods_count": 2,
    },
    {
      "id": 8,
      "goods_name": "依魅人2020休闲运动衣套装女秋季新款秋季韩版宽松卫衣 时尚两件套",
      "goods_img": "assets/8.webp",
      "goods_price": 178,
      "goods_count": 1,
    },
    {
      "id": 9,
      "goods_name": "芷臻(zhizhen)加厚卫衣2020春秋季女长袖韩版宽松短款加绒春秋装连帽开衫外套冬",
      "goods_img": "assets/9.webp",
      "goods_price": 128,
      "goods_count": 1,
    },
    {
      "id": 10,
      "goods_name": "Semir森马卫衣女冬装2019新款可爱甜美大撞色小清新连帽薄绒女士套头衫",
      "goods_img": "assets/10.webp",
      "goods_price": 153,
      "goods_count": 1,
    }
  ]

  build() {
    List({ space: 20 }) {
      ForEach(this.list, (item: GoodItem) => {
        ListItem() {
          Row({ space: 10 }) {
            Image(item.goods_img)
              .borderRadius(8)
              .width(120)
              .height(200)
            Column() {
              Text(item.goods_name)
                .fontWeight(FontWeight.Bold)
              Text("¥ "+item.goods_price.toString())
                .fontColor(Color.Red)
                .fontWeight(FontWeight.Bold)
            }
            .padding({
              top: 5,
              bottom: 5
            })
            .alignItems(HorizontalAlign.Start)
            .justifyContent(FlexAlign.SpaceBetween)
            .height(200)
            .layoutWeight(1)
          }
          .width('100%')
        }
      })
    }
    .padding(20)
  }
}
  • 练习两列布局
    warning
    image.png
import { GoodItem } from './models'
import { SegmentButton, SegmentButtonOptions } from '@ohos.arkui.advanced.SegmentButton'

@Entry
@Component
struct ForEachGoodCase {
  @State tabOptions: SegmentButtonOptions = SegmentButtonOptions.tab({
    buttons: [{ text: '单列' }, { text: '双列' }]
  })
  @State
  tabSelectedIndexes: number[] = [0]
  @State list: GoodItem[] = [
    {
      "id": 1,
      "goods_name": "班俏BANQIAO超火ins潮卫衣女士2020秋季新款韩版宽松慵懒风薄款外套带帽上衣",
      "goods_img": "assets/1.webp",
      "goods_price": 108,
      "goods_count": 1,
    },
    {
      "id": 2,
      "goods_name": "嘉叶希连帽卫衣女春秋薄款2020新款宽松bf韩版字母印花中长款外套ins潮",
      "goods_img": "assets/2.webp",
      "goods_price": 129,
      "goods_count": 1,
    },
    {
      "id": 3,
      "goods_name": "思蜜怡2020休闲运动套装女春秋季新款时尚大码宽松长袖卫衣两件套",
      "goods_img": "assets/3.webp",
      "goods_price": 198,
      "goods_count": 1,
    },
    {
      "id": 4,
      "goods_name": "思蜜怡卫衣女加绒加厚2020秋冬装新款韩版宽松上衣连帽中长款外套",
      "goods_img": "assets/4.webp",
      "goods_price": 99,
      "goods_count": 1,
    },
    {
      "id": 5,
      "goods_name": "幂凝早秋季卫衣女春秋装韩版宽松中长款假两件上衣薄款ins盐系外套潮",
      "goods_img": "assets/5.webp",
      "goods_price": 156,
      "goods_count": 1,
    },
    {
      "id": 6,
      "goods_name": "ME&CITY女装冬季新款针织抽绳休闲连帽卫衣女",
      "goods_img": "assets/6.webp",
      "goods_price": 142.8,
      "goods_count": 1,
    },
    {
      "id": 7,
      "goods_name": "幂凝假两件女士卫衣秋冬女装2020年新款韩版宽松春秋季薄款ins潮外套",
      "goods_img": "assets/7.webp",
      "goods_price": 219,
      "goods_count": 2,
    },
    {
      "id": 8,
      "goods_name": "依魅人2020休闲运动衣套装女秋季新款秋季韩版宽松卫衣 时尚两件套",
      "goods_img": "assets/8.webp",
      "goods_price": 178,
      "goods_count": 1,
    },
    {
      "id": 9,
      "goods_name": "芷臻(zhizhen)加厚卫衣2020春秋季女长袖韩版宽松短款加绒春秋装连帽开衫外套冬",
      "goods_img": "assets/9.webp",
      "goods_price": 128,
      "goods_count": 1,
    },
    {
      "id": 10,
      "goods_name": "Semir森马卫衣女冬装2019新款可爱甜美大撞色小清新连帽薄绒女士套头衫",
      "goods_img": "assets/10.webp",
      "goods_price": 153,
      "goods_count": 1,
    }
  ]

  build() {
    Column() {
      SegmentButton({ options: this.tabOptions, selectedIndexes: $tabSelectedIndexes  })
      if(this.tabSelectedIndexes.includes(0)) {
        List({ space: 20 }) {
          ForEach(this.list, (item: GoodItem) => {
            ListItem() {
              Row({ space: 10 }) {
                Image(item.goods_img)
                  .borderRadius(8)
                  .width(120)
                  .height(200)
                Column() {
                  Text(item.goods_name)
                    .fontWeight(FontWeight.Bold)
                  Text("¥ "+item.goods_price.toString())
                    .fontColor(Color.Red)
                    .fontWeight(FontWeight.Bold)
                }
                .padding({
                  top: 5,
                  bottom: 5
                })
                .alignItems(HorizontalAlign.Start)
                .justifyContent(FlexAlign.SpaceBetween)
                .height(200)
                .layoutWeight(1)
              }
              .width('100%')
            }
          })
        }
        .padding(20)
      }
      else if(this.tabSelectedIndexes.includes(1)) {
        Grid() {
          ForEach(this.list, (item: GoodItem) => {

            GridItem() {
              Column() {
                Image(item.goods_img)
                  .height(200)
                Row({ space: 2 }) {
                  Text("¥ " +item.goods_price)
                    .fontWeight(FontWeight.Bold)
                    .fontColor(Color.Red)
                  Text(item.goods_name)
                    .maxLines(1)
                    .layoutWeight(1)
                    .fontWeight(FontWeight.Bold)
                }
              }
            }
          })
        }
        .columnsGap(20)
        .rowsGap(20)
        .columnsTemplate("1fr 1fr")
        .padding(20)
      }
    }
  }
}

// 750 -UIios  30px * (720/750)
// 720 1080
  • key的推荐建议

ForEach的第三个属性是一个回调,它是生成唯一key的, 不传的话会帮助我们生成独一无二的key
index_ + JSON.stringify(item)

鸿蒙更新的原理:循环的比较-比较你的key存在不,0_“zhangsan” 如果存在相同的key,则不更新
只改动了某一条数据,可能所有列表都会更新
ForEach的第三个参数 宁可不给 也不要瞎给

warning
image.png

info
如果数组会发生插入,删除的操作,不要使用index作为key的值,不给key渲染也是正常的,但是可能会造成渲染性能降低

  • 下面是使用Index作为key的案例
@Entry
@Component
struct Parent {
  @State simpleList: Array<string> = ['one', 'two', 'three'];

  build() {
    Column() {
      Button() {
        Text('在第1项后插入新项').fontSize(30)
      }
      .onClick(() => {
        this.simpleList.splice(1, 0, 'new item');
      })

      ForEach(this.simpleList, (item: string) => {
        ChildItem({ item: item })
      }, (item: string, index: number) => index.toString())
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .height('100%')
    .backgroundColor(0xF1F3F5)
  }
}

@Component
struct ChildItem {
  @Prop item: string;

  build() {
    Text(this.item)
      .fontSize(30)
  }
}

info
我们发现虽然数据是正确的,但是由于鸿蒙针对组件的创建原则,因为key值是索引,发现key值存在的组件存在,并不会重新创建,所以会导致数据渲染非预期限制,要想改正这个问题,要保证key的唯一性,去掉key生成器可以保证唯一性

今日案例-知乎评论

image.png

拆解组件

  • 新建ZhiHu的文件夹,新建page
  • 新建components, 里面新建 HmNavBar和HmCommentItem
@Preview
@Component
struct HmNavBar {
  title: string = "标题"
  build() {
    Row() {
      // 返回键
      Row() {
        Image($r('app.media.ic_public_left_arrow'))
          .width(16)
          .height(16)
      }
      .width(30)
      .height(30)
      .borderRadius(15)
      .backgroundColor("#f4f4f4")
      .justifyContent(FlexAlign.Center)
      .margin({
        left: 20
      })
      Text(this.title)
        .layoutWeight(1)
        .textAlign(TextAlign.Center)
        .margin({
          right: 50
        })
    }
    .width('100%')
    .height(50)
    .border({
      color: "#f4f5f6",
      width: {
        bottom: 1
      }
    })
  }
}
export { HmNavBar }
  • HmCommentItem
@Component
struct HmCommentItem {
  build() {
    Row({ space: 10 }) {
      Image("https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fsafe-img.xhscdn.com%2Fbw1%2F1bad8264-7428-44cf-a92d-3016a2de537b%3FimageView2%2F2%2Fw%2F1080%2Fformat%2Fjpg&refer=http%3A%2F%2Fsafe-img.xhscdn.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=auto?sec=1711626934&t=5478cb3adef5d3e29e6952934797ca39")
        .width(40)
        .height(40)
        .borderRadius(20)

      Column({ space: 10 }) {
        Text("周杰伦")
          .fontColor("#303a43")
          .fontSize(18)
          .fontWeight(FontWeight.Bold)

        Text("黄河江最近一代都带蓝牙,意大利拌面必须使用42👌钢筋混凝土量子力学")
          .fontColor("#2f3642")
          .lineHeight(22)
        Row() {
          Text("10-21 .IP属地北京")
            .fontColor("#cacaca")
            .fontSize(12)

          Row({ space: 4 }) {
            Image($r("app.media.ic_public_like"))
              .width(12)
              .height(12)
              .fillColor("#cacaca")

            Text("100")
              .fontColor("#cacaca")
              .fontSize(12)

          }
        }
        .justifyContent(FlexAlign.SpaceBetween)
        .width('100%')
      }
      .alignItems(HorizontalAlign.Start)
      .layoutWeight(1)

    }
    .alignItems(VerticalAlign.Top)
    .padding(20)
    .width('100%')
  }
}
export { HmCommentItem }
  • 在components/index.ets统一导出
export * from './HmCommentItem'
export * from './HmNavBar'
  • 在ZhiHU/ZHihu.ets中使用
import { HmCommentItem, HmNavBar } from './components'

@Entry
@Component
struct ZhiHu {
  build() {
   Column() {
     HmNavBar({ title: '评论回复'  })
     HmCommentItem()
     Divider().strokeWidth(6)
     Row() {
       Text("评论数50")
     }
     .width('100%')
     .height(50)
     .padding({
       left: 20
     })
     .border({
       color: '#f3f4f5',
       width: {
         bottom: 1
       }
     })
     ForEach([1,2,3,4,5,6], () => {
       HmCommentItem()
     })
   }
  }
}

image.png

需要出现滚动区域。
使用了List组件,子组件必须有ListItem/ListItemGroup

 List() {
       ForEach([1,2,3,4,5,6], () => {
         ListItem() {
           HmCommentItem()
         }
       })
     }.layoutWeight(1)

image.png

评论列表

在ZhiHu/models/index.ets下建立如下类型

  • 定义一个评论的interface
export interface ReplyItem {
  avatar: ResourceStr // 头像
  author: string   // 作者
  id: number  // 评论的id
  content: string // 评论内容
  time: string // 发表时间
  area: string // 地区
  likeNum: number // 点赞数量
  likeFlag: boolean | null // 当前用户是否点过赞
}

info
用i2c生成对应的class

  • 我们前面的循环为什么没有用i2c,直接用的接口呢,因为这里涉及到后续的点赞和其他业务,所以这里直接用class更新起来会更方便
export class ReplyItemModel implements ReplyItem {
  id: number = 0
  avatar: string | Resource = ''
  author: string = ''
  content: string = ''
  time: string = ''
  area: string = ''
  likeNum: number = 0
  likeFlag: boolean | null = null

  constructor(model: ReplyItem) {
    this.id = model.id
    this.avatar = model.avatar
    this.author = model.author
    this.content = model.content
    this.time = model.time
    this.area = model.area
    this.likeNum = model.likeNum
    this.likeFlag = model.likeFlag
  }
}
  • 定义一个评论列表数据- 在Entry组件中
    info
    因为我们需要的是class对象,所以每个对象都需要new一下
 @State commentList: ReplyItemModel[] = [
   new ReplyItemModel({
     id: 1,
     avatar: 'https://picx.zhimg.com/027729d02bdf060e24973c3726fea9da_l.jpg?source=06d4cd63',
     author: '偏执狂-妄想家',
     content: '更何况还分到一个摩洛哥[惊喜]',
     time: '11-30',
     area: '海南',
     likeNum: 34,
     likeFlag: false
   }) ,
    new ReplyItemModel({
      id: 2,
      avatar: 'https://pic1.zhimg.com/v2-5a3f5190369ae59c12bee33abfe0c5cc_xl.jpg?source=32738c0c',
      author: 'William',
      content: '当年希腊可是把1:0发挥到极致了',
      time: '11-29',
      area: '北京',
      likeNum: 58,
      likeFlag: false
    }),
    new ReplyItemModel({
      id: 3,
      avatar: 'https://picx.zhimg.com/v2-e6f4605c16e4378572a96dad7eaaf2b0_l.jpg?source=06d4cd63',
      author: 'Andy Garcia',
      content: '欧洲杯其实16队球队打正赛已经差不多,24队打正赛意味着正赛阶段在小组赛一样有弱队。',
      time: '11-28',
      area: '上海',
      likeNum: 10,
      likeFlag: false
    }),
    new ReplyItemModel({
      id: 4,
      avatar: 'https://picx.zhimg.com/v2-53e7cf84228e26f419d924c2bf8d5d70_l.jpg?source=06d4cd63',
      author: '正宗好鱼头',
      content: '确实眼红啊,亚洲就没这种球队,让中国队刷',
      time: '11-27',
      area: '香港',
      likeNum: 139,
      likeFlag: false
    }),
    new ReplyItemModel({
      id: 5,
      avatar: 'https://pic1.zhimg.com/v2-eeddfaae049df2a407ff37540894c8ce_l.jpg?source=06d4cd63',
      author: '柱子哥',
      content: '我是支持扩大的,亚洲杯欧洲杯扩到32队,世界杯扩到64队才是好的,世界上有超过200支队伍,欧洲区55支队伍,亚洲区47支队伍,即使如此也就六成出现率',
      time: '11-27',
      area: '旧金山',
      likeNum: 29,
      likeFlag: false
    }),
    new ReplyItemModel({
      id: 6,
      avatar: 'https://picx.zhimg.com/v2-fab3da929232ae911e92bf8137d11f3a_l.jpg?source=06d4cd63',
      author: '飞轩逸',
      content: '禁止欧洲杯扩军之前,应该先禁止世界杯扩军,或者至少把亚洲名额一半给欧洲。',
      time: '11-26',
      area: '里约',
      likeNum: 100,
      likeFlag: false
    })
  ]

  • 在主页中渲染
 List() {
       ForEach(this.commentList, (item: ReplyItemModel) => {
         ListItem() {
           HmCommentItem({ item })
         }
       })
     }.layoutWeight(1)
  • CommentItem组件接收传入数据
import { ReplyItem, ReplyItemModel } from '../models'

@Component
struct HmCommentItem {
  // 接收渲染的选项
  item: ReplyItemModel = new ReplyItemModel({} as ReplyItem) // 初始值 只是为了语法不报错

  build() {
    Row({ space: 10 }) {
      Image(this.item.avatar)
        .width(40)
        .height(40)
        .borderRadius(20)

      Column({ space: 10 }) {
        Text(this.item.author)
          .fontColor("#303a43")
          .fontSize(18)
          .fontWeight(FontWeight.Bold)

        Text(this.item.content)
          .fontColor("#2f3642")
          .lineHeight(22)
        Row() {
          Text(`${this.item.time} .IP属地${this.item.area}`)
            .fontColor("#cacaca")
            .fontSize(12)

          Row({ space: 4 }) {
            Image($r("app.media.ic_public_like"))
              .width(12)
              .height(12)
              .fillColor("#cacaca")

            Text(this.item.likeNum.toString())
              .fontColor("#cacaca")
              .fontSize(12)

          }
        }
        .justifyContent(FlexAlign.SpaceBetween)
        .width('100%')
      }
      .alignItems(HorizontalAlign.Start)
      .layoutWeight(1)

    }
    .alignItems(VerticalAlign.Top)
    .padding(20)
    .width('100%')
  }
}
export { HmCommentItem }
  • 顶部组件同样需要new 对象传入过去
 HmCommentItem({
        item: new ReplyItemModel({
          id: 999,
          author: '周杰伦',
          avatar: $r("app.media.10"),
          likeNum: 10,
          likeFlag: false,
          time: '03-02',
          area: '北京',
          content: '人到了一定的年龄新陈代谢就慢了,吃了胖不吃瘦了皱纹就多,要靠锻炼 '
        })
     })

image.png

底部回复按钮

封装底部的回复组件
Zhihu/components/HmReplyInput.ets

@Component
struct HmReplyInput {
  @State
  content: string = ""
  build() {
   Row({ space: 10 }) {
     TextInput({ text: $$this.content, placeholder: '~请留下您的神评论' })
       .layoutWeight(1)
       .height(40)
     Button("发布")
   }
   .padding({ left: 10, right: 10 })
    .width('100%')
    .height(60)
  }
}
export { HmReplyInput }
  • 组件在主页中使用
 List() {
       ForEach(this.commentList, (item: ReplyItemModel) => {
         ListItem() {
           HmCommentItem({ item })
         }
       })
     }.layoutWeight(1)
     HmReplyInput()  // 在此刻显示

实现点赞

子组件如何调用父组件的函数

  • 子组件要声明一个函数
test: () => void = () => {}  // test:变量类型 = 初始值
  • 父组件需要给子组件传入这个参数
Child({
  test: () => {
    this.abc()
  }
})
  • 实现点赞
  changeLike: () => void = () => {}

  • 注册点击事件
 Row({ space: 4 }) {
            Image($r("app.media.ic_public_like"))
              .width(12)
              .height(12)
              .fillColor(this.item.likeFlag ? Color.Red : "#cacaca")

            Text(this.item.likeNum.toString())
              .fontColor(this.item.likeFlag ? Color.Red :"#cacaca")
              .fontSize(12)
          }
          .onClick(() => {
            this.changeLike()
          })
  • 父组件定义函数,传入函数
changeLike (item: ReplyItemModel) {
    // 需要拿到点击的数据 拿到数据更新数据即可
   // item.likeNum
   // item.likeFlag
    if(item.likeFlag) {
      // 点过赞
      item.likeNum--
    }
    else {
      // 没有点过赞
      item.likeNum++
    }
    item.likeFlag = !item.likeFlag // 取反
    // State的修饰符的更新机制
    // 只能监测到第一层
   const index = this.commentList.findIndex(obj => obj.id === item.id)
   //this.commentList[index] = item // 这么写为什么不行 // 引用类型 基础类型
    this.commentList[index] = new ReplyItemModel(item)
  }

  • 传入子组件
List() {
       ForEach(this.commentList, (item: ReplyItemModel) => {
         ListItem() {
           HmCommentItem({
             item,
             changeLike: () => {
               this.changeLike(item)
             }
           })
         }
       })
     }.layoutWeight(1)

更新的秘密

  • 鸿蒙里面的所有的更新都只能监测到一层的更新
  • 如果要更新数组里面的某一项的话
  • this.list[index] = 新值
  • this.list.splice(index, 1, 新值)
  • 关于key的秘密

尝试给了一个id作为key,为什么没有更新
因为鸿蒙会根据key的不同来更新的内容,如果key前后一样,它认为你没有变,那就不更新

image.png

提交代码

顶部的点赞

  • 将顶部的数据抽提出来
  @State
  showTop: boolean = true
@State
  currentComment: ReplyItemModel = new ReplyItemModel({
    id: 999,
    author: '周杰伦',
    avatar: $r("app.media.10"),
    likeNum: 10,
    likeFlag: false,
    time: '03-02',
    area: '北京',
    content: '人到了一定的年龄新陈代谢就慢了,吃了胖不吃瘦了皱纹就多,要靠锻炼 '
  })
  • 赋值
 if(this.showTop) {
       HmCommentItem({
         item: this.currentComment,
         changeLike: () => {
           this.changeLike(this.currentComment, "top")
           this.showTop = false
           setTimeout(() => {
             this.showTop = true
           }, 100)
         }
       })
     }
  • 改造点赞的方法
changeLike (item: ReplyItemModel, type?: "top" | "bottom") {
    // 需要拿到点击的数据 拿到数据更新数据即可
   // item.likeNum
   // item.likeFlag
    if(item.likeFlag) {
      // 点过赞
      item.likeNum--
    }
    else {
      // 没有点过赞
      item.likeNum++
    }
    item.likeFlag = !item.likeFlag // 取反
    // promptAction.showToast({ message: JSON.stringify(item), duration: 300000 })
    if(type !== "top") {
      // State的修饰符的更新机制
      // 只能监测到第一层
      const index = this.commentList.findIndex(obj => obj.id === item.id)
      //this.commentList[index] = item // 这么写为什么不行 // 引用类型 基础类型
      //  this.commentList[index] = new ReplyItemModel(item)
      this.commentList.splice(index, 1, new ReplyItemModel(item))
    }
  }

提交代码

回复评论

  • 底部输入组件双向绑定
import { promptAction } from '@kit.ArkUI'

@Component
struct HmReplyInput {
  @State
  content: string = ""
  publishComment: (content: string) => void = () => {}
  build() {
   Row({ space: 10 }) {
     TextInput({ text: $$this.content, placeholder: '~请留下您的神评论' })
       .layoutWeight(1)
       .height(40)
       .onSubmit(() => {
         // 键盘的确定事件
         if(this.content) {
           this.publishComment(this.content)
           this.content = ""
         }
       })
     Button("发布")
       .onClick(() => {
           if(this.content) {
             this.publishComment(this.content)
             this.content = ""
           }
       })
   }
   .padding({ left: 10, right: 10 })
    .width('100%')
    .height(60)
  }
}

export { HmReplyInput }
  • 调用父组件传入的publishComment的方法
  • 父组件实现的方法
addComment(content: string) {
     this.commentList.unshift(new ReplyItemModel({
       id: Math.random() ,
       avatar: 'https://foruda.gitee.com/avatar/1705232317138324256/1759638_itcast_panpu_1705232317.png',
       author: '老潘',
       content,
       time: `${(new Date().getMonth() + 1).toString().padStart(2, "0")}-${new Date().getDate().toString().padStart(2,  "0")}`,
       area: '北京',
       likeNum: 0,
       likeFlag: false
     }))

    // 控制滚动条
    this.scroller.scrollEdge(Edge.Top)
  }
  • 实现传入方法
  HmReplyInput({
       publishComment: (content: string) => {
          this.addComment(content)
       }
     })
  • 实现滚动顶部
//创建scroller
scroller:Scroller = new Scroller()
//传入scroller
List({ scroller: this.scroller })

image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1984155.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CCPC 2024, Shanghai

2024.8.5 12:00————17:00 CCPC 2024, Shanghai [A - 无线网络整点栅格统计](https://atcoder.jp/contests/math-and-algorithm/tasks/abc204_d?langen)[E - 无线软件日](https://atcoder.jp/contests/abc265/tasks/abc265_a?langen)[J - 极简合数序列](https://atcoder.…

【面试题】合井K个升序链表

合井K个升序链表 仅供面试学习 给定一个链表数组&#xff0c;每个链表都已经按升序排列。将所有链表合并到一个升序链表中&#xff0c;并返回合并后的链表。以下是详细的解题步骤和 Python 代码示例。 一、问题描述 将多个升序链表合并为一个升序链表。要求算法能有效处理链表…

Codeforces Round 871 (Div. 4)(A~H)

比赛链接 Dashboard - Codeforces Round 871 (Div. 4) - Codeforces A. Love Story 找到与codeforces 有多少个不同的字符。 #include<bits/stdc.h> #define int long long #define TEST int T; cin >> T; while (T--) #define ios ios::sync_with_stdio(fals…

人大金仓安装图文

1.下载 通过百度网盘分享的文件&#xff1a;人大金仓安装图文 链接&#xff1a;https://pan.baidu.com/s/1imt0KsyVXQALp_icEMoDpQ 提取码&#xff1a;zwef --来自百度网盘超级会员V3的分享 2.安装

CTFHub~RCE远程代码执行

0X01eval执行 # 进来我们直接看到了代码&#xff0c;大致意思就是传入一个参数cmd# 所以我们直接使用蚁剑进行连接&#xff0c;看一下能否成功# 接下来我们直接找出flag0X02文件包含 # 打开页面发现一大堆代码 大体意思是如果file中没有flag字符串就执行下面的include $_GET[…

【linux深入剖析】线程控制 | 多线程

&#x1f341;你好&#xff0c;我是 RO-BERRY &#x1f4d7; 致力于C、C、数据结构、TCP/IP、数据库等等一系列知识 &#x1f384;感谢你的陪伴与支持 &#xff0c;故事既有了开头&#xff0c;就要画上一个完美的句号&#xff0c;让我们一起加油 目录 1. 创建线程2. POSIX线程…

基于Kubernetes v1.25.0和Docker部署高可用集群(说明篇)

目录描述 Kubernetes组件说明 特定接口CRI、CNI、CSI Kubernetes v1.25集群创建方案选择 一、Kubernetes组件说明 Docker 的运行机制 运行机制详解&#xff1a; docker client&#xff1a;命令行输入的docker命令就是一个docker客户端 docker Engine&#xff1a;Engine也…

Java: 死锁问题详解(5000字)

文章目录 死锁的出现场景1. 一个线程一把锁,这个线程针对这把锁,连续加锁了两次2. 两个线程,两把锁3. N个线程 , M个锁4. 内存可见性为什么会出现内存可见性问题呢?解决方法 volatile关键字 总结synchronized:死锁的四个必要条件(缺一不可)[重点]:内存可见性问题: 死锁的出现场…

PCL安装与配置(PCL1.9.1+MSVC2017)

为了和我的VS的版本VS 2017对应&#xff0c;PCL下载的也是msvc_2017,PCL msvc2017最新的则是1.901版本&#xff0c;我们就以PCL 1.9.1为例了。&#xff08;如果你的vs是2019和2022&#xff0c;一定要注意PCL的版本&#xff09;。 一、下载PCL 我们打开PCL的github下载地址&am…

GDB调试器

GDB调试器 GDB的主要功能 常见命令 3、实战 1、生成能调试的执行文件&#xff08;一定要加-g&#xff09; 第一个是不能调试的 第二个这样加了-g才能进行调试 如果没加-g 执行gdb 执行文件&#xff08;会报下面这个 &#xff09; 像这样才是正常的 执行 gdb a_yes_g 这…

SSM计算机组成原理课程平台-计算机毕设定制-附项目源码(可白嫖)50168

摘 要 21世纪的今天&#xff0c;随着社会的不断发展与进步&#xff0c;人们对于信息科学化的认识&#xff0c;已由低层次向高层次发展&#xff0c;由原来的感性认识向理性认识提高&#xff0c;管理工作的重要性已逐渐被人们所认识&#xff0c;科学化的管理&#xff0c;使信息存…

金融行业到底该选择什么样的FTP替代方案?

2018年以来&#xff0c;受“华为、中兴事件”影响&#xff0c;我国科技尤其是上游核心技术受制于人的现状对我 国经济发展提出了严峻考验。在全球产业从工业经济向数字经济升级的关键时期&#xff0c;中国明确 “数字中国”建设战略&#xff0c; 抢占数字经济产业链制高点。 在…

Python开发工具PyCharm入门指南 - 用户界面主题更改

JetBrains PyCharm是一种Python IDE&#xff0c;其带有一整套可以帮助用户在使用Python语言开发时提高其效率的工具。此外&#xff0c;该IDE提供了一些高级功能&#xff0c;以用于Django框架下的专业Web开发。 界面主题定义了窗口、对话框、按钮和用户界面的所有可视元素的外观…

vscode开发avalonia

安装 安装.net 8 安装avalonia模板 dotnet new install Avalonia.Templates创建项目 dotnet new avalonia.app -o GetStartedApp安装c# dev kit插件和Avalonia for VSCode Community dotnet run运行 修改代码 MainWindow.axaml <Window xmlns"https://githu…

企业层面经济政策不确定性感知数据(2001-2023年)

1.指标介绍 企业经济政策不确定性感知指的是企业在面对政府经济政策变动时所感受到的风险和不确定性程度&#xff0c;这种感知会影响企业的投资决策、生产计划和市场策略 文章根据上市公司披露的MD&A文本&#xff0c;提取指标衡量企业个体面临的经济政策不确定性。 2.参…

了解线性回归、岭回归和套索回归

逐步对 Linear、Ridge 和 Lasso 回归进行数学理解。 ​ LASSO&#xff08;左&#xff09;和岭回归&#xff08;右&#xff09;的约束区域 一、说明 在本文中&#xff0c;我们将深入探讨机器学习中两种基本正则化技术的基础和应用&#xff1a;Ridge 回归和 Lasso 回归。这些方…

脊髓损伤小伙伴的活力重启秘籍! 让我们一起动起来,拥抱不一样的精彩生活✨

Hey小伙伴们~&#x1f44b; 今天咱们来聊聊一个超级重要又温暖的话题——脊髓损伤后的锻炼大法来啦&#xff01;&#x1f389; 记住&#xff0c;无论遇到什么挑战&#xff0c;我们都要像打不死的小强一样&#xff0c;活力满满地面对每一天&#xff01;&#x1f4aa; 首先&#…

基础进阶-搭建pxe网络安装环境实现服务器自动部署

目录 原理解释 ​编辑 开机界面解释 搭建步骤 下载环境需要用到的基本程序 查看帮助 帮助内容解释 环境搭建 修改 DHCP 修改 default 文件 测试 原理解释 开机界面解释 在开机界面中&#xff0c;圈起来的部分显示的就是光盘&#xff0c;我们需要将光盘转换成网络的 在…

.NET内网实战:模拟Installer关闭Defender

01基本介绍 02编码实现 原理上通过Windows API函数将当前进程的权限提升至TrustedInstaller&#xff0c;从而实现了对Windows Defender服务的控制。通常可以利用Windows API中的OpenSCManager、OpenProcessToken、ImpersonateLoggedOnUser以及ControlService等函数协同工作&am…

Modbus-Ascii详解

目录 Modbus-Ascii详解 Modbus-Ascii帧结构 LRC效验 将数据字节转成ASCII 将返回帧转为数据字节 Modbus-Ascii的实现 使用NModbus4实现Modbus-Ascii 实例 Modbus-Ascii详解 Modbus ASCII是一种将二进制数据转换为可打印的ASCII字符的通信协议&#xff0c;‌每个8位数据需要两…