Mobile web/Coding conventions/JavaScript/Views
Handling DOM events in JS views
[edit]Delegated declarative dom events
[edit]The preferred way of handling dom events on a view is by using the events map.
In the view properties, we just define a events
map (just like we do with defaults
and others).
var CounterView = View.extend({
// ...
events: {
'click .increment': 'increment',
'click .reset': 'reset',
},
postRender: function() {
// View display logic
},
increment: function() {
this.counter++;
},
reset: function() {
this.counter=0;
},
// ...
})
The events events
map consists of:
{
'event selector': handler
}
Where:
event
is any valid jQuery event.selector
is an optional valid jQuery selector. If ommited, the event will be bound to the viewsthis.$el
.handler
is either afunction
, or astring
(name of a method on the view). The handler is defined and will be invoked on the view'sthis
.
Advantages
[edit]- More efficient. Events are bound with delegate, effectively attaching one DOM listener per event type on the view's root DOM element. (Read http://api.jquery.com/on/#direct-and-delegated-events)
- Easier to understand. To see which DOM events the view binds and reacts to, just need to find the events map of the view.
- Event handling code (usually pieces of logic of the view) gets a name, and it's encapsulated on its own method giving us:
- Reuse of such logic/functionality
- Easier and better testability
- Cleaner postRender methods
Good practices and conventions
[edit]Naming handlers
[edit]When creating event handlers, if they are doing DOM stuff, like preventDefault, or extracting data from the DOM node, manipulating the DOM, etc, you should name the handler onEvent
, for example onThanksClick
or onHeaderToggle
.
If possible, extract functionality that makes sense independently into methods with a good name and call them from the handlers as needed.
When creating methods on a view, like the increment
example above, that happen to be called when an event happens but make sense independently, you should probably give that method a name increment
makes more sense than a handler name onIncrementClick
, but that is open to common sense and taste.
For example, if I had to extract what to increment from a DOM element, and then increment it, then I would use both approaches:
// ...
events: {
'click .increment': 'onIncrementClick'
},
onIncrementClick: function(ev) {
var amount = parseFloat(this.$('.amount').val());
this.increment(amount);
},
increment: function(amount) {
this.counter += amount || 1;
},
// ...
This way we can cleanly test our increment
function without touching the DOM, and it can be reused from other parts of the view (or called from external consumers of the view).
Migrating from the manual way
[edit]Getting the DOM element that generated the event
[edit]A common practice you can see with the manual approach is inside the event handler to access the DOM element that initiated the event by doing var $button = $(this)
. With the events map approach the handlers/methods are at view level and are consistently bound to the view's this
, so how do we access the DOM element of the event?
Handler methods get an jQuery.Event
as a parameter, so:
onIncrementClick: function(ev) {
var $incButton = $(ev.target);
// ...
http://api.jquery.com/category/events/event-object/
Implementing events map on a View in an inheritance chain
[edit]A View that .extends
from another View class
[edit]If the View for which you are implementing the events map inherits/extends from another View class, for safety when declaring the events map declare it extending from the parent events map, like this:
var CounterView = Panel.extend({
// ...
events: $.extend({}, Panel.prototype.events, {
'click .increment': 'increment',
'click .reset': 'reset',
}),
// ...
})
Careful with child views
[edit]When implementing events map on a View, be careful with the children classes. If you implement the events map, and a children class implements it also, without extending as explained above, then the child view will be overriding the parent events.
As a quick rule, when implementing it on a view, have a look at sub-views (search for "MyParentView.extends
") and if any of the sub-views uses events map, modify it to extend from the parent events map as explained in the example above.
How does it work
[edit]Internally, this is just using jQuery to delegate the events and the selectors to specified handlers bound to the view, no black magic.
This is inspired and brought from Backbone.js, a very simple and clean JS library, similar in philosophy to our own.
Manual approach with jQuery
[edit]To bind events to the view manually, we bind them on the postRender
method, like so:
var CounterView = View.extend({
// ...
postRender: function() {
// View display logic
this.$('.increment').click(function(ev) {
// ...
})
this.$('.reset').click(function(ev) {
// ...
})
},
// ...
})
After the view has been rendered, we use the this.$
shortcut to select elements within the view's this.$el
and bind events as we would do with jQuery. You can use also delegated events with this approach this.$el.on('click', 'td', onCellClick);
.
Gotchas
[edit]- Everytime the view renders, we bind events (multiple of them), this is inefficient, there are better ways.
- The postRender method on views grows a lot in size, event handling declarations and method code is mixed with display logic.
- The complexity of the view (arguably) increases, and with it the ability to read and modify the source confidently.