These are just a couple of the API improvements. There are many more... read on!
Or clone from GitHub.
Multiple Listeners
Listening to multiple events from an object is a common need, so watchable
has made
this simple:
watchable.on({
foo () {
},
bar () {
}
});
The object passed to on()
is called a "listener manifest". That same object can be
passed to un()
to remove all of the listeners just added, but there is an easier way:
let token = watchable.on({
foo () {
},
bar () {
}
});
token.destroy();
By destroying the returned token, all of the corresponding listeners will be removed.
The same applies to listener methods:
let token = watchable.on({
foo: 'onFoo',
bar: 'onBar',
this: watcher
});
token.destroy();
The special key this
in the listener manifest is understood to be the target object
of the listener methods.
A listener manifest can contain a mixture of method names and functions, but it is
generally best to use a consistent form (at least on a per-call basis).
There is also the unAll
method which removes all listeners from a watchable instance:
watchable.unAll();
Scope Resolution
When using listener methods, it can be convenient to have a default or named scope.
Consider these two cases:
watchable.on({
foo: 'onFoo',
bar: 'onBar'
});
watchable.on({
foo: 'onFoo',
bar: 'onBar',
this: 'parent'
});
To enable the above, the watchable instance must implement resolveListenerScope
:
class MyClass extends Watchable {
resolveListenerScope (scope) {
return (scope === 'parent') ? this.parent : this;
}
}
let watchable = new Watchable({
resolveScope (scope) {
}
});
When using an instance in the second part above, the resolveScope
property of the config
object is used to set the resolveListenerScope
method on the watchable instance.
The full parameter list passed to resolveListenerScope
is as below:
scope
: The value of this
on the listener manifest (if any).fn
: The handler function or method name (e.g., 'onFoo'
).listener
: The internal object tracking the on()
request.
The listener
argument is an Array
that holds those values needed by the watchable
mechanism. The object can be useful to the resolveListenerScope
method for holding
cached results on behalf of this particular listener. The resolveListenerScope
method,
however, should not do any of the following with the listeners
object:
- Add or remove array elements.
- Change any of the array element values.
- Depend on any of the array element values.
- Overwrite any of the array prototype methods.
Basically, watchable
treats the listeners
as the array it is and so long as that view
onto the object is preserved and handled as read-only, the resolveListenerScope
implementor is free to place expando properties on the same object for its own benefit.
Relaying Events
When relaying one event between watchable instances, there is always the manual solution:
let watchable1 = new Watchable();
let watchable2 = new Watchable();
watchable1.on({
foo (...args) {
watchable2.fire('foo', ...args);
}
});
To relay all events fired by watchable1
, however, requires a different approach. The
solution provided by watchable
is an event relayer:
const relayEvents = require('@epiphanysoft/watchable/relay');
The above require
returns a function that can be used to create event relayers but it
also enables the latent relayEvents
method which is already defined on all watchable
objects.
These are equivalent:
relayEvents(watchable1, watchable2);
watchable1.relayEvents(watchable2);
They both create an event relayer and register it with watchable1
. The second form is
generally preferred since most of the operations provided by watchable
are instance
methods. Basically, as long as some module issues a require('.../watchable/relay)
then
the relayEvent
method on all watchable instance will work properly.
The valid arguments to relayEvents
are:
watchable.relayEvents(target);
watchable.relayEvents(target, String...);
watchable.relayEvents(target, String[]);
watchable.relayEvents(target, Object);
watchable.relayEvents(target, Function);
watchable.relayEvents(relayer);
Removing a relayer is similar to removing a listener manifest:
let token = watchable1.relayEvents(watchable2);
token.destroy();
To relay multiple events, but not all events:
watchable1.relayEvents(watchable2, 'foo', 'bar');
watchable1.relayEvents(watchable2, [ 'foo', 'bar' ]);
The options
object form accepts an object whose keys are event names. The following is
equivalent to the above:
watchable1.relayEvents(watchable2, {
foo: true,
bar: true
});
To relay all events except bar
:
watchable1.relayEvents(watchable2, {
'*': true,
bar: false
});
The special '*'
pseudo event is used to change the default mapping of events not given
in the options
object.
The values in the options
object can be used to rename or transform individual events.
To relay foo
without modification but rename the bar
event to barish
:
watchable1.relayEvents(watchable2, {
foo: true,
bar: 'barish'
});
To instead transform the bar
event:
watchable1.relayEvents(watchable2, {
foo: true,
bar (event, args) {
return watchable2.fire('barish', ...args);
}
});
To relay all events and only transform bar
:
watchable1.relayEvents(watchable2, {
'*': true,
bar (event, args) {
return watchable2.fire('barish', ...args);
}
});
To transform all fired events, you can specify a relay function:
watchable1.relayEvents(null, (event, args) => {
return watchable2.fire(event, ...args);
});
The null
argument is required to differentiate this case from a watchable class (which
is just a constructor function). Further, the relayer function does not have to be an =>
function:
function relayer (event, args) {
return this.target.fire(event, ...args);
}
watchable1.relayEvents(watchable2, relayer);
watchable1.relayEvents(watchable3, relayer);
In this case, this
in the relayer
function refers to the relayer instance that is
created to hold the target
of the relay.
For maximum flexibility, a custom relayer class can be written and an instance passed as
the first and only parameter to relayEvents
:
const { Relayer } = require('@epiphanysoft/watchable/relay');
class MyRelayer extends Relayer {
constructor (target) {
super();
this.target = target;
}
relay (event, args) {
}
}
watchable1.relayEvents(new MyRelayer(watchable2));
In this case, watchable1
will call the relay()
method for all events it fires. The
relay
method can then decide the particulars.
The filtering and renaming features described above can be leveraged by instead
implementing doRelay
(as long as the constructor
passes the options
object to its
super()
):
const { Relayer } = require('@epiphanysoft/watchable/relay');
class MyRelayer extends Relayer {
constructor (target) {
super();
this.target = target;
}
doRelay (event, args) {
this.target.fire(event, ...args);
}
}
watchable1.relayEvents(new MyRelayer(watchable2));
Event Logging
Logging is a special form of relaying that (by default) logs events to the console
:
const logEvents = require('@epiphanysoft/watchable/log');
logEvents(watchable);
watchable.fire('foo', 42, 'abc');
The above will generate console.log()
calls for all events fired by the watchable
:
> foo 42 "abc"
There are also logOptions
that can be specified:
Using level
By default, events are logged using console.log()
. Some events, however, may be more
appropriate to log using console.error()
or other "level" method. This is handled by
specifying a level
option:
logEvents(watchable, {
level: {
foo: 'error'
}
});
watchable.fire('foo', 42, 'abc');
Given the above logEvents()
call, all events will still use console.log()
except for
the foo
event. This event will use console.error()
. Instead of 'error'
, the value
of the properties in the level
object are any console
API that is callably equivalent
to console.log()
and console.error()
. For example, 'warn'
and 'info'
.
Using mask
In some cases, perhaps only the event name should be logged, or maybe certain event
arguments are just noise in the log. The mask
option can be used to tune the output for
all events or on a per-event basis:
logEvents(watchable, {
mask: 0b011
});
watchable.fire('foo', 42, 'abc', window);
The value of the mask
is a number whose bits are matched to the arguments. The least
significant bit controls arguments[0]
, the next least bit controls arguments[1]
and
so on. In the above 0b011
(an ES6 binary literal) has the two least significant bits set
and so only the first two arguments will be logged:
> foo 42 'abc'
A mask
of 0
will prevent all argument logging. To assign a mask
value based on the
event name, mask
can be an object:
logEvents(watchable, {
mask: {
'*': 0,
foo: 0b011
}
});
In the above case, the default mask (using the '*'
key) is 0
while foo
events have
a value of 0b011
. When mask
is a number, it is equivalent to an object with '*'
as
the only key holding that value.
Using prefix
To make logged events more readable, each line can be decorated with a prefix
:
logEvents(watchable, {
prefix: 'w1:'
});
watchable.fire('foo', 42, 'abc');
Using prefix
each call to console.log()
will be more explicit:
> w1:foo 42 "abc"
Using to
Unit tests and the like can benefit by logging to an array:
let events = [];
logEvents(watchable, events);
watchable.fire('foo', 42, 'abc');
expect(events).to.equal([
[ 'foo', [ 42, 'abc' ]]
]);
To combine other options, the array can be passed as the to
property:
let events = [];
logEvents(watchable1, {
prefix: 'w1:',
to: events
});
logEvents(watchable2, {
prefix: 'w2:',
to: events
});
watchable1.fire('foo', 42, 'abc');
watchable2.fire('foo', 42, 'abc');
expect(events).to.equal([
[ 'w1:foo', [ 42, 'abc' ]],
[ 'w2:foo', [ 42, 'abc' ]]
]);
Utility Methods
The watchable
module provides several helper functions that are directly exported. These
are mostly to mimic the event-emitter
API since many equivalent capabilities are available
as described above.
These methods are:
const { hasListeners, is, unAll } = require('@epiphanysoft/watchable');
const pipe = require('@epiphanysoft/watchable/pipe');
const unify = require('@epiphanysoft/watchable/unify');
The hasListeners
and unAll
methods are also available as instance methods of watchable
objects.
hasListeners
hasListeners (watchable, event);
Returns true
if the watchable
instance has one ore more listeners for event
.
is
is (candidate);
Returns true
if the candidate
is a watchable object.
pipe
pipe (watchable1, watchable2);
Relays all events fired on watchable1
to watchable2
. This is an event-emitter
name
for the relayEvents
method described above:
pipe (watchable1, watchable2) {
return watchable1.relayEvents(watchable2);
}
unAll
unAll (watchable);
Removes all listeners on the watchable
instance.
unify
unify (watchable1, watchable2);
This (non-reversibly) connects the listeners of the two watchable instances. This is done
by sharing the listener registrations. This means that listeners registered on one
instance will be in fact be registered on both. This has the effect that which ever of
these instances is used to fire()
an event, all listeners will be invoked, regardless of
the instance on which they appear to be registered.
To unify()
multiple watchable instances, it is important to always pass one of the
current group members as the first argument:
unify (watchable1, watchable2);
unify (watchable1, watchable3);
unify (watchable1, watchable2);
unify (watchable3, watchable2);
This is because preference is given to the first watchable object when merging.
Extending Watchable
For all of its features, watchable
is rather small so there is room for enhancement by
other modules.
To facilitate such enhancements, the entire Mocha test suite is exported to allow such
modules to verify compliance with the full watchable
interface contract:
const { Watchable } = require('@epiphanysoft/watchable');
class MyWatchable extends Watchable {
}
const watchableTestSuite = require('@epiphanysoft/watchable/test/suite');
describe('MyWatchable', function () {
watchableTestSuite(MyWatchable);
});
Design Goals
If you have ideas, I am wide open to hear them. Some things to keep in mind when deciding
if an idea belongs in watchable
proper (or separate module) are sketched out in this
section.
Small Module Size
While there many features in the watchable
module, the core (as of 1.1.0) is only about
350 lines of code. The most advanced features (like unify
, relaying and logging) are all
implemented as "sub-modules" (separately required files in the same npm
module).
Elegant API
As much as possible, the watchable
API tries to behave "intuitively". This means that
when a call signature looks intuitive, watchable
tries to support it. This principal
can be seen in the on()
method and its support for these call signatures:
watchable.on('foo', x => { });
watchable.on({
foo (x) {
}
});
Compatibility with event-emitter
The event-emitter
module is the de facto standard for this type of problem, so while the
API can be extended and improved, watchable
strives to be as close as possible to a
drop-in replacement.
License