js/listener.js

55 lines
1.4 KiB
JavaScript
Raw Normal View History

/*
A somewhat refined Proxy generator that lets you selectively listen to changed
properties and react accordingly.
Example:
const l = listener()
l.listen("contract", contract => speaker.handle(contract))
l.contract = new Contract()
*/
2022-02-15 14:50:20 +00:00
const registry = new Map()
2021-11-23 15:30:11 +00:00
export const listener = (target={}) => {
2021-06-23 14:04:33 +00:00
let callbacks = new Map()
function listen(prop, callback) {
if ("object" == typeof prop && "forEach" in prop) {
2021-06-23 14:04:33 +00:00
prop.forEach(prop => this.listen(prop, callback))
} else if (callback) {
let list = callbacks.get(prop)
if (!list) callbacks.set(prop, list=[])
list.push(callback)
} else {
2021-06-23 14:04:33 +00:00
callbacks.delete(prop)
}
2021-06-23 14:04:33 +00:00
}
let proxy = new Proxy(target, {
set: (target, prop, value) => {
if (callbacks.has("*")) callbacks.get("*").forEach(callback => callback(value, prop, target[prop]))
if (callbacks.has(prop)) callbacks.get(prop).forEach(callback => callback(value, prop, target[prop]))
2021-06-23 14:04:33 +00:00
return Reflect.set(target, prop, value)
},
2022-02-15 14:50:20 +00:00
get: (target, prop, value) => prop=="listen"
? listen
: target[prop]
2021-06-23 14:04:33 +00:00
})
2022-02-15 14:50:20 +00:00
registry.set(proxy, target)
2021-06-23 14:04:33 +00:00
return proxy
}
2021-11-23 15:30:11 +00:00
2022-02-15 14:50:20 +00:00
listener.raw = proxy => registry.get(proxy)
2021-12-26 16:45:17 +00:00
export const text = (listener, prop) => {
if (prop) {
const node = document.createTextNode(listener[prop])
listener.listen(prop, data => node.data = data)
return node
} else {
return new Proxy(listener, {
get: (target, prop, receiver) => text(listener, prop)
})
}
}
2021-11-23 15:30:11 +00:00
export default listener