An extension to the Node.js events.EventEmitter with an important factor, "bubbling".
hap's EventEmitter inherits all methods from Node.js's events.EventEmitter, but has a new method #fire()
. It works the same as #emit()
but sets off a chain of events that pass around a facade and bubble up to any #parent
EventEmitters.
Let's jump in to an example:
var EventEmitter = require('hap').EventEmitter,
util = require('util'),
emitter;
function MyTestEmitter(){
EventEmitter.apply(this, arguments);
}
util.inherits(MyTestEmitter, EventEmitter);
emitter = new MyTestEmitter();
emitter.on('test event', function(e){
console.log(e);
});
emitter.fire('test event');
The above example is no different to any normal observer pattern... and the facade would be logged to the console. Now let's see the bubbling in action:
var EventEmitter = require('hap').EventEmitter,
util = require('util'),
emitter, parentEmitter;
function MyTestEmitter(){
EventEmitter.apply(this, arguments);
}
util.inherits(MyTestEmitter, EventEmitter);
parentEmitter = new MyTestEmitter();
emitter = new MyTestEmitter();
emitter.setParent(parentEmitter);
parentEmitter.on('test event', function(e){
console.log('Parent');
});
emitter.on('test event', function(e){
console.log('Child');
});
parentEmitter.fire('test event');
// => Child
// => Parent
In the above example the event seems to start at the child object, even though it was called in the parent. That's very similar to what happens in the DOM for example. First, there is a "capture" event that starts at the target element and then fires within each child. Once it hits the bottom, the event then bubbles up to the top.
Here's a quote from quirksmode:
When you use event capturing
| | ---------------| |----------------- | element1 | | | | -----------| |----------- | | |element2 \ / | | | ------------------------- | | Event CAPTURING | -----------------------------------
the event handler of element1 fires first, the event handler of element2 fires last.
When you use event bubbling
/ \ ---------------| |----------------- | element1 | | | | -----------| |----------- | | |element2 | | | | | ------------------------- | | Event BUBBLING | -----------------------------------
the event handler of element2 fires first, the event handler of element1 fires last.
W3C has very sensibly decided to take a middle position in this struggle. Any event taking place in the W3C event model is first captured until it reaches the target element and then bubbles up again.
| | / \ -----------------| |--| |----------------- | element1 | | | | | | -------------| |--| |----------- | | |element2 \ / | | | | | -------------------------------- | | W3C event model | ------------------------------------------
hap uses the "W3C model". First there is a capturing phase of which you can attach to with the #before()
method. There the bubbling takes place of which you can listen to with the #on()
method. And finally, to add some extra power, the event will bubble from bottom to top one more time of which you can listen to with the #after()
method.
| | / \ / \
-----------------| |--| |--|-|------------
| parent | | | | | | |
| -------------| |--| |--|-|------ |
| |child \ / | | | | | |
| -------------------------------- |
| hap event model |
------------------------------------------
Here's a an example using an Express.js like framework:
// app.js
var express = require('express'),
hap = require('hap'),
http = require('http'),
util = require('util'),
app = express(),
nav = require('./nav'); // See below
util._extend(app, hap.EventEmitter.prototype);
// This event is fired by express when an express instance is used by another
// express instance.
app.on('mount', function(parent){
this.setParent(parent);
});
// "nav" is a child app
app.use(nav());
// When our app is ready to render the nav, we want to add some items to it.
app.on('nav', function(e){
e.val().push({
content : 'Item 1',
id : 'nav-item-1'
});
});
app.get('/', function(req, res){
res.send(emitter.fire('nav'));
// => <nav><ul><li id="nav-item-1">Item 1</li></ul></nav>
});
http.createServer(app).listen(8080);
// nav.js
var express = require('express'),
hap = require('hap'),
Html = require('tag').Html,
util = require('util');
module.exports = function(){
var app = express();
util._extend(app, hap.EventEmitter.prototype);
// The nav var needs to be created before (the
// capturng phase) the nav is rendered.
app.before('nav', function(e){
e.val([]);
});
// Add a listener to the end of event process to render the nav.
app.after('nav', function(e){
var ul = Html.factory('ul'),
nav = Html.factory('nav');
e.val().forEach(function(item){
var content = item.content,
li;
delete item.content;
li = Html.factory('li', item, content);
ul.appendChild(li);
});
nav.appendChild(ul);
e.val(nav.toHTML());
});
};
npm i hap
Adds an emitter as a child to this object.
- emitter {hap.EventEmitter} The child emitter
Adds a listener to the final bubling stage.
- eventName {String} The event to attach to
- fn {Function} Callback function
Adds a listener to the capturing stage.
- eventName {String} The event to attach to
- fn {Function} Callback function
A protected method used to trigger the bubbling sequence.
- eventName {String} The event to bubble
- e {hap.EventFacade} The event facade
A protected method use to trigger the capturing sequence.
- eventName {String} The event to start capturing
- e {hap.EventFacade} The event facade
- see events.EventEmitter
Similar to the #emit()
method but dispatches the capturing/bubbling sequence passing a facade object
and return a modified value.
- eventName {String} The event to emit
- attrs {Object | EventFacade} A hash of attributes or an event facade
- see events.EventEmitter
- see events.EventEmitter
Similar to #once()
but attaches the listener to the final bubbling sequence.
- eventName {String} The event to emit
- fn {Function} The listener callback
Similary to #once()
but attached the capturing phase.
- eventName {String} The event to emit
- fn {Function} The listener callback
- see events.EventEmitter
Adds an emitter as a parent to this object.
- emitter {hap.EventEmitter} The parent emitter
Sets or gets the current value. This value might be modified by any listener.
- value {Mixed} A value to set.