Skip to content

Commit eb22a74

Browse files
authored
fix(hmr): register inlined assets as a dependency of CSS file (#18979)
1 parent 56ad2be commit eb22a74

File tree

8 files changed

+134
-31
lines changed

8 files changed

+134
-31
lines changed

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

+48-27
Original file line numberDiff line numberDiff line change
@@ -374,20 +374,20 @@ export function cssPlugin(config: ResolvedConfig): Plugin {
374374
const resolveUrl = (url: string, importer?: string) =>
375375
idResolver(environment, url, importer)
376376

377-
const urlReplacer: CssUrlReplacer = async (url, importer) => {
377+
const urlResolver: CssUrlResolver = async (url, importer) => {
378378
const decodedUrl = decodeURI(url)
379379
if (checkPublicFile(decodedUrl, config)) {
380380
if (encodePublicUrlsInCSS(config)) {
381-
return publicFileToBuiltUrl(decodedUrl, config)
381+
return [publicFileToBuiltUrl(decodedUrl, config), undefined]
382382
} else {
383-
return joinUrlSegments(config.base, decodedUrl)
383+
return [joinUrlSegments(config.base, decodedUrl), undefined]
384384
}
385385
}
386386
const [id, fragment] = decodedUrl.split('#')
387387
let resolved = await resolveUrl(id, importer)
388388
if (resolved) {
389389
if (fragment) resolved += '#' + fragment
390-
return fileToUrl(this, resolved)
390+
return [await fileToUrl(this, resolved), resolved]
391391
}
392392
if (config.command === 'build') {
393393
const isExternal = config.build.rollupOptions.external
@@ -406,7 +406,7 @@ export function cssPlugin(config: ResolvedConfig): Plugin {
406406
)
407407
}
408408
}
409-
return url
409+
return [url, undefined]
410410
}
411411

412412
const {
@@ -419,7 +419,7 @@ export function cssPlugin(config: ResolvedConfig): Plugin {
419419
id,
420420
raw,
421421
preprocessorWorkerController!,
422-
urlReplacer,
422+
urlResolver,
423423
)
424424
if (modules) {
425425
moduleCache.set(id, modules)
@@ -1059,17 +1059,20 @@ export function cssAnalysisPlugin(config: ResolvedConfig): Plugin {
10591059
// main import to hot update
10601060
const depModules = new Set<string | EnvironmentModuleNode>()
10611061
for (const file of pluginImports) {
1062-
depModules.add(
1063-
isCSSRequest(file)
1064-
? moduleGraph.createFileOnlyEntry(file)
1065-
: await moduleGraph.ensureEntryFromUrl(
1066-
await fileToDevUrl(
1067-
this.environment,
1068-
file,
1069-
/* skipBase */ true,
1070-
),
1071-
),
1072-
)
1062+
if (isCSSRequest(file)) {
1063+
depModules.add(moduleGraph.createFileOnlyEntry(file))
1064+
} else {
1065+
const url = await fileToDevUrl(
1066+
this.environment,
1067+
file,
1068+
/* skipBase */ true,
1069+
)
1070+
if (url.startsWith('data:')) {
1071+
depModules.add(moduleGraph.createFileOnlyEntry(file))
1072+
} else {
1073+
depModules.add(await moduleGraph.ensureEntryFromUrl(url))
1074+
}
1075+
}
10731076
}
10741077
moduleGraph.updateModuleInfo(
10751078
thisModule,
@@ -1268,7 +1271,7 @@ async function compileCSS(
12681271
id: string,
12691272
code: string,
12701273
workerController: PreprocessorWorkerController,
1271-
urlReplacer?: CssUrlReplacer,
1274+
urlResolver?: CssUrlResolver,
12721275
): Promise<{
12731276
code: string
12741277
map?: SourceMapInput
@@ -1278,7 +1281,7 @@ async function compileCSS(
12781281
}> {
12791282
const { config } = environment
12801283
if (config.css.transformer === 'lightningcss') {
1281-
return compileLightningCSS(id, code, environment, urlReplacer)
1284+
return compileLightningCSS(id, code, environment, urlResolver)
12821285
}
12831286

12841287
const { modules: modulesOptions, devSourcemap } = config.css
@@ -1387,10 +1390,11 @@ async function compileCSS(
13871390
)
13881391
}
13891392

1390-
if (urlReplacer) {
1393+
if (urlResolver) {
13911394
postcssPlugins.push(
13921395
UrlRewritePostcssPlugin({
1393-
replacer: urlReplacer,
1396+
resolver: urlResolver,
1397+
deps,
13941398
logger: environment.logger,
13951399
}),
13961400
)
@@ -1724,6 +1728,12 @@ async function resolvePostcssConfig(
17241728
return result
17251729
}
17261730

1731+
type CssUrlResolver = (
1732+
url: string,
1733+
importer?: string,
1734+
) =>
1735+
| [url: string, id: string | undefined]
1736+
| Promise<[url: string, id: string | undefined]>
17271737
type CssUrlReplacer = (
17281738
url: string,
17291739
importer?: string,
@@ -1740,7 +1750,8 @@ export const importCssRE =
17401750
const cssImageSetRE = /(?<=image-set\()((?:[\w-]{1,256}\([^)]*\)|[^)])*)(?=\))/
17411751

17421752
const UrlRewritePostcssPlugin: PostCSS.PluginCreator<{
1743-
replacer: CssUrlReplacer
1753+
resolver: CssUrlResolver
1754+
deps: Set<string>
17441755
logger: Logger
17451756
}> = (opts) => {
17461757
if (!opts) {
@@ -1764,8 +1775,13 @@ const UrlRewritePostcssPlugin: PostCSS.PluginCreator<{
17641775
const isCssUrl = cssUrlRE.test(declaration.value)
17651776
const isCssImageSet = cssImageSetRE.test(declaration.value)
17661777
if (isCssUrl || isCssImageSet) {
1767-
const replacerForDeclaration = (rawUrl: string) => {
1768-
return opts.replacer(rawUrl, importer)
1778+
const replacerForDeclaration = async (rawUrl: string) => {
1779+
const [newUrl, resolvedId] = await opts.resolver(rawUrl, importer)
1780+
// only register inlined assets to avoid frequent full refresh (#18979)
1781+
if (newUrl.startsWith('data:') && resolvedId) {
1782+
opts.deps.add(resolvedId)
1783+
}
1784+
return newUrl
17691785
}
17701786
if (isCssUrl && isCssImageSet) {
17711787
promises.push(
@@ -3173,7 +3189,7 @@ async function compileLightningCSS(
31733189
id: string,
31743190
src: string,
31753191
environment: PartialEnvironment,
3176-
urlReplacer?: CssUrlReplacer,
3192+
urlResolver?: CssUrlResolver,
31773193
): ReturnType<typeof compileCSS> {
31783194
const { config } = environment
31793195
const deps = new Set<string>()
@@ -3285,11 +3301,16 @@ async function compileLightningCSS(
32853301
let replaceUrl: string
32863302
if (skipUrlReplacer(dep.url)) {
32873303
replaceUrl = dep.url
3288-
} else if (urlReplacer) {
3289-
replaceUrl = await urlReplacer(
3304+
} else if (urlResolver) {
3305+
const [newUrl, resolvedId] = await urlResolver(
32903306
dep.url,
32913307
dep.loc.filePath.replace(NULL_BYTE_PLACEHOLDER, '\0'),
32923308
)
3309+
// only register inlined assets to avoid frequent full refresh (#18979)
3310+
if (newUrl.startsWith('data:') && resolvedId) {
3311+
deps.add(resolvedId)
3312+
}
3313+
replaceUrl = newUrl
32933314
} else {
32943315
replaceUrl = dep.url
32953316
}

packages/vite/src/node/server/hmr.ts

+3
Original file line numberDiff line numberDiff line change
@@ -789,6 +789,9 @@ function propagateUpdate(
789789
// PostCSS plugins) it should be considered a dead end and force full reload.
790790
if (
791791
!isCSSRequest(node.url) &&
792+
// we assume .svg is never an entrypoint and does not need a full reload
793+
// to avoid frequent full reloads when an SVG file is referenced in CSS files (#18979)
794+
!node.file?.endsWith('.svg') &&
792795
[...node.importers].every((i) => isCSSRequest(i.url))
793796
) {
794797
return true

playground/assets/__tests__/assets.spec.ts

+27-3
Original file line numberDiff line numberDiff line change
@@ -293,12 +293,36 @@ describe('css url() references', () => {
293293
})
294294

295295
test('url() with svg', async () => {
296-
expect(await getBg('.css-url-svg')).toMatch(/data:image\/svg\+xml,.+/)
296+
const bg = await getBg('.css-url-svg')
297+
expect(bg).toMatch(/data:image\/svg\+xml,.+/)
298+
expect(bg).toContain('blue')
299+
expect(bg).not.toContain('red')
300+
301+
if (isServe) {
302+
editFile('nested/fragment-bg-hmr.svg', (code) =>
303+
code.replace('fill="blue"', 'fill="red"'),
304+
)
305+
await untilUpdated(() => getBg('.css-url-svg'), 'red')
306+
}
297307
})
298308

299309
test('image-set() with svg', async () => {
300310
expect(await getBg('.css-image-set-svg')).toMatch(/data:image\/svg\+xml,.+/)
301311
})
312+
313+
test('url() with svg in .css?url', async () => {
314+
const bg = await getBg('.css-url-svg-in-url')
315+
expect(bg).toMatch(/data:image\/svg\+xml,.+/)
316+
expect(bg).toContain('blue')
317+
expect(bg).not.toContain('red')
318+
319+
if (isServe) {
320+
editFile('nested/fragment-bg-hmr2.svg', (code) =>
321+
code.replace('fill="blue"', 'fill="red"'),
322+
)
323+
await untilUpdated(() => getBg('.css-url-svg'), 'red')
324+
}
325+
})
302326
})
303327

304328
describe('image', () => {
@@ -552,8 +576,8 @@ test.runIf(isBuild)('manifest', async () => {
552576

553577
for (const file of listAssets('foo')) {
554578
if (file.endsWith('.css')) {
555-
// ignore icons-*.css as it's imported with ?url
556-
if (file.includes('icons-')) continue
579+
// ignore icons-*.css and css-url-url-*.css as it's imported with ?url
580+
if (file.includes('icons-') || file.includes('css-url-url-')) continue
557581
expect(entry.css).toContain(`assets/${file}`)
558582
} else if (!file.endsWith('.js')) {
559583
expect(entry.assets).toContain(`assets/${file}`)

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

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
.css-url-svg-in-url {
2+
background: url(../nested/fragment-bg-hmr2.svg);
3+
background-size: 10px;
4+
}

playground/assets/css/css-url.css

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

playground/assets/index.html

+9
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,9 @@ <h2>CSS url references</h2>
151151
<div class="css-url-svg">
152152
<span style="background: #fff">CSS SVG background</span>
153153
</div>
154+
<div class="css-url-svg-in-url">
155+
<span style="background: #fff">CSS (?url) SVG background</span>
156+
</div>
154157

155158
<div class="css-image-set-svg">
156159
<span style="background: #fff">CSS SVG background with image-set</span>
@@ -531,6 +534,12 @@ <h3>assets in template</h3>
531534
import cssUrl from './css/icons.css?url'
532535
text('.url-css', cssUrl)
533536

537+
import cssUrlUrl from './css/css-url-url.css?url'
538+
const linkTag = document.createElement('link')
539+
linkTag.href = cssUrlUrl
540+
linkTag.rel = 'stylesheet'
541+
document.body.appendChild(linkTag)
542+
534543
// const url = new URL('non_existent_file.png', import.meta.url)
535544
const metaUrl = new URL('./nested/asset.png', import.meta.url)
536545
text('.import-meta-url', metaUrl)
Loading
Loading

0 commit comments

Comments
 (0)