diff --git a/.changeset/sharp-shoes-look.md b/.changeset/sharp-shoes-look.md
new file mode 100644
index 0000000000..2caf6d421b
--- /dev/null
+++ b/.changeset/sharp-shoes-look.md
@@ -0,0 +1,5 @@
+---
+'svelte': patch
+---
+
+fix: take into account registration state when setting custom element props
diff --git a/packages/svelte/src/internal/client/dom/elements/attributes.js b/packages/svelte/src/internal/client/dom/elements/attributes.js
index 0276069eee..8f6588e94a 100644
--- a/packages/svelte/src/internal/client/dom/elements/attributes.js
+++ b/packages/svelte/src/internal/client/dom/elements/attributes.js
@@ -201,7 +201,7 @@ export function set_xlink_attribute(dom, attribute, value) {
}
/**
- * @param {any} node
+ * @param {HTMLElement} node
* @param {string} prop
* @param {any} value
*/
@@ -216,10 +216,21 @@ export function set_custom_element_data(node, prop, value) {
set_active_reaction(null);
set_active_effect(null);
try {
- if (get_setters(node).includes(prop)) {
+ if (
+ // Don't compute setters for custom elements while they aren't registered yet,
+ // because during their upgrade/instantiation they might add more setters.
+ // Instead, fall back to a simple "an object, then set as property" heuristic.
+ setters_cache.has(node.nodeName) || customElements.get(node.tagName.toLowerCase())
+ ? get_setters(node).includes(prop)
+ : value && typeof value === 'object'
+ ) {
+ // @ts-expect-error
node[prop] = value;
} else {
- set_attribute(node, prop, value);
+ // We did getters etc checks already, stringify before passing to set_attribute
+ // to ensure it doesn't invoke the same logic again, and potentially populating
+ // the setters cache too early.
+ set_attribute(node, prop, value == null ? value : String(value));
}
} finally {
set_active_reaction(previous_reaction);
@@ -384,8 +395,9 @@ function get_setters(element) {
var setters = setters_cache.get(element.nodeName);
if (setters) return setters;
setters_cache.set(element.nodeName, (setters = []));
+
var descriptors;
- var proto = get_prototype_of(element);
+ var proto = element; // In the case of custom elements there might be setters on the instance
var element_proto = Element.prototype;
// Stop at Element, from there on there's only unnecessary setters we're not interested in
diff --git a/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/_config.js b/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/_config.js
new file mode 100644
index 0000000000..0679c820ba
--- /dev/null
+++ b/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/_config.js
@@ -0,0 +1,25 @@
+import { flushSync } from 'svelte';
+import { test } from '../../assert';
+
+const tick = () => Promise.resolve();
+
+// Check that rendering a custom element and setting a property before it is registered
+// does not break the "when to set this as a property" logic
+export default test({
+ async test({ assert, target }) {
+ target.innerHTML = '';
+ await tick();
+ await tick();
+
+ const ce_root = target.querySelector('custom-element').shadowRoot;
+
+ ce_root.querySelector('button')?.click();
+ flushSync();
+ await tick();
+ await tick();
+
+ const inner_ce_root = ce_root.querySelectorAll('set-property-before-mounted');
+ assert.htmlEqual(inner_ce_root[0].shadowRoot.innerHTML, 'object|{"foo":"bar"}');
+ assert.htmlEqual(inner_ce_root[1].shadowRoot.innerHTML, 'object|{"foo":"bar"}');
+ }
+});
diff --git a/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/main.svelte b/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/main.svelte
new file mode 100644
index 0000000000..09a139e642
--- /dev/null
+++ b/packages/svelte/tests/runtime-browser/custom-elements-samples/late-ce-mount/main.svelte
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+
+
+{#if property}
+
+{/if}