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

参考

相关推荐
人工智能训练师15 小时前
Ubuntu22.04如何安装新版本的Node.js和npm
linux·运维·前端·人工智能·ubuntu·npm·node.js
Seveny0715 小时前
pnpm相对于npm,yarn的优势
前端·npm·node.js
yddddddy16 小时前
css的基本知识
前端·css
昔人'16 小时前
css `lh`单位
前端·css
Nan_Shu_61418 小时前
Web前端面试题(2)
前端
知识分享小能手18 小时前
React学习教程,从入门到精通,React 组件核心语法知识点详解(类组件体系)(19)
前端·javascript·vue.js·学习·react.js·react·anti-design-vue
蚂蚁RichLab前端团队19 小时前
🚀🚀🚀 RichLab - 花呗前端团队招贤纳士 - 【转岗/内推/社招】
前端·javascript·人工智能
孩子 你要相信光19 小时前
css之一个元素可以同时应用多个动画效果
前端·css
huangql52019 小时前
npm 发布流程——从创建组件到发布到 npm 仓库
前端·npm·node.js
Days205020 小时前
LeaferJS好用的 Canvas 引擎
前端·开源