mirror of
https://github.com/withastro/astro.git
synced 2024-12-30 22:03:56 -05:00
Fix SVG Component sprite ids (#12511)
* fix: sprite ids per page * update tests to cover the failed scenario * add changeset
This commit is contained in:
parent
a326c2f1da
commit
d023682d6c
3 changed files with 28 additions and 20 deletions
5
.changeset/red-paws-juggle.md
Normal file
5
.changeset/red-paws-juggle.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix SVG Component sprite references
|
|
@ -16,21 +16,13 @@ export interface SvgComponentProps {
|
||||||
/**
|
/**
|
||||||
* Make sure these IDs are kept on the module-level so they're incremented on a per-page basis
|
* Make sure these IDs are kept on the module-level so they're incremented on a per-page basis
|
||||||
*/
|
*/
|
||||||
const ids = new WeakMap<SSRResult, number>();
|
const countersByPage = new WeakMap<SSRResult, number>();
|
||||||
let counter = 0;
|
|
||||||
|
|
||||||
export function createSvgComponent({ meta, attributes, children }: SvgComponentProps) {
|
export function createSvgComponent({ meta, attributes, children }: SvgComponentProps) {
|
||||||
const rendered = new WeakSet<Response>();
|
const renderedIds = new WeakMap<SSRResult, string>();
|
||||||
|
|
||||||
const Component = createComponent((result, props) => {
|
const Component = createComponent((result, props) => {
|
||||||
let id;
|
let counter = countersByPage.get(result) ?? 0;
|
||||||
if (ids.has(result)) {
|
|
||||||
id = ids.get(result)!;
|
|
||||||
} else {
|
|
||||||
counter += 1;
|
|
||||||
ids.set(result, counter);
|
|
||||||
id = counter;
|
|
||||||
}
|
|
||||||
id = `a:${id}`;
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
title: titleProp,
|
title: titleProp,
|
||||||
|
@ -41,12 +33,15 @@ export function createSvgComponent({ meta, attributes, children }: SvgComponentP
|
||||||
const title = titleProp ? unescapeHTML(`<title>${titleProp}</title>`) : '';
|
const title = titleProp ? unescapeHTML(`<title>${titleProp}</title>`) : '';
|
||||||
|
|
||||||
if (mode === 'sprite') {
|
if (mode === 'sprite') {
|
||||||
// On the first render, include the symbol definition
|
// On the first render, include the symbol definition and bump the counter
|
||||||
let symbol: any = '';
|
let symbol: any = '';
|
||||||
if (!rendered.has(result.response)) {
|
let id = renderedIds.get(result);
|
||||||
|
if (!id) {
|
||||||
|
countersByPage.set(result, ++counter);
|
||||||
|
id = `a:${counter}`;
|
||||||
// We only need the viewBox on the symbol definition, we can drop it everywhere else
|
// We only need the viewBox on the symbol definition, we can drop it everywhere else
|
||||||
symbol = unescapeHTML(`<symbol${spreadAttributes({ viewBox, id })}>${children}</symbol>`);
|
symbol = unescapeHTML(`<symbol${spreadAttributes({ viewBox, id })}>${children}</symbol>`);
|
||||||
rendered.add(result.response);
|
renderedIds.set(result, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
return render`<svg${spreadAttributes(normalizedProps)}>${title}${symbol}<use href="#${id}" /></svg>`;
|
return render`<svg${spreadAttributes(normalizedProps)}>${title}${symbol}<use href="#${id}" /></svg>`;
|
||||||
|
|
|
@ -322,8 +322,9 @@ describe('astro:assets - SVG Components', () => {
|
||||||
assert.equal($symbol.length, 1);
|
assert.equal($symbol.length, 1);
|
||||||
let $use = $('.one svg > use');
|
let $use = $('.one svg > use');
|
||||||
assert.equal($use.length, 2);
|
assert.equal($use.length, 2);
|
||||||
let defId = $('.one.def svg > use').attr('id');
|
const defId = $('.one.def svg > symbol').attr('id');
|
||||||
let useId = $('.one.use svg > use').attr('id');
|
const useId = $('.one.use svg > use').attr('href').replace('#', '');
|
||||||
|
assert.ok(defId);
|
||||||
assert.equal(defId, useId);
|
assert.equal(defId, useId);
|
||||||
|
|
||||||
// Second SVG
|
// Second SVG
|
||||||
|
@ -333,9 +334,10 @@ describe('astro:assets - SVG Components', () => {
|
||||||
assert.equal($symbol.length, 1);
|
assert.equal($symbol.length, 1);
|
||||||
$use = $('.two svg > use');
|
$use = $('.two svg > use');
|
||||||
assert.equal($use.length, 2);
|
assert.equal($use.length, 2);
|
||||||
defId = $('.two.def svg > use').attr('id');
|
const defId2 = $('.two.def svg > symbol').attr('id');
|
||||||
useId = $('.two.use svg > use').attr('id');
|
const useId2 = $('.two.use svg > use').attr('href').replace('#', '');
|
||||||
assert.equal(defId, useId);
|
assert.ok(defId2);
|
||||||
|
assert.equal(defId2, useId2);
|
||||||
|
|
||||||
// Third SVG
|
// Third SVG
|
||||||
$svg = $('.three svg');
|
$svg = $('.three svg');
|
||||||
|
@ -344,6 +346,12 @@ describe('astro:assets - SVG Components', () => {
|
||||||
assert.equal($symbol.length, 1);
|
assert.equal($symbol.length, 1);
|
||||||
$use = $('.three svg > use');
|
$use = $('.three svg > use');
|
||||||
assert.equal($use.length, 1);
|
assert.equal($use.length, 1);
|
||||||
|
const defId3 = $('.three.def svg > symbol').attr('id');
|
||||||
|
assert.ok(defId3);
|
||||||
|
|
||||||
|
// Assert IDs are different
|
||||||
|
assert.equal(new Set([defId, defId2, defId3]).size, 3);
|
||||||
|
assert.equal(new Set([useId, useId2]).size, 2);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue