Skip to content

Commit d8e78f7

Browse files
paoloricciutijhwz
andauthored
fix: muted reactive without bind and select/autofocus attributes … (#15326)
* fix: `muted` reactive without `bind` and select/autofocus attributes working with function calls Co-authored-by: Joel Howse <[email protected]> * chore: add comments --------- Co-authored-by: Joel Howse <[email protected]>
1 parent 9091e7e commit d8e78f7

File tree

8 files changed

+85
-10
lines changed

8 files changed

+85
-10
lines changed

.changeset/silver-ties-itch.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'svelte': patch
3+
---
4+
5+
fix: `muted` reactive without `bind` and select/autofocus attributes working with function calls

packages/svelte/src/compiler/phases/3-transform/client/visitors/RegularElement.js

+28-10
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,8 @@ import {
2929
build_render_statement,
3030
build_template_chunk,
3131
build_update_assignment,
32-
get_expression_id
32+
get_expression_id,
33+
memoize_expression
3334
} from './shared/utils.js';
3435
import { visit_event_attribute } from './shared/events.js';
3536

@@ -532,18 +533,30 @@ function build_element_attribute_update_assignment(
532533
const is_svg = context.state.metadata.namespace === 'svg' || element.name === 'svg';
533534
const is_mathml = context.state.metadata.namespace === 'mathml';
534535

536+
const is_autofocus = name === 'autofocus';
537+
535538
let { value, has_state } = build_attribute_value(attribute.value, context, (value, metadata) =>
536-
metadata.has_call ? get_expression_id(state, value) : value
539+
metadata.has_call
540+
? // if it's autofocus we will not add this to a template effect so we don't want to get the expression id
541+
// but separately memoize the expression
542+
is_autofocus
543+
? memoize_expression(state, value)
544+
: get_expression_id(state, value)
545+
: value
537546
);
538547

539-
if (name === 'autofocus') {
548+
if (is_autofocus) {
540549
state.init.push(b.stmt(b.call('$.autofocus', node_id, value)));
541550
return false;
542551
}
543552

544553
// Special case for Firefox who needs it set as a property in order to work
545554
if (name === 'muted') {
546-
state.init.push(b.stmt(b.assignment('=', b.member(node_id, b.id('muted')), value)));
555+
if (!has_state) {
556+
state.init.push(b.stmt(b.assignment('=', b.member(node_id, b.id('muted')), value)));
557+
return false;
558+
}
559+
state.update.push(b.stmt(b.assignment('=', b.member(node_id, b.id('muted')), value)));
547560
return false;
548561
}
549562

@@ -660,8 +673,18 @@ function build_custom_element_attribute_update_assignment(node_id, attribute, co
660673
*/
661674
function build_element_special_value_attribute(element, node_id, attribute, context) {
662675
const state = context.state;
676+
const is_select_with_value =
677+
// attribute.metadata.dynamic would give false negatives because even if the value does not change,
678+
// the inner options could still change, so we need to always treat it as reactive
679+
element === 'select' && attribute.value !== true && !is_text_attribute(attribute);
680+
663681
const { value, has_state } = build_attribute_value(attribute.value, context, (value, metadata) =>
664-
metadata.has_call ? get_expression_id(state, value) : value
682+
metadata.has_call
683+
? // if is a select with value we will also invoke `init_select` which need a reference before the template effect so we memoize separately
684+
is_select_with_value
685+
? memoize_expression(context.state, value)
686+
: get_expression_id(state, value)
687+
: value
665688
);
666689

667690
const inner_assignment = b.assignment(
@@ -674,11 +697,6 @@ function build_element_special_value_attribute(element, node_id, attribute, cont
674697
)
675698
);
676699

677-
const is_select_with_value =
678-
// attribute.metadata.dynamic would give false negatives because even if the value does not change,
679-
// the inner options could still change, so we need to always treat it as reactive
680-
element === 'select' && attribute.value !== true && !is_text_attribute(attribute);
681-
682700
const update = b.stmt(
683701
is_select_with_value
684702
? b.sequence([
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { test } from '../../test';
2+
3+
export default test({
4+
async test({ assert, errors }) {
5+
assert.deepEqual(errors, []);
6+
}
7+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
<svelte:options runes />
2+
<script>
3+
function test(){}
4+
</script>
5+
6+
<input autofocus={test()} />
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { flushSync } from 'svelte';
2+
import { ok, test } from '../../test';
3+
4+
export default test({
5+
async test({ assert, target, logs }) {
6+
const btn = target.querySelector('button');
7+
ok(btn);
8+
flushSync(() => {
9+
btn.click();
10+
});
11+
assert.deepEqual(logs, [true]);
12+
}
13+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<script>
2+
let muted = $state(false);
3+
4+
function volume_change(node){
5+
node.addEventListener("volumechange", () => {
6+
console.log(node.muted);
7+
});
8+
}
9+
</script>
10+
11+
<audio use:volume_change {muted}></audio>
12+
<button onclick={() => (muted = !muted)}></button>
13+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { test } from '../../test';
2+
3+
export default test({
4+
async test({ assert, errors }) {
5+
assert.deepEqual(errors, []);
6+
}
7+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
<svelte:options runes />
2+
<script>
3+
function test(){}
4+
</script>
5+
6+
<select value={test()}></select>

0 commit comments

Comments
 (0)