DropletJS.PubSub is an advanced JavaScript event library, specifically designed for building highly complex web apps.
There are literally dozens of JavaScript event libraries and frameworks, and most of them do the same basic things: you publish events, listeners handle the events. Pretty simple.
So why should you use DropletJS.PubSub? Well, if all you need is publishers and listeners, you shouldn't.
Instead, you should use Backbone, or jQuery, or EventEmitter, or any of the other popular solutions. They're more widely implemented, they're probably faster, and they may already be available in your app (like if you're already using Backbone or jQuery anyway).
Or you can use some events-only micro-framework with a negligible file size and dead simple API.
However, if you've used any of those solutions and have found them lacking, then you should take DropletJS.PubSub out for a spin.
DropletJS.PubSub is available via a number of popular package managers:
npm install dropletjs.pubsub
jam install DropletJS.PubSub
bower install DropletJS.PubSub
Or you can download the latest tag from https://github.com/wmbenedetto/DropletJS.PubSub/tags
There are a few core principles behind DropletJS.PubSub that may make it the right choice for your app:
DropletJS.PubSub doesn't have any dependencies, and clocks in at a slim ~1.8k (minified and gzipped) despite its robust feature set. If size matters in your app, you might not want to incur the download overhead of a larger library like Backbone or jQuery, when all you really need is the event functionality.
Many event libraries (i.e. Backbone) work by modifying or extending existing objects with additional methods. This can cause collisions with existing methods, leading to unintended consequences or unexpected bugs.
In contrast, DropletJS.PubSub is a standalone event aggregator. It doesn't alter your objects in any way. In fact, it doesn't even know or care about your objects. All it cares about are messages and callback functions (listeners).
That means you don't have to worry about memory leaks caused by some lingering object references tied up in the internals of your event library.
DropletJS.PubSub doesn't force you to commit to any particular event naming scheme, nor does it limit you to a restricted list of possible events. Instead, it allows you to publish and listen for any arbitrary string, so you can use whatever event names that are right for your app.
Naturally, DropletJS.PubSub offers all the basic functionality you'd expect from an event framework. You can:
- publish messages
- listen for messages, executing a callback each time a message is published
- listen once for a message, only executing a callback the first time the message is published
- stop listening for messages
- clear all listeners
DropletJS.PubSub offers a number of advanced features that are especially useful if you're building highly complex JavaScript apps:
One of DropletJS.PubSub's most powerful features is its built-in message syntax that allows you to listen for messages using wildcards.
For example, say you publish Foo.issue.detected.SYNTAX_ERROR
and Bar.issue.detected.ILLEGAL_INPUT
in various parts of your app.
Using wildcards, you can listen for *.issue.detected.SYNTAX_ERROR
and capture SYNTAX_ERROR issues from both Foo and Bar.
Or you can listen for Foo.issue.detected.*
and capture both SYNTAX_ERROR and ILLEGAL_INPUT errors coming only from Foo.
Or you can listen for *.issue.detected.*
and capture all issues detected by Foo, Bar, or any other originators.
This flexibility insulates your app against tight coupling. Your main app can listen for messages from modules without needing to know exactly what the modules are called, and you can add new messages without necessarily needing to add corresponding listeners.
Sometimes you need to respond to every message that is published, regardless of what the message is. For example, you might want to pump all messages through a reporting or logging system. Global subscribers allow you to capture all messages flowing through your system through a single, isolated mechanism.
You can specify whether global subscribers should be called before or after the message-specific listeners are executed. "Before" filters are helpful when you need to transform messages coming from external systems before they are published to your app. Likewise, "after" filters can transform messages before they go out to third-party systems (i.e. reporting.)
For example, say you're listening for messages coming from a Flash app or Java applet. There's a good chance that those apps were written by different developers with different event-naming rules. Rather than bake their alternate naming scheme into your app, you can use a "before" filter to transform the message to something your app already recognizes.
Namespaces increase the control you have over adding and removing listeners and subscribers by allowing you to remove listeners from subsets of messages without clobbering all listeners for the message.
For example, you can listen for Foo:click
and Bar:click
. Both listeners will respond to a click
event -- the namespace is ignored for the purposes of triggering the listener.
Where the namespace comes into play is when you want to remove listeners. Without the namespace, you'd be forced to remove all click listeners, then re-add some of them. With namespaces, you can remove just the Foo:click
listeners, leaving the Bar:click
listeners untouched.
These callbacks are executed after all message handlers are executed. They can be used to, for example, perform cleanup tasks or update the UI (i.e. removing a loading spinner).
These callbacks are executed after each message handler is executed. They can be used to fire off high-priority functions that can't wait for the entire listener stack to finish executing, especially if the listeners are asynchronous functions that can take a while to complete.
Alternately, onPublish callbacks can be used to receive the results of listeners (see below).
Listeners can be flagged as asynchronous. Asynchronous listeners allow you to return results of async functions (i.e. AJAX requests) via callbacks without blocking your entire app.
DropletJS.PubSub's listen
, once
, publish
, and stop
methods all accept arrays of messages, so you can simultaneously publish multiple messages, or you can listen for multiple messages using the same handler.
DropletJS.PubSub refers to events as messages. Messages are published; listeners and subscribers handle published messages.
A message can be any arbitrary string. It can be something as simple click
or keyup
, or as complex as DAMN_USER_DONE_DID_SOMETHING_STUPID
. Whatever your app needs.
That said, there is a built-in message syntax that can be very helpful for structuring complex applications.
DropletJS.PubSub messages follow the following pattern:
Originator.subject.verb.DESCRIPTOR
-
Originator describes where the message was published from. This will usually be the name of a module or class.
-
Subject and verb combine to explain what caused the message to be published, i.e.
button.clicked
,form.submitted
,error.detected
, etc. By convention, verbs should be past tense, since the message generally describes something that just happened. However, that's just convention -- if it makes more sense in your app to use present tense, or mix-and-match present and past tense, that's fine too. -
DESCRIPTOR is an optional string that can be used to disambiguate similar events. For example,
button.clicked.SUBMIT
vsbutton.clicked.CANCEL
vsbutton.clicked.NO_THANKS
. By convention, descriptors are ALL_CAPS with underscores used in place of spaces.
When messages are published, best practice is to always use an originator, subject, and verb. The descriptor is optional.
For example:
DropletJS.PubSub.publish('ShoppingCart.item.added.SKU23426081350716');
DropletJS.PubSub.publish('ShoppingCart.form.submitted');
DropletJS.PubSub.publish('ShoppingCart.error.detected.INVALID_CREDIT_CARD');
DropletJS.PubSub.publish('CommentBox.key.pressed');
DropletJS.PubSub.publish('CommentBox.comment.submitted');
DropletJS.PubSub.publish('CommentBox.result.returned.SUCCESS');
When listening for messages, each segment of the message is optional -- a wildcard can be used instead. The more wildcards you use, the more messages the listener can/will respond to.
For example, you can listen for a very specific message. The following listener will only fire when an item with a specific SKU (which appears as the descriptor) is added to the shopping cart:
DropletJS.PubSub.listen('ShoppingCart.item.added.SKU23426081350716');
That's probably not terribly useful though. You probably want something a little more generic, that can respond whenever any item is added to the cart. This listener will fire for any ShoppingCart.item.added
message, regardless of the descriptor.
DropletJS.PubSub.listen('ShoppingCart.item.added.*');
We can actually simplify even more. When a wildcard appears at the end of a message, it can be omitted entirely. Because DropletJS.PubSub is expecting a 4-segment message, it will automatically replace any missing segments with wildcards.
// This will respond to any item being added.
DropletJS.PubSub.listen('ShoppingCart.item.added.*');
// So will this. It's functionally identical to the example above.
DropletJS.PubSub.listen('ShoppingCart.item.added');
// This will respond to anything that happens with an item, regardless of verb
DropletJS.PubSub.listen('ShoppingCart.item');
// This will respond to anything that is published from ShoppingCart
DropletJS.PubSub.listen('ShoppingCart');
So when do you need to use a wildcard? When it's not at the end of the message.
For example, you might want to listen for error messages from any originator:
DropletJS.PubSub.listen('*.error.detected');
Or maybe you want to listen for any time something is clicked in the shopping cart:
DropletJS.PubSub.listen('ShoppingCart.*.clicked');
The listen
message tells DropletJS.PubSub which message(s) to listen for, and which function to use when handling the message. There are two valid ways to call listen
:
- [REQUIRED] someMessage: Message string or array of messages to listen for
- [REQUIRED] messageHandler: Function to call when message is published
// Listen for one message
DropletJS.PubSub.listen('ShoppingCart.item.added',function(message,payload){
console.log('Item added');
});
// Listen for array of messages
var messages = [
'ShoppingCart.item.added',
'WishList.item.added',
'Favorites.item.added'
];
DropletJS.PubSub.listen(messages,function(message,payload){
console.log('Item added');
});
configObj
is an object literal with the following properties:
- [REQUIRED] message: Message string or array of messages to listen for
- [REQUIRED] handler: Function to call when message is published
- [OPTIONAL] async: Boolean. False by default. Set to true if handler is asynchronous.
DropletJS.PubSub.listen({
message : 'ShoppingCart.item.added', // can also be an array of messages
handler : function(message,payload){
console.log('Item added')
},
async : true
});
When the handler function is called, it will be passed several arguments ...
messageHandler(message,payload)
... where message
is the message that triggered the handler, and payload
is an arbitrary value (usually an object literal) passed by the publish()
function.
In addition, there is a third argument (callback
) that will be passed to the handler when async
is true. This can be used to pass results from the asynchronous handler back to the originator of the message.
DropletJS.PubSub.listen({
message : 'ShoppingCart.item.added',
async : true,
handler : function(message,payload,callback){
// Async AJAX request
$.ajax("example.php").done(function(result){
// Pass result of async request to callback
callback(result);
});
}
});
// The onPublish() function is the callback that gets passed to an asynchronous listener
DropletJS.PubSub.publish({
message : 'ShoppingCart.item.added',
onPublish : function(result){
console.log('The result of the listener is ', result);
}
});
The once
message tells DropletJS.PubSub to listen for a message (or messages) one time, and which function to use when handling the message. After the once
handler has been called, the message is no longer listened for.
once
is called exactly like listen
above: either once('someMessage',messageHandler)
or once(configObj)
.
The publish
method broadcasts a message, executing all the handlers which are listening for that message. There are two valid ways to call publish
:
- [REQUIRED] someMessage: Message string or array of messages to publish
- [OPTIONAL] payload: Data object passed to listener
// Publish one message
DropletJS.PubSub.publish('ShoppingCart.item.added',{
sku : 1234567890,
accountID : 123
});
// Publish an array of messages
var messages = [
'ShoppingCart.item.added',
'ShoppingCart.button.clicked.ADD'
];
DropletJS.PubSub.publish(messages,{
sku : 1234567890,
accountID : 123
});
configObj
is an object literal with the following properties:
- [REQUIRED] message: Message string or array of messages to publish
- [OPTIONAL] payload: Data object passed to listener
- [OPTIONAL] onPublish: Function to call after each listener is executed. The results of the listener are returned to this function. It will be called asynchronously if the listener's
async
property istrue
. - [OPTIONAL] onComplete: Function to call once all listeners have been executed
DropletJS.PubSub.publish({
message : 'ShoppingCart.item.added', // can also be an array of messages
payload : {
sku : 1234567890,
accountID : 123
},
onPublish : function(result){
console.log('Handler result:',result);
},
onComplete : function(){
console.log('DONE!')
}
});
The stop
method tells DropletJS.PubSub to stop listening for a message or messages.
- [REQUIRED] message: The message(s) to stop listening for
// Stop listening for one message
DropletJS.PubSub.stop('ShoppingCart.item.added');
// Stop listening for an for array of messages
DropletJS.PubSub.stop([
'ShoppingCart.item.added',
'WishList.item.added',
'Favorites.item.added'
]);
The subscribe
method allows you to specify a handler which will be called every time a message is published. There are two valid ways to call subscribe
:
- [REQUIRED] handler: Function to call every time a message is published. Called after all listeners have been executed.
DropletJS.PubSub.subscribe(function(message,payload){
// do stuff here
});
The handler is passed two arguments:
- [REQUIRED] message: Message that was published
- [REQUIRED] payload: Data object accompanying message
configObj
is an object literal with the following properties:
- [REQUIRED] handler: Function to call every time a message is published
- [OPTIONAL] namespace: Namespace string. Can be used with
unsubscribe
to later remove only subscribers in the namespace. - [OPTIONAL] phase: Either before or after. If "before", handler is executed before the message listeners are executed. If "after" (or if omitted), handler is executed after the message listeners are executed.
- [OPTIONAL] async: Set to true if handler is asynchronous
DropletJS.PubSub.subscribe({
handler : function(message,payload,callback){
//do stuff
callback();
},
namespace : 'SomeNamespace',
phase : 'before',
async : true
});
The handler is passed two or three arguments:
- [REQUIRED] message: Message that was published
- [REQUIRED] payload: Data object accompanying message
- [OPTIONAL] callback: Function to call once handler is complete, when
async
is true
The unsubscribe
method removes subscribers. There are two valid ways to call unsubscribe
:
- [OPTIONAL] namespace: If specified, removes only subscribers in that namespace. If no namespace is specified, then any subscribers added without a namespace are removed.
- [OPTIONAL] phase: If specified, only subscribers in that phase will be removed. If no phase is specified, then subscribers in both phases will be removed.
DropletJS.PubSub.unsubscribe('SomeNamespace','before');
configObj
is an object literal with the following properties:
- [OPTIONAL] namespace: If specified, removes only subscribers in that namespace. If no namespace is specified, then any subscribers added without a namespace are removed.
- [OPTIONAL] phase: If specified, only subscribers in that phase will be removed. If no phase is specified, then subscribers in both phases will be removed.
DropletJS.PubSub.unsubscribe({
namespace : 'SomeNamespace',
phase : 'before'
});
The clear
method removes all handlers and subscribers.
DropletJS.PubSub.clear();
coming soon
Please submit all bugs, questions, and suggestions via the Issues section so everyone can benefit from the answer.
If you need to contact me directly, email [email protected].
Copyright (c) 2013 Warren Benedetto [email protected]
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.