Skip to content

Commit 992aee2

Browse files
committed
fix: image-set with base64 images (fix #8028) (#8035)
1 parent 84496f8 commit 992aee2

File tree

5 files changed

+76
-6
lines changed

5 files changed

+76
-6
lines changed

packages/playground/assets/__tests__/assets.spec.ts

+16-1
Original file line numberDiff line numberDiff line change
@@ -118,8 +118,23 @@ describe('css url() references', () => {
118118
})
119119
})
120120

121-
// not supported in browser now
121+
test('image-set with base64', async () => {
122+
const imageSet = await getBg('.css-image-set-base64')
123+
expect(imageSet).toMatch(
124+
`-webkit-image-set(url("") 1x, url("") 2x)`
125+
)
126+
})
127+
128+
// TODO: not supported in chrome
122129
// https://drafts.csswg.org/css-images-4/#image-set-notation
130+
//
131+
// test('image-set with multiple descriptor', async () => {
132+
// const imageSet = await getBg('.css-image-set-gradient')
133+
// expect(imageSet).toMatch(
134+
// `-webkit-image-set(url("") 1x, linear-gradient(#e66465, #9198e5) 2x)`
135+
// )
136+
// })
137+
//
123138
// test('image-set with multiple descriptor', async () => {
124139
// const imageSet = await getBg('.css-image-set-multiple-descriptor')
125140
// imageSet.split(', ').forEach((s) => {

packages/playground/assets/css/css-url.css

+19
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/playground/assets/index.html

+10
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,16 @@ <h2>CSS url references</h2>
5656
CSS background image-set() (mix var and url)
5757
</span>
5858
</div>
59+
<div class="css-image-set-base64">
60+
<span style="background: #fff">
61+
CSS background image-set() (with base64)
62+
</span>
63+
</div>
64+
<div class="css-image-set-gradient">
65+
<span style="background: #fff">
66+
CSS background image-set() (with gradient)
67+
</span>
68+
</div>
5969
<div class="css-image-set-multiple-descriptor">
6070
<span style="background: #fff">
6171
CSS background image-set() (with multiple descriptor)

packages/vite/src/node/plugins/css.ts

+11-3
Original file line numberDiff line numberDiff line change
@@ -1052,18 +1052,26 @@ function rewriteImportCss(
10521052
})
10531053
}
10541054

1055-
function rewriteCssImageSet(
1055+
// TODO: image and cross-fade could contain a "url" that needs to be processed
1056+
// https://drafts.csswg.org/css-images-4/#image-notation
1057+
// https://drafts.csswg.org/css-images-4/#cross-fade-function
1058+
const cssNotProcessedRE = /(gradient|element|cross-fade|image)\(/
1059+
1060+
async function rewriteCssImageSet(
10561061
css: string,
10571062
replacer: CssUrlReplacer
10581063
): Promise<string> {
1059-
return asyncReplace(css, cssImageSetRE, async (match) => {
1064+
return await asyncReplace(css, cssImageSetRE, async (match) => {
10601065
const [, rawUrl] = match
10611066
const url = await processSrcSet(rawUrl, async ({ url }) => {
10621067
// the url maybe url(...)
10631068
if (cssUrlRE.test(url)) {
10641069
return await rewriteCssUrls(url, replacer)
10651070
}
1066-
return await doUrlReplace(url, url, replacer)
1071+
if (!cssNotProcessedRE.test(url)) {
1072+
return await doUrlReplace(url, url, replacer)
1073+
}
1074+
return url
10671075
})
10681076
return url
10691077
})

packages/vite/src/node/utils.ts

+20-2
Original file line numberDiff line numberDiff line change
@@ -560,8 +560,7 @@ export async function processSrcSet(
560560
srcs: string,
561561
replacer: (arg: ImageCandidate) => Promise<string>
562562
): Promise<string> {
563-
const imageCandidates: ImageCandidate[] = srcs
564-
.split(',')
563+
const imageCandidates: ImageCandidate[] = splitSrcSet(srcs)
565564
.map((s) => {
566565
const src = s.replace(escapedSpaceCharacters, ' ').trim()
567566
const [url] = imageSetUrlRE.exec(src) || []
@@ -589,6 +588,25 @@ export async function processSrcSet(
589588
}, '')
590589
}
591590

591+
function splitSrcSet(srcs: string) {
592+
const parts: string[] = []
593+
// There could be a ',' inside of url(data:...), linear-gradient(...) or "data:..."
594+
const cleanedSrcs = srcs.replace(
595+
/(?:url|image|gradient|cross-fade)\([^\)]*\)|"([^"]|(?<=\\)")*"|'([^']|(?<=\\)')*'/g,
596+
blankReplacer
597+
)
598+
let startIndex = 0
599+
let splitIndex: number
600+
do {
601+
splitIndex = cleanedSrcs.indexOf(',', startIndex)
602+
parts.push(
603+
srcs.slice(startIndex, splitIndex !== -1 ? splitIndex : undefined)
604+
)
605+
startIndex = splitIndex + 1
606+
} while (splitIndex !== -1)
607+
return parts
608+
}
609+
592610
function escapeToLinuxLikePath(path: string) {
593611
if (/^[A-Z]:/.test(path)) {
594612
return path.replace(/^([A-Z]):\//, '/windows/$1/')

0 commit comments

Comments
 (0)