Skip to content

Commit 65f97bd

Browse files
authoredSep 23, 2022
fix: don't duplicate styles with dynamic import (fix #9967) (#9970)
1 parent 533d13c commit 65f97bd

File tree

10 files changed

+179
-3
lines changed

10 files changed

+179
-3
lines changed
 

‎packages/vite/src/node/plugins/importAnalysisBuild.ts

+18-2
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,8 @@ function preload(
6565
return baseModule()
6666
}
6767

68+
const links = document.getElementsByTagName('link')
69+
6870
return Promise.all(
6971
deps.map((dep) => {
7072
// @ts-ignore
@@ -75,10 +77,24 @@ function preload(
7577
seen[dep] = true
7678
const isCss = dep.endsWith('.css')
7779
const cssSelector = isCss ? '[rel="stylesheet"]' : ''
78-
// @ts-ignore check if the file is already preloaded by SSR markup
79-
if (document.querySelector(`link[href="${dep}"]${cssSelector}`)) {
80+
const isBaseRelative = !!importerUrl
81+
82+
// check if the file is already preloaded by SSR markup
83+
if (isBaseRelative) {
84+
// When isBaseRelative is true then we have `importerUrl` and `dep` is
85+
// already converted to an absolute URL by the `assetsURL` function
86+
for (let i = links.length - 1; i >= 0; i--) {
87+
const link = links[i]
88+
// The `links[i].href` is an absolute URL thanks to browser doing the work
89+
// for us. See https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#reflecting-content-attributes-in-idl-attributes:idl-domstring-5
90+
if (link.href === dep && (!isCss || link.rel === 'stylesheet')) {
91+
return
92+
}
93+
}
94+
} else if (document.querySelector(`link[href="${dep}"]${cssSelector}`)) {
8095
return
8196
}
97+
8298
// @ts-ignore
8399
const link = document.createElement('link')
84100
// @ts-ignore
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,121 @@
1+
import type { InlineConfig } from 'vite'
2+
import { build, createServer, preview } from 'vite'
3+
import { expect, test } from 'vitest'
4+
import { getColor, isBuild, isServe, page, ports, rootDir } from '~utils'
5+
6+
const baseOptions = [
7+
{ base: '', label: 'relative' },
8+
{ base: '/', label: 'absolute' }
9+
]
10+
11+
const getConfig = (base: string): InlineConfig => ({
12+
base,
13+
root: rootDir,
14+
logLevel: 'silent',
15+
preview: { port: ports['css/dynamic-import'] },
16+
build: { assetsInlineLimit: 0 }
17+
})
18+
19+
async function withBuild(base: string, fn: () => Promise<void>) {
20+
const config = getConfig(base)
21+
await build(config)
22+
const server = await preview(config)
23+
24+
try {
25+
await page.goto(server.resolvedUrls.local[0])
26+
await fn()
27+
} finally {
28+
server.httpServer.close()
29+
}
30+
}
31+
32+
async function withServe(base: string, fn: () => Promise<void>) {
33+
const config = getConfig(base)
34+
const server = await createServer(config)
35+
await server.listen()
36+
await new Promise((r) => setTimeout(r, 500))
37+
38+
try {
39+
await page.goto(server.resolvedUrls.local[0])
40+
await fn()
41+
} finally {
42+
await server.close()
43+
}
44+
}
45+
46+
async function getLinks() {
47+
const links = await page.$$('link')
48+
return await Promise.all(
49+
links.map((handle) => {
50+
return handle.evaluate((link) => ({
51+
pathname: new URL(link.href).pathname,
52+
rel: link.rel,
53+
as: link.as
54+
}))
55+
})
56+
)
57+
}
58+
59+
baseOptions.forEach(({ base, label }) => {
60+
test.runIf(isBuild)(
61+
`doesn't duplicate dynamically imported css files when built with ${label} base`,
62+
async () => {
63+
await withBuild(base, async () => {
64+
await page.waitForSelector('.loaded', { state: 'attached' })
65+
66+
expect(await getColor('.css-dynamic-import')).toBe('green')
67+
expect(await getLinks()).toEqual([
68+
{
69+
pathname: expect.stringMatching(/^\/assets\/index\..+\.css$/),
70+
rel: 'stylesheet',
71+
as: ''
72+
},
73+
{
74+
pathname: expect.stringMatching(/^\/assets\/dynamic\..+\.css$/),
75+
rel: 'preload',
76+
as: 'style'
77+
},
78+
{
79+
pathname: expect.stringMatching(/^\/assets\/dynamic\..+\.js$/),
80+
rel: 'modulepreload',
81+
as: 'script'
82+
},
83+
{
84+
pathname: expect.stringMatching(/^\/assets\/dynamic\..+\.css$/),
85+
rel: 'stylesheet',
86+
as: ''
87+
},
88+
{
89+
pathname: expect.stringMatching(/^\/assets\/static\..+\.js$/),
90+
rel: 'modulepreload',
91+
as: 'script'
92+
},
93+
{
94+
pathname: expect.stringMatching(/^\/assets\/index\..+\.js$/),
95+
rel: 'modulepreload',
96+
as: 'script'
97+
}
98+
])
99+
})
100+
}
101+
)
102+
103+
test.runIf(isServe)(
104+
`doesn't duplicate dynamically imported css files when served with ${label} base`,
105+
async () => {
106+
await withServe(base, async () => {
107+
await page.waitForSelector('.loaded', { state: 'attached' })
108+
109+
expect(await getColor('.css-dynamic-import')).toBe('green')
110+
// in serve there is no preloading
111+
expect(await getLinks()).toEqual([
112+
{
113+
pathname: '/dynamic.css',
114+
rel: 'preload',
115+
as: 'style'
116+
}
117+
])
118+
})
119+
}
120+
)
121+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// this is automatically detected by playground/vitestSetup.ts and will replace
2+
// the default e2e test serve behavior
3+
4+
// The server is started in the test, so we need to have a custom serve
5+
// function or a default server will be created
6+
export async function serve() {
7+
return {
8+
close: () => Promise.resolve()
9+
}
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
.css-dynamic-import {
2+
color: green;
3+
}
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
import './dynamic.css'
2+
3+
export const lazyLoad = async () => {
4+
await import('./static.js')
5+
document.body.classList.add('loaded')
6+
}
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
<p class="css-dynamic-import">This should be green</p>
2+
3+
<script type="module" src="./index.js"></script>
+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import './static.js'
2+
3+
const link = document.head.appendChild(document.createElement('link'))
4+
link.rel = 'preload'
5+
link.as = 'style'
6+
link.href = new URL('./dynamic.css', import.meta.url).href
7+
8+
import('./dynamic.js').then(async ({ lazyLoad }) => {
9+
await lazyLoad()
10+
})
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
.css-dynamic-import {
2+
color: red;
3+
}
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import './static.css'
2+
3+
export const foo = 'foo'

‎playground/test-utils.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,8 @@ export const ports = {
2929
'ssr-vue': 9604,
3030
'ssr-webworker': 9605,
3131
'css/postcss-caching': 5005,
32-
'css/postcss-plugins-different-dir': 5006
32+
'css/postcss-plugins-different-dir': 5006,
33+
'css/dynamic-import': 5007
3334
}
3435
export const hmrPorts = {
3536
'optimize-missing-deps': 24680,

0 commit comments

Comments
 (0)
Please sign in to comment.