mirror of
https://github.com/tailwindlabs/tailwindcss.git
synced 2025-12-08 21:36:08 +00:00
* temporarily disable workflows
* add oxide
Our Rust related parts
* use oxide
- Setup the codebase to be able to use the Rust parts based on an
environment variable: `OXIDE=1`.
- Setup some tests that run both the non-Rust and Rust version in the
same test.
- Sort the candidates in a consistent way, to guarantee the order for
now (especially in tests).
- Reflect sorting related changes in tests.
- Ensure tests run in both the Rust and non-Rust version. (Some tests
are explicitly skipped when using the Rust version since we haven't
implemented those features yet. These include: custom prefix,
transformers and extractors).
- `jest`
-`OXIDE=1 jest`
* remove into_par_iter where it doesn't make sense
* cargo fmt
* wip
* enable tracing based on `DEBUG` env
* improve CI for the Oxide build
* sort test output
This happened because the sorting happens in this branch, but changes
happened on the `master` branch.
* add failing tests
I noticed that some of the tests were failing, and while looking at
them, it happened because the tests were structured like this:
```html
<div
class="
backdrop-filter
backdrop-filter-none
backdrop-blur-lg
backdrop-brightness-50
backdrop-contrast-0
backdrop-grayscale
backdrop-hue-rotate-90
backdrop-invert
backdrop-opacity-75
backdrop-saturate-150
backdrop-sepia
"
></div>
```
This means that the class names themselves eventually end up like this: `backdrop-filter-none\n`
-> (Notice the `\n`)
/cc @thecrypticace
* fix range to include `\n`
* Include only unique values for tests
Really, what we care about most is that the list contains every expected candidate. Not necessarily how many times it shows up because while many candidates will show up A LOT in a source text we’ll unique them before passing them back to anything that needs them
* Fix failing tests
* Don’t match empty arbitrary values
* skip tests in oxide mode regarding custom separators in arbitrary variants
* re-enable workflows
* use `@tailwindcss/oxide` dependency
* publish `tailwindcss@oxide`
* drop prepublishOnly
I don't think we actually need this anymore (or even want because this
is trying to do things in CI that we don't want to happen. Aka, build
the Oxide Rust code, it is already a dependency).
* WIP
* Defer to existing CLI for Oxide
* Include new compiled typescript stuff when publishing
* Move TS to ./src/oxide
* Update scripts
* Clean up tests for TS
* copy `cli` to `oxide/cli`
* make CLI files TypeScript files
* drop --postcss flag
* setup lightningcss
* Remove autoprefixer and cssnano from oxide CLI
* cleanup Rust code a little bit
- Drop commented out code
- Drop 500 fixture templates
* sort test output
* re-add `prepublishOnly` script
* bump SWC dependencies in package-lock.json
* pin `@swc` dependencies
* ensure to install and build oxide
* update all GitHub Workflows to reflect Oxide required changes
* sort `content-resolution` integration tests
* add `Release Insiders — Oxide`
* setup turbo repo + remote caching
* use `npx` to invoke `turbo`
* setup unique/proper package names for integration tests
* add missing `isomorphic-fetch` dependency
* setup integration tests to use `turborepo`
* scope tailwind tasks to root workspace
* re-enable `node_modules` cache for integration tests
* re-enable `node_modules` cache for main CI workflow
* split cache for `main` and `oxide` node_modules
* fix indent
* split install dependencies so that they can be cached individually
* improve GitHub actions caching
* use correct path for oxide node_modules (crates/node)
* ensure that `cargo install` always succeeds
cargo install X, on CI will fail if it already exists.
* figure out integration tests with turbo
* tmp: use `npm` instead of `turbo`
* disable `fail-fast`
This will allow us to run integration tests so that it still caches the
succesful ones.
* YAML OH YAML, Y U WHITESPACE SENSITIVE
* copy the oxide-ci workflow to release-oxide
* make `oxide-ci` a normal CI workflow
Without publishing
* try to cache cargo and node_modules for the oxide build
* configure turbo to run scripts in the root
* explicitly skip failing test for the Oxide version
* run oxide tests in CI
* only use build script for root package
* sync package-lock.json
* do not cache node_modules for each individual integration
* look for hoisted `.bin`
* use turbo for caching build tailwind css in integration tests
* Robin...
* try to use the local binary first
* skip installing integration test dependencies
Should already be installed due to workspace usage
* Robin...
* drop `output.clean`
* explicitly add `mini-css-extract-plugin`
* drop oxide-ci, this is tested by proxy
* ensure oxide build is used in integration tests
This will ensure the `@tailwindcss/oxide` dependency is available
(whether we use it or not).
* setup Oxide shim in insiders release
* add browserslist dependency
* use `install:all` script name
Just using `install` as a script name will be called when running
`npm install`.
Now that we marked the repo as a `workspace`, `npm install` will run
install in all workspaces which is... not ideal.
* tmp: enable insiders release in PRs
Just to check if everything works before merging. Can be removed once
tested.
* don't cache node_modules?
I feel there is some catch 22 going on here.
We require `npm install` to build the `oxide/crates/node` version.
But we also require `oxide/crates/node` for the `npm install` becaus of
the dependency: `"@tailwindcss/oxide": "file:oxide/creates/node"`
* try to use `oxide/crates/node` as part of the workspace
* let's think about this
Let's try and cache the `node_modules` and share as much as possible.
However, some scripts still need to be installed specific to the OS.
Running `npm install` locally doesn't throw away your `node_modules`,
so if we just cache `node_modules` but also run `npm install` that
should keep as much as possible and still improve install times since
`node_modules` is already there.
I think.
* ensure generated `index.js` and `index.d.ts` files are considered outputs
* use `npx napi` instead of `napi` directly
* include all `package-lock.json` files
* normalize caching further in all workflows
* drop nested `package-lock.json` files
* `npm uninstall mini-css-extract-plugin && npm install mini-css-extract-plugin --save-dev`
* bump webpack-5 integration tests dependencies
* only release insiders on `master` branch
* tmp: let's figure out release insiders oxide
* fix little typo
* use Node 18 for Oxide Insiders
* syncup package-lock.json
* let's try node 16
Node 18 currently fails on `Build x86_64-unknown-linux-gnu (OXIDE)`
Workflow.
Install Node.JS output:
```
Environment details
Warning: /__t/node/18.13.0/x64/bin/node: /lib64/libm.so.6: version `GLIBC_2.27' not found (required by /__t/node/18.13.0/x64/bin/node)
/__t/node/18.13.0/x64/bin/node: /lib64/libc.so.6: version `GLIBC_2.25' not found (required by /__t/node/18.13.0/x64/bin/node)
/__t/node/18.13.0/x64/bin/node: /lib64/libc.so.6: version `GLIBC_2.28' not found (required by /__t/node/18.13.0/x64/bin/node)
/__t/node/18.13.0/x64/bin/node: /lib64/libstdc++.so.6: version `CXXABI_1.3.9' not found (required by /__t/node/18.13.0/x64/bin/node)
/__t/node/18.13.0/x64/bin/node: /lib64/libstdc++.so.6: version `GLIBCXX_3.4.20' not found (required by /__t/node/18.13.0/x64/bin/node)
/__t/node/18.13.0/x64/bin/node: /lib64/libstdc++.so.6: version `GLIBCXX_3.4.21' not found (required by /__t/node/18.13.0/x64/bin/node)
Warning: node: /lib64/libm.so.6: version `GLIBC_2.27' not found (required by node)
node: /lib64/libc.so.6: version `GLIBC_2.25' not found (required by node)
node: /lib64/libc.so.6: version `GLIBC_2.28' not found (required by node)
node: /lib64/libstdc++.so.6: version `CXXABI_1.3.9' not found (required by node)
node: /lib64/libstdc++.so.6: version `GLIBCXX_3.4.20' not found (required by node)
node: /lib64/libstdc++.so.6: version `GLIBCXX_3.4.21' not found (required by node)
```
* bump some Node versions
* only release oxide insiders on `master` branch
* don't cache `npm`
* bump napi-rs
Co-authored-by: Jordan Pittman <jordan@cryptica.me>
Co-authored-by: Adam Wathan <4323180+adamwathan@users.noreply.github.com>
1138 lines
28 KiB
JavaScript
1138 lines
28 KiB
JavaScript
import fs from 'fs'
|
|
import path from 'path'
|
|
import postcss from 'postcss'
|
|
|
|
import { run, css, html, defaults } from './util/run'
|
|
|
|
test('variants', () => {
|
|
let config = {
|
|
darkMode: 'class',
|
|
content: [path.resolve(__dirname, './variants.test.html')],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind base;
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
let expectedPath = path.resolve(__dirname, './variants.test.css')
|
|
let expected = fs.readFileSync(expectedPath, 'utf8')
|
|
|
|
expect(result.css).toMatchFormattedCss(expected)
|
|
})
|
|
})
|
|
|
|
test('order matters and produces different behaviour', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="hover:file:bg-pink-600"></div>
|
|
<div class="file:hover:bg-pink-600"></div>
|
|
`,
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.file\:hover\:bg-pink-600:hover::file-selector-button {
|
|
--tw-bg-opacity: 1;
|
|
background-color: rgb(219 39 119 / var(--tw-bg-opacity));
|
|
}
|
|
.hover\:file\:bg-pink-600::file-selector-button:hover {
|
|
--tw-bg-opacity: 1;
|
|
background-color: rgb(219 39 119 / var(--tw-bg-opacity));
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
describe('custom advanced variants', () => {
|
|
test('prose-headings usage on its own', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="prose-headings:text-center"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('prose-headings', ':where(&) :is(h1, h2, h3, h4)')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
:where(.prose-headings\:text-center) :is(h1, h2, h3, h4) {
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('prose-headings with another "simple" variant', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="hover:prose-headings:text-center"></div>
|
|
<div class="prose-headings:hover:text-center"></div>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('prose-headings', ':where(&) :is(h1, h2, h3, h4)')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
:where(.hover\:prose-headings\:text-center) :is(h1, h2, h3, h4):hover {
|
|
text-align: center;
|
|
}
|
|
|
|
:where(.prose-headings\:hover\:text-center:hover) :is(h1, h2, h3, h4) {
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('prose-headings with another "complex" variant', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="group-hover:prose-headings:text-center"></div>
|
|
<div class="prose-headings:group-hover:text-center"></div>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('prose-headings', ':where(&) :is(h1, h2, h3, h4)')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.group:hover :where(.group-hover\:prose-headings\:text-center) :is(h1, h2, h3, h4) {
|
|
text-align: center;
|
|
}
|
|
|
|
:where(.group:hover .prose-headings\:group-hover\:text-center) :is(h1, h2, h3, h4) {
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('using variants with multi-class selectors', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="screen:parent screen:child"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant, addComponents }) {
|
|
addComponents({
|
|
'.parent .child': {
|
|
foo: 'bar',
|
|
},
|
|
})
|
|
addVariant('screen', '@media screen')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media screen {
|
|
.screen\:parent .child {
|
|
foo: bar;
|
|
}
|
|
.parent .screen\:child {
|
|
foo: bar;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('using multiple classNames in your custom variant', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="my-variant:underline test"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('my-variant', '&:where(.one, .two, .three)')
|
|
},
|
|
],
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
|
|
@layer components {
|
|
.test {
|
|
@apply my-variant:italic;
|
|
}
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.test:where(.one, .two, .three) {
|
|
font-style: italic;
|
|
}
|
|
|
|
.my-variant\:underline:where(.one, .two, .three) {
|
|
text-decoration-line: underline;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('variant format string must include at-rule or & (1)', async () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="wtf-bbq:text-center"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('wtf-bbq', 'lol')
|
|
},
|
|
],
|
|
}
|
|
|
|
await expect(run('@tailwind components;@tailwind utilities', config)).rejects.toThrowError(
|
|
"Your custom variant `wtf-bbq` has an invalid format string. Make sure it's an at-rule or contains a `&` placeholder."
|
|
)
|
|
})
|
|
|
|
test('variant format string must include at-rule or & (2)', async () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="wtf-bbq:text-center"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('wtf-bbq', () => 'lol')
|
|
},
|
|
],
|
|
}
|
|
|
|
await expect(run('@tailwind components;@tailwind utilities', config)).rejects.toThrowError(
|
|
"Your custom variant `wtf-bbq` has an invalid format string. Make sure it's an at-rule or contains a `&` placeholder."
|
|
)
|
|
})
|
|
})
|
|
|
|
test('stacked peer variants', async () => {
|
|
let config = {
|
|
content: [{ raw: 'peer-disabled:peer-focus:peer-hover:border-blue-500' }],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind base;
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
`
|
|
|
|
let expected = css`
|
|
.peer:disabled:focus:hover ~ .peer-disabled\:peer-focus\:peer-hover\:border-blue-500 {
|
|
--tw-border-opacity: 1;
|
|
border-color: rgb(59 130 246 / var(--tw-border-opacity));
|
|
}
|
|
`
|
|
|
|
let result = await run(input, config)
|
|
expect(result.css).toIncludeCss(expected)
|
|
})
|
|
|
|
it('should properly handle keyframes with multiple variants', async () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: 'animate-spin hover:animate-spin focus:animate-spin hover:animate-bounce focus:animate-bounce',
|
|
},
|
|
],
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
`
|
|
|
|
let result = await run(input, config)
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
@keyframes spin {
|
|
to {
|
|
transform: rotate(360deg);
|
|
}
|
|
}
|
|
.animate-spin {
|
|
animation: spin 1s linear infinite;
|
|
}
|
|
@keyframes bounce {
|
|
0%,
|
|
100% {
|
|
transform: translateY(-25%);
|
|
animation-timing-function: cubic-bezier(0.8, 0, 1, 1);
|
|
}
|
|
50% {
|
|
transform: none;
|
|
animation-timing-function: cubic-bezier(0, 0, 0.2, 1);
|
|
}
|
|
}
|
|
.hover\:animate-bounce:hover {
|
|
animation: bounce 1s infinite;
|
|
}
|
|
@keyframes spin {
|
|
to {
|
|
transform: rotate(360deg);
|
|
}
|
|
}
|
|
.hover\:animate-spin:hover {
|
|
animation: spin 1s linear infinite;
|
|
}
|
|
@keyframes bounce {
|
|
0%,
|
|
100% {
|
|
transform: translateY(-25%);
|
|
animation-timing-function: cubic-bezier(0.8, 0, 1, 1);
|
|
}
|
|
50% {
|
|
transform: none;
|
|
animation-timing-function: cubic-bezier(0, 0, 0.2, 1);
|
|
}
|
|
}
|
|
.focus\:animate-bounce:focus {
|
|
animation: bounce 1s infinite;
|
|
}
|
|
@keyframes spin {
|
|
to {
|
|
transform: rotate(360deg);
|
|
}
|
|
}
|
|
.focus\:animate-spin:focus {
|
|
animation: spin 1s linear infinite;
|
|
}
|
|
`)
|
|
})
|
|
|
|
test('custom addVariant with more complex media query params', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="magic:text-center"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('magic', '@media screen and (max-width: 600px)')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media screen and (max-width: 600px) {
|
|
.magic\:text-center {
|
|
text-align: center;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('custom addVariant with nested media & format shorthand', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html` <div class="magic:text-center"></div> `,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('magic', '@supports (hover: hover) { @media print { &:disabled } }')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@supports (hover: hover) {
|
|
@media print {
|
|
.magic\:text-center:disabled {
|
|
text-align: center;
|
|
}
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('before and after variants are a bit special, and forced to the end', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="before:hover:text-center"></div>
|
|
<div class="hover:before:text-center"></div>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.before\:hover\:text-center:hover::before {
|
|
content: var(--tw-content);
|
|
text-align: center;
|
|
}
|
|
|
|
.hover\:before\:text-center:hover::before {
|
|
content: var(--tw-content);
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('before and after variants are a bit special, and forced to the end (2)', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="before:prose-headings:text-center"></div>
|
|
<div class="prose-headings:before:text-center"></div>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant }) {
|
|
addVariant('prose-headings', ':where(&) :is(h1, h2, h3, h4)')
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
:where(.before\:prose-headings\:text-center) :is(h1, h2, h3, h4)::before {
|
|
content: var(--tw-content);
|
|
text-align: center;
|
|
}
|
|
|
|
:where(.prose-headings\:before\:text-center) :is(h1, h2, h3, h4)::before {
|
|
content: var(--tw-content);
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('returning non-strings and non-selectors in addVariant', () => {
|
|
/** @type {import('../types/config').Config} */
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="peer-aria-expanded:text-center"></div>
|
|
<div class="peer-aria-expanded-2:text-center"></div>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [
|
|
function ({ addVariant, e }) {
|
|
addVariant('peer-aria-expanded', ({ modifySelectors, separator }) =>
|
|
// Returning anything other string | string[] | undefined here is not supported
|
|
// But we're trying to be lenient here and just throw it out
|
|
modifySelectors(
|
|
({ className }) =>
|
|
`.peer[aria-expanded="true"] ~ .${e(`peer-aria-expanded${separator}${className}`)}`
|
|
)
|
|
)
|
|
|
|
addVariant('peer-aria-expanded-2', ({ modifySelectors, separator }) => {
|
|
let nodes = modifySelectors(
|
|
({ className }) => `.${e(`peer-aria-expanded-2${separator}${className}`)}`
|
|
)
|
|
|
|
return [
|
|
// Returning anything other than strings here is not supported
|
|
// But we're trying to be lenient here and just throw it out
|
|
nodes,
|
|
'.peer[aria-expanded="false"] ~ &',
|
|
]
|
|
})
|
|
},
|
|
],
|
|
}
|
|
|
|
return run('@tailwind components;@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.peer[aria-expanded='true'] ~ .peer-aria-expanded\:text-center {
|
|
text-align: center;
|
|
}
|
|
.peer[aria-expanded='false'] ~ .peer-aria-expanded-2\:text-center {
|
|
text-align: center;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('should not generate variants of user css if it is not inside a layer', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="hover:foo"></div>` }],
|
|
plugins: [],
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
|
|
.foo {
|
|
color: red;
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.foo {
|
|
color: red;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('should be possible to use responsive modifiers that are defined with special characters', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="<sm:underline"></div>` }],
|
|
theme: {
|
|
screens: {
|
|
'<sm': { max: '399px' },
|
|
},
|
|
},
|
|
plugins: [],
|
|
}
|
|
|
|
return run('@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (max-width: 399px) {
|
|
.\<sm\:underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('including just the base layer should not produce variants', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="sm:container sm:underline"></div>` }],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
return run('@tailwind base', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(
|
|
css`
|
|
${defaults}
|
|
`
|
|
)
|
|
})
|
|
})
|
|
|
|
it('variants for components should not be produced in a file without a components layer', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="sm:container sm:underline"></div>` }],
|
|
}
|
|
|
|
return run('@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.sm\:underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('variants for utilities should not be produced in a file without a utilities layer', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="sm:container sm:underline"></div>` }],
|
|
}
|
|
|
|
return run('@tailwind components', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.sm\:container {
|
|
width: 100%;
|
|
}
|
|
@media (min-width: 640px) {
|
|
.sm\:container {
|
|
max-width: 640px;
|
|
}
|
|
}
|
|
@media (min-width: 768px) {
|
|
.sm\:container {
|
|
max-width: 768px;
|
|
}
|
|
}
|
|
@media (min-width: 1024px) {
|
|
.sm\:container {
|
|
max-width: 1024px;
|
|
}
|
|
}
|
|
@media (min-width: 1280px) {
|
|
.sm\:container {
|
|
max-width: 1280px;
|
|
}
|
|
}
|
|
@media (min-width: 1536px) {
|
|
.sm\:container {
|
|
max-width: 1536px;
|
|
}
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('The visited variant removes opacity support', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<a class="visited:border-red-500 visited:bg-red-500 visited:text-red-500"
|
|
>Look, it's a link!</a
|
|
>
|
|
`,
|
|
},
|
|
],
|
|
plugins: [],
|
|
}
|
|
|
|
return run('@tailwind utilities', config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
.visited\:border-red-500:visited {
|
|
border-color: rgb(239 68 68);
|
|
}
|
|
.visited\:bg-red-500:visited {
|
|
background-color: rgb(239 68 68);
|
|
}
|
|
.visited\:text-red-500:visited {
|
|
color: rgb(239 68 68);
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('appends variants to the correct place when using postcss documents', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="underline sm:underline"></div>` }],
|
|
plugins: [],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
const doc = postcss.document()
|
|
doc.append(postcss.parse(`a {}`))
|
|
doc.append(postcss.parse(`@tailwind base`))
|
|
doc.append(postcss.parse(`@tailwind utilities`))
|
|
doc.append(postcss.parse(`b {}`))
|
|
|
|
const result = doc.toResult()
|
|
|
|
return run(result, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
a {
|
|
}
|
|
${defaults}
|
|
.underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
@media (min-width: 640px) {
|
|
.sm\:underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
}
|
|
b {
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('variants support multiple, grouped selectors (html)', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="sm:base1 sm:base2"></div>` }],
|
|
plugins: [],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
.base1 .foo,
|
|
.base1 .bar {
|
|
color: red;
|
|
}
|
|
|
|
.base2 .bar .base2-foo {
|
|
color: red;
|
|
}
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.sm\:base1 .foo,
|
|
.sm\:base1 .bar {
|
|
color: red;
|
|
}
|
|
|
|
.sm\:base2 .bar .base2-foo {
|
|
color: red;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('variants support multiple, grouped selectors (apply)', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="baz"></div>` }],
|
|
plugins: [],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
.base .foo,
|
|
.base .bar {
|
|
color: red;
|
|
}
|
|
}
|
|
.baz {
|
|
@apply sm:base;
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.baz .foo,
|
|
.baz .bar {
|
|
color: red;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('variants only picks the used selectors in a group (html)', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="sm:b"></div>` }],
|
|
plugins: [],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
.a,
|
|
.b {
|
|
color: red;
|
|
}
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.sm\:b {
|
|
color: red;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
it('variants only picks the used selectors in a group (apply)', () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="baz"></div>` }],
|
|
plugins: [],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
.a,
|
|
.b {
|
|
color: red;
|
|
}
|
|
}
|
|
.baz {
|
|
@apply sm:b;
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
return expect(result.css).toMatchFormattedCss(css`
|
|
@media (min-width: 640px) {
|
|
.baz {
|
|
color: red;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('hoverOnlyWhenSupported adds hover and pointer media features by default', () => {
|
|
let config = {
|
|
future: {
|
|
hoverOnlyWhenSupported: true,
|
|
},
|
|
content: [
|
|
{ raw: html`<div class="hover:underline group-hover:underline peer-hover:underline"></div>` },
|
|
],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind base;
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
${defaults}
|
|
|
|
@media (hover: hover) and (pointer: fine) {
|
|
.hover\:underline:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.group:hover .group-hover\:underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
.peer:hover ~ .peer-hover\:underline {
|
|
text-decoration-line: underline;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('multi-class utilities handle selector-mutating variants correctly', () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`<div
|
|
class="after:foo after:bar after:baz hover:foo hover:bar hover:baz group-hover:foo group-hover:bar group-hover:baz peer-checked:foo peer-checked:bar peer-checked:baz"
|
|
></div>`,
|
|
},
|
|
{
|
|
raw: html`<div
|
|
class="after:foo1 after:bar1 after:baz1 hover:foo1 hover:bar1 hover:baz1 group-hover:foo1 group-hover:bar1 group-hover:baz1 peer-checked:foo1 peer-checked:bar1 peer-checked:baz1"
|
|
></div>`,
|
|
},
|
|
],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
.foo.bar.baz {
|
|
color: red;
|
|
}
|
|
.foo1 .bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
}
|
|
`
|
|
|
|
// The second set of ::after cases (w/ descendant selectors)
|
|
// are clearly "wrong" BUT you can't have a descendant of a
|
|
// pseudo - element so the utilities `after:foo1` and
|
|
// `after:bar1` are non-sensical so this is still
|
|
// perfectly fine behavior
|
|
|
|
return run(input, config).then((result) => {
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
.after\:foo.bar.baz::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.after\:bar.foo.baz::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.after\:baz.foo.bar::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.after\:foo1 .bar1 .baz1::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.foo1 .after\:bar1 .baz1::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.foo1 .bar1 .after\:baz1::after {
|
|
content: var(--tw-content);
|
|
color: red;
|
|
}
|
|
.hover\:foo:hover.bar.baz {
|
|
color: red;
|
|
}
|
|
.hover\:bar:hover.foo.baz {
|
|
color: red;
|
|
}
|
|
.hover\:baz:hover.foo.bar {
|
|
color: red;
|
|
}
|
|
.hover\:foo1:hover .bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .hover\:bar1:hover .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .bar1 .hover\:baz1:hover {
|
|
color: red;
|
|
}
|
|
.group:hover .group-hover\:foo.bar.baz {
|
|
color: red;
|
|
}
|
|
.group:hover .group-hover\:bar.foo.baz {
|
|
color: red;
|
|
}
|
|
.group:hover .group-hover\:baz.foo.bar {
|
|
color: red;
|
|
}
|
|
.group:hover .group-hover\:foo1 .bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .group:hover .group-hover\:bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .bar1 .group:hover .group-hover\:baz1 {
|
|
color: red;
|
|
}
|
|
.peer:checked ~ .peer-checked\:foo.bar.baz {
|
|
color: red;
|
|
}
|
|
.peer:checked ~ .peer-checked\:bar.foo.baz {
|
|
color: red;
|
|
}
|
|
.peer:checked ~ .peer-checked\:baz.foo.bar {
|
|
color: red;
|
|
}
|
|
.peer:checked ~ .peer-checked\:foo1 .bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .peer:checked ~ .peer-checked\:bar1 .baz1 {
|
|
color: red;
|
|
}
|
|
.foo1 .bar1 .peer:checked ~ .peer-checked\:baz1 {
|
|
color: red;
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('class inside pseudo-class function :has', () => {
|
|
let config = {
|
|
content: [
|
|
{ raw: html`<div class="foo hover:foo sm:foo"></div>` },
|
|
{ raw: html`<div class="bar hover:bar sm:bar"></div>` },
|
|
{ raw: html`<div class="baz hover:baz sm:baz"></div>` },
|
|
],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
@layer utilities {
|
|
:where(.foo) {
|
|
color: red;
|
|
}
|
|
:matches(.foo, .bar, .baz) {
|
|
color: orange;
|
|
}
|
|
:is(.foo) {
|
|
color: yellow;
|
|
}
|
|
html:has(.foo) {
|
|
color: green;
|
|
}
|
|
}
|
|
`
|
|
|
|
return run(input, config).then((result) => {
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
:where(.foo) {
|
|
color: red;
|
|
}
|
|
:matches(.foo, .bar, .baz) {
|
|
color: orange;
|
|
}
|
|
:is(.foo) {
|
|
color: yellow;
|
|
}
|
|
html:has(.foo) {
|
|
color: green;
|
|
}
|
|
|
|
:where(.hover\:foo:hover) {
|
|
color: red;
|
|
}
|
|
:matches(.hover\:foo:hover, .bar, .baz) {
|
|
color: orange;
|
|
}
|
|
:matches(.foo, .hover\:bar:hover, .baz) {
|
|
color: orange;
|
|
}
|
|
:matches(.foo, .bar, .hover\:baz:hover) {
|
|
color: orange;
|
|
}
|
|
:is(.hover\:foo:hover) {
|
|
color: yellow;
|
|
}
|
|
html:has(.hover\:foo:hover) {
|
|
color: green;
|
|
}
|
|
@media (min-width: 640px) {
|
|
:where(.sm\:foo) {
|
|
color: red;
|
|
}
|
|
:matches(.sm\:foo, .bar, .baz) {
|
|
color: orange;
|
|
}
|
|
:matches(.foo, .sm\:bar, .baz) {
|
|
color: orange;
|
|
}
|
|
:matches(.foo, .bar, .sm\:baz) {
|
|
color: orange;
|
|
}
|
|
:is(.sm\:foo) {
|
|
color: yellow;
|
|
}
|
|
html:has(.sm\:foo) {
|
|
color: green;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
})
|
|
|
|
test('variant functions returning arrays should output correct results when nesting', async () => {
|
|
let config = {
|
|
content: [{ raw: html`<div class="test:foo" />` }],
|
|
corePlugins: { preflight: false },
|
|
plugins: [
|
|
function ({ addUtilities, addVariant }) {
|
|
addVariant('test', () => ['@media (test)'])
|
|
addUtilities({
|
|
'.foo': {
|
|
display: 'grid',
|
|
'> *': {
|
|
'grid-column': 'span 2',
|
|
},
|
|
},
|
|
})
|
|
},
|
|
],
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
`
|
|
|
|
let result = await run(input, config)
|
|
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
@media (test) {
|
|
.test\:foo {
|
|
display: grid;
|
|
}
|
|
.test\:foo > * {
|
|
grid-column: span 2;
|
|
}
|
|
}
|
|
`)
|
|
})
|
|
|
|
test('arbitrary variant selectors should not re-order scrollbar pseudo classes', async () => {
|
|
let config = {
|
|
content: [
|
|
{
|
|
raw: html`
|
|
<div class="[&::-webkit-scrollbar:hover]:underline" />
|
|
<div class="[&::-webkit-scrollbar-button:hover]:underline" />
|
|
<div class="[&::-webkit-scrollbar-thumb:hover]:underline" />
|
|
<div class="[&::-webkit-scrollbar-track:hover]:underline" />
|
|
<div class="[&::-webkit-scrollbar-track-piece:hover]:underline" />
|
|
<div class="[&::-webkit-scrollbar-corner:hover]:underline" />
|
|
<div class="[&::-webkit-resizer:hover]:underline" />
|
|
`,
|
|
},
|
|
],
|
|
corePlugins: { preflight: false },
|
|
}
|
|
|
|
let input = css`
|
|
@tailwind utilities;
|
|
`
|
|
|
|
let result = await run(input, config)
|
|
|
|
expect(result.css).toMatchFormattedCss(css`
|
|
.\[\&\:\:-webkit-resizer\:hover\]\:underline::-webkit-resizer:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar-button\:hover\]\:underline::-webkit-scrollbar-button:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar-corner\:hover\]\:underline::-webkit-scrollbar-corner:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar-thumb\:hover\]\:underline::-webkit-scrollbar-thumb:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar-track-piece\:hover\]\:underline::-webkit-scrollbar-track-piece:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar-track\:hover\]\:underline::-webkit-scrollbar-track:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
.\[\&\:\:-webkit-scrollbar\:hover\]\:underline::-webkit-scrollbar:hover {
|
|
text-decoration-line: underline;
|
|
}
|
|
`)
|
|
})
|