You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
I was wrong in bug 15543. The effect of invoking event listeners with AT_TARGET at each shadow host along the event path is that even non-bubbling events appear to bubble.
This is especially evident in situations where logical units of an application are composed using Shadow DOM. Here's a simple example:
I am building a grocery list app. Suppose I am currently making the grocery list editor. The EDIT ITEM dialog box is in its own shadow tree. Inside of this box, I have input fields, as well as a CANCEL button and a DELETE button.
Clicking CANCEL fires a simple 'cancel' event on the EDIT ITEM's shadow host. The event is meant to be listened to by the owner of the EDIT ITEM dialog box (to close it, presumably).
Clicking DELETE opens another CONFIRM DELETE dialog (also in a shadow tree) on top of the EDIT ITEM dialog. That dialog has OK and CANCEL as well.
Suppose that I repeat my event pattern on the CONFIRM DELETE dialog. Clicking CANCEL there fires a simple 'cancel' event. The idea is that my EDIT ITEM dialog can listen to this event and close the CONFIRM DELETE dialog.
Sounds simplistic enough, right? Here's where things get interesting.
Unfortunately for me, when I click on CANCEL in the CONFIRM DELETE dialog, my EDIT ITEM dialog goes away as well!
The reason for this strange occurrence is because, despite 'cancel' being a simple event, the AT_TARGET listener invocation occurs both at the CONFIRM DELETE shadow host and at the EDIT ITEM shadow host.
In other words, the bubbling event listener that is only supposed to hear EDIT ITEM's 'cancel' event will also hear the CONFIRM DELETE's 'cancel' event -- even though I, the developer, specifically told this event not to bubble.
Now I have no choice but work around this problem by attempting to distinguish different 'cancel' events that are coming through.
I think the solution to this conundrum is the same as the one for the bug 20247.
No, events don't appear as bubbling in the scope of the host. What happens
under the host doesn't matter. If a "cancel" is dispatched there, then
it is supposed to look like its target is actually the host.
Capturing event listeners should fire during capture phase, and
phase should be at_target is currentTarget == target.
bubbling event listeners would then fire during bubble phase, and
phase would be at_target when currentTarget == target.
I don't quite see what bubbling actually have to do with your example.
What if web page uses capturing listeners? You'd have the same issue even
if bubbling would stop earlier.
What we may need is some way to say that events x,y,z shouldn't propagate out from the shadow dom, either in capture or bubble phase.
No, events don't appear as bubbling in the scope of the host. What happens
under the host doesn't matter. If a "cancel" is dispatched there, then
it is supposed to look like its target is actually the host.
Capturing event listeners should fire during capture phase, and
phase should be at_target is currentTarget == target.
bubbling event listeners would then fire during bubble phase, and
phase would be at_target when currentTarget == target.
Yes, this is all good. I was there, remember? :)
I don't quite see what bubbling actually have to do with your example.
What if web page uses capturing listeners? You'd have the same issue even
if bubbling would stop earlier.
Take your browser eng hat off for a minute and put your average webdev hat on.
As an average web dev, you only have a vague idea what the hell capturing listener is. You mostly do element.addEventListener(name, handler).
So you have this app structure (idealized/simplified):
grocery-app.html:
...
#shadow-root
...
#shadow-root
...
...
<script>
...
var editGroceryItem = document.querySelector('edit-grocery-item-dialog');
editGroceryItem.addEventListener('cancel', function() {
console.log('CANCEL: closing edit grocery item!');
...
});
...
var confirmDelete = editGroceryItem.shadowRoot.querySelector('confirm-delete-dialog');
confirmDelete.addEventListener('cancel', function() {
console.log('CANCEL: closing confirm!');
});
...
// when CANCEL button in the confirm delete dialog is clicked...
buttonInsideConfirmDeleteDialog.dispatchEvent(new CustomEvent('cancel'));
</script>
Now, as a webdev, you'll be surprised (and probably frustrated) to find out that you ALWAYS get both event handlers invoked when you click on the cancel delete button. You'll be like "whaaa? I just wanted to close the confirm delete dialog. You stupid SOB, why are also closing my edit item dialog?"
The web developer's expectation that the event doesn't bubble, and that whole AT_TARGET mumbo jumbo that you and I are painfully aware of is but a faint nagging feeling that something's gone terribly wrong.
What we may need is some way to say that events x,y,z shouldn't propagate
out from the shadow dom, either in capture or bubble phase.
What we may need is some way to say that events x,y,z shouldn't propagate
out from the shadow dom, either in capture or bubble phase.
Yes! That's what I said in the first comment! :)
I don't see it mentioned in the first comment ;)
I'm thinking some API like
interface ShadowRoot {
...
void setEncapsulatedEvents(sequence types);
}
That would prevent propagation of those events in case the event is originally
from somewhere under the shadow tree (i.e. shadowhost.contains(event.target) would return false or so)
Title: [Shadow]: Consider not using AT_TARGET more than once in the event path (bugzilla: 26892)
Migrated from: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892
comment: 0
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c0
Dimitri Glazkov wrote on 2014-09-23 15:36:32 +0000.
I was wrong in bug 15543. The effect of invoking event listeners with AT_TARGET at each shadow host along the event path is that even non-bubbling events appear to bubble.
This is especially evident in situations where logical units of an application are composed using Shadow DOM. Here's a simple example:
I am building a grocery list app. Suppose I am currently making the grocery list editor. The EDIT ITEM dialog box is in its own shadow tree. Inside of this box, I have input fields, as well as a CANCEL button and a DELETE button.
Clicking CANCEL fires a simple 'cancel' event on the EDIT ITEM's shadow host. The event is meant to be listened to by the owner of the EDIT ITEM dialog box (to close it, presumably).
Clicking DELETE opens another CONFIRM DELETE dialog (also in a shadow tree) on top of the EDIT ITEM dialog. That dialog has OK and CANCEL as well.
Suppose that I repeat my event pattern on the CONFIRM DELETE dialog. Clicking CANCEL there fires a simple 'cancel' event. The idea is that my EDIT ITEM dialog can listen to this event and close the CONFIRM DELETE dialog.
Sounds simplistic enough, right? Here's where things get interesting.
Unfortunately for me, when I click on CANCEL in the CONFIRM DELETE dialog, my EDIT ITEM dialog goes away as well!
The reason for this strange occurrence is because, despite 'cancel' being a simple event, the AT_TARGET listener invocation occurs both at the CONFIRM DELETE shadow host and at the EDIT ITEM shadow host.
In other words, the bubbling event listener that is only supposed to hear EDIT ITEM's 'cancel' event will also hear the CONFIRM DELETE's 'cancel' event -- even though I, the developer, specifically told this event not to bubble.
Now I have no choice but work around this problem by attempting to distinguish different 'cancel' events that are coming through.
I think the solution to this conundrum is the same as the one for the bug 20247.
comment: 1
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c1
Olli Pettay wrote on 2014-09-23 16:26:29 +0000.
No, events don't appear as bubbling in the scope of the host. What happens
under the host doesn't matter. If a "cancel" is dispatched there, then
it is supposed to look like its target is actually the host.
Capturing event listeners should fire during capture phase, and
phase should be at_target is currentTarget == target.
bubbling event listeners would then fire during bubble phase, and
phase would be at_target when currentTarget == target.
I don't quite see what bubbling actually have to do with your example.
What if web page uses capturing listeners? You'd have the same issue even
if bubbling would stop earlier.
What we may need is some way to say that events x,y,z shouldn't propagate out from the shadow dom, either in capture or bubble phase.
comment: 2
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c2
Dimitri Glazkov wrote on 2014-09-24 16:23:18 +0000.
(In reply to Olli Pettay from comment #1)
Yes, this is all good. I was there, remember? :)
Take your browser eng hat off for a minute and put your average webdev hat on.
As an average web dev, you only have a vague idea what the hell capturing listener is. You mostly do element.addEventListener(name, handler).
So you have this app structure (idealized/simplified):
grocery-app.html:
<script> ... var editGroceryItem = document.querySelector('edit-grocery-item-dialog'); editGroceryItem.addEventListener('cancel', function() { console.log('CANCEL: closing edit grocery item!'); ... }); ... var confirmDelete = editGroceryItem.shadowRoot.querySelector('confirm-delete-dialog'); confirmDelete.addEventListener('cancel', function() { console.log('CANCEL: closing confirm!'); }); ... // when CANCEL button in the confirm delete dialog is clicked... buttonInsideConfirmDeleteDialog.dispatchEvent(new CustomEvent('cancel')); </script>...
#shadow-root
...
#shadow-root
...
...
Now, as a webdev, you'll be surprised (and probably frustrated) to find out that you ALWAYS get both event handlers invoked when you click on the cancel delete button. You'll be like "whaaa? I just wanted to close the confirm delete dialog. You stupid SOB, why are also closing my edit item dialog?"
The web developer's expectation that the event doesn't bubble, and that whole AT_TARGET mumbo jumbo that you and I are painfully aware of is but a faint nagging feeling that something's gone terribly wrong.
Yes! That's what I said in the first comment! :)
comment: 3
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c3
Olli Pettay wrote on 2014-09-24 16:33:09 +0000.
(In reply to Dimitri Glazkov from comment #2)
I'm thinking some API like
interface ShadowRoot {
...
void setEncapsulatedEvents(sequence types);
}
That would prevent propagation of those events in case the event is originally
from somewhere under the shadow tree (i.e. shadowhost.contains(event.target) would return false or so)
comment: 4
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c4
Olli Pettay wrote on 2014-09-24 16:34:07 +0000.
And yes, the method name in the example is horrible, something better should be used.
comment: 5
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c5
Hayato Ito wrote on 2015-04-22 22:29:23 +0000.
If we are pursuing something like "void setEncapsulatedEvents(sequence types)", can we merge this issue to https://www.w3.org/Bugs/Public/show_bug.cgi?id=20247 ?
I think both topics are related. I'd like to have one good idea which resolves both.
comment: 6
comment_url: https://www.w3.org/Bugs/Public/show_bug.cgi?id=26892#c6
Dimitri Glazkov wrote on 2015-04-23 18:21:42 +0000.
*** Bug 28246 has been marked as a duplicate of this bug. ***
The text was updated successfully, but these errors were encountered: