>[success] # Snabbdom
1. **Vue 2.x** 内部使用的 `Virtual DOM` 就是改造的 `Snabbdom`,官网对其介绍Snabbdom 则**极其简单、高效并且可拓展,同时核心代码 ≈ 200 行**。我们提供了一个**具有丰富功能同时支持自定义拓展的模块化结构**。为了使核心代码更简洁,所有**非必要的功能都将模块化引入**
>[info] ## Snabbdom 基本使用
1. `yarn init -y` 初始化项目
2. `yarn add snabbdom` 安装snabbdom
3. `yarn add webpack webpack-cli` 安装打包工具方便看运行效果
* 项目结构目录
~~~
|-- dist
| | -- index.html
| | -- main.js // webpack 打包后产出打包文件
|-- package.json
|-- pnpm-lock.yaml
|-- src
| |-- index.js
~~~
>[danger] ##### 简单案例
~~~
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app"></div>
<script src="./src/index.js"></script>
</body>
</html>
~~~
* src/index.js
~~~
import {
init,
classModule,
propsModule,
styleModule,
eventListenersModule,
h,
} from 'snabbdom'
// 注册使用模块
let patch = init([
classModule, // 开启 classes 功能
propsModule, // 支持传入 props
styleModule, // 支持内联样式同时支持动画
eventListenersModule, // 添加事件监听
])
// 创建 虚拟node 节点
let vnode = h('div#container.cls', 'hello word')
let app = document.querySelector('#app')
// 将虚拟node 节点插入指定容器中
let oldVnode = patch(app, vnode)
setTimeout(() => {
// 假设的时刻改变后的内容
vnode = h(
'div',
{
style: { color: 'yellow' },
},
'Hello Snabbdom'
)
// patch 传入新老的虚拟dom对象,进行比较重新渲染
patch(oldVnode, vnode)
}, 1000)
~~~
>[info] ## 使用 Snabbdom 简单说明
1. **Snabbdom 的核心库**并不能处理元素的**属性/样式/事件**等,如果需要处理的话,可以使用模块注册进入
2. 官方提供了六个模块
* `attributes设置 DOM 元素的属性`,使用`setAttribute ()`处理布尔类型的属性
* `props和 attributes 模块相似`,**设置 DOM 元素的属性中attributes设置 DOM 元素的属性**,使用 `setAttribute ()`处理布尔类型的属性`props`是`element[attr] = value`**不处理布尔类型的属性**
* `class切换类样式`,给元素设置类样式是通过 sel 选择器
* `dataset设置 data-* `的自定义属性
3. 决定使用模块进行导入模块,在`init()` 中注册模块使用 `h()` 创建`Vnode`,`init`函数返回的`patch`
* 关于`patch`函数 会接受两个参数,**一个 DOM 元素**或者 **一个表示当前视图的 vnode**,第一次创建时候第一个参数一般往往都是**一个 DOM 元素**,因为如果第一个参数传入一个包含父节点的 `DOM `元素,那么新的 `vnode `将转换为一个 `DOM `节点并替换传入的元素,后续为了进行比较新`老虚拟dom` 来优化渲染,所以新老的 `Vnode` 都要经过 `patch` 函数
* 案例点击时候切换dom
~~~
import { init, eventListenersModule, styleModule,h } from 'snabbdom'
// 2. 注册模块
let patch = init([styleModule, eventListenersModule])
// 3. 使用 h() 函数的第二个参数传入模块需要的数据(对象)
let vnode = h(
'div',
{
style: {
backgroundColor: 'red',
},
on: {
click: eventHandler,
},
},
[h('h1', 'Hello Snabbdom'), h('p', '这是p标签')]
)
let app = document.querySelector('#app')
let oldVnode = patch(app, vnode)
function eventHandler() {
vnode = h('div', 'hello')
patch(oldVnode, vnode)
}
~~~
>[info] ## 项目结构目录
~~~
│ h.ts h()函数,用来创建 VNode
│ hooks.ts 所有钩子函数的定义
│ htmldomapi.ts 对 DOM API 的包装
│ is.ts 判断数组和原始值的函数
│ jsx-global.d.ts jsx 的类型声明文件
│ jsx.ts 处理 jsx
│ snabbdom.bundle.ts 入口,已经注册了模块
│ snabbdom.ts 初始化,返回 init/h/thunk
│ thunk.ts 优化处理,对复杂视图不可变值得优化
│ tovnode.ts DOM 转换成 VNode
│ vnode.ts 虚拟节点定义
│
├─helpers
│ attachto.ts 定义了 vnode.ts 中 AttachData 的数据结构
│
└─modules 所有模块定义 / 钩子函数
attributes.ts
class.ts
dataset.ts
eventlisteners.ts
hero.ts example 中使用到的自定义钩子
module.ts 定义了模块中用到的钩子函数
props.ts
style.ts
~~~
>[info] ## 参考
[参考一个全程带注释链接](https://github.com/tramp-xu/codeAnalysis/blob/master/snabbdom-js/snabbdom.js)
- 工程化 -- Node
- vscode -- 插件
- vscode -- 代码片段
- 前端学会调试
- 谷歌浏览器调试技巧
- 权限验证
- 包管理工具 -- npm
- 常见的 npm ci 指令
- npm -- npm install安装包
- npm -- package.json
- npm -- 查看包版本信息
- npm - package-lock.json
- npm -- node_modules 层级
- npm -- 依赖包规则
- npm -- install 安装流程
- npx
- npm -- 发布自己的包
- 包管理工具 -- pnpm
- 模拟数据 -- Mock
- 页面渲染
- 渲染分析
- core.js && babel
- core.js -- 到底是什么
- 编译器那些术语
- 词法解析 -- tokenize
- 语法解析 -- ast
- 遍历节点 -- traverser
- 转换阶段、生成阶段略
- babel
- babel -- 初步上手之了解
- babel -- 初步上手之各种配置(preset-env)
- babel -- 初步上手之各种配置@babel/helpers
- babel -- 初步上手之各种配置@babel/runtime
- babel -- 初步上手之各种配置@babel/plugin-transform-runtime
- babel -- 初步上手之各种配置(babel-polyfills )(未来)
- babel -- 初步上手之各种配置 polyfill-service
- babel -- 初步上手之各种配置(@babel/polyfill )(过去式)
- babel -- 总结
- 各种工具
- 前端 -- 工程化
- 了解 -- Yeoman
- 使用 -- Yeoman
- 了解 -- Plop
- node cli -- 开发自己的脚手架工具
- 自动化构建工具
- Gulp
- 模块化打包工具为什么出现
- 模块化打包工具(新) -- webpack
- 简单使用 -- webpack
- 了解配置 -- webpack.config.js
- webpack -- loader 浅解
- loader -- 配置css模块解析
- loader -- 图片和字体(4.x)
- loader -- 图片和字体(5.x)
- loader -- 图片优化loader
- loader -- 配置解析js/ts
- webpack -- plugins 浅解
- eslit
- plugins -- CleanWebpackPlugin(4.x)
- plugins -- CleanWebpackPlugin(5.x)
- plugin -- HtmlWebpackPlugin
- plugin -- DefinePlugin 注入全局成员
- webapck -- 模块解析配置
- webpack -- 文件指纹了解
- webpack -- 开发环境运行构建
- webpack -- 项目环境划分
- 模块化打包工具 -- webpack
- webpack -- 打包文件是个啥
- webpack -- 基础配置项用法
- webpack4.x系列学习
- webpack -- 常见loader加载器
- webpack -- 移动端px转rem处理
- 开发一个自己loader
- webpack -- plugin插件
- webpack -- 文件指纹
- webpack -- 压缩css和html构建
- webpack -- 清里构建包
- webpack -- 复制静态文件
- webpack -- 自定义插件
- wepack -- 关于静态资源内联
- webpack -- source map 对照包
- webpack -- 环境划分构建
- webpack -- 项目构建控制台输出
- webpack -- 项目分析
- webpack -- 编译提速优护体积
- 提速 -- 编译阶段
- webpack -- 项目优化
- webpack -- DefinePlugin 注入全局成员
- webpack -- 代码分割
- webpack -- 页面资源提取
- webpack -- import按需引入
- webpack -- 摇树
- webpack -- 多页面打包
- webpack -- eslint
- webpack -- srr打包后续看
- webpack -- 构建一个自己的配置后续看
- webpack -- 打包组件和基础库
- webpack -- 源码
- webpack -- 启动都做了什么
- webpack -- cli做了什么
- webpack - 5
- 模块化打包工具 -- Rollup
- 工程化搭建代码规范
- 规范化标准--Eslint
- eslint -- 扩展配置
- eslint -- 指令
- eslint -- vscode
- eslint -- 原理
- Prettier -- 格式化代码工具
- EditorConfig -- 编辑器编码风格
- 检查提交代码是否符合检查配置
- 整体流程总结
- 微前端
- single-spa
- 简单上手 -- single-spa
- 快速理解systemjs
- single-sap 不使用systemjs
- monorepo -- 工程
- Vue -- 响应式了解
- Vue2.x -- 源码分析
- 发布订阅和观察者模式
- 简单 -- 了解响应式模型(一)
- 简单 -- 了解响应式模型(二)
- 简单 --了解虚拟DOM(一)
- 简单 --了解虚拟DOM(二)
- 简单 --了解diff算法
- 简单 --了解nextick
- Snabbdom -- 理解虚拟dom和diff算法
- Snabbdom -- h函数
- Snabbdom - Vnode 函数
- Snabbdom -- init 函数
- Snabbdom -- patch 函数
- 手写 -- 虚拟dom渲染
- Vue -- minVue
- vue3.x -- 源码分析
- 分析 -- reactivity
- 好文
- grpc -- 浏览器使用gRPC
- grcp-web -- 案例
- 待续