鸿蒙开发:动态添加节点

关键词:组件,节点,上下树

首先要看官网介绍,然后我们开始尝试动态添加节点

首先,我们先创建一个节点类MyNodeController

kotlin 复制代码
class MyNodeController extends NodeController {
  // 将 meg 设为实例属性
  private meg: string = "传入了具体的数据,使用的时候会自动解析";
  private buttonNode: BuilderNode<[string]> | null = null;
  private wrapBuilder: WrappedBuilder<[string]> = wrapBuilder(textBuilder);

  constructor(uiContext: UIContext, meg?: string) {
    super();
    if (meg) {
      this.meg = meg;
      this.makeNode(uiContext);
    }
  }

  makeNode(uiContext: UIContext): FrameNode | null {
    if (this.buttonNode === null) {
      this.buttonNode = new BuilderNode(uiContext);
      this.buttonNode.build(this.wrapBuilder, this.meg);
    }
    return this.buttonNode.getFrameNode();
  }
}

我们将要塞入的自定义结构就是用WrapBuilder封起来,里面放入自定义的组件

全局Builder函数如下

scss 复制代码
@Builder
function textBuilder(message: string) {
  Text(message)
    .fontSize(20)
    .fontWeight(FontWeight.Bold)
}

现在 我们准备创建一个页面,用于渲染节点,通过 NodeContainer进行渲染

因为我这想要让本地数据覆盖类中的数据,更加自定义一些

所以,页面如下

typescript 复制代码
import { BuilderNode, NodeController, UIContext } from '@kit.ArkUI'

@Entry
@Component
struct NodePage {
  @State msg: string = "本地化的数据传入"
  private myNodeController: MyNodeController = new MyNodeController(this.getUIContext(),this.msg)

  build() {
    Column({ space: 10 }) {
      NodeContainer(this.myNodeController)
    }
    .size({ width: "100%", height: "100%" })
    .backgroundColor(Color.Orange)
    .justifyContent(FlexAlign.Center)
    .opacity(0.5)
  }
}

页面如下

但是目前还不够,我们需要数据上的交互才能算完整了功能

这里,我们的思路就是更新节点的数据

类中先更改,添加方法changgeMsg

kotlin 复制代码
// 提供公共方法修改 meg
changeMsg(msg: string) {
  this.meg = msg;
  // 若 buttonNode 已创建,更新节点内容
  if (this.buttonNode) {
    this.buttonNode.build(this.wrapBuilder, this.meg);
  }
}

所以这里单独说的就是,如果直接更改类中的数据,节点数据是不会刷新的

很好,现在,直接在页面中添加按钮,分别是,增加数据的内容,和直接更改数据内容,更新后的所有代码如下

kotlin 复制代码
import { BuilderNode, NodeController, UIContext } from '@kit.ArkUI'

@Entry
@Component
struct NodePage {
  @State msg: string = "本地化的数据传入"
  private myNodeController: MyNodeController = new MyNodeController(this.getUIContext(),this.msg)

  build() {
    Column({ space: 10 }) {
      NodeContainer(this.myNodeController)

      Button("改变文字 加一个好字").onClick(() => {
        this.msg = this.msg + "好 "
        this.myNodeController.changeMsg(this.msg )
      })

      Button("新按钮").onClick(() => {
        this.msg = "new msg"
        this.myNodeController.changeMsg(this.msg)
      })
    }
    .size({ width: "100%", height: "100%" })
    .backgroundColor(Color.Orange)
    .justifyContent(FlexAlign.Center)
    .opacity(0.5)
  }
}

@Builder
function textBuilder(message: string) {
  Text(message)
    .fontSize(20)
    .fontWeight(FontWeight.Bold)
}


class MyNodeController extends NodeController {
  // 将 meg 设为实例属性
  private meg: string = "传入了具体的数据,使用的时候会自动解析";
  private buttonNode: BuilderNode<[string]> | null = null;
  private wrapBuilder: WrappedBuilder<[string]> = wrapBuilder(textBuilder);

  constructor(uiContext: UIContext, meg?: string) {
    super();
    if (meg) {
      this.meg = meg;
      this.makeNode(uiContext);
    }
  }

  // 提供公共方法修改 meg
  changeMsg(msg: string) {
    this.meg = msg;
    // 若 buttonNode 已创建,更新节点内容
    if (this.buttonNode) {
      this.buttonNode.build(this.wrapBuilder, this.meg);
    }
  }

  makeNode(uiContext: UIContext): FrameNode | null {
    if (this.buttonNode === null) {
      this.buttonNode = new BuilderNode(uiContext);
      this.buttonNode.build(this.wrapBuilder, this.meg);
    }
    return this.buttonNode.getFrameNode();
  }
}

代码效果展示如下

到了这里,我们基本就知道应该如何动态添加节点了,不过,我们的知识面还不完整

这里就要介绍一下FrameNode

通过FrameNode我们可以添加节点、删除和获取子节点

我们通过typeNode初始化各种样式组件 然后 配置给frameNode

实现代码如下

scala 复制代码
import { FrameNode, NodeController, typeNode } from '@kit.ArkUI'

@Entry
@Component
struct DemoPage {
  private myNodeController: MyNodeController = new MyNodeController()

  build() {
    Column() {
      NodeContainer(this.myNodeController)
    }
    .height('100%')
    .width('100%')
    .justifyContent(FlexAlign.Center)
  }
}


class MyNodeController extends NodeController {
  makeNode(uiContext: UIContext): FrameNode {
    let rootNode = new FrameNode(uiContext)
    //创建一个Column组件
    let column = typeNode.createNode(uiContext, "Column")
    column.initialize()
      .width("100%")
      .height("100%")
      .justifyContent(FlexAlign.Center)
    rootNode.appendChild(column)
    //创建一个Text组件
    let text = typeNode.createNode(uiContext, "Text")
    text.initialize("文本组件")
      .fontColor(Color.Red)
      .fontSize(20)
      .padding(10)
      .border({ width: 1, color: Color.Red })
    column.appendChild(text)
    return rootNode
  }
}

当然,这些都是基本的使用和认识,要想熟悉具体的内容,还是离不开官方文档

鸿蒙官方文档链接 developer.huawei.com/consumer/cn...

相关推荐
前端付豪几秒前
1、为什么浏览器要有渲染流程? ——带你一口气吃透 Critical Rendering Path
前端·后端·浏览器
前端付豪3 分钟前
3、Node.js异步编程彻底吃透
前端·后端·node.js
孤鸿玉8 分钟前
[Flutter小试牛刀] 低配版signals,添加局部刷新
前端·flutter
亦黑迷失9 分钟前
轻量级 Express 服务器:用 Pug 模板引擎实现动态参数传递
前端·javascript·后端
吃瓜群众i1 小时前
理解Javascript闭包
前端·javascript
安大桃子1 小时前
Mapbox GL + Deck.gl 三维实战:Mapbox 加载 Tileset3D 倾斜摄影模型
前端·webgl
yede1 小时前
多行文本省略号显示,更多按钮展开全部
前端
就是我1 小时前
React 应用性能优化实战
前端·react.js·性能优化
G扇子1 小时前
深入解析XSS攻击:从原理到防御的全方位指南
前端·安全