Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(css): allow scoping css to importers exports #19418

Merged
merged 13 commits into from
Feb 21, 2025

Conversation

sapphi-red
Copy link
Member

@sapphi-red sapphi-red commented Feb 13, 2025

Description

closes #4389
closes #15497 (doesn't exactly fix it, but unblocks downstream to fix this)

This PR is based on #16058 but solves #16058 (comment)

Quoting the description from #16058 with some edits:

Frameworks like Vue, Svelte, and Astro has the concept of scoped CSS. Internally, they are transformed as imports to a virtual CSS module. However, the general CSS bundling flow will assume the CSS to be global, so even if the framework component is used but treeshaken, the CSS will still be kept because Vite assumes its global.

To resolve this, Vite plugins can mark certain CSS to be scoped to certain modules' exports. For example, /src/Foo.vue?vue&type=style&lang.css is scoped to /src/App.vue's default export.

The API this PR exposes is via meta.vite.cssScopeTo['/src/Foo.vue'] = ['default'] meta.vite.cssScopeTo: ['/src/Foo.vue', 'default'] set on the /src/Foo.vue?vue&type=style&lang.css modules in Vite plugins. Now, if Foo.vue is not used, the styles are also treeshaken.

@sapphi-red sapphi-red added feat: css p2-to-be-discussed Enhancement under consideration (priority) labels Feb 13, 2025
Comment on lines 327 to 337
/**
* If this is a CSS Rollup module, you can scope to its importer's exports
* so that if those exports are treeshaken away, the CSS module will also
* be treeshaken.
*
* Example config if the CSS id is `/src/App.vue?vue&type=style&lang.css`:
* ```js
* cssScopeTo: ['/src/App.vue', 'default']
* ```
*/
cssScopeTo?: [string, string | undefined]
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I changed the interface to the one I felt easier to implement by proposal for now.
@bluwy What was the reason of your interface? I'm wondering the usage of allowing multiple files here.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't have a concrete usecase. Was thinking that maybe there's some sort of unocss-like tool that generates scoped css derived from multiple modules somehow that could benefit from this. So I futureproofed it as a Record.

@@ -323,6 +323,20 @@ export interface Plugin<A = any> extends RollupPlugin<A> {
>
}

export interface CustomPluginOptionsVite {
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I moved it here because it seemed CustomPluginOptions was changed to a type from an interface.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this might probably be unintentional from Rollup side due to rollup/rollup#5591 🤔

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Made a PR here 👍
rollup/rollup#5850

@@ -0,0 +1,7 @@
<h1>treeshake-scoped (another)</h1>
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@bluwy This file and the files in the barrel directory covers this case (#16058 (comment)).

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reading the implementation, I believe you have the scoped CSS side effects set to false to fix this, but keep track of the scoped css styles map separately to bring it back? That's a neat trick. I guess a limitation of this is that it's harder to scope to multiple modules like you mentioned, but I think it's fine for now.

@zeakd
Copy link

zeakd commented Feb 17, 2025

Maybe this PR could help to close #11855

@sapphi-red
Copy link
Member Author

@zeakd No, this PR is not related to that issue. This PR improves the behavior of "already" scoped styles. That issue asks for "a way to" scope the styles.

@sapphi-red sapphi-red marked this pull request as ready for review February 18, 2025 06:31
* cssScopeTo: ['/src/App.vue', 'default']
* ```
*
* @experimental
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added @experimental for now. The plan is to release this in 6.2 and stabilize it in 6.3 or 6.4.

@sapphi-red sapphi-red requested a review from bluwy February 20, 2025 05:13
// a css module contains JS, so it makes this not a pure css chunk
if (cssModuleRE.test(id)) {
isPureCssChunk = false
}
}
} else if (styles.hasContentsScopedTo(id)) {
const renderedExports = chunk.modules[id]!.renderedExports
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think RenderedModule.renderedExports isn't implemented yet on rolldown. Should we check with them if this is a simple addition on rolldown side?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, good catch. Let me ask them.

Copy link
Member

@bluwy bluwy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks great!

HPCesia added a commit to HPCesia/astral-halo that referenced this pull request Apr 5, 2025
- Remove Twikoo's local style.
- Remove local package of Artalk and Waline to minify styles

This commit is because of unused comment styles in build packaged css.

vitejs/vite#19418 might have helped, but Astro didn't use this experimental feature (withastro/astro#13420).
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
feat: css p2-to-be-discussed Enhancement under consideration (priority)
Projects
Archived in project
Development

Successfully merging this pull request may close these issues.

Svelte CSS treeshaking not working CSS files cannot be treeshaken with side effects
5 participants