-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmyvue2.js
151 lines (147 loc) · 5 KB
/
myvue2.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
class Vue{
constructor(options){
this.$options = options;
this._data = options.data;
this.observer(this._data);
this.compile();
}
observer(data){
for(let key in data){
let value = data[key];
let _this = this;
let dep = new Dep(); //搜集watcher
Object.defineProperty(data,key,{
configurable:true,
enumerable:true,
get(){
console.log("get...");
if(Dep.target){
// 搜集watcher
dep.addSub(Dep.target);
}
console.log(dep);
return value;
},
set(newValue){
console.log("set...");
// 发布;
// 触发watcher里的update方法;
dep.notify(newValue);
// let event = new CustomEvent(key,{
// detail: newValue
// });
// _this.dispatchEvent(event)
value = newValue;
}
})
}
}
compile(){
let ele = document.querySelector(this.$options.el);
this.compileNodes(ele);
}
compileNodes(ele){
let childNodes = ele.childNodes;
// console.log(childNodes);
childNodes.forEach(node=>{
// console.log(node);
// 文本 元素节点;
if( node.nodeType ===1){
// 元素节点
let attrs = node.attributes;
// console.log(attrs);
[...attrs].forEach(attr=>{
// console.log(attr);
let attrName = attr.name;
let attrValue = attr.value;
// console.log(attrName,attrValue);
if(attrName==="v-text"){
node.innerText = this._data[attrValue];
} else if (attrName ==="v-model"){
node.value = this._data[attrValue];
node.addEventListener("input",e=>{
// console.log(e.target.value);
// 触发视图更新
this._data[attrValue] = e.target.value;
});
} else if (attrName ==="v-html"){
node.innerHTML = this._data[attrValue];
new Watcher(this._data, attrValue, (newValue) => {
// console.log("cb...",newValue)
// 更新视图逻辑
node.innerHTML = newValue;
});
}
})
if(node.childNodes.length>0){
this.compileNodes(node);
}
}else if(node.nodeType===3){
// 文本
// 匹配 ”{{}}“的文本
let reg = /\{\{\s*([^\{\}\s]+)\s*\}\}/g;
let textContent = node.textContent;
console.log("??",textContent);
if(reg.test(textContent)){
console.log("有大括号表达式");
// 获取数据的下标
// RegExp Function Object
let $1 = RegExp.$1;
// RegExp.$2
// console.log($1);
let rData = this._data[$1];
console.log(rData);
// console.log(this.$options.data);
// 将数据渲染到视图;
node.textContent = node.textContent.replace(reg,rData);
// this.addEventListener($1,e=>{
// console.log("自定义事件",e.detail);
// // 重新渲染模板;获取新值;
// let oldValue = this._data[$1];
// let newValue = e.detail;
// let reg = new RegExp(oldValue);
// node.textContent = node.textContent.replace(reg,newValue);
// })
// 实例化watcher --> 重新编译模板
new Watcher(this._data,$1,(newValue)=>{
// console.log("cb...",newValue)
// 更新视图逻辑
let oldValue = this._data[$1];
let reg = new RegExp(oldValue);
node.textContent = node.textContent.replace(reg,newValue);
});
}
}
})
}
}
// 依赖收集器
class Dep{
constructor(){
this.subs = [];
}
addSub(sub){
this.subs.push(sub);
}
// 发布;
notify(newValue){
this.subs.forEach(sub=>{
sub.update(newValue);
})
}
}
// 订阅者;
class Watcher{
constructor(data,key,cb){
Dep.target = this;
// 触发get 添加到dep里;
data[key];
this.cb = cb;
Dep.target = null
}
update(newValue){
console.log("update...")
this.cb(newValue);
}
}
// 作业: 在我的基础上实现 v-html 指令;