@@ -29,7 +29,8 @@ import {
29
29
build_render_statement ,
30
30
build_template_chunk ,
31
31
build_update_assignment ,
32
- get_expression_id
32
+ get_expression_id ,
33
+ memoize_expression
33
34
} from './shared/utils.js' ;
34
35
import { visit_event_attribute } from './shared/events.js' ;
35
36
@@ -532,18 +533,30 @@ function build_element_attribute_update_assignment(
532
533
const is_svg = context . state . metadata . namespace === 'svg' || element . name === 'svg' ;
533
534
const is_mathml = context . state . metadata . namespace === 'mathml' ;
534
535
536
+ const is_autofocus = name === 'autofocus' ;
537
+
535
538
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
537
546
) ;
538
547
539
- if ( name === 'autofocus' ) {
548
+ if ( is_autofocus ) {
540
549
state . init . push ( b . stmt ( b . call ( '$.autofocus' , node_id , value ) ) ) ;
541
550
return false ;
542
551
}
543
552
544
553
// Special case for Firefox who needs it set as a property in order to work
545
554
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 ) ) ) ;
547
560
return false ;
548
561
}
549
562
@@ -660,8 +673,18 @@ function build_custom_element_attribute_update_assignment(node_id, attribute, co
660
673
*/
661
674
function build_element_special_value_attribute ( element , node_id , attribute , context ) {
662
675
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
+
663
681
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
665
688
) ;
666
689
667
690
const inner_assignment = b . assignment (
@@ -674,11 +697,6 @@ function build_element_special_value_attribute(element, node_id, attribute, cont
674
697
)
675
698
) ;
676
699
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
-
682
700
const update = b . stmt (
683
701
is_select_with_value
684
702
? b . sequence ( [
0 commit comments