-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathvue.js
140 lines (111 loc) · 2.76 KB
/
vue.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
class Vue {
constructor({ el, data, beforeCreate, created, mounted, methods }) {
// Before Create
beforeCreate?.bind(this)();
this.$el = document.querySelector(el);
this.$data = data;
walkDataProps(this);
walkMethods(this, methods);
// Create
created?.bind(this)();
const render = renderVue(this);
walkDataProps(this, render);
render();
// Mount
if (mounted) {
this.$mount = mounted;
this.$mount();
}
}
}
//#region regex
const regex = {
vueOn: /(@|v-on:)\w+="([0-z.?]+)\(?\)?"/,
eventAtt: /vue-event(=""?)/,
mostach: /\{\{((?:.|\r?\n)+?)\}\}/,
};
//#endregion
//#region rendering
/**
* read the Vue.$el and replace the HTML to the vue's data.
* @param {Vue} vue
* @returns {() => void}
*/
function renderVue(vue) {
const originalTemplate = addAttributes(vue.$el.cloneNode(true));
return () => {
const { $data } = vue;
vue.$el.innerHTML = originalTemplate.innerHTML.replace(
new RegExp(regex.mostach, "g"),
(_, val) => $data[val.trim()]
);
addEvents(vue);
};
}
function addEvents(vue) {
vue.$el.querySelectorAll("[vue-event]").forEach((el) => {
// extract name attr and method of v-on:[event]=[method] or @[event]=[method]
const { name, value: method } = el.attributes[1];
// get event from v-on:[event] or @[event]
const event = /@/.test(name) ? name.slice(1) : name.split(":")[1];
el.addEventListener(event, vue[method].bind(vue.$data));
clearElement(el, ["vue-event",`v-on:${event}`, `@${event}`])
});
}
function clearElement(el, attributes) {
attributes.forEach(attr => el.removeAttribute(attr));
}
function addAttributes(el) {
el.innerHTML = el.innerHTML.replace(
new RegExp(regex.vueOn, "g"),
(match) => `vue-event ${match}`
);
return el;
}
//#endregion
//#region state
/**
* read all props of a object and call defineReactive.
* @param {Vue} vue
* @param {Function} cb
*/
function walkDataProps(vue, cb) {
for (const key in vue.$data) {
defineReactive(vue, key);
defineReactive(vue, key, cb);
}
}
/**
* change set behavior and define props for Vue and Vue.data from Vue.data
* @param {Vue} obj
* @param {string} key
* @param {Function} cb
* @param {boolean} redefineData
*/
function defineReactive(obj, key, cb) {
let value = obj.$data[key];
Object.defineProperty(cb ? obj.$data : obj, key, {
configurable: true,
get() {
return value;
},
set(newValue) {
if (value === newValue) return;
value = newValue;
if (cb) {
obj[key] = value;
cb();
} else {
obj.$data[key] = value;
}
},
});
}
//#endregion
//#region methods
function walkMethods(vue, methods) {
for (const key in methods) {
vue[key] = methods[key];
}
}
//#endregion