diff --git a/.changeset/fast-penguins-matter.md b/.changeset/fast-penguins-matter.md
new file mode 100644
index 000000000000..523aaf8a125c
--- /dev/null
+++ b/.changeset/fast-penguins-matter.md
@@ -0,0 +1,5 @@
+---
+"svelte": patch
+---
+
+fix: apply modifiers to bubbled events
diff --git a/packages/svelte/src/compiler/phases/3-transform/client/visitors/template.js b/packages/svelte/src/compiler/phases/3-transform/client/visitors/template.js
index 89eb32f429c3..3a7e52ac0c65 100644
--- a/packages/svelte/src/compiler/phases/3-transform/client/visitors/template.js
+++ b/packages/svelte/src/compiler/phases/3-transform/client/visitors/template.js
@@ -1170,8 +1170,11 @@ function serialize_render_stmt(state) {
* @param {import('../types.js').ComponentContext} context
*/
function serialize_event_handler(node, { state, visit }) {
+ /** @type {import('estree').Expression} */
+ let handler;
+
if (node.expression) {
- let handler = node.expression;
+ handler = node.expression;
// Event handlers can be dynamic (source/store/prop/conditional etc)
const dynamic_handler = () =>
@@ -1209,34 +1212,34 @@ function serialize_event_handler(node, { state, visit }) {
} else {
handler = /** @type {import('estree').Expression} */ (visit(handler));
}
-
- if (node.modifiers.includes('stopPropagation')) {
- handler = b.call('$.stopPropagation', handler);
- }
- if (node.modifiers.includes('stopImmediatePropagation')) {
- handler = b.call('$.stopImmediatePropagation', handler);
- }
- if (node.modifiers.includes('preventDefault')) {
- handler = b.call('$.preventDefault', handler);
- }
- if (node.modifiers.includes('self')) {
- handler = b.call('$.self', handler);
- }
- if (node.modifiers.includes('trusted')) {
- handler = b.call('$.trusted', handler);
- }
-
- return handler;
} else {
state.analysis.needs_props = true;
// Function + .call to preserve "this" context as much as possible
- return b.function(
+ handler = b.function(
null,
[b.id('$$arg')],
b.block([b.stmt(b.call('$.bubble_event.call', b.this, b.id('$$props'), b.id('$$arg')))])
);
}
+
+ if (node.modifiers.includes('stopPropagation')) {
+ handler = b.call('$.stopPropagation', handler);
+ }
+ if (node.modifiers.includes('stopImmediatePropagation')) {
+ handler = b.call('$.stopImmediatePropagation', handler);
+ }
+ if (node.modifiers.includes('preventDefault')) {
+ handler = b.call('$.preventDefault', handler);
+ }
+ if (node.modifiers.includes('self')) {
+ handler = b.call('$.self', handler);
+ }
+ if (node.modifiers.includes('trusted')) {
+ handler = b.call('$.trusted', handler);
+ }
+
+ return handler;
}
/**
diff --git a/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/_config.js b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/_config.js
new file mode 100644
index 000000000000..c30af43043d2
--- /dev/null
+++ b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/_config.js
@@ -0,0 +1,12 @@
+import { ok, test } from '../../test';
+
+export default test({
+ async test({ assert, component, target }) {
+ const button = target.querySelector('button');
+ ok(button);
+
+ await button.click();
+
+ assert.ok(component.default_was_prevented);
+ }
+});
diff --git a/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/button.svelte b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/button.svelte
new file mode 100644
index 000000000000..d94a505c0c82
--- /dev/null
+++ b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/button.svelte
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/main.svelte b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/main.svelte
new file mode 100644
index 000000000000..d5134e28fa1f
--- /dev/null
+++ b/packages/svelte/tests/runtime-legacy/samples/event-handler-modifier-bubble/main.svelte
@@ -0,0 +1,11 @@
+
+
+
\ No newline at end of file