harmony 鸿蒙Creating a Custom Component

  • 2023-06-24
  • 浏览 (1009)

Creating a Custom Component

In ArkUI, components are what’s displayed on the UI. They can be classified as built-in components – those directly provided by the ArkUI framework, and custom components – those defined by developers. Defining the entire application UI with just built-in components would lead to a monolithic design, low code maintainability, and poor execution performance. A good UI is the result of a well-thought-out development process, with such factors as code reusability, separation of service logic from the UI, and version evolution carefully considered. Creating custom components that encapsulate the UI and some business logic is a critical step in this process.

The custom component has the following features:

  • Combinable: allows you to combine built-in components and other components, as well as their attributes and methods.

  • Reusable: can be reused by other components and used as different instances in different parent components or containers.

  • Data-driven update: holds some state and triggers UI re-rendering with the change of state variables.

The following example shows the basic usage of a custom component.

@Component
struct HelloComponent {
  @State message: string = 'Hello, World!';

  build() {
    // The HelloComponent custom component combines the <Row> and <Text> built-in components.
    Row() {
      Text(this.message)
        .onClick(() => {
          // The change of the state variable message drives the UI to be re-rendered. As a result, the text changes from "Hello, World!" to "Hello, ArkUI!".
          this.message = 'Hello, ArkUI!';
        })
    }
  }
}

Multiple HelloComponent instances can be created in the build() function of other custom components. In this way, HelloComponent is reused by those custom components.

class HelloComponentParam {
  message: string = ""
}

@Entry
@Component
struct ParentComponent {
  param: HelloComponentParam = {
    message: 'Hello, World!'
  }

  build() {
    Column() {
      Text('ArkUI message')
      HelloComponent(param);
      Divider()
      HelloComponent(param);
    }
  }
}

To fully understand the preceding example, a knowledge of the following concepts is essential:

Basic Structure of a Custom Component

  • struct: The definition of a custom component must start with the \@Component struct followed by the component name, and then component body enclosed by curly brackets {….}. No inheritance is allowed. You can omit the new operator when instantiating a struct. > NOTE > > The name or its class or function name of a custom component must be different from that of any built-in components.

  • \@Component: The \@Component decorator can decorate only the structs declared by the struct keyword. After being decorated by \@Component, a struct has the componentization capability. It must implement the build function to describe the UI. One struct can be decorated by only one \@Component. > NOTE > > Since API version 9, this decorator is supported in ArkTS widgets.

  @Component
  struct MyComponent {
  }
  • build(): The build() function is used to define the declarative UI description of a custom component. Every custom component must define a build() function.
  @Component
  struct MyComponent {
    build() {
    }
  }
  • \@Entry: A custom component decorated with \@Entry is used as the default entry component of the page. At most one component can be decorated with \@Entry in a single source file. The \@Entry decorator accepts an optional parameter of type LocalStorage.

NOTE

Since API version 9, this decorator is supported in ArkTS widgets.

Since API version 10, the \@Entry decorator accepts an optional parameter of type LocalStorage or type EntryOptions.

  @Entry
  @Component
  struct MyComponent {
  }

### EntryOptions10+

Describes the named route options.

Name Type Mandatory Description
routeName string No Name of the target named route.
storage LocalStorage No Storage of page-level UI state.
  @Entry({ routeName : 'myPage' })
  @Component
  struct MyComponent {
  }
  • \@Reusable: Custom components decorated by \@Reusable can be reused.

NOTE

Since API version 10, this decorator is supported in ArkTS widgets.

  @Reusable
  @Component
  struct MyComponent {
  }

Member Functions/Variables

In addition to the mandatory build() function, a custom component may implement other member functions with the following restrictions:

  • Static functions are not supported.

  • Access to the member functions is always private.

A custom component can also implement member variables with the following restrictions:

  • Static member variables are not supported.

  • Access to the member variables is always private. The access rules of member variables are the same as those of member functions.

  • Local initialization is optional for some member variables and mandatory for others. For details about whether local initialization or initialization from the parent component is required, see State Management.

Rules of for Custom Component Parameters

As can be learnt from preceding examples, a custom component can be created from a build or @Builder function, and during the creation, parameters can be supplied to the component.

@Component
struct MyComponent {
  private countDownFrom: number = 0;
  private color: Color = Color.Blue;

  build() {
  }
}

@Entry
@Component
struct ParentComponent {
  private someColor: Color = Color.Pink;

  build() {
    Column() {
      // Create an instance of MyComponent and initialize its countDownFrom variable with the value 10 and its color variable with the value this.someColor.
      MyComponent({ countDownFrom: 10, color: this.someColor })
    }
  }
}

build Function

All languages declared in the build function are called UI description languages. The UI description languages must comply with the following rules:

  • For an \@Entry decorated custom component, exactly one root component is required under the build function. This root component must be a container component. ForEach is not allowed at the top level. For an \@Component decorated custom component, exactly one root component is required under the build function. This root component is not necessarily a container component. ForEach is not allowed at the top level.
  @Entry
  @Component
  struct MyComponent {
    build() {
      // Exactly one root component is required, and it must be a container component.
      Row() {
        ChildComponent() 
      }
    }
  }

  @Component
  struct ChildComponent {
    build() {
      // Exactly one root component is required, and it is not necessarily a container component.
      Image('test.jpg')
    }
  }
  • Local variable declaration is not allowed. The following example is invalid:
  build() {
    // Invalid: Local variable declaration is not allowed.
    let a: number = 1;
  }
  • console.info cannot be directly used in the UI description, but can be used in methods or functions. The following is an example:
  build() {
    // Invalid: Use of console.info is not allowed.
    console.info('print debug log');
  }
  • Creation of a local scope is not allowed. The following example is invalid:
  build() {
    // Invalid: Creation of local scope is not allowed.
    {
      ...
    }
  }
  • Calling a function other than the \@Builder decorated is not allowed. The parameters of built-in components can be the return values of TS methods.
  @Component
  struct ParentComponent {
    doSomeCalculations() {
    }

    calcTextValue(): string {
      return 'Hello World';
    }

    @Builder doSomeRender() {
      Text(`Hello World`)
    }

    build() {
      Column() {
        // Invalid: No function calls except @Builder functions.
        this.doSomeCalculations();
        // Valid: The function can be called.
        this.doSomeRender();
        // Valid: The parameter can be the return value of a TS method.
        Text(this.calcTextValue())
      }
    }
  }
  • The switch syntax is not allowed. Use if instead. The following example is invalid:
  build() {
    Column() {
      // Invalid: The switch syntax is not allowed.
      switch (expression) {
        case 1:
          Text('...')
          break;
        case 2:
          Image('...')
          break;
        default:
          Text('...')
          break;
      }
    }
  }
  • Expressions are not allowed. The following example is invalid:
  build() {
    Column() {
      // Invalid: Expressions are not allowed.
      (this.aVar > 10) ? Text('...') : Image('...')
    }
  }

Universal Style of a Custom Component

The universal style of a custom component is configured by invoking chainable attribute methods.

@Component
struct MyComponent2 {
  build() {
    Button(`Hello World`)
  }
}

@Entry
@Component
struct MyComponent {
  build() {
    Row() {
      MyComponent2()
        .width(200)
        .height(300)
        .backgroundColor(Color.Red)
    }
  }
}

NOTE

When ArkUI sets styles for custom components, an invisible container component is set for MyComponent2. These styles are set on the container component instead of the <Button> component of MyComponent2. As seen from the rendering result, the red background color is not directly applied to the button. Instead, it is applied to the container component that is invisible to users where the button is located.

你可能感兴趣的鸿蒙文章

harmony 鸿蒙Quick Start

harmony 鸿蒙app.json5 Configuration File

harmony 鸿蒙Internal Structure of the app Tag

harmony 鸿蒙Application Configuration File Overview (FA Model)

harmony 鸿蒙Application Configuration File Overview (Stage Model)

harmony 鸿蒙Application Installation and Uninstallation Process

harmony 鸿蒙Application Package Overview

harmony 鸿蒙Application Package Structure in FA Model

harmony 鸿蒙Application Package Structure in Stage Model

harmony 鸿蒙Application Package Update Process

0  赞