Fix set:html
behavior with null
(#2790)
* feat: improve set:html behavior for null/undefined * chore: add changeset * refactor: improve set:html and set:text documentation * test: improve set:html tests * refactor: better types for server API
This commit is contained in:
parent
de2b246237
commit
6b34840d3d
5 changed files with 57 additions and 6 deletions
5
.changeset/serious-guests-matter.md
Normal file
5
.changeset/serious-guests-matter.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Improve `set:html` behavior for `null` and `undefined` values
|
|
@ -1,6 +1,7 @@
|
||||||
const entities = { '"': 'quot', '&': 'amp', "'": 'apos', '<': 'lt', '>': 'gt' } as const;
|
const entities = { '"': 'quot', '&': 'amp', "'": 'apos', '<': 'lt', '>': 'gt' } as const;
|
||||||
|
|
||||||
export const escapeHTML = (string: any) => string.replace(/["'&<>]/g, (char: keyof typeof entities) => '&' + entities[char] + ';');
|
// This util is only ever run on expression values that we already know are of type `string`
|
||||||
|
export const escapeHTML = (str: string) => str.replace(/["'&<>]/g, (char: string) => '&' + entities[char as keyof typeof entities] + ';');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* RawString is a "blessed" version of String
|
* RawString is a "blessed" version of String
|
||||||
|
@ -11,7 +12,14 @@ export class UnescapedString extends String {}
|
||||||
/**
|
/**
|
||||||
* unescapeHTML marks a string as raw, unescaped HTML.
|
* unescapeHTML marks a string as raw, unescaped HTML.
|
||||||
* This should only be generated internally, not a public API.
|
* This should only be generated internally, not a public API.
|
||||||
*
|
|
||||||
* Need to cast the return value `as unknown as string` so TS doesn't yell at us.
|
|
||||||
*/
|
*/
|
||||||
export const unescapeHTML = (str: any) => new UnescapedString(str) as unknown as string;
|
export const unescapeHTML = (value: any) => {
|
||||||
|
// Cast any `string` values to `UnescapedString` to mark them as ignored
|
||||||
|
// The `as unknown as string` is necessary for TypeScript to treat this as `string`
|
||||||
|
if (typeof value === 'string') {
|
||||||
|
return new UnescapedString(value) as unknown as string;
|
||||||
|
}
|
||||||
|
// Just return values that are `number`, `null`, `undefined` etc
|
||||||
|
// The compiler will recursively stringify these correctly
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
|
@ -418,7 +418,7 @@ export async function renderEndpoint(mod: EndpointHandler, params: any) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calls a component and renders it into a string of HTML
|
// Calls a component and renders it into a string of HTML
|
||||||
export async function renderToString(result: SSRResult, componentFactory: AstroComponentFactory, props: any, children: any) {
|
export async function renderToString(result: SSRResult, componentFactory: AstroComponentFactory, props: any, children: any): Promise<string> {
|
||||||
const Component = await componentFactory(result, props, children);
|
const Component = await componentFactory(result, props, children);
|
||||||
let template = await renderAstroComponent(Component);
|
let template = await renderAstroComponent(Component);
|
||||||
|
|
||||||
|
@ -439,7 +439,7 @@ const uniqueElements = (item: any, index: number, all: any[]) => {
|
||||||
|
|
||||||
// Renders a page to completion by first calling the factory callback, waiting for its result, and then appending
|
// Renders a page to completion by first calling the factory callback, waiting for its result, and then appending
|
||||||
// styles and scripts into the head.
|
// styles and scripts into the head.
|
||||||
export async function renderHead(result: SSRResult) {
|
export async function renderHead(result: SSRResult): Promise<string> {
|
||||||
const styles = Array.from(result.styles)
|
const styles = Array.from(result.styles)
|
||||||
.filter(uniqueElements)
|
.filter(uniqueElements)
|
||||||
.map((style) => {
|
.map((style) => {
|
||||||
|
|
|
@ -17,4 +17,30 @@ describe('Directives', async () => {
|
||||||
expect($('script#inline')).to.have.lengthOf(1);
|
expect($('script#inline')).to.have.lengthOf(1);
|
||||||
expect($('script#inline').toString()).to.include('let foo = "bar"');
|
expect($('script#inline').toString()).to.include('let foo = "bar"');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('set:html', async () => {
|
||||||
|
const html = await fixture.readFile('/set-html/index.html');
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect($('#text')).to.have.lengthOf(1);
|
||||||
|
expect($('#text').text()).to.equal('a');
|
||||||
|
|
||||||
|
expect($('#zero')).to.have.lengthOf(1);
|
||||||
|
expect($('#zero').text()).to.equal('0');
|
||||||
|
|
||||||
|
expect($('#number')).to.have.lengthOf(1);
|
||||||
|
expect($('#number').text()).to.equal('1');
|
||||||
|
|
||||||
|
expect($('#undefined')).to.have.lengthOf(1);
|
||||||
|
expect($('#undefined').text()).to.equal('');
|
||||||
|
|
||||||
|
expect($('#null')).to.have.lengthOf(1);
|
||||||
|
expect($('#null').text()).to.equal('');
|
||||||
|
|
||||||
|
expect($('#false')).to.have.lengthOf(1);
|
||||||
|
expect($('#false').text()).to.equal('');
|
||||||
|
|
||||||
|
expect($('#true')).to.have.lengthOf(1);
|
||||||
|
expect($('#true').text()).to.equal('true');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
12
packages/astro/test/fixtures/astro-directives/src/pages/set-html.astro
vendored
Normal file
12
packages/astro/test/fixtures/astro-directives/src/pages/set-html.astro
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
<html>
|
||||||
|
<head></head>
|
||||||
|
<body>
|
||||||
|
<div id="text" set:html={"a"} />
|
||||||
|
<div id="zero" set:html={0} />
|
||||||
|
<div id="number" set:html={1} />
|
||||||
|
<div id="false" set:html={false} />
|
||||||
|
<div id="true" set:html={true} />
|
||||||
|
<div id="undefined" set:html={undefined} />
|
||||||
|
<div id="null" set:html={null} />
|
||||||
|
</body>
|
||||||
|
</html>
|
Loading…
Reference in a new issue