fix: correctly call exported state (#10114)

fixes #10104
also cleans up related code and adds support for destructuring `$state.frozen`
pull/10121/head
Simon H 1 year ago committed by GitHub
parent 92408e1506
commit b3d185da29
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: correctly call exported state

@ -7,7 +7,7 @@ import { global_visitors } from './visitors/global.js';
import { javascript_visitors } from './visitors/javascript.js'; import { javascript_visitors } from './visitors/javascript.js';
import { javascript_visitors_runes } from './visitors/javascript-runes.js'; import { javascript_visitors_runes } from './visitors/javascript-runes.js';
import { javascript_visitors_legacy } from './visitors/javascript-legacy.js'; import { javascript_visitors_legacy } from './visitors/javascript-legacy.js';
import { serialize_get_binding } from './utils.js'; import { is_state_source, serialize_get_binding } from './utils.js';
import { remove_types } from '../typescript.js'; import { remove_types } from '../typescript.js';
/** /**
@ -242,9 +242,7 @@ export function client_component(source, analysis, options) {
const properties = analysis.exports.map(({ name, alias }) => { const properties = analysis.exports.map(({ name, alias }) => {
const binding = analysis.instance.scope.get(name); const binding = analysis.instance.scope.get(name);
const is_source = const is_source = binding !== null && is_state_source(binding, state);
(binding?.kind === 'state' || binding?.kind === 'frozen_state') &&
(!state.analysis.immutable || binding.reassigned);
// TODO This is always a getter because the `renamed-instance-exports` test wants it that way. // TODO This is always a getter because the `renamed-instance-exports` test wants it that way.
// Should we for code size reasons make it an init in runes mode and/or non-dev mode? // Should we for code size reasons make it an init in runes mode and/or non-dev mode?

@ -45,6 +45,18 @@ export function get_assignment_value(node, { state, visit }) {
} }
} }
/**
* @param {import('#compiler').Binding} binding
* @param {import('./types').ClientTransformState} state
* @returns {boolean}
*/
export function is_state_source(binding, state) {
return (
(binding.kind === 'state' || binding.kind === 'frozen_state') &&
(!state.analysis.immutable || binding.reassigned || state.analysis.accessors)
);
}
/** /**
* @param {import('estree').Identifier} node * @param {import('estree').Identifier} node
* @param {import('./types').ClientTransformState} state * @param {import('./types').ClientTransformState} state
@ -93,8 +105,7 @@ export function serialize_get_binding(node, state) {
} }
if ( if (
((binding.kind === 'state' || binding.kind === 'frozen_state') && is_state_source(binding, state) ||
(!state.analysis.immutable || state.analysis.accessors || binding.reassigned)) ||
binding.kind === 'derived' || binding.kind === 'derived' ||
binding.kind === 'legacy_reactive' binding.kind === 'legacy_reactive'
) { ) {
@ -491,33 +502,6 @@ export function get_prop_source(binding, state, name, initial) {
return b.call('$.prop', ...args); return b.call('$.prop', ...args);
} }
/**
* Creates the output for a state declaration.
* @param {import('estree').VariableDeclarator} declarator
* @param {import('../../scope').Scope} scope
* @param {import('estree').Expression} value
*/
export function create_state_declarators(declarator, scope, value) {
// in the simple `let count = $state(0)` case, we rewrite `$state` as `$.source`
if (declarator.id.type === 'Identifier') {
return [b.declarator(declarator.id, b.call('$.mutable_source', value))];
}
const tmp = scope.generate('tmp');
const paths = extract_paths(declarator.id);
return [
b.declarator(b.id(tmp), value), // TODO inject declarator for opts, so we can use it below
...paths.map((path) => {
const value = path.expression?.(b.id(tmp));
const binding = scope.get(/** @type {import('estree').Identifier} */ (path.node).name);
return b.declarator(
path.node,
binding?.kind === 'state' ? b.call('$.mutable_source', value) : value
);
})
];
}
/** @param {import('estree').Expression} node */ /** @param {import('estree').Expression} node */
export function should_proxy_or_freeze(node) { export function should_proxy_or_freeze(node) {
if ( if (

@ -1,7 +1,33 @@
import { is_hoistable_function } from '../../utils.js'; import { is_hoistable_function } from '../../utils.js';
import * as b from '../../../../utils/builders.js'; import * as b from '../../../../utils/builders.js';
import { extract_paths } from '../../../../utils/ast.js'; import { extract_paths } from '../../../../utils/ast.js';
import { create_state_declarators, get_prop_source, serialize_get_binding } from '../utils.js'; import { get_prop_source, serialize_get_binding } from '../utils.js';
/**
* Creates the output for a state declaration.
* @param {import('estree').VariableDeclarator} declarator
* @param {import('../../../scope.js').Scope} scope
* @param {import('estree').Expression} value
*/
function create_state_declarators(declarator, scope, value) {
if (declarator.id.type === 'Identifier') {
return [b.declarator(declarator.id, b.call('$.mutable_source', value))];
}
const tmp = scope.generate('tmp');
const paths = extract_paths(declarator.id);
return [
b.declarator(b.id(tmp), value),
...paths.map((path) => {
const value = path.expression?.(b.id(tmp));
const binding = scope.get(/** @type {import('estree').Identifier} */ (path.node).name);
return b.declarator(
path.node,
binding?.kind === 'state' ? b.call('$.mutable_source', value) : value
);
})
];
}
/** @type {import('../types.js').ComponentVisitors} */ /** @type {import('../types.js').ComponentVisitors} */
export const javascript_visitors_legacy = { export const javascript_visitors_legacy = {

@ -2,8 +2,8 @@ import { get_rune } from '../../../scope.js';
import { is_hoistable_function, transform_inspect_rune } from '../../utils.js'; import { is_hoistable_function, transform_inspect_rune } from '../../utils.js';
import * as b from '../../../../utils/builders.js'; import * as b from '../../../../utils/builders.js';
import * as assert from '../../../../utils/assert.js'; import * as assert from '../../../../utils/assert.js';
import { create_state_declarators, get_prop_source, should_proxy_or_freeze } from '../utils.js'; import { get_prop_source, is_state_source, should_proxy_or_freeze } from '../utils.js';
import { unwrap_ts_expression } from '../../../../utils/ast.js'; import { extract_paths, unwrap_ts_expression } from '../../../../utils/ast.js';
/** @type {import('../types.js').ComponentVisitors} */ /** @type {import('../types.js').ComponentVisitors} */
export const javascript_visitors_runes = { export const javascript_visitors_runes = {
@ -223,66 +223,79 @@ export const javascript_visitors_runes = {
} }
const args = /** @type {import('estree').CallExpression} */ (init).arguments; const args = /** @type {import('estree').CallExpression} */ (init).arguments;
let value = const value =
args.length === 0 args.length === 0
? b.id('undefined') ? b.id('undefined')
: /** @type {import('estree').Expression} */ (visit(args[0])); : /** @type {import('estree').Expression} */ (visit(args[0]));
if (declarator.id.type === 'Identifier') { if (rune === '$state' || rune === '$state.frozen') {
if (rune === '$state') { /**
const binding = /** @type {import('#compiler').Binding} */ ( * @param {import('estree').Identifier} id
state.scope.get(declarator.id.name) * @param {import('estree').Expression} value
); */
const create_state_declarator = (id, value) => {
const binding = /** @type {import('#compiler').Binding} */ (state.scope.get(id.name));
if (should_proxy_or_freeze(value)) { if (should_proxy_or_freeze(value)) {
value = b.call('$.proxy', value); value = b.call(rune === '$state' ? '$.proxy' : '$.freeze', value);
} }
if (is_state_source(binding, state)) {
if (!state.analysis.immutable || state.analysis.accessors || binding.reassigned) {
value = b.call('$.source', value); value = b.call('$.source', value);
} }
} else if (rune === '$state.frozen') { return value;
const binding = /** @type {import('#compiler').Binding} */ ( };
state.scope.get(declarator.id.name)
);
if (should_proxy_or_freeze(value)) {
value = b.call('$.freeze', value);
}
if (binding.reassigned) { if (declarator.id.type === 'Identifier') {
value = b.call('$.source', value); declarations.push(
} b.declarator(declarator.id, create_state_declarator(declarator.id, value))
);
} else { } else {
value = b.call('$.derived', b.thunk(value)); const tmp = state.scope.generate('tmp');
const paths = extract_paths(declarator.id);
declarations.push(
b.declarator(b.id(tmp), value),
...paths.map((path) => {
const value = path.expression?.(b.id(tmp));
const binding = state.scope.get(
/** @type {import('estree').Identifier} */ (path.node).name
);
return b.declarator(
path.node,
binding?.kind === 'state' || binding?.kind === 'frozen_state'
? create_state_declarator(binding.node, value)
: value
);
})
);
} }
declarations.push(b.declarator(declarator.id, value));
continue; continue;
} }
if (rune === '$derived') { if (rune === '$derived') {
const bindings = state.scope.get_bindings(declarator); if (declarator.id.type === 'Identifier') {
const id = state.scope.generate('derived_value'); declarations.push(b.declarator(declarator.id, b.call('$.derived', b.thunk(value))));
declarations.push( } else {
b.declarator( const bindings = state.scope.get_bindings(declarator);
b.id(id), const id = state.scope.generate('derived_value');
b.call( declarations.push(
'$.derived', b.declarator(
b.thunk( b.id(id),
b.block([ b.call(
b.let(declarator.id, value), '$.derived',
b.return(b.array(bindings.map((binding) => binding.node))) b.thunk(
]) b.block([
b.let(declarator.id, value),
b.return(b.array(bindings.map((binding) => binding.node)))
])
)
) )
) )
) );
); for (let i = 0; i < bindings.length; i++) {
for (let i = 0; i < bindings.length; i++) { bindings[i].expression = b.member(b.call('$.get', b.id(id)), b.literal(i), true);
bindings[i].expression = b.member(b.call('$.get', b.id(id)), b.literal(i), true); }
} }
continue; continue;
} }
declarations.push(...create_state_declarators(declarator, state.scope, value));
} }
if (declarations.length === 0) { if (declarations.length === 0) {

@ -1,13 +1,11 @@
import { test } from '../../test'; import { test } from '../../test';
export default test({ export default test({
html: `<button>0</button>`, html: `<button>0 / 0</button>`,
async test({ assert, target, window }) { async test({ assert, target }) {
const btn = target.querySelector('button'); const btn = target.querySelector('button');
const clickEvent = new window.Event('click', { bubbles: true }); await btn?.click();
await btn?.dispatchEvent(clickEvent); assert.htmlEqual(target.innerHTML, `<button>1 / 1</button>`);
assert.htmlEqual(target.innerHTML, `<button>1</button>`);
} }
}); });

@ -2,6 +2,7 @@
import { setup } from './utils.js'; import { setup } from './utils.js';
let { num } = $state(setup()); let { num } = $state(setup());
let { num: num_frozen } = $state(setup());
</script> </script>
<button on:click={() => num++}>{num}</button> <button on:click={() => { num++; num_frozen++; }}>{num} / {num_frozen}</button>

@ -0,0 +1,12 @@
import { test } from '../../test';
export default test({
async test({ assert, target }) {
assert.htmlEqual(target.innerHTML, `0 0 <button>0 / 0</button>`);
const [btn] = target.querySelectorAll('button');
btn?.click();
await Promise.resolve();
assert.htmlEqual(target.innerHTML, '0 1 <button>0 / 1</button>');
}
});

@ -0,0 +1,7 @@
<script>
import Sub from './sub.svelte'
let sub = $state();
</script>
<Sub bind:this={sub} />
<button on:click={() => sub.increment()}>{sub?.count1.value} / {sub?.count2.value}</button>

@ -0,0 +1,15 @@
<script>
export const count1 = $state.frozen({value: 0});
export const count2 = $state({value: 0});
export function increment() {
count2.value += 1;
}
</script>
{count1.value}
{count2.value}
<!-- so that count1/2 become sources -->
<svelte:options accessors />
Loading…
Cancel
Save