鸿蒙开发:动态添加节点

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

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

首先,我们先创建一个节点类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...

相关推荐
qq. 28040339845 小时前
CSS层叠顺序
前端·css
喝拿铁写前端6 小时前
SmartField AI:让每个字段都找到归属!
前端·算法
猫猫不是喵喵.6 小时前
vue 路由
前端·javascript·vue.js
烛阴6 小时前
JavaScript Import/Export:告别混乱,拥抱模块化!
前端·javascript
bin91536 小时前
DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加行拖拽排序功能示例12,TableView16_12 拖拽动画示例
前端·javascript·vue.js·ecmascript·deepseek
GISer_Jing7 小时前
[Html]overflow: auto 失效原因,flex 1却未设置min-height &overflow的几个属性以及应用场景
前端·html
程序员黄同学7 小时前
解释 Webpack 中的模块打包机制,如何配置 Webpack 进行项目构建?
前端·webpack·node.js
拉不动的猪7 小时前
vue自定义“权限控制”指令
前端·javascript·vue.js
再学一点就睡7 小时前
浏览器页面渲染机制深度解析:从构建 DOM 到 transform 高效渲染的底层逻辑
前端·css
拉不动的猪7 小时前
刷刷题48 (setState常规问答)
前端·react.js·面试