fix: reset `reset_element` in `render` to prevent runtime error (#13669)

pull/13675/head
Paolo Ricciuti 3 months ago committed by GitHub
parent 9f930f4e9b
commit 0598f2bbe1
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: reset `reset_element` in `render` to prevent runtime error

@ -53,7 +53,11 @@ function print_error(payload, parent, child) {
} }
export function reset_elements() { export function reset_elements() {
let old_parent = parent;
parent = null; parent = null;
return () => {
parent = old_parent;
};
} }
/** /**

@ -101,9 +101,11 @@ export function render(component, options = {}) {
on_destroy = []; on_destroy = [];
payload.out += BLOCK_OPEN; payload.out += BLOCK_OPEN;
let reset_reset_element;
if (DEV) { if (DEV) {
// prevent parent/child element state being corrupted by a bad render // prevent parent/child element state being corrupted by a bad render
reset_elements(); reset_reset_element = reset_elements();
} }
if (options.context) { if (options.context) {
@ -118,6 +120,10 @@ export function render(component, options = {}) {
pop(); pop();
} }
if (reset_reset_element) {
reset_reset_element();
}
payload.out += BLOCK_CLOSE; payload.out += BLOCK_CLOSE;
for (const cleanup of on_destroy) cleanup(); for (const cleanup of on_destroy) cleanup();
on_destroy = prev_on_destroy; on_destroy = prev_on_destroy;

@ -0,0 +1,7 @@
<script>
let count = $state(0);
</script>
<button onclick={() => (count += 1)}>
clicks: {count}
</button>

@ -0,0 +1,23 @@
import { flushSync } from 'svelte';
import { test } from '../../test';
export default test({
props: {
browser: true
},
server_props: {
browser: false
},
compileOptions: {
dev: true
},
html: `<div><div><button>clicks: 0</button></div></div>`,
test({ target, assert }) {
const button = target.querySelector('button');
flushSync(() => button?.click());
assert.htmlEqual(target.innerHTML, `<div><div><button>clicks: 1</button></div></div>`);
}
});

@ -0,0 +1,22 @@
<script>
import { createRawSnippet, hydrate } from 'svelte';
import { render } from 'svelte/server';
import Child from './Child.svelte';
let { browser } = $props();
let count = $state(0);
const hello = createRawSnippet((count) => ({
render: () => `
<div>${browser ? '' : render(Child).body}</div>
`,
setup(target) {
hydrate(Child, { target })
}
}));
</script>
<div>
{@render hello()}
</div>
Loading…
Cancel
Save