mini-react 动态渲染复杂的DOM结构

目标

实现一个更通用的虚拟DOM创建和渲染机制,能够动态渲染复杂的DOM结构

分析

  1. 创建虚拟DOM元素的函数 createElementcreateTextNodeElement
  2. 使用递归的方式将虚拟DOM转换为真实DOM并渲染到页面上
  3. 支持多层嵌套的虚拟DOM结构

关键点

  • 递归渲染函数 renderElement,根据虚拟DOM的类型创建对应的真实DOM节点
  • 处理文本节点和普通节点的区别
  • 遍历子节点并递归调用渲染函数

代码实现

创建文本虚拟DOM元素

javascript 复制代码
/**
 * 创建一个文本虚拟DOM元素
 * @param {*} text 
 * @returns 
 */
const createTextNodeElement = (text) => {
    return {
        type: 'TEXT_ELEMENT',
        props: {
            nodeValue: text,
            children: []
        }
    }
}

创建虚拟DOM元素

javascript 复制代码
/**
 * 创建一个虚拟DOM元素
 * @param {*} type 
 * @param {*} props 
 * @param  {...any} children 
 * @returns 
 */
const createElement = (type, props, ...children) => {
    return {
        type,
        props: {
            ...props,
            children: children.map(child => {
                // 如果子节点是对象,直接返回;如果是文本,则创建文本虚拟DOM元素
                return typeof child === 'object' ? child : createTextNodeElement(child);
            })
        }
    }
}

递归渲染虚拟DOM到真实DOM

javascript 复制代码
/**
 * 递归渲染虚拟DOM到真实DOM
 * 渲染算法实现逻辑:
 * 1. 判断虚拟DOM的类型,如果是文本节点则创建文本节点,否则创建对应的元素节点
 * 2. 设置元素节点的属性
 * 3. 遍历子节点,递归调用渲染函数,并将子节点添加到父节点下
 * 4. 返回创建的真实DOM节点
 * @param {*} element 
 * @returns 
 */
const renderElement = (element) => {
    let dom;
    if (element.type === 'TEXT_ELEMENT') {
        dom = document.createTextNode('');
        dom.nodeValue = element.props.nodeValue;
    } else {
        dom = document.createElement(element.type);
        // 设置属性
        for (const prop in element.props) {
            if (prop !== 'children') {
                dom[prop] = element.props[prop];
            }
        }
    }

    // 递归渲染子节点
    element.props?.children.forEach(child => {
        dom.appendChild(renderElement(child));
    });

    return dom;
}

创建复杂的虚拟DOM结构并渲染

javascript 复制代码
// 创建一个复杂的虚拟DOM结构, 可以无限的添加
const App = createElement(
    'div',
    { id: 'app' },
    createElement(
        'div',
        { id: 'div1' },
        createElement(
            'p',
            { id: 'p1' },
            createElement(
                'span', {},
                createTextNodeElement('Nested span inside paragraph.')
            )
        ),
        createTextNodeElement('Hello Mini React V5')
    ),
    "This is Mini React V5",
    createElement(
        'p',
        {},
        createTextNodeElement('This is a paragraph in Mini React V5.')
    )
);

// 渲染虚拟DOM到页面上
const dom = renderElement(App);
document.querySelector('#root').appendChild(dom);

console.log(dom);

实现总结

以上代码实现了一个更通用的虚拟DOM创建和渲染机制,包含以下几个步骤:

  1. 创建文本虚拟DOM元素函数 createTextNodeElement
  2. 创建普通虚拟DOM元素函数 createElement
  3. 递归渲染函数 renderElement,根据虚拟DOM的类型创建对应的真实DOM节点,并处理子节点的递归渲染
  4. 创建一个复杂的虚拟DOM结构 App,并将其渲染到页面上,最终在页面上显示出相应的内容

注意事项

当前实现方式支持多层嵌套的虚拟DOM结构,并能够动态渲染复杂的DOM结构

存在问题

并没有真正意义上的动态创建虚拟DOM元素对象,后续版本会改进这个问题,实现更完善的虚拟DOM创建和渲染机制。

相关推荐
柠檬味的Cat几秒前
使用腾讯云COS作为WordPress图床的实践
前端·github·腾讯云
Hilaku3 分钟前
卷AI、卷算法、2026 年的前端工程师到底在卷什么?
前端·javascript·面试
非凡ghost5 分钟前
AIMP(音乐播放软件)
前端·windows·音视频·firefox
xiaotao1317 分钟前
Vite 完全学习指南
前端·vite·前端打包
军军君0121 分钟前
Three.js基础功能学习十五:智能黑板实现实例二
开发语言·前端·javascript·vue.js·3d·threejs·三维
IT枫斗者28 分钟前
构建具有执行功能的 AI Agent:基于工作记忆的任务规划与元认知监控架构
android·前端·vue.js·spring boot·后端·架构
hotlinhao29 分钟前
Nginx rewrite last 与 redirect 的区别——Vue history 模式短链接踩坑记录
前端·vue.js·nginx
ZC跨境爬虫32 分钟前
海南大学交友平台开发实战day7(实现核心匹配算法+解决JSON请求报错问题)
前端·python·算法·html·json
下北沢美食家35 分钟前
CSS面试题2
前端·css
weixin_4617694041 分钟前
npm create vue@latest 错误
前端·vue.js·npm