@ -12,7 +12,7 @@ import { namespaces } from '../../../../utils/namespaces';
import AttributeWrapper from './Attribute' ;
import AttributeWrapper from './Attribute' ;
import StyleAttributeWrapper from './StyleAttribute' ;
import StyleAttributeWrapper from './StyleAttribute' ;
import SpreadAttributeWrapper from './SpreadAttribute' ;
import SpreadAttributeWrapper from './SpreadAttribute' ;
import { dimensions, start_newline } from '../../../../utils/patterns' ;
import { regex_ dimensions, regex_ starts_with _newline, regex_backslashes } from '../../../../utils/patterns' ;
import Binding from './Binding' ;
import Binding from './Binding' ;
import add_to_set from '../../../utils/add_to_set' ;
import add_to_set from '../../../utils/add_to_set' ;
import { add_event_handler } from '../shared/add_event_handlers' ;
import { add_event_handler } from '../shared/add_event_handlers' ;
@ -34,12 +34,16 @@ interface BindingGroup {
bindings : Binding [ ] ;
bindings : Binding [ ] ;
}
}
const regex_contains_radio_or_checkbox_or_file = /radio|checkbox|file/ ;
const regex_contains_radio_or_checkbox_or_range_or_file = /radio|checkbox|range|file/ ;
const events = [
const events = [
{
{
event_names : [ 'input' ] ,
event_names : [ 'input' ] ,
filter : ( node : Element , _name : string ) = >
filter : ( node : Element , _name : string ) = >
node . name === 'textarea' ||
node . name === 'textarea' ||
node . name === 'input' && ! /radio|checkbox|range|file/ . test ( node . get_static_attribute_value ( 'type' ) as string )
node . name === 'input' &&
! regex_contains_radio_or_checkbox_or_range_or_file . test ( node . get_static_attribute_value ( 'type' ) as string )
} ,
} ,
{
{
event_names : [ 'input' ] ,
event_names : [ 'input' ] ,
@ -51,7 +55,8 @@ const events = [
event_names : [ 'change' ] ,
event_names : [ 'change' ] ,
filter : ( node : Element , _name : string ) = >
filter : ( node : Element , _name : string ) = >
node . name === 'select' ||
node . name === 'select' ||
node . name === 'input' && /radio|checkbox|file/ . test ( node . get_static_attribute_value ( 'type' ) as string )
node . name === 'input' &&
regex_contains_radio_or_checkbox_or_file . test ( node . get_static_attribute_value ( 'type' ) as string )
} ,
} ,
{
{
event_names : [ 'change' , 'input' ] ,
event_names : [ 'change' , 'input' ] ,
@ -62,7 +67,7 @@ const events = [
{
{
event_names : [ 'elementresize' ] ,
event_names : [ 'elementresize' ] ,
filter : ( _node : Element , name : string ) = >
filter : ( _node : Element , name : string ) = >
dimensions. test ( name )
regex_ dimensions. test ( name )
} ,
} ,
// media events
// media events
@ -136,6 +141,8 @@ const events = [
] ;
] ;
const CHILD_DYNAMIC_ELEMENT_BLOCK = 'child_dynamic_element' ;
const CHILD_DYNAMIC_ELEMENT_BLOCK = 'child_dynamic_element' ;
const regex_invalid_variable_identifier_characters = /[^a-zA-Z0-9_$]/g ;
const regex_minus_signs = /-/g ;
export default class ElementWrapper extends Wrapper {
export default class ElementWrapper extends Wrapper {
node : Element ;
node : Element ;
@ -182,7 +189,7 @@ export default class ElementWrapper extends Wrapper {
this . var = {
this . var = {
type : 'Identifier' ,
type : 'Identifier' ,
name : node.name.replace ( /[^a-zA-Z0-9_$]/g , '_' )
name : node.name.replace ( regex_invalid_variable_identifier_characters , '_' )
} ;
} ;
this . void = is_void ( node . name ) ;
this . void = is_void ( node . name ) ;
@ -1100,7 +1107,7 @@ export default class ElementWrapper extends Wrapper {
const snippet = expression . manipulate ( block ) ;
const snippet = expression . manipulate ( block ) ;
let cached_snippet ;
let cached_snippet ;
if ( should_cache ) {
if ( should_cache ) {
cached_snippet = block . get_unique_name ( ` style_ ${ name . replace ( /-/g , '_' ) } ` ) ;
cached_snippet = block . get_unique_name ( ` style_ ${ name . replace ( regex_minus_signs , '_' ) } ` ) ;
block . add_variable ( cached_snippet , snippet ) ;
block . add_variable ( cached_snippet , snippet ) ;
}
}
@ -1138,6 +1145,9 @@ export default class ElementWrapper extends Wrapper {
}
}
}
}
const regex_backticks = /`/g ;
const regex_dollar_signs = /\$/g ;
function to_html ( wrappers : Array < ElementWrapper | TextWrapper | MustacheTagWrapper | RawMustacheTagWrapper > , block : Block , literal : any , state : any , can_use_raw_text? : boolean ) {
function to_html ( wrappers : Array < ElementWrapper | TextWrapper | MustacheTagWrapper | RawMustacheTagWrapper > , block : Block , literal : any , state : any , can_use_raw_text? : boolean ) {
wrappers . forEach ( wrapper = > {
wrappers . forEach ( wrapper = > {
if ( wrapper instanceof TextWrapper ) {
if ( wrapper instanceof TextWrapper ) {
@ -1155,9 +1165,9 @@ function to_html(wrappers: Array<ElementWrapper | TextWrapper | MustacheTagWrapp
) ;
) ;
state . quasi . value . raw += ( raw ? wrapper.data : escape_html ( wrapper . data ) )
state . quasi . value . raw += ( raw ? wrapper.data : escape_html ( wrapper . data ) )
. replace ( /\\/g , '\\\\' )
. replace ( regex_backslashes , '\\\\' )
. replace ( /`/g , '\\`' )
. replace ( regex_backticks , '\\`' )
. replace ( /\$/g , '\\$' ) ;
. replace ( regex_dollar_signs , '\\$' ) ;
} else if ( wrapper instanceof MustacheTagWrapper || wrapper instanceof RawMustacheTagWrapper ) {
} else if ( wrapper instanceof MustacheTagWrapper || wrapper instanceof RawMustacheTagWrapper ) {
literal . quasis . push ( state . quasi ) ;
literal . quasis . push ( state . quasi ) ;
literal . expressions . push ( wrapper . node . expression . manipulate ( block ) ) ;
literal . expressions . push ( wrapper . node . expression . manipulate ( block ) ) ;
@ -1192,7 +1202,7 @@ function to_html(wrappers: Array<ElementWrapper | TextWrapper | MustacheTagWrapp
// Two or more leading newlines are required to restore the leading newline immediately after `<pre>`.
// Two or more leading newlines are required to restore the leading newline immediately after `<pre>`.
// see https://html.spec.whatwg.org/multipage/grouping-content.html#the-pre-element
// see https://html.spec.whatwg.org/multipage/grouping-content.html#the-pre-element
const first = wrapper . fragment . nodes [ 0 ] ;
const first = wrapper . fragment . nodes [ 0 ] ;
if ( first && first . node . type === 'Text' && start_newline. test ( first . node . data ) ) {
if ( first && first . node . type === 'Text' && regex_ starts_with _newline. test ( first . node . data ) ) {
state . quasi . value . raw += '\n' ;
state . quasi . value . raw += '\n' ;
}
}
}
}
@ -1204,7 +1214,7 @@ function to_html(wrappers: Array<ElementWrapper | TextWrapper | MustacheTagWrapp
// Two or more leading newlines are required to restore the leading newline immediately after `<textarea>`.
// Two or more leading newlines are required to restore the leading newline immediately after `<textarea>`.
// see https://html.spec.whatwg.org/multipage/syntax.html#element-restrictions
// see https://html.spec.whatwg.org/multipage/syntax.html#element-restrictions
const first = value_attribute . node . chunks [ 0 ] ;
const first = value_attribute . node . chunks [ 0 ] ;
if ( first && first . type === 'Text' && start_newline. test ( first . data ) ) {
if ( first && first . type === 'Text' && regex_ starts_with _newline. test ( first . data ) ) {
state . quasi . value . raw += '\n' ;
state . quasi . value . raw += '\n' ;
}
}
to_html_for_attr_value ( value_attribute , block , literal , state ) ;
to_html_for_attr_value ( value_attribute , block , literal , state ) ;