From 46c33928feef3cc61c881160cb682bea8cfc2f06 Mon Sep 17 00:00:00 2001 From: Stein Strindhaug Date: Tue, 29 Aug 2023 12:29:51 +0200 Subject: [PATCH 1/2] test: add failing tests for external svg files in use-tags --- test/resources/svg-use-tag-external/image | 1 + test/resources/svg-use-tag-external/node.html | 9 +++++++ .../svg-use-tag-external/sprites.svg | 15 +++++++++++ test/resources/svg-use-tag-external/style.css | 27 +++++++++++++++++++ test/spec/helper.ts | 2 +- test/spec/svg.spec.ts | 13 ++++++++- 6 files changed, 65 insertions(+), 2 deletions(-) create mode 100644 test/resources/svg-use-tag-external/image create mode 100644 test/resources/svg-use-tag-external/node.html create mode 100644 test/resources/svg-use-tag-external/sprites.svg create mode 100644 test/resources/svg-use-tag-external/style.css diff --git a/test/resources/svg-use-tag-external/image b/test/resources/svg-use-tag-external/image new file mode 100644 index 00000000..5e5f4054 --- /dev/null +++ b/test/resources/svg-use-tag-external/image @@ -0,0 +1 @@ + diff --git a/test/resources/svg-use-tag-external/node.html b/test/resources/svg-use-tag-external/node.html new file mode 100644 index 00000000..b49f12ef --- /dev/null +++ b/test/resources/svg-use-tag-external/node.html @@ -0,0 +1,9 @@ +
+
+ + + +
+
diff --git a/test/resources/svg-use-tag-external/sprites.svg b/test/resources/svg-use-tag-external/sprites.svg new file mode 100644 index 00000000..0cbd2768 --- /dev/null +++ b/test/resources/svg-use-tag-external/sprites.svg @@ -0,0 +1,15 @@ + + + + home + + + + diff --git a/test/resources/svg-use-tag-external/style.css b/test/resources/svg-use-tag-external/style.css new file mode 100644 index 00000000..b51cf808 --- /dev/null +++ b/test/resources/svg-use-tag-external/style.css @@ -0,0 +1,27 @@ +#dom-node { + width: 100px; + overflow: hidden; +} + +#root { + border: 1px solid red; + position: relative; + height: 100px; +} + +svg { + width: 100%; + height: 100%; +} + +.icon { + display: inline-block; + width: 0.9285714285714285em; + height: 1em; + stroke-width: 0; + stroke: currentColor; + fill: currentColor; + vertical-align: middle; + top: -1px; + position: relative; +} diff --git a/test/spec/helper.ts b/test/spec/helper.ts index 03e8a432..7f49c33e 100644 --- a/test/spec/helper.ts +++ b/test/spec/helper.ts @@ -154,7 +154,7 @@ export function compareToRefImage(sourceData: ImageData, threshold = 0.1) { pixelmatch(sourceData.data, refData.data, null, ref.width, ref.height, { threshold, }), - ).toBeLessThan(100) + ).toBeLessThan(2) // Temporarily changed this threshold from 100 to 2 to so that failing tests actually fail } export async function getSvgDocument(dataUrl: string): Promise { diff --git a/test/spec/svg.spec.ts b/test/spec/svg.spec.ts index f4595a11..9e005d53 100644 --- a/test/spec/svg.spec.ts +++ b/test/spec/svg.spec.ts @@ -47,7 +47,7 @@ describe('work with svg element', () => { .catch(done) }) - it('should render SVG use tags', function (done) { + it('should render embedded SVG use tags', function (done) { bootstrap( 'svg-use-tag/node.html', 'svg-use-tag/style.css', @@ -57,4 +57,15 @@ describe('work with svg element', () => { .then(done) .catch(done) }) + + it('should render external SVG use tags', function (done) { + bootstrap( + 'svg-use-tag-external/node.html', + 'svg-use-tag-external/style.css', + 'svg-use-tag-external/image', + ) + .then(renderAndCheck) + .then(done) + .catch(done) + }) }) From 225e357cd971880c36542279960876d30d1651ae Mon Sep 17 00:00:00 2001 From: Stein Strindhaug Date: Tue, 29 Aug 2023 12:37:45 +0200 Subject: [PATCH 2/2] fix: make external svg files referred to in use tags work (#425 and #392) --- src/clone-node.ts | 44 ++++++++++++++++++++++++++++++++++++++------ 1 file changed, 38 insertions(+), 6 deletions(-) diff --git a/src/clone-node.ts b/src/clone-node.ts index 500ce332..741b3034 100644 --- a/src/clone-node.ts +++ b/src/clone-node.ts @@ -133,11 +133,11 @@ function cloneCSSStyle(nativeNode: T, clonedNode: T) { ) { value = 'block' } - + if (name === 'd' && clonedNode.getAttribute('d')) { value = `path(${clonedNode.getAttribute('d')})` } - + targetStyle.setProperty( name, value, @@ -193,13 +193,45 @@ async function ensureSVGSymbols( const processedDefs: { [key: string]: HTMLElement } = {} for (let i = 0; i < uses.length; i++) { const use = uses[i] - const id = use.getAttribute('xlink:href') + const href = + use.getAttribute('href') ?? // check href first as this is preferred and will be used if both are set + use.getAttribute('xlink:href') + + if (!href) continue // skip blank hrefs + + const [svgUrl, id] = href.split('#') + if (id) { - const exist = clone.querySelector(id) - const definition = document.querySelector(id) as HTMLElement - if (!exist && definition && !processedDefs[id]) { + const query = `#${id}` + // const definition = ownerDocument?.querySelector(`svg ${ query }`) + const exist = clone.querySelector(query) + const definition = document.querySelector(query) as HTMLElement + + if (svgUrl) { + // change the href attribute to use a local symbol on this cloned use-node + use.setAttribute('href', query) + // No need to set xlink:href since this is ignored when href is set + } + + if (exist || processedDefs[query]) continue // already exists in defs + + if (definition) { + // found local embedded definition // eslint-disable-next-line no-await-in-loop processedDefs[id] = (await cloneNode(definition, options, true))! + } else if (svgUrl) { + // no local definition but found an url + // try to fetch the svg and append it to the svgDefsElement + try { + // wrapped in try/catch since this is network calls that is likely to fail + // eslint-disable-next-line no-await-in-loop + const response = await fetch(svgUrl) + // eslint-disable-next-line no-await-in-loop + const svgData = await response.text() + clone.insertAdjacentHTML('beforeend', svgData) + } catch (error) { + // TODO log errors here? + } } } }