-
-
Notifications
You must be signed in to change notification settings - Fork 6.4k
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
Vite injects css assets in wrong order with dynamic import and css modules. #3924
Comments
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
How did you solve it |
This comment was marked as spam.
This comment was marked as spam.
4 similar comments
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
This comment was marked as spam.
Any updates?( Maybe @sodatea will provide some thoughts where to find problem in project and i will try to make pull request to vite? |
I'm seeing this issue as well, now that I've started to use storybook's new dynamic loading feature, my styles are broken in production stories. I've found that the css files are directly after the component in the network tab. So, in one example, I have a RadioButton component that imports Label (which has its own styles that I want to override), and then it imports some css module with the overrides. However, when building now, I see this: Which puts the RadioButton styles higher in the cascade, and they are overwritten by the Label styles, which is backwards from what I want. |
Here is a minimal stackblitz reproduction: https://stackblitz.com/edit/vite-gu874q?file=main.jsx vite: 2.7.1 |
Great work @IanVS! Really hope this helps resolving the issue. I would help, but have no idea on where to start unfortunately. |
I finally found the culprit, and FOR a while I thought My temporary solution is to load influenced page synchronously in router configuration. |
any idea when will #6301 be merged? |
As far as I know it's not ready, @mgiraldo. This is a sticky problem to solve. If you have any ideas or can help out, please do! |
i understand. thanks for all this work! |
I wonder whether Vite could maybe look for a With Webpack's Webpack Configuration
|
For those who (like me) had to ship and need a temporary workaround: Depending on your project, you may be able to use |
sadly this didnt work out for me @laurentvd 😔 i might need to refactor my components 🤔 |
This seems to still be an issue. Really killing me. Any updates? |
is there any update on this? |
Applying styles in the wrong order is a deal breaker for using vite. |
Because it isn't fixed yet. Have you considered that this may be a difficult problem?
There are solutions to manage cascade order that do not depend on CSS order. Have you considered reading the thread that you're commenting on? |
Speaking of cascade layers, if you're using css modules (which is likely if you're hitting this issue), you could try out https://github.com/DefinedNet/postcss-assign-layer to assign all of your components to a separate layer from your global / utility layers. This doesn't work with the cascade layers polyfill, but I think browser support is good enough you shouldn't need the polyfill anymore. Another option for avoiding this issue is to not code-split your app. There are tradeoffs there, but it's something to consider. |
|
UPDATE: I just want to retract my addition to this issue. Mine was an embarrassingly newbie mistake of leaving a comment open >.< |
I've also encountered this issue as we upgrade a legacy project. The load order of CSS is important. Why is there no option to specify the order within the manifest.json? CSS order is, unfortunately, a real concern. Seeing how many people are interacting with this issue, I wonder why there hasn't been a solution proposed. |
I've got a really hacky solution in case it helps anyone, but it is very possible that it only works for my particular project: it consists of moving the styles from the <script>
document.addEventListener('DOMContentLoaded', function () {
const head = document.head;
const body = document.body;
const styleTags = head.querySelectorAll('style[type="text/css"]');
styleTags.forEach(function (styleTag) {
head.removeChild(styleTag);
body.appendChild(styleTag);
});
const linkTags = head.querySelectorAll('link[rel="stylesheet"]');
linkTags.forEach(function (linkTag) {
head.removeChild(linkTag);
body.appendChild(linkTag);
});
});
</script> However, keep in mind that in my project:
Also let me know if this is a terrible idea and why, please 🙏🏼 |
Still not a solution, but https://admin.defined.net uses https://developer.mozilla.org/en-US/docs/Web/CSS/@layer (finally part of the "Baseline") + https://github.com/DefinedNet/postcss-assign-layer to ensure our cascade is enforced programmatically. /* establish a layer order up-front, from lowest to highest priority */
@layer defaults, theme, libraries, patterns, components, utilities, overrides;
/** Defaults */
@import url('defaults.css');
/** Theme */
@import url('theme/colors.css');
@import url('theme/typography.css');
/** Patterns */
@import url('../../components/atoms/forms/border-focusable.css');
@import url('../../components/patterns/menus/menu.css');
/** Components are assigned by a PostCSS plugin for all `*.module.css` files */
/** Utilities */
@import url('utilities/layout.css');
@import url('utilities/color.css');
@import url('utilities/borders.css');
@import url('utilities/typography.css'); The PostCSS plugin rewrites all of our This still requires caution when it comes to specificity within each cascade layer, but it's much easier to avoid that. This enabled us to setup fully async react-router-dom routes, code-splitting our app finally! |
Hey want to get people's thoughts on the new React 19 docs, specifically this section, as it relates to this issue: https://react.dev/blog/2024/04/25/react-19?ck_subscriber_id=2099611990#support-for-stylesheets It looks like you can inline style sheets directly in the components, themselves, which unless I am mistaken, should solve pretty much everyone's issues here? Please correct me if I am wrong. Thanks. EDIT: Should have asked: "Does this solve all of the React users' issues?" |
I use svelte, so that doesn't help me. |
That's ok. Nobody's perfect |
Hey, is there any update on same? This issue is happening in latest vite version also with css module . ( when css code split is set to true) |
@jasikpark's workaround works in my build.
vite.config.ts import postcssAssignLayer from 'postcss-assign-layer';
//...
postcss: {
plugins: [
postcssAssignLayer([
{
include: '**/*.module.scss',
layerName: 'components'
}
])
]
}
//... Index.scss @layer base, components;
@layer base {
@import './scss/normalize';
@import './scss/variables';
@import './scss/fonts';
@import './scss/mixins';
@import './scss/globals';
@import './scss/typography';
} Router.tsx import { Suspense, lazy } from 'react';
import { createBrowserRouter, RouterProvider, LoaderFunctionArgs, Outlet } from 'react-router-dom';
const Home = lazy(() => import('../components/Home'));
const About = lazy(() => import('../components/About'));
export const Router = () => {
const routeConfig = createBrowserRouter([
{
path: '/',
element: (
<Suspense fallback={<div>Loading...</div>}>
<Home />
</Suspense>
)
},
{
path: '/about',
element: (
<Suspense fallback={<div>Loading...</div>}>
<About />
</Suspense>
)
},
])
return (
<RouterProvider
fallbackElement={<div>Loading...</div>}
router={routeConfig}
/>
);
} |
Hey @novym , will the above solution solve the order issue in same layer like component ? In my case I have 2 components module css which is not being imported in correct order. |
This was impacting our code as well, a react based app which uses Seems like a bug to me, CSS stylesheets should be appended to in the order that you reference/import them, right? If you import 'main_styles.css' right back at the start in the entry point, or reference it in your tag of a page, that should stay placed before anything else imported later on, in my opinion, and that seems to be the behaviour of HMR preview. At first my work around for this was basically to import the CSS for the global styles as 'inline', and add the styles to the page with a function that ensures the styles get placed before any other styles: import styles from "./../assets/css/styles.css?inline";
// Inject CSS styles into head before any other style elements.
function injectCSS(css) {
const styleElement = document.createElement('style');
styleElement.textContent = css;
const head = document.head;
const firstStyleOrLinkElement = head.querySelector('style, link');
if (firstStyleOrLinkElement) {
head.insertBefore(styleElement, firstStyleOrLinkElement);
} else {
head.appendChild(styleElement);
}
}
injectCSS(styles); Fixed the order of the only style sheet that was in the wrong placement. Very hacky solution though so I wasn't a fan of it. My new solution is to import our global style sheet before the per-component style sheet in every React component. import './../assets/css/styles.css';
import './../assets/css/components/ThisComponent.css'; And that seems to do the trick to ensure that the order of the styles is as each component expects, but has the downside of needing to import our CSS styles on every single component. Bit of a pain, but it seems to work at least. |
Looking into the issue... Can anyone provide a minimal reproduction of the bug on StackBlitz using the latest Vite? If anyone is experiencing this issue, please try checking if using https://github.com/privatenumber/vite-css-modules fixes the problem. |
I can't unfortunately reproduce with minimum code, the codebase is very large and private and the issue just manifested itself yesterday. Not using CSS modules here and not using any code splitting. All CSS imports are simply
The app components always import the component library code before the local styles.
When running
I am not using CSS modules at all, and this seems to be aimed at solvin other problems (dependency duplication and Hot Module Replacement (HMR) issues, according to the GH page). With 3 months of hard work invested in this project I am willing to try it, though. On the other hand, reading this ... makes me wonder if I would be running into even more problems. After all, my idea was to use the simpleset C(ascade)SS based product, but that seems to be broken in Vite as well 😢 |
The correct loading order is probably already implemented for JavaScript modules, and it should ideally work the same way for CSS modules. Here's an example: // file: Card.tsx
import styles from './card.css';
export function Card() { ... }
// file: CoolCard.tsx:
import { Card } from 'card';
import coolStyles from './coolCard.css';
export function CoolCard() { ... }
// file: consumer.tsx:
import { CoolCard } from 'CoolCard'; // <-- *
import { Card } from 'Card';
import localStyles from './consumer.css';
export function Consumer() { ... } Assuming that:
I expect
The expected load sequence would be:
Does this make sense, or am I missing anything? |
One caveat: styles and scripts must be inserted in sorted order when lazy-loading multiple modules to avoid redundant loading. Here's an example scenario:
In the wrong implementation: <head>
<!-- Lazy-loaded Page 01 -->
<link rel="stylesheet" href="card.css">
<link rel="stylesheet" href="coolCard.css">
<link rel="stylesheet" href="page01.css">
<script src="card.js"></script>
<script src="coolCard.js"></script>
<script src="page01.js"></script>
<!-- Lazy-loaded Page 02 -->
<link rel="stylesheet" href="card.css"> <!-- Duplicate -->
<link rel="stylesheet" href="page02.css">
<script src="card.js"></script> <!-- Duplicate -->
<script src="page02.js"></script>
</head> In the correct implementation: <head>
<!-- Base module -->
<link rel="stylesheet" href="card.css"> <!-- Loaded once -->
<script src="card.js"></script>
<!-- Lazy-loaded Page 01 -->
<link rel="stylesheet" href="coolCard.css">
<script src="coolCard.js"></script>
<link rel="stylesheet" href="page01.css">
<script src="page01.js"></script>
<!-- Lazy-loaded Page 02 -->
<link rel="stylesheet" href="page02.css">
<script src="page02.js"></script>
</head> |
Describe the bug
Vite injects css assets in wrong order with dynamic import and css modules.
Reproduction
Repo to reproduce
For example:
P.S. You can reproduce this but with cssCodeSplit: true or false.
System Info
Output of
npx envinfo --system --npmPackages vite,@vitejs/plugin-vue --binaries --browsers
:Used package manager:
Logs
Before submitting the issue, please make sure you do the following
The text was updated successfully, but these errors were encountered: