I prepared a little example of the various event registration models, event properties and event orders. Thus you can get a quick overview of the possibilities and restrictions of event handling.
On the Introduction to events page I asked a question that at first sight seems incomprehensible: “If an element and one of its ancestors have an event handler for the same event, which one should fire first?” Not surprisingly, this depends on the browser.
The basic problem is very simple. Suppose you have a element inside an element
----------------------------------- | element1 | | ------------------------- | | |element2 | | | ------------------------- | | | -----------------------------------
and both have an onClick event handler. If the user clicks on element2 he causes a click event in both element1 and element2. But which event fires first? Which event handler should be executed first? What, in other words, is the event order?
Not surprisingly, back in the bad old days Netscape and Microsoft came to different conclusions.
The two event orders are radically opposed. Netscape 4 only supports event capturing, Explorer only supports event bubbling. Netscape 6 and Konqueror support both, while Opera and iCab support neither.
Unfortunately Netscape 4’s implementation of its own event capturing model is strange and usually buggy. The Microsoft model has always worked as advertised, giving it a huge advantage over Netscape. The old Netscape model isn’t used any more, so that event bubbling has become the default event order in all browsers except for Netscape 4.
When you use event capturing
| | ---------------| |----------------- | element1 | | | | -----------| |----------- | | |element2 \ / | | | ------------------------- | | Event CAPTURING | -----------------------------------
the event handler of element1 fires first, the event handler of element2 fires last. In Netscape 4 capturing doesn’t happen automatically. We’ll take a look at this later.
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 | ------------------------------------------
You, the web developer, can choose whether to register an event handler in the capturing or
in the bubbling phase. This is done through the addEventListener()
method explained on the Advanced models page.
If its last argument is true
the event handler is set for the capturing phase, if it is
false
the event handler is set for the bubbling phase.
Suppose you do
element1.addEventListener('click',doSomething2,true) element2.addEventListener('click',doSomething,false)
If the user clicks on element2 the following happens:
click
event starts in the capturing phase. The event looks if any
ancestor element of element2 has a onclick
event handler for the capturing
phase.doSomething2()
is
executed.doSomething()
,
which is registered to element2 for the bubbling phase.The reverse would be
element1.addEventListener('click',doSomething2,false) element2.addEventListener('click',doSomething,false)
Now if the user clicks on element2 the following happens:
click
event starts in the capturing phase. The event looks if any
ancestor element of element2 has a onclick
event handler for the capturing
phase and doesn’t find any.doSomething()
,
which is registered to element2 for the bubbling phase.doSomething2()
is executed.In the browsers that support the W3C DOM, a traditional event registration
element1.onclick = doSomething2;
is seen as a registration in the bubbling phase.
Few web developers consciously use event capturing or bubbling. In Web pages as they are made today, it is simply not necessary to let a bubbling event be handled by several different event handlers. Users might get confused by several things happening after one mouse click, and usually you want to keep your event handling scripts separated. When the user clicks on an element, something happens, when he clicks on another element, something else happens.
Of course this might change in the future, and it’s good to have models available that are forward compatible. But the main practical use of event capturing and bubbling today is the registration of default functions.
What you first need to understand is that event capturing or bubbling always happens, except in Netscape 4. If you define a general onclick event handler for your entire document
document.onclick = doSomething; if (document.captureEvents) document.captureEvents(Event.CLICK);
any click
event on any element in the document will eventually bubble up to the document and
thus fire this general event handler.
Only when a previous event handling script explicitly orders the event to stop bubbling, it will not propagate to the
document.
Something similar (but not the same!) happens in Netscape 4. Any event is captured by the document first, since it is the highest element in the page. Therefore any click event on any element will first fire the general event handler. Only when the script explicitly orders it, the event will be captured by other HTML elements.
Later on I’ll explain how to stop events from propagating. As to Netscape 4, it’ll get its own page later on.
Because any event ends up on the document, default event handlers become possible. Suppose you have this page:
------------------------------------ | document | | --------------- ------------ | | | element1 | | element2 | | | --------------- ------------ | | | ------------------------------------ element1.onclick = doSomething; element2.onclick = doSomething; document.onclick = defaultFunction;
Now if the user clicks on element1 or 2, doSomething()
is executed. You can
stop the event propagation here, if you wish. If you don’t the event bubbles up to
defaultFunction()
.
If the user clicks anywhere else defaultFunction()
is also executed. This might be
useful sometimes.
Setting document–wide event handlers is necessary in drag–and–drop
scripts. Typically a mousedown
event on a layer selects this layer and makes
it respond to the mousemove
event. Though the
mousedown
is usually registered on the layer to avoid browser bugs, both
other event handlers must be document–wide.
Remember the First Law of Browserology: anything can happen, and it usually does when you’re least prepared for it. So it may happen that the user moves his mouse very wildly and the script doesn’t keep up so that the mouse is not over the layer any more.
onmousemove
event handler is registered to the layer,
the layer doesn’t react to the mouse movement any more, causing confusion.onmouseup
event handler is registered on the layer,
this event isn’t caught either so that the layer keeps reacting to the mouse
movements even after the user thinks he dropped the layer. This causes even more confusion.So in this case event bubbling is very useful because registering your event handlers on document level makes sure they’re always executed.
But usually you want to turn all capturing and bubbling off to keep functions from interfering with each other. Besides, if your document structure is very complex (lots of nested tables and such) you may save system resources by turning off bubbling. The browser has to go through every single ancestor element of the event target to see if it has an event handler. Even if none are found, the search still takes time.
In the Microsoft model you must set the event’s cancelBubble
property to true.
window.event.cancelBubble = true
In the W3C model you must call the event’s stopPropagation()
method.
e.stopPropagation()
This stops all propagation of the event in the bubbling phase. Stopping event propagation in the capturing phase is impossible. One wonders why.
For a complete cross-browser experience do
function doSomething(e) { if (!e) var e = window.event; e.cancelBubble = true; if (e.stopPropagation) e.stopPropagation(); }
Setting the cancelBubble
property in browsers that don’t support it doesn’t hurt.
The browser shrugs and creates the property. Of course it doesn’t actually
cancel the bubbling, but the assignment itself is safe.
As I said before, in Netscape 4 event capturing automatically stops unless you specifically order the browser to continue. I’ll give the necessary code later on.
As we’ve seen earlier, an event has a target
or srcElement
that
contains a reference to the element the event happened on. In our example this is element2,
since the user clicked on it.
It is very important to understand that during the capturing and bubbling phases (if any) this target does not change: it always remains a reference to element2.
But suppose we register these event handlers:
element1.onclick = doSomething; element2.onclick = doSomething;
If the user clicks on element2 doSomething()
is executed twice. But how
do you know which HTML element is currently handling the event? target/srcElement
don’t give a clue, they always refer to element2 since it is the original source of the
event.
To solve this problem W3C has added the currentTarget
property. It contains a
reference to the HTML element the event is currently being handled by: exactly what we need.
Unfortunately the Microsoft model doesn’t contain a similar property.
You can also use
the this
keyword.
In the example above it refers to the HTML element the event is handled on, just like
currentTarget
.
But when you use the Microsoft event registration model
the this
keyword doesn’t refer to the HTML element. Combined with the
lack of a currentTarget
–like property in the Microsoft model,
this means that if you do
element1.attachEvent('onclick',doSomething) element2.attachEvent('onclick',doSomething)
you cannot know which HTML element currently handles the event. This is the most serious problem with the Microsoft event registration model and for me it’s reason enough never to use it, not even in IE/Win only applications.
I hope Microsoft will soon add a currentTarget
–like property — or maybe
even follow the standard? Web developers need this information.
If you wish to go through all event pages in order, you should now continue with the Mouse events page.