fix: more robust handling of var declarations (#12949)

* fix: more robust handling of var declarations

- fixes #12807: state declared with var is now retrieved using a new `safe_get` function, which works like `get` but checks for undefined
- fixes #12900: ensure we're not creating another new scope for block statements of function declarations, which resulted in var declarations getting "swallowed" (because they were hoisted to the function declaration scope and never seen by our logic)

* simplify

---------

Co-authored-by: Rich Harris <rich.harris@vercel.com>
pull/12960/head
Simon H 5 months ago committed by GitHub
parent 70ed8cd666
commit 01d32e53a9
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: more robust handling of var declarations

@ -284,7 +284,7 @@ export function VariableDeclaration(node, context) {
} }
/** /**
* Creates the output for a state declaration. * Creates the output for a state declaration in legacy mode.
* @param {VariableDeclarator} declarator * @param {VariableDeclarator} declarator
* @param {Scope} scope * @param {Scope} scope
* @param {Expression} value * @param {Expression} value

@ -23,7 +23,7 @@ export function add_state_transformers(context) {
binding.kind === 'legacy_reactive' binding.kind === 'legacy_reactive'
) { ) {
context.state.transform[name] = { context.state.transform[name] = {
read: get_value, read: binding.declaration_kind === 'var' ? (node) => b.call('$.safe_get', node) : get_value,
assign: (node, value) => { assign: (node, value) => {
let call = b.call('$.set', node, value); let call = b.call('$.set', node, value);

@ -461,8 +461,20 @@ export function create_scopes(ast, root, allow_reactive_declarations, parent) {
ForStatement: create_block_scope, ForStatement: create_block_scope,
ForInStatement: create_block_scope, ForInStatement: create_block_scope,
ForOfStatement: create_block_scope, ForOfStatement: create_block_scope,
BlockStatement: create_block_scope,
SwitchStatement: create_block_scope, SwitchStatement: create_block_scope,
BlockStatement(node, context) {
const parent = context.path.at(-1);
if (
parent?.type === 'FunctionDeclaration' ||
parent?.type === 'FunctionExpression' ||
parent?.type === 'ArrowFunctionExpression'
) {
// We already created a new scope for the function
context.next();
} else {
create_block_scope(node, context);
}
},
ClassDeclaration(node, { state, next }) { ClassDeclaration(node, { state, next }) {
if (node.id) state.scope.declare(node.id, 'normal', 'let', node); if (node.id) state.scope.declare(node.id, 'normal', 'let', node);

@ -127,6 +127,7 @@ export {
export { set_text } from './render.js'; export { set_text } from './render.js';
export { export {
get, get,
safe_get,
invalidate_inner_signals, invalidate_inner_signals,
flush_sync, flush_sync,
tick, tick,

@ -734,6 +734,16 @@ export function get(signal) {
return signal.v; return signal.v;
} }
/**
* Like `get`, but checks for `undefined`. Used for `var` declarations because they can be accessed before being declared
* @template V
* @param {Value<V> | undefined} signal
* @returns {V | undefined}
*/
export function safe_get(signal) {
return signal && get(signal);
}
/** /**
* Invokes a function and captures all signals that are read during the invocation, * Invokes a function and captures all signals that are read during the invocation,
* then invalidates them. * then invalidates them.

@ -0,0 +1,7 @@
import { test } from '../../test';
export default test({
test({ assert, logs }) {
assert.deepEqual(logs, [undefined, undefined, 10, 20, 0, 1]);
}
});

@ -0,0 +1,24 @@
<script>
console.log(foo, double);
var foo = $state(10);
var double = $derived(foo * 2);
console.log(foo, double);
function wrap(initial) {
var _value = $state(initial);
return {
get() {
return _value;
},
set(state) {
_value = state;
}
};
}
var wrapped = wrap(0);
console.log(wrapped.get());
wrapped.set(1);
console.log(wrapped.get());
</script>
Loading…
Cancel
Save