-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbind.js
125 lines (98 loc) · 2.69 KB
/
bind.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
const keyUp = "keyup";
export function bind(templateId, object) {
const template = getTemplateById(templateId);
const propertiesNodesDict = getPropertyAndNodesDict(object, template);
const proxy = createProxy(propertiesNodesDict, object);
const nodesAndListeners = createInputKeyUpListeners(
propertiesNodesDict,
proxy
);
initDefaultValues(propertiesNodesDict, proxy);
return {
proxy,
destroy: () => {
nodesAndListeners.forEach(({ node, listener }) =>
node.removeEventListener(keyUp, listener)
);
},
};
}
function getTemplateById(templateId) {
const template = document.getElementById(templateId);
if (!template) {
throw new Error("Not found html template with id: ", templateId);
}
return template;
}
function getPropertyAndNodesDict(object, template) {
return Object.keys(object).reduce((acc, property) => {
const nodes = template.querySelectorAll(`[data-bind=${property}]`);
if (nodes?.length) {
acc[property] = nodes;
}
return acc;
}, {});
}
function createProxy(propertiesNodesDict, object) {
const proxy = {
set(obj, prop, value) {
updateHtml(propertiesNodesDict, prop, value);
if (!isFunction(obj[prop])) {
Reflect.set(...arguments);
calcAllFuctions(obj, propertiesNodesDict);
}
return true;
},
};
return new Proxy(object, proxy);
}
function calcAllFuctions(obj, propertiesNodesDict) {
Object.entries(obj)
.filter(([_, value]) => isFunction(value))
.forEach(([key, value]) => {
updateHtml(propertiesNodesDict, key, value.bind(obj)());
});
}
function updateHtml(propertiesNodesDict, prop, value) {
if (propertiesNodesDict[prop]) {
propertiesNodesDict[prop].forEach((node) => {
if (isInput(node)) {
if (node.value !== value) {
node.value = value;
}
return;
}
node.innerText = value;
});
}
}
function createInputKeyUpListeners(propertiesNodesDict, proxy) {
return Object.entries(propertiesNodesDict).reduce(
(acc, [property, nodes]) => {
nodes.forEach((node) => {
if (isInput(node)) {
const listener = () => {
proxy[property] = node.value;
};
node.addEventListener(keyUp, listener);
acc.push({ node, listener });
}
});
return acc;
},
[]
);
}
function initDefaultValues(propertiesNodesDict, proxy) {
Object.keys(propertiesNodesDict).forEach((property) => {
if (!isFunction(proxy[property])) {
proxy[property] = proxy[property];
}
});
}
function isFunction(value) {
return typeof value === "function";
}
function isInput(node) {
return node.nodeName === "INPUT";
}