组件(Components)

组件是什么?

组件是 Yew 的基石。它们管理自己的状态,并可以渲染为 DOM。组件是通过实现描述组件生命周期的 Component trait 来创建的。

生命周期

组件(Components) - 图1备注

为我们的文档做出贡献:添加组件的生命周期图示

生命周期方法

Create

当一个组件被创建时,它会从其父组件以及一个 ComponentLink 接收属性(properties)。属性(properties)可用于初始化组件的状态,“link”可用于注册回调或向组件发送消息。

通常将 props 和 link 存储在组件的结构体中,如下所示:

  1. pub struct MyComponent {
  2. props: Props,
  3. link: ComponentLink<Self>,
  4. }
  5. impl Component for MyComponent {
  6. type Properties = Props;
  7. // ...
  8. fn create(props: Self::Properties, link: ComponentLink<Self>) -> Self {
  9. MyComponent { props, link }
  10. }
  11. // ...
  12. }

View

组件在 view() 方法中声明它的布局。Yew 提供了 html! 宏来声明 HTML 和 SVG 节点和它们的监听器及其子组件。这个宏的行为很像 React 中的 JSX,但是使用的是 Rust 表达式而不是 JavaScript。

  1. impl Component for MyComponent {
  2. // ...
  3. fn view(&self) -> Html {
  4. let onclick = self.link.callback(|_| Msg::Click);
  5. html! {
  6. <button {onclick}>{ self.props.button_text }</button>
  7. }
  8. }
  9. }

有关用法的详细信息,请查看 html! 宏指南]

Mounted

mounted() 组件生命周期方法调用是在 view() 被处理并且 Yew 已经把组件挂载到 DOM 上之后,浏览器刷新页面之前。组件通常希望实现此方法以执行只能在组件渲染元素之后才能执行的操作。如果你想在做出一些更改后重新渲染组件,返回 true 就可以了。

  1. use stdweb::web::html_element::InputElement;
  2. use stdweb::web::IHtmlElement;
  3. use yew::prelude::*;
  4. pub struct MyComponent {
  5. node_ref: NodeRef,
  6. }
  7. impl Component for MyComponent {
  8. // ...
  9. fn view(&self) -> Html {
  10. html! {
  11. <input ref={self.node_ref.clone()} type="text" />
  12. }
  13. }
  14. fn mounted(&mut self) -> ShouldRender {
  15. if let Some(input) = self.node_ref.cast::<InputElement>() {
  16. input.focus();
  17. }
  18. false
  19. }
  20. }

组件(Components) - 图2备注

请注意,此生命周期方法不要求必须实现,默认情况下不会执行任何操作。

Update

组件是动态的,可以注册以接收异步信息。update() 生命周期方法对于每个消息都会被调用。这使得组件可以根据消息的内容来更新自身,并决定是否需要重新渲染自己。消息可以由 HTML 元素监听器触发,或者由子组件,Agents,Services 或 Futures 发送。

update() 可能看起来像下面这个例子:

  1. pub enum Msg {
  2. SetInputEnabled(bool)
  3. }
  4. impl Component for MyComponent {
  5. type Message = Msg;
  6. // ...
  7. fn update(&mut self, msg: Self::Message) -> ShouldRender {
  8. match msg {
  9. Msg::SetInputEnabled(enabled) => {
  10. if self.input_enabled != enabled {
  11. self.input_enabled = enabled;
  12. true // 重新渲染
  13. } else {
  14. false
  15. }
  16. }
  17. }
  18. }
  19. }

Change

组件可能被其父节点重新渲染。发生这种情况时,它们可以接收新的属性(properties)并选择重新渲染。这种设计通过更改属性(properties)来促进父子组件之间的通信。你不是必须实现 change(),但是如果想在组件被创建后通过 props 来更新组件,则可能要这么做。

一个原始的实现可能看起来像:

  1. impl Component for MyComponent {
  2. // ...
  3. fn change(&mut self, props: Self::Properties) -> ShouldRender {
  4. self.props = props;
  5. true // 当提供了新的 props 将始终重新渲染。
  6. }
  7. }

Destroy

组件从 DOM 上被卸载后,Yew 调用 destroy() 生命周期方法来支持任何必要的清理操作。这个方法是可选的,默认情况下不执行任何操作。

关联类型

Component trait 有两个关联类型:MessageProperties

  1. impl Component for MyComponent {
  2. type Message = Msg;
  3. type Properties = Props;
  4. // ...
  5. }

Message 表示组件可以处理以触发某些副作用的各种消息。例如,你可能有一条 Click 消息,该消息触发 API 请求或者切换 UI 组件的外观。通常的做法是在组件模块中创建一个叫做 Msg 的枚举并将其用作组件中的消息类型。通常将“message”缩写为“msg”。

  1. enum Msg {
  2. Click,
  3. }

Properties 表示从父级传递到组件的信息。此类型必须实现 Properties trait(通常通过派生),并且可以指定某些属性(properties)是必需的还是可选的。创建和更新组件时使用此类型。通常的做法是在组件模块中创建一个叫做 Props 的结构体并将其用作组件的 Properties 类型。通常将“properties”缩写为“props”。由于 props 是从父组件传递下来的,因此应用程序的根组件通常有一个类型为 ()Properties。如果你希望为根组件指定属性(properties),请使用 App::mount_with_props 方法。