建立你自己的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)

参考

相关推荐
dasseinzumtode5 分钟前
nestJS 使用ExcelJS 实现数据的excel导出功能
前端·后端·node.js
子兮曰8 分钟前
🔥C盘告急!WSL磁盘暴增?三招秒清20GB+空间
前端·windows·docker
Jinuss9 分钟前
Vue3源码reactivity响应式篇之EffectScope
前端·vue3
stoneship12 分钟前
网页截图API-Npm工具包分享
前端
Jedi Hongbin23 分钟前
Three.js shader内置矩阵注入
前端·javascript·three.js
etcix24 分钟前
dmenux.c: integrate dmenu project as one file
c语言·前端·算法
光影少年37 分钟前
react16到react19更新及底层实现是什么以及区别
前端·react.js·前端框架
超人不会飛40 分钟前
vue3 markdown组件|大模型应用专用
前端·vue.js·人工智能
じòぴé南冸じょうげん41 分钟前
微信小程序如何进行分包处理?
前端·小程序
Jolyne_1 小时前
Table自定义单元格渲染分享
前端