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

参考

相关推荐
豐儀麟阁贵1 分钟前
9.5格式化字符串
java·开发语言·前端·面试
春生野草8 分钟前
Ruoyi前端基于vue的脚手架的目录解析
前端·javascript·vue.js
m0_7400437320 分钟前
Axios拦截器 -- 请求拦截器和响应拦截器
开发语言·前端·javascript
不会代码的小猴1 小时前
C++的第十一天笔记
java·前端·jvm
北辰alk1 小时前
React Native vs React Web:深度对比与架构解析
react native·react.js
风止何安啊1 小时前
递归 VS 动态规划:从 “无限套娃计算器” 到 “积木式解题神器”
前端·javascript·算法
ohyeah1 小时前
使用 Vue 3 实现大模型流式输出:从零搭建一个简易对话 Demo
前端·vue.js·openai
GPTMirrors镜像系统1 小时前
JS 实现指定 UA 访问网站跳转弹窗提醒,解决夸克等浏览器兼容性问题
前端·javascript
踢球的打工仔1 小时前
前端html(2)
前端·算法·html
Rysxt_2 小时前
Vue文件下载功能完整指南:从基础实现到进阶实战
前端·javascript·vue.js