fix: only escape attribute values for elements, not components (#9456)

* only escape attribute values for elements, not components - closes #9454

* changeset

---------

Co-authored-by: Rich Harris <rich.harris@vercel.com>
pull/9457/head
Rich Harris 1 year ago committed by GitHub
parent 9abfb52f67
commit 6f6c3a0787
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: only escape attribute values for elements, not components

@ -611,9 +611,15 @@ const javascript_visitors_runes = {
* @param {true | Array<import('#compiler').Text | import('#compiler').ExpressionTag>} attribute_value * @param {true | Array<import('#compiler').Text | import('#compiler').ExpressionTag>} attribute_value
* @param {import('./types').ComponentContext} context * @param {import('./types').ComponentContext} context
* @param {boolean} trim_whitespace * @param {boolean} trim_whitespace
* @param {boolean} is_component
* @returns {import('estree').Expression} * @returns {import('estree').Expression}
*/ */
function serialize_attribute_value(attribute_value, context, trim_whitespace = false) { function serialize_attribute_value(
attribute_value,
context,
trim_whitespace = false,
is_component = false
) {
if (attribute_value === true) { if (attribute_value === true) {
return b.true; return b.true;
} }
@ -629,7 +635,8 @@ function serialize_attribute_value(attribute_value, context, trim_whitespace = f
if (trim_whitespace) { if (trim_whitespace) {
data = data.replace(regex_whitespaces_strict, ' ').trim(); data = data.replace(regex_whitespaces_strict, ' ').trim();
} }
return b.literal(escape_html(data, true));
return b.literal(is_component ? data : escape_html(data, true));
} else { } else {
return /** @type {import('estree').Expression} */ (context.visit(value.expression)); return /** @type {import('estree').Expression} */ (context.visit(value.expression));
} }
@ -777,12 +784,12 @@ function serialize_inline_component(node, component_name, context) {
} else if (attribute.type === 'Attribute') { } else if (attribute.type === 'Attribute') {
if (attribute.name === 'slot') continue; if (attribute.name === 'slot') continue;
if (attribute.name.startsWith('--')) { if (attribute.name.startsWith('--')) {
const value = serialize_attribute_value(attribute.value, context); const value = serialize_attribute_value(attribute.value, context, false, true);
custom_css_props.push(b.init(attribute.name, value)); custom_css_props.push(b.init(attribute.name, value));
continue; continue;
} }
const value = serialize_attribute_value(attribute.value, context); const value = serialize_attribute_value(attribute.value, context, false, true);
push_prop(b.prop('init', b.key(attribute.name), value)); push_prop(b.prop('init', b.key(attribute.name), value));
} else if (attribute.type === 'BindDirective') { } else if (attribute.type === 'BindDirective') {
// TODO this needs to turn the whole thing into a while loop because the binding could be mutated eagerly in the child // TODO this needs to turn the whole thing into a while loop because the binding could be mutated eagerly in the child

@ -0,0 +1,5 @@
<script>
const { prop } = $props();
</script>
{prop}

@ -0,0 +1,5 @@
import { test } from '../../test';
export default test({
html: `&quot;`
});

@ -0,0 +1,5 @@
<script>
import Child from './Child.svelte';
</script>
<Child prop='"'/>
Loading…
Cancel
Save