Vue 底層編譯實現

kvue源碼

// 期待用法
// new KVue({
// data:{msg:'hello'}
// })
class KVue {
constructor(options) {
this.$options = options;
//處理data選項
this.$data = options.data;
// 響應化
this.observe(this.$data);
// new Watcher();
// this.$data.test;
// new Watcher();
// this.$data.foo.bar;
new Compile(options.el, this);
if (options.created) {
options.created.call(this);
}
}

observe(value) {
if (!value || typeof value !== 'object') {
return;
}
// 遍歷對象
Object.keys(value).forEach(key => {
this.defineReactive(value, key, value[key])
// 代理到vm上
this.proxyData(key);
})
}

proxyData(key) {
Object.defineProperty(this, key, {
get(){
return this.$data[key];
},
set(newVal){
this.$data[key] = newVal;
}
})
}
defineReactive(obj, key, val) {
const dep = new Dep();
Object.defineProperty(obj, key, {
get(){
// 將Dep.target添加到dep中
Dep.target && dep.addDep(Dep.target)
return val;
},
set(newVal){
if (newVal !== val) {
val = newVal;
// console.log(`${key}更新了:${newVal}`);
dep.notify();
}
}
})
// 遞歸
this.observe(val);
}
}
class Dep {
constructor(){
this.deps = [];
}
addDep(dep) {
this.deps.push(dep)
}
notify() {
this.deps.forEach(dep => dep.update())
}
}
class Watcher {
constructor(vm, key, cb) {
this.vm = vm;
this.key = key;
this.cb = cb;
Dep.target = this;
this.vm[this.key];// 添加watcher到dep
Dep.target = null;
}
update() {
// console.log('屬性更新了');
this.cb.call(this.vm, this.vm[this.key])
}
}

compile源碼

// new Compile(el, vm)
class Compile {
constructor(el, vm) {
this.$vm = vm;
this.$el = document.querySelector(el);
if (this.$el) {
// 提取宿主中模板內容到Fragment標籤,dom操做會提升效率
this.$fragment = this.node2Fragment(this.$el);
// 編譯模板內容,同時進行依賴收集
this.compile(this.$fragment);
this.$el.appendChild(this.$fragment);
}
}
node2Fragment(el) {
const fragment = document.createDocumentFragment();
let child;
while ((child = el.firstChild)) {
fragment.appendChild(child);
}
return fragment;
}
compile(el) {
const childNodes = el.childNodes;
Array.from(childNodes).forEach(node => {
// 判斷節點類型
if (node.nodeType === 1) {
// element節點
// console.log('編譯元素節點'+node.nodeName);
this.compileElement(node);
} else if (this.isInterpolation(node)) {
// 插值表達式
// console.log('編譯插值文本'+node.textContent);
this.compileText(node);
}
// 遞歸子節點
if (node.childNodes && node.childNodes.length > 0) {
this.compile(node);
}
});
}
isInterpolation(node) {
// 是文本且符合{{}}
return node.nodeType === 3 && /\{\{(.*)\}\}/.test(node.textContent);
}
compileElement(node) {
// <div k-model="foo" k-text="test" @click="onClick">
let nodeAttrs = node.attributes;
Array.from(nodeAttrs).forEach(attr => {
const attrName = attr.nameconst exp = attr.value;
if (this.isDirective(attrName)) {
const dir = attrName.substring(2);
this[dir] && this[dir](node, this.$vm, exp);
}
if (this.isEvent(attrName)) const dir = attrName.substring(1);
this.eventHandler(node, this.$vm, exp, dir);
}
});
}
isDirective(attr) {
return attr.indexOf("k-") === 0;
}
isEvent(attr) {
return attr.indexOf("@") === 0;
}

compileText(node) {
console.log(RegExp.$1);
this.update(node, this.$vm, RegExp.$1, "text");
}
update(node, vm, exp, dir) let updatrFn = this[dir + "Updater"];
updatrFn && updatrFn(node, vm[exp]);
// 依賴收集
new Watcher(vm, exp, function(value) {
updatrFn && updatrFn(node, value);
});
}
text(node, vm, exp) {
this.update(node, vm, exp, "text");
}
textUpdater(node, val) {
node.textContent = val;
}
eventHandler(node, vm, exp, dir) {
const fn = vm.$options.methods && vm.$options.methods[exp];
if (dir && fn) {
node.addEventListener(dir, fn.bind(vm));
}
}
html(node, vm, exp) {
this.update(node, vm, exp, "html");
}
model(node, vm, exp) {
// data -> view
this.update(node, vm, exp, "model");
// view -> data
node.addEventListener("input", e => {
vm[exp] = e.target.value;
});
}
htmlUpdater(node, value) {
node.innerHTML = value;
}
modelUpdater(node, value) {
node.value = value;
}
}

測試代碼

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge"<title>Document</title>
</head>
<body>
<div id="app">
<p>{{name}}</p>
<p k-text="name"></p>
<p>{{age}}</p>
<p>
{{doubleAge}}
</p>
<input type="text" k-model="name">
開課吧web全棧架構師<button @click="changeName">呵呵</button>
<div k-html="html"></div>
</div>
<script src='./kvue.js'></script>
<script src='./compile.js'></script>
<script>
const kaikeba = new KVue({
el: '#app',
data: {
name: "I am test.",
age: 12,
html: '<button>這是一個按鈕</button>'
},
created() {
console.log('開始啦')
setTimeout(() => {
this.name = '我是測試'
}, 1500)
},
methods: {
changeName() {
this.name = '哈嘍,開課吧'
this.age = 1
}
}
})
</script>
<!-- <script src="./kvue.js"></script>
<script>
const app = new KVue({
data: {
test: 'kaikeba',
foo: {bar:'bar'}
}
})
app.$data.test = '我變了'
app.$data.foo.bar = '我變了app.test = '我又變了'
</script> -->
</body>
</html>
相關文章
相關標籤/搜索