ArkUI通过自定义组件
的build()函数和@builder装饰器
中的声明式UI描述语句构建相应的UI。在声明式描述语句中开发者除了使用系统组件外,还可以使用渲染控制语句来辅助UI的构建,这些渲染控制语句包括控制组件是否显示的条件渲染语句,基于数组数据快速生成组件的循环渲染语句以及针对大数据量场景的数据懒加载语句。
ArkTS提供了渲染控制的能力。条件渲染可根据应用的不同状态,使用if、else和else if渲染对应状态下的UI内容。
当if、else if后跟随的状态判断中使用的状态变量值变化时,条件渲染语句会进行更新,更新步骤如下:
条件可以包括Typescript表达式。对于构造函数中的表达式,此类表达式不得更改应用程序状态。
- @Entry
- @Component
- struct ViewA {
- @State count: number = 0;
-
- build() {
- Column() {
- Text(`count=${this.count}`)
-
- if (this.count > 0) {
- Text(`count is positive`)
- .fontColor(Color.Green)
- }
-
- Button('increase count')
- .onClick(() => {
- this.count++;
- })
-
- Button('decrease count')
- .onClick(() => {
- this.count--;
- })
- }
- }
- }
if语句的每个分支都包含一个构建函数。此类构建函数必须创建一个或多个子组件。在初始渲染时,if语句会执行构建函数,并将生成的子组件添加到其父组件中。
每当if或else if条件语句中使用的状态变量发生变化时,条件语句都会更新并重新评估新的条件值。如果条件值评估发生了变化,这意味着需要构建另一个条件分支。此时ArkUI框架将:
在以上示例中,如果count从0增加到1,那么if语句更新,条件count > 0将重新评估,评估结果将从false更改为true。因此,将执行条件为真分支的构造函数,创建一个Text组件,并将它添加到父组件Column中。如果后续count更改为0,则Text组件将从Column组件中删除。由于没有else分支,因此不会执行新的构造函数。
以下示例包含if ... else ...语句与拥有@State装饰变量的子组件。
- @Component
- struct CounterView {
- @State counter: number = 0;
- label: string = 'unknown';
-
- build() {
- Row() {
- Text(`${this.label}`)
- Button(`counter ${this.counter} +1`)
- .onClick(() => {
- this.counter += 1;
- })
- }
- }
- }
-
- @Entry
- @Component
- struct MainView {
- @State toggle: boolean = true;
-
- build() {
- Column() {
- if (this.toggle) {
- CounterView({ label: 'CounterView #positive' })
- } else {
- CounterView({ label: 'CounterView #negative' })
- }
- Button(`toggle ${this.toggle}`)
- .onClick(() => {
- this.toggle = !this.toggle;
- })
- }
- }
- }
CounterView(label为 'CounterView #positive')子组件在初次渲染时创建。此子组件携带名为counter的状态变量。当修改CounterView.counter状态变量时,CounterView(label为 'CounterView #positive')子组件重新渲染时并保留状态变量值。当MainView.toggle状态变量的值更改为false时,MainView父组件内的if语句将更新,随后将删除CounterView(label为 'CounterView #positive')子组件。与此同时,将创建新的CounterView(label为 'CounterView #negative')实例。而它自己的counter状态变量设置为初始值0。
说明
CounterView(label为 'CounterView #positive')和CounterView(label为 'CounterView #negative')是同一自定义组件的两个不同实例。if分支的更改,不会更新现有子组件,也不会保留状态。
以下示例展示了条件更改时,若需要保留counter值所做的修改。
- @Component
- struct CounterView {
- @Link counter: number;
- label: string = 'unknown';
-
- build() {
- Row() {
- Text(`${this.label}`)
- Button(`counter ${this.counter} +1`)
- .onClick(() => {
- this.counter += 1;
- })
- }
- }
- }
-
- @Entry
- @Component
- struct MainView {
- @State toggle: boolean = true;
- @State counter: number = 0;
-
- build() {
- Column() {
- if (this.toggle) {
- CounterView({ counter: $counter, label: 'CounterView #positive' })
- } else {
- CounterView({ counter: $counter, label: 'CounterView #negative' })
- }
- Button(`toggle ${this.toggle}`)
- .onClick(() => {
- this.toggle = !this.toggle;
- })
- }
- }
- }
此处,@State counter变量归父组件所有。因此,当CounterView组件实例被删除时,该变量不会被销毁。CounterView组件通过@Link装饰器引用状态。状态必须从子级移动到其父级(或父级的父级),以避免在条件内容或重复内容被销毁时丢失状态。
条件语句的嵌套对父组件的相关规则没有影响。
- @Entry
- @Component
- struct CompA {
- @State toggle: boolean = false;
- @State toggleColor: boolean = false;
-
- build() {
- Column() {
- Text('Before')
- .fontSize(15)
- if (this.toggle) {
- Text('Top True, positive 1 top')
- .backgroundColor('#aaffaa').fontSize(20)
- // 内部if语句
- if (this.toggleColor) {
- Text('Top True, Nested True, positive COLOR Nested ')
- .backgroundColor('#00aaaa').fontSize(15)
- } else {
- Text('Top True, Nested False, Negative COLOR Nested ')
- .backgroundColor('#aaaaff').fontSize(15)
- }
- } else {
- Text('Top false, negative top level').fontSize(20)
- .backgroundColor('#ffaaaa')
- if (this.toggleColor) {
- Text('positive COLOR Nested ')
- .backgroundColor('#00aaaa').fontSize(15)
- } else {
- Text('Negative COLOR Nested ')
- .backgroundColor('#aaaaff').fontSize(15)
- }
- }
- Text('After')
- .fontSize(15)
- Button('Toggle Outer')
- .onClick(() => {
- this.toggle = !this.toggle;
- })
- Button('Toggle Inner')
- .onClick(() => {
- this.toggleColor = !this.toggleColor;
- })
- }
- }
- }