Skip to content

Commit

Permalink
fix: image-set with base64 images (fix #8028) (#8035)
Browse files Browse the repository at this point in the history
  • Loading branch information
patak-dev committed May 11, 2022
1 parent 84496f8 commit 992aee2
Show file tree
Hide file tree
Showing 5 changed files with 76 additions and 6 deletions.
17 changes: 16 additions & 1 deletion packages/playground/assets/__tests__/assets.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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) => {
Expand Down
19 changes: 19 additions & 0 deletions packages/playground/assets/css/css-url.css

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

10 changes: 10 additions & 0 deletions packages/playground/assets/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,16 @@ <h2>CSS url references</h2>
CSS background image-set() (mix var and url)
</span>
</div>
<div class="css-image-set-base64">
<span style="background: #fff">
CSS background image-set() (with base64)
</span>
</div>
<div class="css-image-set-gradient">
<span style="background: #fff">
CSS background image-set() (with gradient)
</span>
</div>
<div class="css-image-set-multiple-descriptor">
<span style="background: #fff">
CSS background image-set() (with multiple descriptor)
Expand Down
14 changes: 11 additions & 3 deletions packages/vite/src/node/plugins/css.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1052,18 +1052,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<string> {
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
})
Expand Down
22 changes: 20 additions & 2 deletions packages/vite/src/node/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -560,8 +560,7 @@ export async function processSrcSet(
srcs: string,
replacer: (arg: ImageCandidate) => Promise<string>
): Promise<string> {
const imageCandidates: ImageCandidate[] = srcs
.split(',')
const imageCandidates: ImageCandidate[] = splitSrcSet(srcs)
.map((s) => {
const src = s.replace(escapedSpaceCharacters, ' ').trim()
const [url] = imageSetUrlRE.exec(src) || []
Expand Down Expand Up @@ -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/')
Expand Down

0 comments on commit 992aee2

Please sign in to comment.