鸿蒙开发:动态添加节点

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

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

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

相关推荐
Wect3 分钟前
LeetCode 4. 寻找两个正序数组的中位数:二分优化思路详解
前端·算法·typescript
李剑一3 分钟前
纯干货,前端字体极致优化!谷歌、阿里、字节、腾讯都在用的终极解决方案,Vue3 + Vite 直接抄,页面提速不妥协!
前端·vue.js·面试
memeflyfly6 分钟前
Vercel 自动部署完全指南:从配置到问题排查
前端·前端工程化
星辰徐哥11 分钟前
C语言Web开发:CGI、FastCGI、Nginx深度解析
c语言·前端·nginx
暗不需求18 分钟前
JavaScript 面向对象探秘:从构造函数到原型链的优雅继承
前端·javascript
圣光SG21 分钟前
奶茶店网页(纯HTML和CSS)
前端·css·html
kyriewen21 分钟前
你还在给每个图片父元素加类名?CSS :has() 让选择器“逆天改命”
前端·css·面试
漫天黄叶远飞24 分钟前
async/await 到底怎么工作的?
前端
ai_xiaogui32 分钟前
PanelAI前端全面升级!私有化部署AI面板控制台+生态市场一键管理详解
前端·人工智能·comfyui一键部署·生态市场算力共享·ai面板控制台·panelai私有化部署·大模型前端管理
Jelena1577958579234 分钟前
1688.item_get_app接口:包装尺寸重量信息深度解析
开发语言·前端·python