Many people have explained that e.stopPropagation()
prevents event bubbling. However, I'm having a hard time finding why one would want or want to prevent event bubbling in the first place.
On my site I have many elements which are called like this:
$(document.body).on('click', ".clickable", function(e){
//e.stopPropagation();
//do something, for example show a pop-up or click a link
});
<body>
<p>outside stuff</p>
<button type="button" class='clickable'>
<img src='/icon.jpg'> Do Something
</button>
</body>
I was thinking to add e.stopPropagation()
because I want to change the event handler to 'touch'
from 'click'
using this awesome touch library, Hammer.js.. This would allow for clicking to happen normally on desktop and for touch events on mobile devices.
The problem with this (please correct me if I'm wrong) is that scrolling on touch devices slows to a halt.
Is this where e.stopPropgation()
is useful? Such that whenever one touches the screen document.body
-event bubbling is NOT happening every time?
Imagine you have a button and want to handle a click event:
Without event propagation, clicking on the image or the text wouldn't trigger the click event handler bound to the button, as the event would never leave the
<img>
or the<span>
elements.A case where you don't want event propagation is where nested elements have their own event handlers. Here's one example:
If you don't stop the event propagation with
.arrow
's event handler, it will trigger the button's event handler as well.There are several ways you can handle events in javascript/jQuery. Two of them are:
If you are using a direct event handler on the object and there are no delegated event handlers configured in the page, then there is no reason for
e.stopPropagation()
.But, if you have delegated event handlers that are using propagation, you sometimes want to make sure that a higher level delegated event handler does not fire on the current event.
In your particular example:
This is a delegated event handler. It is looking for any click event that propagates up to the
document.body
object, yet originated on ana.ajaxLink
object. Here, there is little advantage toe.stopPropagation()
because the event is almost entirely propagated already (it will also go up to thedocument
, but unless you also have a handler forclick
on thedocument
object, then there's no reason toe.stopPropagation()
in this handler.Where it would make a lot of sense would be when you have both a top level delegated event handler (like the one in your example) and you have lower level event handlers (either directly on the objects or using delegated event handling, but at a level below the
document.body
object. In that case, if you only want the lower level event handler to get the event, then you would calle.stopPropagation()
in it's handler so that thedocument.body
handler never saw the event.Note: Using
return false
from a jQuery event handler triggers bothe.stopPropagation()
ande.preventDefault()
. But, if you are in a delegated event handler, thee.preventDefault()
doesn't do anything because the default behavior (if there was one) has already fired when the target object first saw the event. Default behaviors happen before event propagation soe.preventDefault()
only works in event handlers directly on the target object.There is no noticeable performance degradation because you allow an event to bubble up because these are user level events and they just don't occur fast enough to matter, not is the bubbling particularly slow when there are no handlers on all the intervening objects. The system already special cases some events like
mousemove
that can happen rapidly to solve that issue. If you have a giant project with hundreds or thousands of event handlers, there are cases where using delegated event handling is more efficient and there are cases where direct event handlers on the actual target objects are more efficient. But, except in giant scenarios, the performance difference is probably not noticeable.Here's an example where bubbling/delegation is more efficient. You have a giant table with thousands of rows and each row has two buttons in it (add/delete say). With delegated event handling, you can handle all the buttons in two simple event handlers that you attach to the table object (a common parent of the buttons). This will be much quicker to install the event handlers rather than installing several thousand event handlers directly on each and every button. These delegated event handlers will also automatically work on newly created rows/objects in the table. This is the ideal scenario for event bubbling/delegated event handlers. Note, in this scenario, there is no reason to stop propagation/bubbling.
Here's an example of where delegated event handlers are very inefficient. Supposed you have a good-sized web page with hundreds of objects and event handlers. You could make every one of the event handlers be a delegated event handler attached to the
document
object. But, here's what happens. A click happens. There's no event handler on the actual object so it bubbles up. Eventually, it gets to the document object. The document object has hundreds of event handlers. The event processing engine (jQuery in this case) has to look through every one of those event handlers and compare the selector in the delegated event handler for each one with the original event target to see if they match. Some of these comparisons are not fast as they can be full-blown CSS selectors. It has to do that for hundreds of delegated events. This is bad for performance. This is exactly why.live()
in jQuery was deprecated because it worked this way. Instead, delegated event handlers should be placed as close to the target object as possible (the closest parent that is practical given the circumstances). And, when there is no need for a delegated event handler, handlers should be put on the actual target object as this is the most efficient at runtime.Back to your original question. There is no time that you want bubbling turned off generally. As I described earlier in my answer, there are specific instances where an event handler further out on the tree wants to handle the event and stop any delegated event handlers higher up in the DOM tree from processing this event. That is a time to
e.stopPropatation()
.Here are several other relevant posts with useful info on this topic (as it has been widely discussed before):
Why not take Javascript event delegation to the extreme?
Should all jquery events be bound to $(document)?
Does jQuery.on() work for elements that are added after the event handler is created?
jQuery on() and stopPropagation()
Best practice to avoid memory or performance issues related to binding a large number of DOM objects to a click event
jQuery .live() vs .on() method for adding a click event after loading dynamic html
Do not use
stopPropagation()
, if possible.Two advantages using
stopPropagation()
are:Although this function seems to be useful, it can be considered bad coding style, especially when you have no full control over the code (e. g. because you use a third-party library).
stopPropagation()
is a take-everything-or-nothing concept. It does not allow fine control flow. If some element between two other nested elements stops the event propagation, none of the parent elements will receive it, although there might be situations, when they should receive it.A more elegant (and not that complex) way to solve this problem, is to always allow the event propagation by never calling
stopPropagation()
at all. Elements, which define their own events which shall not be automatically executed from a child element, can use the target and currentTarget properties to check from where the initial event comes and execute own event functions only in situations when it is wanted.In following example, there are 3 nested DIVs. Clicking the lowest (blue) DIV shall propagate the onClick event up through the entire DOM structure but without calling the onClick event of the green DIV, which lies between:
So calling
stopPropagation()
in onClick event function of DIV #3 is not necessary required to prevent firing the onClick event if DIV #2.Also note, that the document structure is as follows:
If an event propagation is not stopped, it will reach the
document
object.event.currentTarget
will then bedocument
, whileevent.target
will be eitherdocument.documentElement
,document.body
or any sub element under the<body>
element.So considering you have following code:
Here is what it looks like and where the different parts of the document are:
Gray is the body area. Green is the actual document "element" (top most styleable part). And behind it, there is the invisible
document
object.If you want to use event functions, which will be executed only for the elements directly under your finger / mouse cursor, you could use following code (example for onClick event):
It works with
document.documentElement
,document.body
or any element on the document without needing to callstopPropagation()
.