diff --git a/.changeset/nine-pigs-approve.md b/.changeset/nine-pigs-approve.md new file mode 100644 index 0000000000..b11daff5a3 --- /dev/null +++ b/.changeset/nine-pigs-approve.md @@ -0,0 +1,5 @@ +--- +'svelte': patch +--- + +fix: allow binding to const with spread in legacy mode diff --git a/packages/svelte/src/internal/client/reactivity/props.js b/packages/svelte/src/internal/client/reactivity/props.js index 1d0f7b239f..1df1933c19 100644 --- a/packages/svelte/src/internal/client/reactivity/props.js +++ b/packages/svelte/src/internal/client/reactivity/props.js @@ -177,6 +177,19 @@ const spread_props_handler = { if (typeof p === 'object' && p !== null && key in p) return p[key]; } }, + set(target, key, value) { + let i = target.props.length; + while (i--) { + let p = target.props[i]; + if (is_function(p)) p = p(); + const desc = get_descriptor(p, key); + if (desc && desc.set) { + desc.set(value); + return true; + } + } + return false; + }, getOwnPropertyDescriptor(target, key) { let i = target.props.length; while (i--) { diff --git a/packages/svelte/src/internal/server/index.js b/packages/svelte/src/internal/server/index.js index 9b5e4ac3e2..9a66095de4 100644 --- a/packages/svelte/src/internal/server/index.js +++ b/packages/svelte/src/internal/server/index.js @@ -270,7 +270,12 @@ export function spread_props(props) { for (let i = 0; i < props.length; i++) { const obj = props[i]; for (key in obj) { - merged_props[key] = obj[key]; + const desc = Object.getOwnPropertyDescriptor(obj, key); + if (desc) { + Object.defineProperty(merged_props, key, desc); + } else { + merged_props[key] = obj[key]; + } } } return merged_props; diff --git a/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/Test.svelte b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/Test.svelte new file mode 100644 index 0000000000..1ec9031319 --- /dev/null +++ b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/Test.svelte @@ -0,0 +1,3 @@ + \ No newline at end of file diff --git a/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/_config.js b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/_config.js new file mode 100644 index 0000000000..0645e5bc7f --- /dev/null +++ b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/_config.js @@ -0,0 +1,9 @@ +import { test } from '../../test'; + +export default test({ + html: `

42

`, + async test({ target, assert }) { + const p = target.querySelector('p'); + assert.equal(p?.innerHTML, '42'); + } +}); diff --git a/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/main.svelte b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/main.svelte new file mode 100644 index 0000000000..8a0a370acb --- /dev/null +++ b/packages/svelte/tests/runtime-legacy/samples/bind-export-const-with-spread/main.svelte @@ -0,0 +1,11 @@ + + + +

{x}

\ No newline at end of file