建立你自己的react(三)----render函数

前言

今天是我们建立自己的react的第三天,昨天我们讲了createElement函数来生成虚拟dom,那么操作完dom以后,我们还要组成真实的dom,渲染到页面上,今天我们就讲渲染成真实dom,渲染到页面上。

render函数

接下来,我们要写我们的ReactDOM.render版本。

js 复制代码
ReactDOM.render(element, container)

现在,我们仅仅关心添加东西到dom里面,稍后我们将进行处理更新和删除

js 复制代码
function render(element, container) {
 // TODO create dom nodes
}
const Didact = {
  render,
}

Didact.render(element, container)

我们开始使用element类型创造DOM节点,然后把新的节点append到container中

js 复制代码
function render(element, container) {
    const dom = document.createElement(element.type)
    container.appendChild(dom)
}

我们递归子元素做同样的事情

js 复制代码
function render(element, container) {
    const dom = document.createElement(element.type)
    element.props.children.forEach(child =>
        render(child, dom)
    )
    container.appendChild(dom)
}

我们也需要处理一下文本元素,如果这个节点类型是TEXT_ELEMENT,那么我们创建一个文本节点而不是普通节点

js 复制代码
const dom =
element.type == "TEXT_ELEMENT"
? document.createTextNode("")
: document.createElement(element.type)

这里需要做的最后一件事儿是,把element的props分配给节点。

js 复制代码
const isProperty = key => key !== "children"
Object.keys(element.props)
.filter(isProperty)
.forEach(name => {
    dom[name] = element.props[name]
})

做到这里,现在我们有一个库可以render JSX到dom里面

js 复制代码
function createElement(type, props, ...children) {
  return {
    type,
    props: {
      ...props,
      children: children.map(child =>
        typeof child === "object"
          ? child
          : createTextElement(child)
      ),
    },
  }
}

function createTextElement(text) {
  return {
    type: "TEXT_ELEMENT",
    props: {
      nodeValue: text,
      children: [],
    },
  }
}

function render(element, container) {
  const dom =
    element.type == "TEXT_ELEMENT"
      ? document.createTextNode("")
      : document.createElement(element.type)

  const isProperty = key => key !== "children"
  Object.keys(element.props)
    .filter(isProperty)
    .forEach(name => {
      dom[name] = element.props[name]
    })

  element.props.children.forEach(child =>
    render(child, dom)
  )
  container.appendChild(dom)
}

const Didact = {
  createElement,
  render,
}

/** @jsx Didact.createElement */
const element = (
  <div id= "foo" >
     <a>bar < /a>
     < b />
  </div>
  )
const container = document.getElementById("root")
Didact.render(element, container)

参考

相关推荐
GIS之路3 分钟前
GDAL 实现矢量合并
前端
hxjhnct6 分钟前
React useContext的缺陷
前端·react.js·前端框架
前端 贾公子30 分钟前
从入门到实践:前端 Monorepo 工程化实战(4)
前端
菩提小狗33 分钟前
Sqlmap双击运行脚本,双击直接打开。
前端·笔记·安全·web安全
前端工作日常44 分钟前
我学习到的AG-UI的概念
前端
韩师傅1 小时前
前端开发消亡史:AI也无法掩盖没有设计创造力的真相
前端·人工智能·后端
XiaoYu20021 小时前
第12章 支付宝SDK
前端
双向331 小时前
RAG的下一站:检索增强生成如何重塑企业知识中枢?
前端
拖拉斯旋风1 小时前
从零开始:使用 Ollama 在本地部署开源大模型并集成到 React 应用
前端·javascript·ollama
栀秋6662 小时前
智能驱动的 Git 提交:基于 Ollama 大模型的规范化提交信息生成方案
react.js·llm·ollama