diff --git a/packages/vite/src/node/plugins/css.ts b/packages/vite/src/node/plugins/css.ts index 3ffd93bb9f9910..c13a4c83052639 100644 --- a/packages/vite/src/node/plugins/css.ts +++ b/packages/vite/src/node/plugins/css.ts @@ -1045,18 +1045,26 @@ function rewriteImportCss( }) } -function rewriteCssImageSet( +// TODO: image and cross-fade could contain a "url" that needs to be processed +// https://drafts.csswg.org/css-images-4/#image-notation +// https://drafts.csswg.org/css-images-4/#cross-fade-function +const cssNotProcessedRE = /(gradient|element|cross-fade|image)\(/ + +async function rewriteCssImageSet( css: string, replacer: CssUrlReplacer ): Promise { - return asyncReplace(css, cssImageSetRE, async (match) => { + return await asyncReplace(css, cssImageSetRE, async (match) => { const [, rawUrl] = match const url = await processSrcSet(rawUrl, async ({ url }) => { // the url maybe url(...) if (cssUrlRE.test(url)) { return await rewriteCssUrls(url, replacer) } - return await doUrlReplace(url, url, replacer) + if (!cssNotProcessedRE.test(url)) { + return await doUrlReplace(url, url, replacer) + } + return url }) return url }) diff --git a/packages/vite/src/node/utils.ts b/packages/vite/src/node/utils.ts index ddcaced9832bc1..0c25b4592adf6d 100644 --- a/packages/vite/src/node/utils.ts +++ b/packages/vite/src/node/utils.ts @@ -560,8 +560,7 @@ export async function processSrcSet( srcs: string, replacer: (arg: ImageCandidate) => Promise ): Promise { - const imageCandidates: ImageCandidate[] = srcs - .split(',') + const imageCandidates: ImageCandidate[] = splitSrcSet(srcs) .map((s) => { const src = s.replace(escapedSpaceCharacters, ' ').trim() const [url] = imageSetUrlRE.exec(src) || [] @@ -589,6 +588,25 @@ export async function processSrcSet( }, '') } +function splitSrcSet(srcs: string) { + const parts: string[] = [] + // There could be a ',' inside of url(data:...), linear-gradient(...) or "data:..." + const cleanedSrcs = srcs.replace( + /(?:url|image|gradient|cross-fade)\([^\)]*\)|"([^"]|(?<=\\)")*"|'([^']|(?<=\\)')*'/g, + blankReplacer + ) + let startIndex = 0 + let splitIndex: number + do { + splitIndex = cleanedSrcs.indexOf(',', startIndex) + parts.push( + srcs.slice(startIndex, splitIndex !== -1 ? splitIndex : undefined) + ) + startIndex = splitIndex + 1 + } while (splitIndex !== -1) + return parts +} + function escapeToLinuxLikePath(path: string) { if (/^[A-Z]:/.test(path)) { return path.replace(/^([A-Z]):\//, '/windows/$1/') diff --git a/playground/assets/__tests__/assets.spec.ts b/playground/assets/__tests__/assets.spec.ts index c63ffc7cc1c0c2..d9f1c8ed761773 100644 --- a/playground/assets/__tests__/assets.spec.ts +++ b/playground/assets/__tests__/assets.spec.ts @@ -118,8 +118,23 @@ describe('css url() references', () => { }) }) - // not supported in browser now + test('image-set with base64', async () => { + const imageSet = await getBg('.css-image-set-base64') + expect(imageSet).toMatch( + `-webkit-image-set(url("") 1x, url("") 2x)` + ) + }) + + // TODO: not supported in chrome // https://drafts.csswg.org/css-images-4/#image-set-notation + // + // test('image-set with multiple descriptor', async () => { + // const imageSet = await getBg('.css-image-set-gradient') + // expect(imageSet).toMatch( + // `-webkit-image-set(url("") 1x, linear-gradient(#e66465, #9198e5) 2x)` + // ) + // }) + // // test('image-set with multiple descriptor', async () => { // const imageSet = await getBg('.css-image-set-multiple-descriptor') // imageSet.split(', ').forEach((s) => { diff --git a/playground/assets/css/css-url.css b/playground/assets/css/css-url.css index 9047cd384e7d38..1446f19afa662e 100644 --- a/playground/assets/css/css-url.css +++ b/playground/assets/css/css-url.css @@ -41,6 +41,25 @@ background-size: 10px; } +.css-image-set-base64 { + background-image: -webkit-image-set( + url() + 1x, + url() + 2x + ); + background-size: 10px; +} + +.css-image-set-gradient { + background-image: -webkit-image-set( + '' + 1x, + linear-gradient(#e66465, #9198e5) 2x + ); + background-size: 10px; +} + .css-image-set-multiple-descriptor { background-image: -webkit-image-set( '../nested/asset.png' type('image/png') 1x, diff --git a/playground/assets/index.html b/playground/assets/index.html index 43eed55236abd3..fcf0e4ce1e81d7 100644 --- a/playground/assets/index.html +++ b/playground/assets/index.html @@ -56,6 +56,16 @@

CSS url references

CSS background image-set() (mix var and url) +
+ + CSS background image-set() (with base64) + +
+
+ + CSS background image-set() (with gradient) + +
CSS background image-set() (with multiple descriptor)