myvue 模拟vue核心原理
// js部分index.js
class Myvue{ constructor(options){ this.data = options.data; this.dep = new Dep(); var id = options.el; this.observe(); var Dom = this.VnodeContainer(document.querySelector(id)); document.querySelector(id).appendChild(Dom); } VnodeContainer(node,flag){ // 虚拟DOM容器 var flag = flag || document.createDocumentFragment(); var child; while(child = node.firstChild){ this.compile(child); flag.appendChild(child); this.VnodeContainer(child,flag); } return flag; } compile(node){ // 编译DOM let reg = /\{\{(.*)\}\}/g; if(node.nodeType === 1){ // 元素类型 let attr = node.attributes; for(let i=0;i<attr.length;i++){ if(attr[i].nodeName === 'v-model'){ let name = attr[i].nodeValue; node.addEventListener('input',(e)=>{ this.data[name] = e.target.value; }); node.value = this.data[name]; this.dep.add(new Watcher(this.data,node,name)); } } } if(node.nodeType === 3){ // text类型节点 if(reg.test(node.nodeValue)){ let name = RegExp.$1; // 匹配到的字符串 name = name.trim(); node.nodeValue = this.data[name]; this.dep.add(new Watcher(this.data,node,name)); } } } observe(){ Object.keys(this.data).forEach((el)=>{ this.definePropertyInit(this.data,el,this.data[el]); }); } definePropertyInit(target,key,value){ // 将data做成响应式的 Object.defineProperty(target,key,{ get:()=>{ return value; }, set:(newVal)=>{ if(newVal === value) return; value = newVal; this.dep.notify(); // 更新 } }); } } class Dep{ // 发布者 constructor(){ this.subs = []; } add(sub){ this.subs.push(sub); } notify(){ this.subs.forEach((el)=>{ el.update(); }); } } class Watcher{ // 观察者(订阅者) constructor(vm,node,name){ Dep.global = this; this.vm = vm; this.node = node; this.name = name; this.update(); } update(){ this.get(); switch (this.node.nodeType) { case 1: // 标签元素 this.node.value = this.value; break; case 3: // 文本 this.node.nodeValue = this.value; break; default: break; } } get(){ this.value = this.vm[this.name]; } } let vm = new Myvue({ el: '#app', data: { msg: 'hello' } })
html部分
<!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"> <input type="text" id="input" v-model='msg'> <div id="box">{{msg}}</div> </div> <script src='./index.js'></script> </body> </html>