Watch an object or array for changes
It works recursively, so it will even detect if you modify a deep property like obj.a.b[0].c = true
.
Uses the Proxy
API.
$ npm install on-change
const onChange = require('on-change');
const object = {
foo: false,
a: {
b: [
{
c: false
}
]
}
};
let i = 0;
const watchedObject = onChange(object, function (path, value, previousValue) {
console.log('Object changed:', ++i);
console.log('this:', this);
console.log('path:', path);
console.log('value:', value);
console.log('previousValue:', previousValue);
});
watchedObject.foo = true;
//=> 'Object changed: 1'
//=> 'this: {
// foo: true,
// a: {
// b: [
// {
// c: false
// }
// ]
// }
// }'
//=> 'path: "foo"'
//=> 'value: true'
//=> 'previousValue: false'
watchedObject.a.b[0].c = true;
//=> 'Object changed: 2'
//=> 'this: {
// foo: true,
// a: {
// b: [
// {
// c: true
// }
// ]
// }
// }'
//=> 'path: "a.b.0.c"'
//=> 'value: true'
//=> 'previousValue: false'
// Access the original object
onChange.target(watchedObject).foo = false;
// Callback isn't called
// Unsubscribe
onChange.unsubscribe(watchedObject);
watchedObject.foo = 'bar';
// Callback isn't called
Returns a version of object
that is watched. It's the exact same object, just with some Proxy
traps.
Type: object
Object to watch for changes.
Type: Function
Function that gets called anytime the object changes.
The function receives three arguments:
- A path to the value that was changed. A change to
c
in the above example would returna.b.0.c
. - The new value at the path.
- The previous value at the path.
The context (this) is set to the original object passed to onChange
(with Proxy).
Type: object
Type: boolean
Default: false
Deep changes will not trigger the callback. Only changes to the immediate properties of the original object.
Type: Function
Default: Object.is
The function receives two arguments to be compared for equality. Should return true
if the two values are determined to be equal. Useful if you only need a more loose form of equality.
Type: boolean
Default: false
Setting properties as Symbol
won't trigger the callback.
Returns the original unwatched object.
Type: object
Object that is already being watched for changes.
Cancels all future callbacks on a watched object and returns the original unwatched object.
Type: object
Object that is already being watched for changes.
I had some code that was like:
const foo = {
a: 0,
b: 0
};
// …
foo.a = 3;
save(foo);
// …
foo.b = 7;
save(foo);
// …
foo.a = 10;
save(foo);
Now it can be simplified to:
const foo = onChange({
a: 0,
b: 0
}, () => save(foo));
// …
foo.a = 3;
// …
foo.b = 7;
// …
foo.a = 10;
- known - Allow only access to known object properties (Uses
Proxy
too) - negative-array - Negative array index support
array[-1]
(UsesProxy
too) - statux - State manager (Uses
Proxy
too) - introspected - Never-ending Proxy with multiple observers (Uses
Proxy
too)