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 modules (early experimental support) #2489

Closed
wants to merge 87 commits into from
Closed
Show file tree
Hide file tree
Changes from 85 commits
Commits
Show all changes
87 commits
Select commit Hold shift + click to select a range
04c0a2f
Initial implementation of CSS modules plugins for both server and bro…
mjackson Feb 23, 2022
73df19a
Merge branch 'dev' into css-modules
chaance Feb 23, 2022
8b76ccc
chore: use correct promise resolve type for AssetsManifestRef
chaance Feb 24, 2022
75c61ab
Add virtual modules plugins for CSS Modules
chaance Feb 24, 2022
42eb07f
add cssmodules file path to assets object
chaance Feb 24, 2022
ea79fd5
modify scope + comments
chaance Feb 24, 2022
0531d10
Merge branch 'dev' of github.com:remix-run/remix into css-modules
chaance Feb 24, 2022
3af5759
refactor CSS modules plugins
chaance Feb 25, 2022
5726ca3
fix css hashing
chaance Feb 25, 2022
fa30edc
chore: update assets manifest types
chaance Mar 1, 2022
aec0023
chore: compiler fixes
chaance Mar 1, 2022
3f38369
chore: add CSS modules imports to gists app
chaance Mar 1, 2022
958ef0d
chore: Merge branch 'dev' into css-modules
chaance Mar 1, 2022
c2e238b
remove virtual module stuff, add modules to browser build
chaance Mar 1, 2022
44fee47
cleanup
chaance Mar 1, 2022
7c0596c
Merge branch 'dev' into css-modules
chaance Mar 4, 2022
df50e1b
create @remix-run/css-modules package
chaance Mar 4, 2022
158ac85
add css-modules package to build
chaance Mar 4, 2022
2016b6d
add css-modules package to scripts
chaance Mar 4, 2022
fa86d32
update manifest types + resolve module imports
chaance Mar 4, 2022
6653f17
css-modules package entrypoints
chaance Mar 4, 2022
01ee119
add css-modules to workspaces
chaance Mar 4, 2022
a66e67c
rm index and fix package.json
chaance Mar 4, 2022
bb050fc
add css-modules to tsconfig
chaance Mar 4, 2022
a607949
add processor for Parcel
chaance Mar 4, 2022
91c0c8e
bundle internal css-modules package
chaance Mar 4, 2022
4952de5
add css modules to gists app
chaance Mar 4, 2022
9054616
check composes feature
chaance Mar 5, 2022
b200036
deps
chaance Mar 5, 2022
e0085c5
fix browser import
chaance Mar 5, 2022
770d62d
types and such
chaance Mar 5, 2022
5ecd50a
types
chaance Mar 5, 2022
0d48c47
clean up gists app
chaance Mar 5, 2022
7449ce3
types
chaance Mar 5, 2022
0915e6c
swap postcss for parcel
chaance Mar 5, 2022
7a7bfd5
cleanup
chaance Mar 5, 2022
9c96f81
Merge branch 'dev' into css-modules
chaance Mar 15, 2022
c12a05f
revert deno in tsconfig
chaance Mar 15, 2022
c399fe2
add placeholder for tests
chaance Mar 15, 2022
fa72bd1
Merge branch 'dev' into css-modules
chaance Mar 15, 2022
9b834b8
Merge branch 'dev' into css-modules
chaance Mar 25, 2022
f41293e
revert gists app change
chaance Mar 25, 2022
b822a30
reduce diffs
chaance Mar 25, 2022
641afa9
typo
chaance Mar 25, 2022
82073ce
tests
chaance Mar 25, 2022
0537a08
Merge branch 'dev' into feature/css-modules
chaance Mar 30, 2022
418a389
deps
chaance Mar 30, 2022
f485536
test stuff
chaance Mar 30, 2022
9c8465c
fix links test
chaance Mar 30, 2022
75dab98
test updates
chaance Mar 30, 2022
31be49a
use composition in test cases
chaance Mar 31, 2022
392e5ca
fix sorting and caching issues
chaance Mar 31, 2022
147f196
Merge branch 'dev' into feature/css-modules
chaance Apr 26, 2022
1e64d70
Merge branch 'dev' into feature/css-modules
chaance Apr 27, 2022
2678ee1
chore: remove obsolete dev scripts
pcattori Apr 28, 2022
43f35c6
test(css-modules): rewrite test for playwright
pcattori Apr 28, 2022
4cd6a76
chore(rollup): use `REMIX_LOCAL_DEV_OUTPUT_DIRECTORY` for `@remix-run…
pcattori Apr 28, 2022
d9d444d
feat(compiler): order css modules output according to dependency order
pcattori Apr 28, 2022
4081507
refactor(css-modules): remove unused handler param from css modules p…
pcattori Apr 28, 2022
0e8def8
update module tests
chaance Apr 29, 2022
6a44223
Add todo tests for ordering
chaance Apr 29, 2022
9dd389b
rename cssModules.ts -> cssModulesPlugi.ts
chaance Apr 29, 2022
1c40806
Merge branch 'dev' into feature/css-modules
chaance May 12, 2022
c626296
fix tests
chaance May 12, 2022
0747d0e
run all tests
chaance May 16, 2022
ec43d23
Merge branch 'dev' into feature/css-modules
chaance May 16, 2022
063e57c
fix test scoping issues
chaance May 16, 2022
952232f
rm placeholder file
chaance May 16, 2022
7f674ac
Merge branch 'dev' into feature/css-modules
chaance May 17, 2022
8e0e071
only run topological sort on files w/ CSS modules
chaance May 17, 2022
c95fb09
reduce array iterations
chaance May 17, 2022
7632c49
Merge branch 'dev' into feature/css-modules
chaance May 17, 2022
f5a1432
use named imports for modules stylesheet
chaance May 17, 2022
859eadf
Merge branch 'dev' into feature/css-modules
chaance May 18, 2022
8026a1f
check for processed css before reading json
chaance May 18, 2022
21e6f0d
null check
chaance May 18, 2022
c01f9ff
rebuild fixes
chaance May 18, 2022
0e9ee7d
run build after CSS modules updates
chaance May 18, 2022
ba0b019
rename module references
chaance May 18, 2022
481fe22
rename compiler references
chaance May 18, 2022
47eb2f5
renaming
chaance May 18, 2022
67eac49
resolve paths based on tsconfig paths
chaance May 18, 2022
ff2f5bb
pin @parcel/css version
chaance May 18, 2022
5ea1116
rm log
chaance May 18, 2022
97823e3
update comments
chaance May 18, 2022
ae984c8
Merge branch 'dev' into feature/css-modules
chaance May 19, 2022
4f0d8e3
Version 0.0.0-experimental-ae984c8e
chaance May 19, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
381 changes: 381 additions & 0 deletions integration/css-modules-test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,381 @@
import { test, expect } from "@playwright/test";
import type * as Playwright from "@playwright/test";

import { PlaywrightFixture } from "./helpers/playwright-fixture";
import type { Fixture, AppFixture } from "./helpers/create-fixture";
import {
createAppFixture,
createFixture,
css,
js,
} from "./helpers/create-fixture";

test.describe("CSS Modules", () => {
// TODO: Finish all todo tests
let testTodo = (
title: string,
testFunction?: (
args: { page: Playwright.Page },
testInfo: Playwright.TestInfo
) => any
) => {};

let fixture: Fixture;
let appFixture: AppFixture;

test.beforeAll(async () => {
fixture = await createFixture({
files: {
"app/root.jsx": js`
import { Links, Outlet, Scripts } from "@remix-run/react";
import { cssModulesStylesheetUrl } from "@remix-run/css-modules";
import stylesHref from "~/styles.css";
export function links() {
return [
{ rel: "stylesheet", href: stylesHref },
{ rel: "stylesheet", href: cssModulesStylesheetUrl },
];
}
export default function Root() {
return (
<html>
<head>
<Links />
</head>
<body>
<div id="content">
<Outlet />
</div>
<Scripts />
</body>
</html>
)
}
`,
"app/styles.css": css`
.reset--button {
appearance: none;
display: inline-block;
border: none;
padding: 0;
text-decoration: none;
background: 0;
color: inherit;
font: inherit;
}
`,

"app/routes/index.jsx": js`
import { Badge } from "~/lib/badge";
export default function() {
return (
<div>
<h1>Index</h1>
<Badge>Hello</Badge>
</div>
);
}
`,

"app/routes/page-a.jsx": js`
import { Badge } from "~/lib/badge";
import { Button } from "~/lib/button";
import { Heading } from "~/lib/heading";
import { Text } from "~/lib/text";
export default function() {
return (
<div>
<Heading level={1}>Route A</Heading>
<Badge>Welcome</Badge>
<Text>This is really good information, eh?</Text>
<Button>Click Me</Button>
</div>
);
}
`,

"app/routes/page-b.jsx": js`
import { Button } from "~/lib/button";
import { Heading } from "~/lib/heading";
import { Text } from "~/lib/text";
export default function() {
return (
<div>
<Heading level={1}>Route B</Heading>
<Text>Here's a red button</Text>
<Button variant="red">Click Me</Button>
</div>
);
}
`,

"app/routes/layout.jsx": js`
import { Outlet } from "@remix-run/react";
import { Container } from "~/lib/container";
import { Heading } from "~/lib/heading";
import { Text } from "~/lib/text";
export default function() {
return (
<div>
<Container>
<Heading level={1}>Layout</Heading>
<Outlet />
</Container>
</div>
);
}
`,

"app/routes/layout/one.jsx": js`
import { Heading } from "~/lib/heading";
import { Input } from "~/lib/input";
export default function() {
return (
<div>
<Heading level={2}>Subpage 1</Heading>
<Input name="email" type="email" defaultValue="hi@remix.run" />
</div>
);
}
`,

"app/lib/button.jsx": js`
import styles from "./button.module.css";
export function Button({ variant, ...props }) {
return (
<button
{...props}
data-ui-button=""
className={variant === "red" ? styles.buttonRed : styles.button}
/>
);
}
`,
"app/lib/button.module.css": css`
.button {
/* TODO: composes: text from "./text.module.css"; */
composes: reset--button from global;
padding: 0.5rem 1rem;
box-shadow: none;
background-color: dodgerblue;
color: white;
font-weight: bold;
}
.buttonRed {
composes: button;
background-color: red;
}
`,

"app/lib/container.jsx": js`
import styles from "~/lib/container.module.css";
export function Container(props) {
return (
<div
{...props}
data-ui-container=""
className={(styles.container + " " + (props.className || "")).trim()}
/>
);
}
`,

"app/lib/container.module.css": css`
.container {
display: block;
max-width: 1000px;
margin: 0 auto;
}
`,

"app/lib/badge.jsx": js`
import styles from "./badge.module.css";
export function Badge(props) {
return (
<div
{...props}
data-ui-badge=""
className={styles.badge + " bold"}
/>
);
}
`,

"app/lib/badge.module.css": css`
.badge {
/* TODO: composes: text from "./text.module.css"; */
display: inline-block;
padding: 0.25rem 0.5rem;
background-color: lightgray;
color: black;
font-size: 0.8rem;
}
:global(.bold):local(.badge) {
text-transform: uppercase;
}
`,

"app/lib/input.jsx": js`
import styles from "./input.module.css";
export function Input(props) {
return (
<input
{...props}
data-ui-input=""
className={(styles.input + " " + (props.className || "")).trim()}
/>
);
}
`,

"app/lib/input.module.css": css`
.input {
width: 100%;
outline-offset: 2px;
appearance: none;
border-radius: 3px;
border: 1px solid #000;
}
`,

"app/lib/text.jsx": js`
import styles from "./text.module.css";
export function Text({ as: Comp = "span", ...props }) {
return (
<Comp
{...props}
data-ui-text=""
className={props.className ? props.className + " " + styles.text : styles.text}
/>
);
}
`,
"app/lib/text.module.css": css`
.text {
font-family: system-ui, sans-serif;
letter-spacing: -0.5px;
}
`,
"app/lib/heading.jsx": js`
import styles from "./heading.module.css";
import { Text } from "~/lib/text";
export function Heading({ level = 2, ...props }) {
return (
<Text
{...props}
as={"h" + level}
data-ui-heading=""
data-ui-heading-level={level}
className={props.className ? props.className + " " + styles.heading : styles.heading}
/>
);
}
`,
"app/lib/heading.module.css": css`
.heading {
font-weight: bold;
}
.heading[data-heading-level="1"] {
font-size: 3rem;
}
.heading[data-heading-level="2"] {
font-size: 2rem;
}
`,
},
});
appFixture = await createAppFixture(fixture);
});

test.afterAll(async () => {
await appFixture.close();
});

test("server renders with hashed classnames", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
await app.goto("/");
let badge = await app.getElement("[data-ui-badge]");
let badgeClasses = badge.attr("class").split(" ");
let found: string[] = [];
for (let className of badgeClasses) {
if (/^badge-module__[\w-]+_badge$/.test(className)) {
found.push(className);
}
}
expect(found.length).toBe(1);
});

test("composes from global classname", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
await app.goto("/page-a");
let button = await app.getElement("[data-ui-button]");
expect(
button
.attr("class")
// .button composes the global .reset--button class
.includes("reset--button")
).toBe(true);
});

test("composes from locally scoped classname", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
await app.goto("/page-b");
let button = await app.getElement("[data-ui-button]");
let buttonClasses = button.attr("class").split(" ");
let found: string[] = [];
for (let className of buttonClasses) {
if (
/^button-module__[\w-]+_button$/.test(className) ||
/^button-module__[\w-]+_buttonRed$/.test(className)
) {
found.push(className);
}
}
expect(found.length).toBe(2);
});

// TODO: Feature not implemented yet
testTodo("composes from imported module classname");

test("composes :global selector with :local selector", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
await app.goto("/");
let badge = await app.getElement("[data-ui-badge]");
let buttonClasses = badge.attr("class").split(" ");
let found: string[] = [];
for (let className of buttonClasses) {
if (
/^badge-module__[\w-]+_badge$/.test(className) ||
// .badge composes the global .bold class
className === "bold"
) {
found.push(className);
}
}
expect(found.length).toBe(2);
});

testTodo("keeps declarations in the correct order", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
await app.goto("/page-a");
// let heading = await app.getElement("[data-ui-heading]"); A couple of
// approaches:
// - get the stylesheet text and check the order
// - check that the heading itself has computed styles from `Text` that are
// not overridden
// - check that the heading does NOT have computed styles from `Text` that
// are overridden
});

test.describe("No javascript", () => {
test.use({ javaScriptEnabled: false });
test("loads the CSS Modules stylesheet", async ({ page }) => {
let app = new PlaywrightFixture(appFixture, page);
let cssResponses = app.collectResponses((url) =>
url.pathname.endsWith(".css")
);
await app.goto("/");
expect(cssResponses.length).toEqual(2);
});
});
});
1 change: 1 addition & 0 deletions jest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ module.exports = {
],
projects: [
"packages/remix-dev",
"packages/remix-css-modules",
"packages/remix-architect",
"packages/remix-express",
"packages/remix-netlify",
Expand Down
Loading