0ad/binaries/data/mods/public/gui/common/Observable.js
Dunedan 93ce94655d
Use @stylistic/brace-style for eslint
Up to now `eslint-plugin-brace-rules` was used to enforce a common brace
style for JavaScript code. This plugin was however updated the last time
over 9 years ago and will be incompatible with ESLint v10, as that
[removes `context.getSourceCode()`][1], the plugin relies on.

To keep the eslint config working with ESLint v10, this replaces
`eslint-plugin-brace-rules` with the [`@stylistic/brace-style`][2] rule
from `@stylistic/eslint-plugin`, a package we already use.

While `@stylistic/brace-style` doesn't offer an option to format braces
in exactly the same way as before, the "allman" style seems to be the
one closest to the existing code.

[1]: https://eslint.org/blog/2025/11/eslint-v10.0.0-alpha.0-released/#removed-deprecated-rule-context-members
[2]: https://eslint.style/rules/brace-style
2026-01-12 21:33:52 +01:00

66 lines
1.5 KiB
JavaScript

/**
* Observable is a self-proxy to enable opt-in reactive programming.
* Properties of an Observable can be watched, and whenever they are changed
* a callback will be fired.
*/
var ObservableMixin = (Parent) => class Observable extends (() => Parent || Object)()
{
constructor()
{
super();
// Stores observers that are fired when the value is changed
Object.defineProperty(this, "_changeObserver", {
"value": {},
"enumerable": false,
});
// Stores observers that are fired when the value is assigned to,
// even if it isn't changed.
Object.defineProperty(this, "_setObserver", {
"value": {},
"enumerable": false,
});
return new Proxy(this, {
"set": (target, key, value) =>
{
let old;
let hasOld = false;
if (Reflect.has(target, key))
{
hasOld = true;
old = Reflect.get(target, key);
}
Reflect.set(target, key, value);
this._trigger(this._setObserver, key, old);
if (!hasOld || old !== value)
this._trigger(this._changeObserver, key, old);
return true;
}
});
}
_trigger(dict, key, old)
{
if (dict[key])
for (const watcher of dict[key])
watcher(key, old);
}
trigger(key, old)
{
this._trigger(this._setObserver, key, old);
this._trigger(this._changeObserver, key, old);
}
watch(watcher, props, onlyChange = true)
{
const dic = onlyChange ? this._changeObserver : this._setObserver;
for (const prop of props)
{
if (!dic[prop])
dic[prop] = [];
dic[prop].push(watcher);
}
}
};
var Observable = ObservableMixin();