我们以一个具体的示例来说明ArkTS的基本组成。如下图所示,当开发者点击按钮时,文本内容从“Hello World”变为“Hello ArkUI”。
类、结构、方法以及变量
,并赋予其特殊的含义。如上述示例中@Entry
、@Component
和@State
都是装饰器,@Component
表示自定义组件,@Entry
表示该自定义组件为入口组件,@State
表示组件中的状态变量,状态变量变化会触发UI刷新。build()
方法中的代码块。@Componen
t装饰的struct Hello。Column、Text、Divider、Button
。fontSize()
、width()
、height()
、backgroundColor()
等。Button
后面的onClick()
。根据组件构造方法的不同,创建组件包含有参数和无参数两种方式。
如果组件的接口定义没有包含必选构造参数,则组件后面的“()”不需要配置任何内容。例如,Divider组件不包含构造参数:
Column() {
Text('item 1')
Divider()
Text('item 2')
}
如果组件的接口定义包含构造参数,则在组件后面的“()”配置相应参数。
Image('https://xyz/test.jpg')
// string类型的参数
Text('test')
// $r形式引入应用资源,可应用于多语言场景
Text($r('app.string.title_value'))
// 无参数形式
Text()
Image(this.imagePath)
Image('https://' + this.imageUrl)
Text(`count: ${this.count}`)
属性方法以“.
”链式调用的方式配置系统组件的样式和其他属性,建议每个属性方法单独写一行。
Text('test')
.fontSize(12)
Image('test.jpg')
.alt('error.jpg')
.width(100)
.height(100)
Text('hello')
.fontSize(this.size)
Image('test.jpg')
.width(this.count % 2 === 0 ? 100 : 200)
.height(this.offset + 100)
Text('hello')
.fontSize(20)
.fontColor(Color.Red)
.fontWeight(FontWeight.Bold)
事件方法以“.
”链式调用的方式配置系统组件支持的事件,建议每个事件方法单独写一行。
Button('Click me')
.onClick(() => {
this.myText = 'ArkUI';
})
bind
,以确保函数体中的this
指向当前组件。Button('add counter')
.onClick(function(){
this.counter += 2;
}.bind(this))
myClickHandler(): void {
this.counter += 2;
}
...
Button('add counter')
.onClick(this.myClickHandler.bind(this))
fn = () => {
console.info(`counter: ${this.counter}`)
this.counter++
}
...
Button('add counter')
.onClick(this.fn)
如果组件支持子组件配置,则需在尾随闭包"{…}"中为组件添加子组件的UI描述。Column、Row、Stack、Grid、List等组件都是容器组件。
容器组件均支持子组件配置,可以实现相对复杂的多级嵌套
Column() {
Row() {
Image('test1.jpg')
.width(100)
.height(100)
Button('click +1')
.onClick(() => {
console.info('+1 clicked!');
})
}
}
在ArkUI中,UI显示的内容均为组件,由框架直接提供的称为系统组件,由开发者定义的称为自定义组件。在进行 UI 界面开发时,通常不是简单的将系统组件进行组合使用,而是需要考虑代码可复用性、业务逻辑与UI分离,后续版本演进等因素。因此,将UI和部分业务逻辑封装成自定义组件是不可或缺的能力。
@Component
struct HelloComponent {
@State message: string = 'Hello, World!';
build() {
// HelloComponent自定义组件组合系统组件Row和Text
Row() {
Text(this.message)
.onClick(() => {
// 状态变量message的改变驱动UI刷新,UI从'Hello, World!'刷新为'Hello, ArkUI!'
this.message = 'Hello, ArkUI!';
})
}
}
}
HelloComponent可以在其他自定义组件中的build()函数中多次创建,实现自定义组件的重用。
相当于其它语言中的函数,方法等等。
通过组件名({参数})
来实现调用。
@Entry
@Component
struct ParentComponent {
build() {
Column() {
Text('ArkUI message')
HelloComponent({ message: 'Hello, World!' });
Divider()
HelloComponent({ message: '你好!' });
}
}
}
struct
:自定义组件基于struct实现,struct + 自定义组件名 + {…}的组合构成自定义组件,不能有继承关系。对于struct的实例化,可以省略new。@Component
:@Component装饰器仅能装饰struct关键字声明的数据结构。struct被@Component装饰后具备组件化的能力,需要实现build方法描述UI,一个struct只能被一个@Component装饰。@Component
struct MyComponent {
}
build()
函数:build()函数用于定义自定义组件的声明式UI描述,自定义组件必须定义build()函数。@Component
struct MyComponent {
build() {
}
}
@Entry
:@Entry装饰的自定义组件将作为UI页面的入口。在单个UI页面中,最多可以使用@Entry装饰一个自定义组件。@Entry可以接受一个可选的LocalStorage的参数。@Entry
@Component
struct MyComponent {
}
自定义组件除了必须要实现build()函数外,还可以实现其他成员函数,成员函数具有以下约束: