@tailwindcss/typography
Advanced tools
Comparing version 0.0.0-insiders.24f3595 to 0.0.0-insiders.25b0f9e
{ | ||
"name": "@tailwindcss/typography", | ||
"version": "0.0.0-insiders.24f3595", | ||
"version": "0.0.0-insiders.25b0f9e", | ||
"description": "A Tailwind CSS plugin for automatically styling plain HTML content with beautiful typographic defaults.", | ||
"main": "src/index.js", | ||
"types": "src/index.d.ts", | ||
"files": [ | ||
"src/*.js", | ||
"src/*.d.ts", | ||
"dist/" | ||
], | ||
"repository": "https://github.com/tailwindcss/typography", | ||
"repository": "https://github.com/tailwindlabs/tailwindcss-typography", | ||
"license": "MIT", | ||
@@ -26,6 +28,8 @@ "publishConfig": { | ||
"export": "next export demo", | ||
"start": "next start demo" | ||
"start": "next start demo", | ||
"release-channel": "node ./scripts/release-channel.js", | ||
"release-notes": "node ./scripts/release-notes.js" | ||
}, | ||
"peerDependencies": { | ||
"tailwindcss": ">=3.0.0 || >= 3.0.0-alpha.1 || insiders" | ||
"tailwindcss": ">=3.0.0 || insiders || >=4.0.0-alpha.20 || >=4.0.0-beta.1" | ||
}, | ||
@@ -38,3 +42,3 @@ "devDependencies": { | ||
"highlight.js": "^10.4.1", | ||
"jest": "^26.6.1", | ||
"jest": "^29.7.0", | ||
"jest-diff": "^27.3.1", | ||
@@ -46,3 +50,3 @@ "next": "^12.0.1", | ||
"react-dom": "^17.0.2", | ||
"tailwindcss": "^3.0.0-alpha.2" | ||
"tailwindcss": "^3.2.2" | ||
}, | ||
@@ -52,3 +56,4 @@ "dependencies": { | ||
"lodash.isplainobject": "^4.0.6", | ||
"lodash.merge": "^4.6.2" | ||
"lodash.merge": "^4.6.2", | ||
"postcss-selector-parser": "6.0.10" | ||
}, | ||
@@ -55,0 +60,0 @@ "jest": { |
469
README.md
<p> | ||
<img alt="Tailwind CSS Typography" width="350" src="./.github/logo.svg"> | ||
<a href="https://tailwindcss.com/docs/typography-plugin" target="_blank"> | ||
<picture> | ||
<source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/tailwindlabs/tailwindcss-typography/HEAD/.github/logo-dark.svg"> | ||
<source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/tailwindlabs/tailwindcss-typography/HEAD/.github/logo-light.svg"> | ||
<img alt="Tailwind CSS Typography" src="https://raw.githubusercontent.com/tailwindlabs/tailwindcss-typography/HEAD/.github/logo-light.svg" width="450" height="70" style="max-width: 100%;"> | ||
</picture> | ||
</a> | ||
</p> | ||
**As of v0.3.0, @tailwindcss/typography is designed for Tailwind CSS v2.0+.** | ||
The official Tailwind CSS Typography plugin provides a set of `prose` classes you can use to add beautiful typographic defaults to any vanilla HTML you don’t control, like HTML rendered from Markdown, or pulled from a CMS. | ||
A plugin that provides a set of `prose` classes you can use to add beautiful typographic defaults to any vanilla HTML you don't control (like HTML rendered from Markdown, or pulled from a CMS). | ||
[View live demo](https://tailwindcss-typography.vercel.app/) | ||
```html | ||
<article class="prose lg:prose-xl"> | ||
{{ markdown }} | ||
</article> | ||
<article class="prose lg:prose-xl">{{ markdown }}</article> | ||
``` | ||
To see what it looks like in action, check out our [live demo](https://play.tailwindcss.com/uj1vGACRJA?layout=preview) on Tailwind Play. | ||
--- | ||
## Installation | ||
@@ -21,8 +25,4 @@ | ||
```sh | ||
# Using npm | ||
npm install @tailwindcss/typography | ||
# Using Yarn | ||
yarn add @tailwindcss/typography | ||
```shell | ||
npm install -D @tailwindcss/typography | ||
``` | ||
@@ -33,3 +33,3 @@ | ||
```js | ||
// tailwind.config.js | ||
/** @type {import('tailwindcss').Config} */ | ||
module.exports = { | ||
@@ -46,34 +46,6 @@ theme: { | ||
### Using a CDN | ||
--- | ||
If you need to pull in these styles via CDN, you can do so using services like UNPKG or jsDeliver: | ||
## Basic usage | ||
```html | ||
<!-- From UNPKG --> | ||
<link | ||
rel="stylesheet" | ||
href="https://unpkg.com/@tailwindcss/typography@0.4.x/dist/typography.min.css" | ||
/> | ||
<!-- From jsDelivr --> | ||
<link | ||
rel="stylesheet" | ||
href="https://cdn.jsdelivr.net/npm/@tailwindcss/typography@0.4.x/dist/typography.min.css" | ||
/> | ||
``` | ||
To use these styles alongside the rest of Tailwind via CDN, we recommend pulling in each layer separately so you can put the styles in the correct order: | ||
```html | ||
<link rel="stylesheet" href="https://unpkg.com/tailwindcss@^1.5/dist/base.min.css" /> | ||
<link rel="stylesheet" href="https://unpkg.com/tailwindcss@^1.5/dist/components.min.css" /> | ||
<link | ||
rel="stylesheet" | ||
href="https://unpkg.com/@tailwindcss/typography@0.2.x/dist/typography.min.css" | ||
/> | ||
<link rel="stylesheet" href="https://unpkg.com/tailwindcss@^1.5/dist/utilities.min.css" /> | ||
``` | ||
## Usage | ||
Now you can use the `prose` classes to add sensible typography styles to any vanilla HTML: | ||
@@ -97,10 +69,37 @@ | ||
### Size modifiers | ||
### Choosing a gray scale | ||
This plugin includes a modifier class for each of the five gray scales Tailwind includes by default so you can easily style your content to match the grays you're using in your project. | ||
```html | ||
<article class="prose prose-slate">{{ markdown }}</article> | ||
``` | ||
Here are the classes that are generated using a totally default Tailwind CSS v2.0 build: | ||
| Class | Gray scale | | ||
| ------------------------ | ---------- | | ||
| `prose-gray` _(default)_ | Gray | | ||
| `prose-slate` | Slate | | ||
| `prose-zinc` | Zinc | | ||
| `prose-neutral` | Neutral | | ||
| `prose-stone` | Stone | | ||
Modifier classes are designed to be used with the [multi-class modifier pattern](http://nicolasgallagher.com/about-html-semantics-front-end-architecture/#component-modifiers) and must be used in conjunction with the base `prose` class. | ||
> [!NOTE] | ||
> Always include the `prose` class when adding a gray scale modifier | ||
```html | ||
<article class="prose prose-stone">{{ markdown }}</article> | ||
``` | ||
To learn about creating your own color themes, read the [adding custom color themes](#adding-custom-color-themes) documentation. | ||
### Applying a type scale | ||
Size modifiers allow you to adjust the overall size of your typography for different contexts. | ||
```html | ||
<article class="prose prose-xl"> | ||
{{ markdown }} | ||
</article> | ||
<article class="prose prose-xl">{{ markdown }}</article> | ||
``` | ||
@@ -110,32 +109,45 @@ | ||
| Class | Body font size | | ||
| ----------- | --------------: | | ||
| `prose-sm` | 0.875rem (14px) | | ||
| `prose` | 1rem (16px) | | ||
| `prose-lg` | 1.125rem (18px) | | ||
| `prose-xl` | 1.25rem (20px) | | ||
| `prose-2xl` | 1.5rem (24px) | | ||
| Class | Body font size | | ||
| ------------------------ | ----------------- | | ||
| `prose-sm` | 0.875rem _(14px)_ | | ||
| `prose-base` _(default)_ | 1rem _(16px)_ | | ||
| `prose-lg` | 1.125rem _(18px)_ | | ||
| `prose-xl` | 1.25rem _(20px)_ | | ||
| `prose-2xl` | 1.5rem _(24px)_ | | ||
Everything about the provided size modifiers has been hand-tuned to look as beautiful as possible, including the relationships between font sizes, heading spacing, code block padding, etc. Just like the Tailwind color palettes, none of these styles are based on naive mathematical formulas, and have been hand-crafted by professional designers. | ||
These can be used in combination with Tailwind's [breakpoint modifiers](https://tailwindcss.com/docs/responsive-design) to change the overall font size of a piece of content at different viewport sizes: | ||
Size modifiers are designed to be used with the [multi-class modifier pattern](http://nicolasgallagher.com/about-html-semantics-front-end-architecture/#component-modifiers) and **must be used in conjunction with the base `prose` class**: | ||
```html | ||
<article class="prose md:prose-lg lg:prose-xl">{{ markdown }}</article> | ||
``` | ||
Everything about the provided size modifiers has been hand-tuned by professional designers to look as beautiful as possible, including the relationships between font sizes, heading spacing, code block padding, and more. | ||
Size modifiers are designed to be used with the [multi-class modifier pattern](http://nicolasgallagher.com/about-html-semantics-front-end-architecture/#component-modifiers) and must be used in conjunction with the base `prose` class. | ||
> [!NOTE] | ||
> Always include the `prose` class when adding a size modifier | ||
```html | ||
<!-- Will not work --> | ||
<article class="prose-lg"> | ||
{{ markdown }} | ||
</article> | ||
<article class="prose prose-lg">{{ markdown }}</article> | ||
``` | ||
<!-- Always add the `prose` class --> | ||
<article class="prose prose-lg"> | ||
{{ markdown }} | ||
</article> | ||
To learn about customizing the included type scales, read the documentation on [customizing the CSS](#customizing-the-css). | ||
### Adapting to dark mode | ||
Each default color theme includes a hand-designed dark mode version that you can trigger by adding the `prose-invert` class: | ||
```html | ||
<article class="prose dark:prose-invert">{{ markdown }}</article> | ||
``` | ||
### Color modifiers | ||
To learn about creating your own color themes, read the [adding custom color themes](#adding-custom-color-themes) documentation. | ||
Color modifiers allow you to "brand" your typography sections by changing the link color. By default, modifiers are generated for every color in your color palette that include a `600` shade except for `gray` since it's the default. | ||
### Element modifiers | ||
Use element modifiers to customize the style of individual elements in your content directly in your HTML: | ||
```html | ||
<article class="prose prose-indigo"> | ||
<article class="prose prose-img:rounded-xl prose-headings:underline prose-a:text-blue-600"> | ||
{{ markdown }} | ||
@@ -145,51 +157,132 @@ </article> | ||
Here are the classes that are generated using a totally default Tailwind CSS v2.0 build: | ||
This makes it easy to do things like style links to match your brand, add a border radius to images, and tons more. | ||
| Class | Link color | | ||
| -------------- | ------------ | | ||
| `prose-red` | `red.600` | | ||
| `prose-yellow` | `yellow.600` | | ||
| `prose-green` | `green.600` | | ||
| `prose-blue` | `blue.600` | | ||
| `prose-indigo` | `indigo.600` | | ||
| `prose-purple` | `purple.600` | | ||
| `prose-pink` | `pink.600` | | ||
Here's a complete list of available element modifiers: | ||
For more control, use the [low-level customization API](#customization). | ||
| Modifier | Target | | ||
| ---------------------------- | ---------------------------- | | ||
| `prose-headings:{utility}` | `h1`, `h2`, `h3`, `h4`, `th` | | ||
| `prose-lead:{utility}` | `[class~="lead"]` | | ||
| `prose-h1:{utility}` | `h1` | | ||
| `prose-h2:{utility}` | `h2` | | ||
| `prose-h3:{utility}` | `h3` | | ||
| `prose-h4:{utility}` | `h4` | | ||
| `prose-p:{utility}` | `p` | | ||
| `prose-a:{utility}` | `a` | | ||
| `prose-blockquote:{utility}` | `blockquote` | | ||
| `prose-figure:{utility}` | `figure` | | ||
| `prose-figcaption:{utility}` | `figcaption` | | ||
| `prose-strong:{utility}` | `strong` | | ||
| `prose-em:{utility}` | `em` | | ||
| `prose-kbd:{utility}` | `kbd` | | ||
| `prose-code:{utility}` | `code` | | ||
| `prose-pre:{utility}` | `pre` | | ||
| `prose-ol:{utility}` | `ol` | | ||
| `prose-ul:{utility}` | `ul` | | ||
| `prose-li:{utility}` | `li` | | ||
| `prose-table:{utility}` | `table` | | ||
| `prose-thead:{utility}` | `thead` | | ||
| `prose-tr:{utility}` | `tr` | | ||
| `prose-th:{utility}` | `th` | | ||
| `prose-td:{utility}` | `td` | | ||
| `prose-img:{utility}` | `img` | | ||
| `prose-video:{utility}` | `video` | | ||
| `prose-hr:{utility}` | `hr` | | ||
### Responsive variants | ||
When stacking these modifiers with other modifiers like `hover`, you most likely want the other modifier to come first: | ||
None of the sizes are automatically responsive, but responsive variants are provided for each size modifier so you can easily change the typography size at different breakpoints: | ||
```html | ||
<article class="prose prose-a:text-blue-600 hover:prose-a:text-blue-500">{{ markdown }}</article> | ||
``` | ||
Read the Tailwind CSS documentation on [ordering stacked modifiers](https://tailwindcss.com/docs/hover-focus-and-other-states#ordering-stacked-modifiers) to learn more. | ||
### Overriding max-width | ||
Each size modifier comes with a baked in `max-width` designed to keep the content as readable as possible. This isn't always what you want though, and sometimes you'll want the content to just fill the width of its container. | ||
In those cases, all you need to do is add `max-w-none` to your content to override the embedded max-width: | ||
```html | ||
<article class="prose prose-sm sm:prose lg:prose-lg xl:prose-xl"> | ||
{{ markdown }} | ||
<div class="grid grid-cols-4"> | ||
<div class="col-span-1"> | ||
<!-- ... --> | ||
</div> | ||
<div class="col-span-3"> | ||
<article class="prose max-w-none">{{ markdown }}</article> | ||
</div> | ||
</div> | ||
``` | ||
--- | ||
## Advanced topics | ||
### Undoing typography styles | ||
If you have a block of markup embedded in some content that shouldn't inherit the `prose` styles, use the `not-prose` class to sandbox it: | ||
```html | ||
<article class="prose"> | ||
<h1>My Heading</h1> | ||
<p>...</p> | ||
<div class="not-prose"> | ||
<!-- Some example or demo that needs to be prose-free --> | ||
</div> | ||
<p>...</p> | ||
<!-- ... --> | ||
</article> | ||
``` | ||
Note that you can't nest new `prose` instances within a `not-prose` block at this time. | ||
## Customization | ||
### Adding custom color themes | ||
> The customization API is currently extremely low-level in order to be as flexible as possible. We will be introducing higher-level configuration options over time as we learn what types of customizations are most common. | ||
You can create your own color theme by adding a new key in the `typography` section of your `tailwind.config.js` file and providing your colors under the `css` key: | ||
To customize the styles provided by this plugin, add your overrides under the `typography` key in the `theme` section of your `tailwind.config.js` file: | ||
```js | ||
// tailwind.config.js | ||
```js {{ filename: 'tailwind.config.js' }} | ||
/** @type {import('tailwindcss').Config} */ | ||
module.exports = { | ||
theme: { | ||
extend: { | ||
typography: { | ||
DEFAULT: { | ||
typography: ({ theme }) => ({ | ||
pink: { | ||
css: { | ||
color: '#333', | ||
a: { | ||
color: '#3182ce', | ||
'&:hover': { | ||
color: '#2c5282', | ||
}, | ||
}, | ||
'--tw-prose-body': theme('colors.pink[800]'), | ||
'--tw-prose-headings': theme('colors.pink[900]'), | ||
'--tw-prose-lead': theme('colors.pink[700]'), | ||
'--tw-prose-links': theme('colors.pink[900]'), | ||
'--tw-prose-bold': theme('colors.pink[900]'), | ||
'--tw-prose-counters': theme('colors.pink[600]'), | ||
'--tw-prose-bullets': theme('colors.pink[400]'), | ||
'--tw-prose-hr': theme('colors.pink[300]'), | ||
'--tw-prose-quotes': theme('colors.pink[900]'), | ||
'--tw-prose-quote-borders': theme('colors.pink[300]'), | ||
'--tw-prose-captions': theme('colors.pink[700]'), | ||
'--tw-prose-code': theme('colors.pink[900]'), | ||
'--tw-prose-pre-code': theme('colors.pink[100]'), | ||
'--tw-prose-pre-bg': theme('colors.pink[900]'), | ||
'--tw-prose-th-borders': theme('colors.pink[300]'), | ||
'--tw-prose-td-borders': theme('colors.pink[200]'), | ||
'--tw-prose-invert-body': theme('colors.pink[200]'), | ||
'--tw-prose-invert-headings': theme('colors.white'), | ||
'--tw-prose-invert-lead': theme('colors.pink[300]'), | ||
'--tw-prose-invert-links': theme('colors.white'), | ||
'--tw-prose-invert-bold': theme('colors.white'), | ||
'--tw-prose-invert-counters': theme('colors.pink[400]'), | ||
'--tw-prose-invert-bullets': theme('colors.pink[600]'), | ||
'--tw-prose-invert-hr': theme('colors.pink[700]'), | ||
'--tw-prose-invert-quotes': theme('colors.pink[100]'), | ||
'--tw-prose-invert-quote-borders': theme('colors.pink[700]'), | ||
'--tw-prose-invert-captions': theme('colors.pink[400]'), | ||
'--tw-prose-invert-code': theme('colors.white'), | ||
'--tw-prose-invert-pre-code': theme('colors.pink[300]'), | ||
'--tw-prose-invert-pre-bg': 'rgb(0 0 0 / 50%)', | ||
'--tw-prose-invert-th-borders': theme('colors.pink[600]'), | ||
'--tw-prose-invert-td-borders': theme('colors.pink[700]'), | ||
}, | ||
}, | ||
} | ||
}), | ||
}, | ||
@@ -204,41 +297,45 @@ }, | ||
Like with all theme customizations in Tailwind, you can also define the `typography` key as a function if you need access to the `theme` helper: | ||
See our internal [style definitions](https://github.com/tailwindlabs/tailwindcss-typography/blob/main/src/styles.js) for some more examples. | ||
```js | ||
// tailwind.config.js | ||
### Changing the default class name | ||
If you need to use a class name other than `prose` for any reason, you can do so using the `className` option when registering the plugin: | ||
```js {{ filename: 'tailwind.config.js' }} | ||
/** @type {import('tailwindcss').Config} */ | ||
module.exports = { | ||
theme: { | ||
extend: { | ||
typography: (theme) => ({ | ||
DEFAULT: { | ||
css: { | ||
color: theme('colors.gray.800'), | ||
// ... | ||
}, | ||
}, | ||
}), | ||
} | ||
// ... | ||
}, | ||
plugins: [ | ||
require('@tailwindcss/typography'), | ||
// ... | ||
], | ||
require('@tailwindcss/typography')({ | ||
className: 'wysiwyg', | ||
}), | ||
] | ||
... | ||
} | ||
``` | ||
Customizations should be applied to a specific modifier like `DEFAULT` or `xl`, and must be added under the `css` property. Customizations are authored in the same [CSS-in-JS syntax](https://tailwindcss.com/docs/plugins#css-in-js-syntax) used to write Tailwind plugins. | ||
Now every instance of `prose` in the default class names will be replaced by your custom class name: | ||
It's important to note that all customizations are **merged** with the defaults. If you'd like to completely override a provided size modifier, you can do so by disabling that modifier so the default styles are not included. | ||
```html | ||
<article class="wysiwyg wysiwyg-slate lg:wysiwyg-xl"> | ||
<h1>My Heading</h1> | ||
<p>...</p> | ||
See [the default styles](./src/styles.js) for this plugin for more in-depth examples of configuring each modifier. | ||
<div class="not-wysiwyg"> | ||
<!-- Some example or demo that needs to be prose-free --> | ||
</div> | ||
### Customizing shared styles | ||
<p>...</p> | ||
<!-- ... --> | ||
</article> | ||
``` | ||
Many styles _(for example colors, font weight, and text decoration)_ are shared between all size modifiers, and are therefore defined only for the `DEFAULT` modifier, since modifiers are designed to be used with the [multi-class modifier pattern](http://nicolasgallagher.com/about-html-semantics-front-end-architecture/#component-modifiers). | ||
### Customizing the CSS | ||
If you'd like to customize these sorts of styles, do so using the `DEFAULT` modifier: | ||
If you want to customize the raw CSS generated by this plugin, add your overrides under the `typography` key in the `theme` section of your `tailwind.config.js` file: | ||
```js | ||
// tailwind.config.js | ||
```js {{ filename: 'tailwind.config.js' }} | ||
/** @type {import('tailwindcss').Config} */ | ||
module.exports = { | ||
@@ -251,10 +348,12 @@ theme: { | ||
color: '#333', | ||
strong: { | ||
fontWeight: '800', | ||
a: { | ||
color: '#3182ce', | ||
'&:hover': { | ||
color: '#2c5282', | ||
}, | ||
}, | ||
// ... | ||
}, | ||
}, | ||
}, | ||
} | ||
}, | ||
}, | ||
@@ -268,23 +367,19 @@ plugins: [ | ||
### Adding new modifiers | ||
Like with all theme customizations in Tailwind, you can also define the `typography` key as a function if you need access to the `theme` helper: | ||
You can add a new modifier by creating a new key in the `typography` section of your theme and providing your own styles under the `css` key: | ||
```js | ||
// tailwind.config.js | ||
```js {{ filename: 'tailwind.config.js' }} | ||
/** @type {import('tailwindcss').Config} */ | ||
module.exports = { | ||
theme: { | ||
extend: { | ||
typography: { | ||
'3xl': { | ||
typography: (theme) => ({ | ||
DEFAULT: { | ||
css: { | ||
fontSize: '1.875rem', | ||
h1: { | ||
fontSize: '4rem', | ||
}, | ||
color: theme('colors.gray.800'), | ||
// ... | ||
}, | ||
}, | ||
}, | ||
} | ||
}), | ||
}, | ||
}, | ||
@@ -298,88 +393,16 @@ plugins: [ | ||
### Overriding max-width | ||
Customizations should be applied to a specific modifier like `DEFAULT` or `xl`, and must be added under the `css` property. Customizations are authored in the same [CSS-in-JS syntax](https://tailwindcss.com/docs/plugins#css-in-js-syntax) used to write Tailwind plugins. | ||
Each size modifier comes with a baked in `max-width` designed to keep the content as readable as possible. This isn't always what you want though, and sometimes you'll want the content to just fill the width of its container. | ||
See [the default styles](https://github.com/tailwindlabs/tailwindcss-typography/blob/main/src/styles.js) for this plugin for more in-depth examples of configuring each modifier. | ||
In those cases, all you need to do is add `max-w-none` to your content to override the embedded max-width: | ||
--- | ||
```html | ||
<div class="grid grid-cols-4"> | ||
<div class="col-span-1"> | ||
<!-- ... --> | ||
</div> | ||
<div class="col-span-3"> | ||
<article class="prose max-w-none"> | ||
{{ markdown }} | ||
</article> | ||
</div> | ||
</div> | ||
``` | ||
## Community | ||
### Disabling size modifiers | ||
For help, discussion about best practices, or any other conversation that would benefit from being searchable: | ||
If you'd like to completely disable any size modifiers (either for file size reasons or because you'd like to completely redefine that modifier), you can do so using the `modifiers` option when including the plugin: | ||
[Discuss the Tailwind CSS Typography plugin on GitHub](https://github.com/tailwindlabs/tailwindcss/discussions) | ||
```js | ||
// tailwind.config.js | ||
module.exports = { | ||
theme: { | ||
// ... | ||
}, | ||
plugins: [ | ||
require('@tailwindcss/typography')({ | ||
modifiers: ['sm', 'lg'], | ||
}), | ||
// ... | ||
], | ||
} | ||
``` | ||
For casual chit-chat with others using the framework: | ||
This option acts as a _safelist_, so you can list only the modifiers you'd actually like included and the others will be removed. | ||
The `DEFAULT` modifier is always included and cannot be disabled. | ||
### Disabling responsive variants | ||
If you'd like to disable the responsive variants for any reason, you can do so by setting the `typography` key to an empty array in the `variants` section of your `tailwind.config.js` file: | ||
```js | ||
// tailwind.config.js | ||
module.exports = { | ||
theme: { | ||
// ... | ||
}, | ||
plugins: [ | ||
require('@tailwindcss/typography'), | ||
// ... | ||
], | ||
variants: { | ||
typography: [], | ||
}, | ||
} | ||
``` | ||
### Changing the default class name | ||
If you need to use a class name other than `prose` for any reason, you can do so using the `className` option when registering the plugin: | ||
```js | ||
// tailwind.config.js | ||
module.exports = { | ||
theme: { | ||
// ... | ||
}, | ||
plugins: [ | ||
require('@tailwindcss/typography')({ | ||
className: 'markdown', | ||
}), | ||
] | ||
... | ||
} | ||
``` | ||
```html | ||
<article class="markdown md:markdown-lg"> | ||
{{ markdown }} | ||
</article> | ||
``` | ||
[Join the Tailwind CSS Discord Server](https://tailwindcss.com/discord) |
@@ -5,2 +5,3 @@ const plugin = require('tailwindcss/plugin') | ||
const styles = require('./styles') | ||
const { commonTrailingPseudos } = require('./utils') | ||
@@ -12,31 +13,16 @@ const computed = { | ||
function inWhere(selector, { className, prefix }) { | ||
function inWhere(selector, { className, modifier, prefix }) { | ||
let prefixedNot = prefix(`.not-${className}`).slice(1) | ||
let selectorPrefix = selector.startsWith('>') | ||
? `${modifier === 'DEFAULT' ? `.${className}` : `.${className}-${modifier}`} ` | ||
: '' | ||
if (selector.endsWith('::before')) { | ||
if (selector.startsWith('>')) { | ||
return `> :where(${selector.slice(2, -8)}):not(:where([class~="${prefixedNot}"] *))::before` | ||
} | ||
return `:where(${selector.slice(0, -8)}):not(:where([class~="${prefixedNot}"] *))::before` | ||
} | ||
// Parse the selector, if every component ends in the same pseudo element(s) then move it to the end | ||
let [trailingPseudo, rebuiltSelector] = commonTrailingPseudos(selector) | ||
if (selector.endsWith('::after')) { | ||
if (selector.startsWith('>')) { | ||
return `> :where(${selector.slice(2, -7)}):not(:where([class~="${prefixedNot}"] *))::after` | ||
} | ||
return `:where(${selector.slice(0, -7)}):not(:where([class~="${prefixedNot}"] *))::after` | ||
if (trailingPseudo) { | ||
return `:where(${selectorPrefix}${rebuiltSelector}):not(:where([class~="${prefixedNot}"],[class~="${prefixedNot}"] *))${trailingPseudo}` | ||
} | ||
if (selector.endsWith('::marker')) { | ||
if (selector.startsWith('>')) { | ||
return `> :where(${selector.slice(2, -8)}):not(:where([class~="${prefixedNot}"] *))::marker` | ||
} | ||
return `:where(${selector.slice(0, -8)}):not(:where([class~="${prefixedNot}"] *))::marker` | ||
} | ||
if (selector.startsWith('>')) { | ||
return `> :where(${selector.slice(2)}):not(:where([class~="${prefixedNot}"] *))` | ||
} | ||
return `:where(${selector}):not(:where([class~="${prefixedNot}"] *))` | ||
return `:where(${selectorPrefix}${selector}):not(:where([class~="${prefixedNot}"],[class~="${prefixedNot}"] *))` | ||
} | ||
@@ -48,3 +34,3 @@ | ||
function configToCss(config = {}, { target, className, prefix }) { | ||
function configToCss(config = {}, { target, className, modifier, prefix }) { | ||
function updateSelector(k, v) { | ||
@@ -62,6 +48,10 @@ if (target === 'legacy') { | ||
if (nested) { | ||
return [k, Object.fromEntries(Object.entries(v).map(([k, v]) => updateSelector(k, v)))] | ||
return [ | ||
inWhere(k, { className, modifier, prefix }), | ||
v, | ||
Object.fromEntries(Object.entries(v).map(([k, v]) => updateSelector(k, v))), | ||
] | ||
} | ||
return [inWhere(k, { className, prefix }), v] | ||
return [inWhere(k, { className, modifier, prefix }), v] | ||
} | ||
@@ -92,5 +82,4 @@ | ||
for (let [name, selector = name] of [ | ||
['headings', 'h1, h2, h3, h4, th'], | ||
['lead', '[class~="lead"]'], | ||
for (let [name, ...selectors] of [ | ||
['headings', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'th'], | ||
['h1'], | ||
@@ -100,2 +89,4 @@ ['h2'], | ||
['h4'], | ||
['h5'], | ||
['h6'], | ||
['p'], | ||
@@ -108,2 +99,3 @@ ['a'], | ||
['em'], | ||
['kbd'], | ||
['code'], | ||
@@ -122,4 +114,13 @@ ['pre'], | ||
['hr'], | ||
['lead', '[class~="lead"]'], | ||
]) { | ||
addVariant(`${className}-${name}`, `& :is(${inWhere(selector, options)})`) | ||
selectors = selectors.length === 0 ? [name] : selectors | ||
let selector = | ||
target === 'legacy' ? selectors.map((selector) => `& ${selector}`) : selectors.join(', ') | ||
addVariant( | ||
`${className}-${name}`, | ||
target === 'legacy' ? selector : `& :is(${inWhere(selector, options)})` | ||
) | ||
} | ||
@@ -134,2 +135,3 @@ | ||
className, | ||
modifier, | ||
prefix, | ||
@@ -136,0 +138,0 @@ } |
@@ -8,4 +8,60 @@ const path = require('path') | ||
let css = String.raw | ||
let javascript = String.raw | ||
let vars = ` | ||
--tw-border-spacing-x: 0; | ||
--tw-border-spacing-y: 0; | ||
--tw-translate-x: 0; | ||
--tw-translate-y: 0; | ||
--tw-rotate: 0; | ||
--tw-skew-x: 0; | ||
--tw-skew-y: 0; | ||
--tw-scale-x: 1; | ||
--tw-scale-y: 1; | ||
--tw-pan-x: ; | ||
--tw-pan-y: ; | ||
--tw-pinch-zoom: ; | ||
--tw-scroll-snap-strictness: proximity; | ||
--tw-ordinal: ; | ||
--tw-slashed-zero: ; | ||
--tw-numeric-figure: ; | ||
--tw-numeric-spacing: ; | ||
--tw-numeric-fraction: ; | ||
--tw-ring-inset: ; | ||
--tw-ring-offset-width: 0px; | ||
--tw-ring-offset-color: #fff; | ||
--tw-ring-color: rgb(59 130 246 / 0.5); | ||
--tw-ring-offset-shadow: 0 0 #0000; | ||
--tw-ring-shadow: 0 0 #0000; | ||
--tw-shadow: 0 0 #0000; | ||
--tw-shadow-colored: 0 0 #0000; | ||
--tw-blur: ; | ||
--tw-brightness: ; | ||
--tw-contrast: ; | ||
--tw-grayscale: ; | ||
--tw-hue-rotate: ; | ||
--tw-invert: ; | ||
--tw-saturate: ; | ||
--tw-sepia: ; | ||
--tw-drop-shadow: ; | ||
--tw-backdrop-blur: ; | ||
--tw-backdrop-brightness: ; | ||
--tw-backdrop-contrast: ; | ||
--tw-backdrop-grayscale: ; | ||
--tw-backdrop-hue-rotate: ; | ||
--tw-backdrop-invert: ; | ||
--tw-backdrop-opacity: ; | ||
--tw-backdrop-saturate: ; | ||
--tw-backdrop-sepia: ; | ||
` | ||
let defaults = css` | ||
*, | ||
::before, | ||
::after { | ||
${vars} | ||
} | ||
::backdrop { | ||
${vars} | ||
} | ||
` | ||
function run(config, plugin = tailwind) { | ||
@@ -80,47 +136,144 @@ let { currentTestName } = expect.getState() | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(ol[type='A']):not(:where([class~='not-prose'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose :where(blockquote p:first-of-type):not(:where([class~='not-prose'] *))::before { | ||
content: open-quote; | ||
} | ||
.prose :where(blockquote p:last-of-type):not(:where([class~='not-prose'] *))::after { | ||
content: close-quote; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose :where(figure > *):not(:where([class~='not-prose'] *)) { | ||
margin: 0; | ||
} | ||
.prose :where(ol > li):not(:where([class~='not-prose'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose > :where(ul > li p):not(:where([class~='not-prose'] *)) { | ||
margin-top: 16px; | ||
margin-bottom: 16px; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'] *))::before { | ||
content: '`'; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'] *))::after { | ||
content: '`'; | ||
} | ||
`) | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(ol[type='A']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose | ||
:where(blockquote p:first-of-type):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::before { | ||
content: open-quote; | ||
} | ||
.prose | ||
:where(blockquote p:last-of-type):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::after { | ||
content: close-quote; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose :where(figure > *):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin: 0; | ||
} | ||
.prose :where(ol > li):not(:where([class~='not-prose'], [class~='not-prose'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose | ||
:where(.prose > ul > li p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin-top: 16px; | ||
margin-bottom: 16px; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'], [class~='not-prose'] *))::before { | ||
content: '`'; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'], [class~='not-prose'] *))::after { | ||
content: '`'; | ||
} | ||
` | ||
) | ||
}) | ||
}) | ||
test('variants', async () => { | ||
let config = { | ||
content: [{ raw: html`<div class="sm:prose hover:prose-lg lg:prose-lg"></div>` }], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: [ | ||
{ | ||
color: 'red', | ||
p: { | ||
color: 'lime', | ||
}, | ||
'> ul > li': { | ||
color: 'purple', | ||
}, | ||
}, | ||
], | ||
}, | ||
lg: { | ||
css: { | ||
color: 'green', | ||
p: { | ||
color: 'tomato', | ||
}, | ||
'> ul > li': { | ||
color: 'blue', | ||
}, | ||
}, | ||
}, | ||
xl: { | ||
css: { | ||
color: 'yellow', | ||
'> ul > li': { | ||
color: 'hotpink', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.hover\:prose-lg:hover { | ||
color: green; | ||
} | ||
.hover\:prose-lg:hover :where(p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: tomato; | ||
} | ||
.hover\:prose-lg:hover | ||
:where(.hover\:prose-lg:hover | ||
> ul | ||
> li):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: blue; | ||
} | ||
@media (min-width: 640px) { | ||
.sm\:prose { | ||
color: red; | ||
} | ||
.sm\:prose :where(p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: lime; | ||
} | ||
.sm\:prose | ||
:where(.sm\:prose > ul > li):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: purple; | ||
} | ||
} | ||
@media (min-width: 1024px) { | ||
.lg\:prose-lg { | ||
color: green; | ||
} | ||
.lg\:prose-lg :where(p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: tomato; | ||
} | ||
.lg\:prose-lg | ||
:where(.lg\:prose-lg > ul > li):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*)) { | ||
color: blue; | ||
} | ||
} | ||
` | ||
) | ||
}) | ||
}) | ||
test('modifiers', async () => { | ||
@@ -187,2 +340,5 @@ let config = { | ||
}, | ||
'> ul > li': { | ||
paddingLeft: '12px', | ||
}, | ||
h1: { | ||
@@ -211,70 +367,83 @@ fontSize: '48px', | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(ol[type='A']):not(:where([class~='not-prose'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose :where(blockquote p:first-of-type):not(:where([class~='not-prose'] *))::before { | ||
content: open-quote; | ||
} | ||
.prose :where(blockquote p:last-of-type):not(:where([class~='not-prose'] *))::after { | ||
content: close-quote; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose :where(figure > *):not(:where([class~='not-prose'] *)) { | ||
margin: 0; | ||
} | ||
.prose :where(ol > li):not(:where([class~='not-prose'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'] *))::before { | ||
content: '`'; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'] *))::after { | ||
content: '`'; | ||
} | ||
.prose-lg { | ||
font-size: 18px; | ||
line-height: 1.75; | ||
} | ||
.prose-lg :where(p):not(:where([class~='not-prose'] *)) { | ||
margin-top: 24px; | ||
margin-bottom: 24px; | ||
} | ||
.prose-lg :where([class~='lead']):not(:where([class~='not-prose'] *)) { | ||
font-size: 22px; | ||
} | ||
.prose-lg :where(blockquote):not(:where([class~='not-prose'] *)) { | ||
margin-top: 40px; | ||
margin-bottom: 40px; | ||
} | ||
.prose-lg :where(h1):not(:where([class~='not-prose'] *)) { | ||
font-size: 48px; | ||
margin-top: 0; | ||
margin-bottom: 40px; | ||
} | ||
.prose-lg :where(h2):not(:where([class~='not-prose'] *)) { | ||
font-size: 30px; | ||
margin-top: 56px; | ||
margin-bottom: 32px; | ||
} | ||
.prose-lg :where(h3):not(:where([class~='not-prose'] *)) { | ||
font-size: 24px; | ||
margin-top: 40px; | ||
margin-bottom: 16px; | ||
} | ||
`) | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(ol[type='A']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose | ||
:where(blockquote p:first-of-type):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::before { | ||
content: open-quote; | ||
} | ||
.prose | ||
:where(blockquote p:last-of-type):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::after { | ||
content: close-quote; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose :where(figure > *):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin: 0; | ||
} | ||
.prose :where(ol > li):not(:where([class~='not-prose'], [class~='not-prose'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'], [class~='not-prose'] *))::before { | ||
content: '`'; | ||
} | ||
.prose :where(code):not(:where([class~='not-prose'], [class~='not-prose'] *))::after { | ||
content: '`'; | ||
} | ||
.prose-lg { | ||
font-size: 18px; | ||
line-height: 1.75; | ||
} | ||
.prose-lg :where(p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin-top: 24px; | ||
margin-bottom: 24px; | ||
} | ||
.prose-lg | ||
:where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-size: 22px; | ||
} | ||
.prose-lg :where(blockquote):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin-top: 40px; | ||
margin-bottom: 40px; | ||
} | ||
.prose-lg | ||
:where(.prose-lg > ul > li):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
padding-left: 12px; | ||
} | ||
.prose-lg :where(h1):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-size: 48px; | ||
margin-top: 0; | ||
margin-bottom: 40px; | ||
} | ||
.prose-lg :where(h2):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-size: 30px; | ||
margin-top: 56px; | ||
margin-bottom: 32px; | ||
} | ||
.prose-lg :where(h3):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-size: 24px; | ||
margin-top: 40px; | ||
margin-bottom: 16px; | ||
} | ||
` | ||
) | ||
}) | ||
@@ -286,3 +455,5 @@ }) | ||
plugins: [typographyPlugin({ target: 'legacy' })], | ||
content: [{ raw: html`<div class="prose"></div>` }], | ||
content: [ | ||
{ raw: html`<div class="prose prose-h1:text-center prose-headings:text-ellipsis"></div>` }, | ||
], | ||
theme: { | ||
@@ -335,40 +506,68 @@ typography: { | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose [class~='lead'] { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose strong { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose ol[type='A'] { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose blockquote p:first-of-type::before { | ||
content: open-quote; | ||
} | ||
.prose blockquote p:last-of-type::after { | ||
content: close-quote; | ||
} | ||
.prose h4 strong { | ||
font-weight: 700; | ||
} | ||
.prose figure > * { | ||
margin: 0; | ||
} | ||
.prose ol > li::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose code::before { | ||
content: '`'; | ||
} | ||
.prose code::after { | ||
content: '`'; | ||
} | ||
`) | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose [class~='lead'] { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose strong { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose ol[type='A'] { | ||
list-style-type: upper-alpha; | ||
} | ||
.prose blockquote p:first-of-type::before { | ||
content: open-quote; | ||
} | ||
.prose blockquote p:last-of-type::after { | ||
content: close-quote; | ||
} | ||
.prose h4 strong { | ||
font-weight: 700; | ||
} | ||
.prose figure > * { | ||
margin: 0; | ||
} | ||
.prose ol > li::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.prose code::before { | ||
content: '`'; | ||
} | ||
.prose code::after { | ||
content: '`'; | ||
} | ||
.prose-headings\:text-ellipsis h1 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis h2 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis h3 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis h4 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis h5 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis h6 { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-headings\:text-ellipsis th { | ||
text-overflow: ellipsis; | ||
} | ||
.prose-h1\:text-center h1 { | ||
text-align: center; | ||
} | ||
` | ||
) | ||
}) | ||
@@ -428,40 +627,56 @@ }) | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
.markdown { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.markdown :where([class~='lead']):not(:where([class~='not-markdown'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.markdown :where(strong):not(:where([class~='not-markdown'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.markdown :where(ol[type='A']):not(:where([class~='not-markdown'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.markdown :where(blockquote p:first-of-type):not(:where([class~='not-markdown'] *))::before { | ||
content: open-quote; | ||
} | ||
.markdown :where(blockquote p:last-of-type):not(:where([class~='not-markdown'] *))::after { | ||
content: close-quote; | ||
} | ||
.markdown :where(h4 strong):not(:where([class~='not-markdown'] *)) { | ||
font-weight: 700; | ||
} | ||
.markdown :where(figure > *):not(:where([class~='not-markdown'] *)) { | ||
margin: 0; | ||
} | ||
.markdown :where(ol > li):not(:where([class~='not-markdown'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.markdown :where(code):not(:where([class~='not-markdown'] *))::before { | ||
content: '`'; | ||
} | ||
.markdown :where(code):not(:where([class~='not-markdown'] *))::after { | ||
content: '`'; | ||
} | ||
`) | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.markdown { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.markdown | ||
:where([class~='lead']):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.markdown :where(strong):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.markdown | ||
:where(ol[type='A']):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
list-style-type: upper-alpha; | ||
} | ||
.markdown | ||
:where(blockquote | ||
p:first-of-type):not(:where([class~='not-markdown'], [class~='not-markdown'] | ||
*))::before { | ||
content: open-quote; | ||
} | ||
.markdown | ||
:where(blockquote | ||
p:last-of-type):not(:where([class~='not-markdown'], [class~='not-markdown'] *))::after { | ||
content: close-quote; | ||
} | ||
.markdown | ||
:where(h4 strong):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
font-weight: 700; | ||
} | ||
.markdown | ||
:where(figure > *):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
margin: 0; | ||
} | ||
.markdown | ||
:where(ol > li):not(:where([class~='not-markdown'], [class~='not-markdown'] *))::marker { | ||
font-weight: 400; | ||
color: var(--tw-prose-counters); | ||
} | ||
.markdown | ||
:where(code):not(:where([class~='not-markdown'], [class~='not-markdown'] *))::before { | ||
content: '`'; | ||
} | ||
.markdown | ||
:where(code):not(:where([class~='not-markdown'], [class~='not-markdown'] *))::after { | ||
content: '`'; | ||
} | ||
` | ||
) | ||
}) | ||
@@ -490,2 +705,3 @@ }) | ||
prose-em:italic | ||
prose-kbd:border-b-2 | ||
prose-code:font-mono | ||
@@ -531,110 +747,136 @@ prose-pre:font-mono | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
:is(:where(hr):not(:where([class~='not-prose'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(229 231 235 / var(--tw-border-opacity)); | ||
} | ||
.prose { | ||
color: var(--tw-prose-body); | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose-headings\:underline | ||
:is(:where(h1, h2, h3, h4, th):not(:where([class~='not-prose'] *))) { | ||
text-decoration: underline; | ||
} | ||
.prose-lead\:italic :is(:where([class~='lead']):not(:where([class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
.prose-h1\:text-3xl :is(:where(h1):not(:where([class~='not-prose'] *))) { | ||
font-size: 1.875rem; | ||
line-height: 2.25rem; | ||
} | ||
.prose-h2\:text-2xl :is(:where(h2):not(:where([class~='not-prose'] *))) { | ||
font-size: 1.5rem; | ||
line-height: 2rem; | ||
} | ||
.prose-h3\:text-xl :is(:where(h3):not(:where([class~='not-prose'] *))) { | ||
font-size: 1.25rem; | ||
line-height: 1.75rem; | ||
} | ||
.prose-h4\:text-lg :is(:where(h4):not(:where([class~='not-prose'] *))) { | ||
font-size: 1.125rem; | ||
line-height: 1.75rem; | ||
} | ||
.prose-p\:text-gray-700 :is(:where(p):not(:where([class~='not-prose'] *))) { | ||
--tw-text-opacity: 1; | ||
color: rgb(55 65 81 / var(--tw-text-opacity)); | ||
} | ||
.prose-a\:font-bold :is(:where(a):not(:where([class~='not-prose'] *))) { | ||
font-weight: 700; | ||
} | ||
.prose-blockquote\:italic :is(:where(blockquote):not(:where([class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
.prose-figure\:mx-auto :is(:where(figure):not(:where([class~='not-prose'] *))) { | ||
margin-left: auto; | ||
margin-right: auto; | ||
} | ||
.prose-figcaption\:opacity-75 :is(:where(figcaption):not(:where([class~='not-prose'] *))) { | ||
opacity: 0.75; | ||
} | ||
.prose-strong\:font-medium :is(:where(strong):not(:where([class~='not-prose'] *))) { | ||
font-weight: 500; | ||
} | ||
.prose-em\:italic :is(:where(em):not(:where([class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
.prose-code\:font-mono :is(:where(code):not(:where([class~='not-prose'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.prose-pre\:font-mono :is(:where(pre):not(:where([class~='not-prose'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.prose-ol\:pl-6 :is(:where(ol):not(:where([class~='not-prose'] *))) { | ||
padding-left: 1.5rem; | ||
} | ||
.prose-ul\:pl-8 :is(:where(ul):not(:where([class~='not-prose'] *))) { | ||
padding-left: 2rem; | ||
} | ||
.prose-li\:my-4 :is(:where(li):not(:where([class~='not-prose'] *))) { | ||
margin-top: 1rem; | ||
margin-bottom: 1rem; | ||
} | ||
.prose-table\:my-8 :is(:where(table):not(:where([class~='not-prose'] *))) { | ||
margin-top: 2rem; | ||
margin-bottom: 2rem; | ||
} | ||
.prose-thead\:border-red-300 :is(:where(thead):not(:where([class~='not-prose'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(252 165 165 / var(--tw-border-opacity)); | ||
} | ||
.prose-tr\:border-red-200 :is(:where(tr):not(:where([class~='not-prose'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(254 202 202 / var(--tw-border-opacity)); | ||
} | ||
.prose-th\:text-left :is(:where(th):not(:where([class~='not-prose'] *))) { | ||
text-align: left; | ||
} | ||
.prose-img\:rounded-lg :is(:where(img):not(:where([class~='not-prose'] *))) { | ||
border-radius: 0.5rem; | ||
} | ||
.prose-video\:my-12 :is(:where(video):not(:where([class~='not-prose'] *))) { | ||
margin-top: 3rem; | ||
margin-bottom: 3rem; | ||
} | ||
.prose-hr\:border-t-2 :is(:where(hr):not(:where([class~='not-prose'] *))) { | ||
border-top-width: 2px; | ||
} | ||
`) | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.prose { | ||
color: var(--tw-prose-body); | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.prose :where(strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.prose :where(h4 strong):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
font-weight: 700; | ||
} | ||
.prose-headings\:underline | ||
:is(:where(h1, h2, h3, h4, h5, h6, th):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))) { | ||
text-decoration-line: underline; | ||
} | ||
.prose-h1\:text-3xl | ||
:is(:where(h1):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 1.875rem; | ||
line-height: 2.25rem; | ||
} | ||
.prose-h2\:text-2xl | ||
:is(:where(h2):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 1.5rem; | ||
line-height: 2rem; | ||
} | ||
.prose-h3\:text-xl | ||
:is(:where(h3):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 1.25rem; | ||
line-height: 1.75rem; | ||
} | ||
.prose-h4\:text-lg | ||
:is(:where(h4):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 1.125rem; | ||
line-height: 1.75rem; | ||
} | ||
.prose-p\:text-gray-700 | ||
:is(:where(p):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
--tw-text-opacity: 1; | ||
color: rgb(55 65 81 / var(--tw-text-opacity)); | ||
} | ||
.prose-a\:font-bold | ||
:is(:where(a):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-weight: 700; | ||
} | ||
.prose-blockquote\:italic | ||
:is(:where(blockquote):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
.prose-figure\:mx-auto | ||
:is(:where(figure):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
margin-left: auto; | ||
margin-right: auto; | ||
} | ||
.prose-figcaption\:opacity-75 | ||
:is(:where(figcaption):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
opacity: 0.75; | ||
} | ||
.prose-strong\:font-medium | ||
:is(:where(strong):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-weight: 500; | ||
} | ||
.prose-em\:italic | ||
:is(:where(em):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
.prose-kbd\:border-b-2 | ||
:is(:where(kbd):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
border-bottom-width: 2px; | ||
} | ||
.prose-code\:font-mono | ||
:is(:where(code):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.prose-pre\:font-mono | ||
:is(:where(pre):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.prose-ol\:pl-6 :is(:where(ol):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
padding-left: 1.5rem; | ||
} | ||
.prose-ul\:pl-8 :is(:where(ul):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
padding-left: 2rem; | ||
} | ||
.prose-li\:my-4 :is(:where(li):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
margin-top: 1rem; | ||
margin-bottom: 1rem; | ||
} | ||
.prose-table\:my-8 | ||
:is(:where(table):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
margin-top: 2rem; | ||
margin-bottom: 2rem; | ||
} | ||
.prose-thead\:border-red-300 | ||
:is(:where(thead):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(252 165 165 / var(--tw-border-opacity)); | ||
} | ||
.prose-tr\:border-red-200 | ||
:is(:where(tr):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(254 202 202 / var(--tw-border-opacity)); | ||
} | ||
.prose-th\:text-left | ||
:is(:where(th):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
text-align: left; | ||
} | ||
.prose-img\:rounded-lg | ||
:is(:where(img):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
border-radius: 0.5rem; | ||
} | ||
.prose-video\:my-12 | ||
:is(:where(video):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
margin-top: 3rem; | ||
margin-bottom: 3rem; | ||
} | ||
.prose-hr\:border-t-2 | ||
:is(:where(hr):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
border-top-width: 2px; | ||
} | ||
.prose-lead\:italic | ||
:is(:where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-style: italic; | ||
} | ||
` | ||
) | ||
}) | ||
@@ -664,2 +906,3 @@ }) | ||
markdown-em:italic | ||
markdown-kbd:border-b-2 | ||
markdown-code:font-mono | ||
@@ -705,110 +948,413 @@ markdown-pre:font-mono | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss(css` | ||
:is(:where(hr):not(:where([class~='not-markdown'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(229 231 235 / var(--tw-border-opacity)); | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.markdown { | ||
color: var(--tw-prose-body); | ||
} | ||
.markdown | ||
:where([class~='lead']):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.markdown :where(strong):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.markdown | ||
:where(h4 strong):not(:where([class~='not-markdown'], [class~='not-markdown'] *)) { | ||
font-weight: 700; | ||
} | ||
.markdown-headings\:underline | ||
:is(:where(h1, h2, h3, h4, h5, h6, th):not(:where([class~='not-markdown'], [class~='not-markdown'] | ||
*))) { | ||
text-decoration-line: underline; | ||
} | ||
.markdown-h1\:text-3xl | ||
:is(:where(h1):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-size: 1.875rem; | ||
line-height: 2.25rem; | ||
} | ||
.markdown-h2\:text-2xl | ||
:is(:where(h2):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-size: 1.5rem; | ||
line-height: 2rem; | ||
} | ||
.markdown-h3\:text-xl | ||
:is(:where(h3):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-size: 1.25rem; | ||
line-height: 1.75rem; | ||
} | ||
.markdown-h4\:text-lg | ||
:is(:where(h4):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-size: 1.125rem; | ||
line-height: 1.75rem; | ||
} | ||
.markdown-p\:text-gray-700 | ||
:is(:where(p):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
--tw-text-opacity: 1; | ||
color: rgb(55 65 81 / var(--tw-text-opacity)); | ||
} | ||
.markdown-a\:font-bold | ||
:is(:where(a):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-weight: 700; | ||
} | ||
.markdown-blockquote\:italic | ||
:is(:where(blockquote):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-style: italic; | ||
} | ||
.markdown-figure\:mx-auto | ||
:is(:where(figure):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
margin-left: auto; | ||
margin-right: auto; | ||
} | ||
.markdown-figcaption\:opacity-75 | ||
:is(:where(figcaption):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
opacity: 0.75; | ||
} | ||
.markdown-strong\:font-medium | ||
:is(:where(strong):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-weight: 500; | ||
} | ||
.markdown-em\:italic | ||
:is(:where(em):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-style: italic; | ||
} | ||
.markdown-kbd\:border-b-2 | ||
:is(:where(kbd):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
border-bottom-width: 2px; | ||
} | ||
.markdown-code\:font-mono | ||
:is(:where(code):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.markdown-pre\:font-mono | ||
:is(:where(pre):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.markdown-ol\:pl-6 | ||
:is(:where(ol):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
padding-left: 1.5rem; | ||
} | ||
.markdown-ul\:pl-8 | ||
:is(:where(ul):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
padding-left: 2rem; | ||
} | ||
.markdown-li\:my-4 | ||
:is(:where(li):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
margin-top: 1rem; | ||
margin-bottom: 1rem; | ||
} | ||
.markdown-table\:my-8 | ||
:is(:where(table):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
margin-top: 2rem; | ||
margin-bottom: 2rem; | ||
} | ||
.markdown-thead\:border-red-300 | ||
:is(:where(thead):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(252 165 165 / var(--tw-border-opacity)); | ||
} | ||
.markdown-tr\:border-red-200 | ||
:is(:where(tr):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(254 202 202 / var(--tw-border-opacity)); | ||
} | ||
.markdown-th\:text-left | ||
:is(:where(th):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
text-align: left; | ||
} | ||
.markdown-img\:rounded-lg | ||
:is(:where(img):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
border-radius: 0.5rem; | ||
} | ||
.markdown-video\:my-12 | ||
:is(:where(video):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
margin-top: 3rem; | ||
margin-bottom: 3rem; | ||
} | ||
.markdown-hr\:border-t-2 | ||
:is(:where(hr):not(:where([class~='not-markdown'], [class~='not-markdown'] *))) { | ||
border-top-width: 2px; | ||
} | ||
.markdown-lead\:italic | ||
:is(:where([class~='lead']):not(:where([class~='not-markdown'], [class~='not-markdown'] | ||
*))) { | ||
font-style: italic; | ||
} | ||
` | ||
) | ||
}) | ||
}) | ||
test('customizing defaults with multiple values does not result in invalid css', async () => { | ||
let config = { | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
textAlign: ['-webkit-match-parent', 'match-parent'], | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toMatchFormattedCss( | ||
css` | ||
${defaults} | ||
.prose { | ||
text-align: -webkit-match-parent; | ||
text-align: match-parent; | ||
} | ||
` | ||
) | ||
}) | ||
}) | ||
it('should be possible to use nested syntax (&) when extending the config', () => { | ||
let config = { | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
extend: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
color: '#000', | ||
a: { | ||
color: '#888', | ||
'&:hover': { | ||
color: '#ff0000', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.prose { | ||
color: #000; | ||
max-width: 65ch; | ||
} | ||
.markdown { | ||
color: var(--tw-prose-body); | ||
} | ||
.markdown :where([class~='lead']):not(:where([class~='not-markdown'] *)) { | ||
color: var(--tw-prose-lead); | ||
} | ||
.markdown :where(strong):not(:where([class~='not-markdown'] *)) { | ||
color: var(--tw-prose-bold); | ||
font-weight: 600; | ||
} | ||
.markdown :where(h4 strong):not(:where([class~='not-markdown'] *)) { | ||
font-weight: 700; | ||
} | ||
.markdown-headings\:underline | ||
:is(:where(h1, h2, h3, h4, th):not(:where([class~='not-markdown'] *))) { | ||
`) | ||
expect(result.css).toIncludeCss(css` | ||
.prose :where(a):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: #888; | ||
text-decoration: underline; | ||
font-weight: 500; | ||
} | ||
.markdown-lead\:italic :is(:where([class~='lead']):not(:where([class~='not-markdown'] *))) { | ||
font-style: italic; | ||
`) | ||
expect(result.css).toIncludeCss(css` | ||
.prose :where(a):not(:where([class~='not-prose'], [class~='not-prose'] *)):hover { | ||
color: #ff0000; | ||
} | ||
.markdown-h1\:text-3xl :is(:where(h1):not(:where([class~='not-markdown'] *))) { | ||
font-size: 1.875rem; | ||
line-height: 2.25rem; | ||
`) | ||
}) | ||
}) | ||
it('should be possible to specify custom h5 and h6 styles', () => { | ||
let config = { | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose prose-h5:text-sm prose-h6:text-xl"></div>`, | ||
}, | ||
], | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.prose-h5\:text-sm :is(:where(h5):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 0.875rem; | ||
line-height: 1.25rem; | ||
} | ||
.markdown-h2\:text-2xl :is(:where(h2):not(:where([class~='not-markdown'] *))) { | ||
font-size: 1.5rem; | ||
line-height: 2rem; | ||
} | ||
.markdown-h3\:text-xl :is(:where(h3):not(:where([class~='not-markdown'] *))) { | ||
.prose-h6\:text-xl :is(:where(h6):not(:where([class~='not-prose'], [class~='not-prose'] *))) { | ||
font-size: 1.25rem; | ||
line-height: 1.75rem; | ||
} | ||
.markdown-h4\:text-lg :is(:where(h4):not(:where([class~='not-markdown'] *))) { | ||
font-size: 1.125rem; | ||
line-height: 1.75rem; | ||
`) | ||
}) | ||
}) | ||
it('should not break with multiple selectors with pseudo elements using variants', () => { | ||
let config = { | ||
darkMode: 'class', | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="dark:prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
'ol li::before, ul li::before': { | ||
color: 'red', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.dark | ||
.dark\:prose | ||
:where(ol li, ul li):not(:where([class~='not-prose'], [class~='not-prose'] *))::before { | ||
color: red; | ||
} | ||
.markdown-p\:text-gray-700 :is(:where(p):not(:where([class~='not-markdown'] *))) { | ||
--tw-text-opacity: 1; | ||
color: rgb(55 65 81 / var(--tw-text-opacity)); | ||
`) | ||
}) | ||
}) | ||
it('lifts all common, trailing pseudo elements when the same across all selectors', () => { | ||
let config = { | ||
darkMode: 'class', | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose dark:prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
'ol li::marker::before, ul li::marker::before': { | ||
color: 'red', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.prose | ||
:where(ol li, ul li):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::marker::before { | ||
color: red; | ||
} | ||
.markdown-a\:font-bold :is(:where(a):not(:where([class~='not-markdown'] *))) { | ||
font-weight: 700; | ||
`) | ||
// TODO: The output here is a bug in tailwindcss variant selector rewriting | ||
// IT should be ::marker::before | ||
expect(result.css).toIncludeCss(css` | ||
.dark | ||
.dark\:prose | ||
:where(ol li, ul li):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::before::marker { | ||
color: red; | ||
} | ||
.markdown-blockquote\:italic :is(:where(blockquote):not(:where([class~='not-markdown'] *))) { | ||
font-style: italic; | ||
`) | ||
}) | ||
}) | ||
it('does not modify selectors with differing pseudo elements', () => { | ||
let config = { | ||
darkMode: 'class', | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose dark:prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
'ol li::before, ul li::after': { | ||
color: 'red', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.prose | ||
:where(ol li::before, ul li::after):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*)) { | ||
color: red; | ||
} | ||
.markdown-figure\:mx-auto :is(:where(figure):not(:where([class~='not-markdown'] *))) { | ||
margin-left: auto; | ||
margin-right: auto; | ||
`) | ||
// TODO: The output here is a bug in tailwindcss variant selector rewriting | ||
expect(result.css).toIncludeCss(css` | ||
.dark | ||
.dark\:prose | ||
:where(ol li, ul li):not(:where([class~='not-prose'], [class~='not-prose'] *))::before, | ||
::after { | ||
color: red; | ||
} | ||
.markdown-figcaption\:opacity-75 | ||
:is(:where(figcaption):not(:where([class~='not-markdown'] *))) { | ||
opacity: 0.75; | ||
`) | ||
}) | ||
}) | ||
it('lifts only the common, trailing pseudo elements from selectors', () => { | ||
let config = { | ||
darkMode: 'class', | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose dark:prose"></div>`, | ||
}, | ||
], | ||
theme: { | ||
typography: { | ||
DEFAULT: { | ||
css: { | ||
'ol li::scroll-thumb::before, ul li::scroll-track::before': { | ||
color: 'red', | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss(css` | ||
.prose | ||
:where(ol li::scroll-thumb, ul | ||
li::scroll-track):not(:where([class~='not-prose'], [class~='not-prose'] *))::before { | ||
color: red; | ||
} | ||
.markdown-strong\:font-medium :is(:where(strong):not(:where([class~='not-markdown'] *))) { | ||
font-weight: 500; | ||
`) | ||
// TODO: The output here is a bug in tailwindcss variant selector rewriting | ||
expect(result.css).toIncludeCss(css` | ||
.dark | ||
.dark\:prose | ||
:where(ol li, ul li):not(:where([class~='not-prose'], [class~='not-prose'] | ||
*))::scroll-thumb, | ||
::scroll-track, | ||
::before { | ||
color: red; | ||
} | ||
.markdown-em\:italic :is(:where(em):not(:where([class~='not-markdown'] *))) { | ||
font-style: italic; | ||
} | ||
.markdown-code\:font-mono :is(:where(code):not(:where([class~='not-markdown'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.markdown-pre\:font-mono :is(:where(pre):not(:where([class~='not-markdown'] *))) { | ||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', | ||
'Courier New', monospace; | ||
} | ||
.markdown-ol\:pl-6 :is(:where(ol):not(:where([class~='not-markdown'] *))) { | ||
padding-left: 1.5rem; | ||
} | ||
.markdown-ul\:pl-8 :is(:where(ul):not(:where([class~='not-markdown'] *))) { | ||
padding-left: 2rem; | ||
} | ||
.markdown-li\:my-4 :is(:where(li):not(:where([class~='not-markdown'] *))) { | ||
margin-top: 1rem; | ||
margin-bottom: 1rem; | ||
} | ||
.markdown-table\:my-8 :is(:where(table):not(:where([class~='not-markdown'] *))) { | ||
margin-top: 2rem; | ||
margin-bottom: 2rem; | ||
} | ||
.markdown-thead\:border-red-300 :is(:where(thead):not(:where([class~='not-markdown'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(252 165 165 / var(--tw-border-opacity)); | ||
} | ||
.markdown-tr\:border-red-200 :is(:where(tr):not(:where([class~='not-markdown'] *))) { | ||
--tw-border-opacity: 1; | ||
border-color: rgb(254 202 202 / var(--tw-border-opacity)); | ||
} | ||
.markdown-th\:text-left :is(:where(th):not(:where([class~='not-markdown'] *))) { | ||
text-align: left; | ||
} | ||
.markdown-img\:rounded-lg :is(:where(img):not(:where([class~='not-markdown'] *))) { | ||
border-radius: 0.5rem; | ||
} | ||
.markdown-video\:my-12 :is(:where(video):not(:where([class~='not-markdown'] *))) { | ||
margin-top: 3rem; | ||
margin-bottom: 3rem; | ||
} | ||
.markdown-hr\:border-t-2 :is(:where(hr):not(:where([class~='not-markdown'] *))) { | ||
border-top-width: 2px; | ||
} | ||
`) | ||
@@ -818,8 +1364,9 @@ }) | ||
test('customizing defaults with multiple values does not result in invalid css', async () => { | ||
it('ignores common non-trailing pseudo-elements in selectors', () => { | ||
let config = { | ||
darkMode: 'class', | ||
plugins: [typographyPlugin()], | ||
content: [ | ||
{ | ||
raw: html`<div class="prose"></div>`, | ||
raw: html`<div class="prose dark:prose"></div>`, | ||
}, | ||
@@ -831,3 +1378,5 @@ ], | ||
css: { | ||
textAlign: ['-webkit-match-parent', 'match-parent'], | ||
'ol li::before::scroll-thumb, ul li::before::scroll-track': { | ||
color: 'red', | ||
}, | ||
}, | ||
@@ -838,10 +1387,51 @@ }, | ||
} | ||
return run(config).then((result) => { | ||
// TODO: Fix this test. It should list both properties but there's a bug in tailwind that's overriding them. | ||
expect(result.css).toMatchFormattedCss(css` | ||
.prose { | ||
text-align: match-parent; | ||
expect(result.css).toIncludeCss(css` | ||
.prose | ||
:where(ol li::before::scroll-thumb, ul | ||
li::before::scroll-track):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: red; | ||
} | ||
`) | ||
// TODO: The output here is a bug in tailwindcss variant selector rewriting | ||
expect(result.css).toIncludeCss(css` | ||
.dark | ||
.dark\:prose | ||
:where(ol li::scroll-thumb, ul | ||
li::scroll-track):not(:where([class~='not-prose'], [class~='not-prose'] *))::before, | ||
::before { | ||
color: red; | ||
} | ||
`) | ||
}) | ||
}) | ||
test('lead styles are inserted after paragraph styles', async () => { | ||
let config = { | ||
content: [{ raw: html`<div class="prose"></div>` }], | ||
} | ||
return run(config).then((result) => { | ||
expect(result.css).toIncludeCss( | ||
css` | ||
.prose { | ||
color: var(--tw-prose-body); | ||
max-width: 65ch; | ||
} | ||
.prose :where(p):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
margin-top: 1.25em; | ||
margin-bottom: 1.25em; | ||
} | ||
.prose :where([class~='lead']):not(:where([class~='not-prose'], [class~='not-prose'] *)) { | ||
color: var(--tw-prose-lead); | ||
font-size: 1.25em; | ||
line-height: 1.6; | ||
margin-top: 1.2em; | ||
margin-bottom: 1.2em; | ||
} | ||
` | ||
) | ||
}) | ||
}) |
@@ -10,2 +10,10 @@ const colors = require('tailwindcss/colors') | ||
const em = (px, base) => `${round(px / base)}em` | ||
const hexToRgb = (hex) => { | ||
hex = hex.replace('#', '') | ||
hex = hex.length === 3 ? hex.replace(/./g, '$&$&') : hex | ||
const r = parseInt(hex.substring(0, 2), 16) | ||
const g = parseInt(hex.substring(2, 4), 16) | ||
const b = parseInt(hex.substring(4, 6), 16) | ||
return `${r} ${g} ${b}` | ||
} | ||
@@ -31,3 +39,3 @@ let defaultModifiers = { | ||
marginBottom: em(24, 18), | ||
paddingLeft: em(20, 18), | ||
paddingInlineStart: em(20, 18), | ||
}, | ||
@@ -61,18 +69,21 @@ h1: { | ||
}, | ||
video: { | ||
picture: { | ||
marginTop: em(24, 14), | ||
marginBottom: em(24, 14), | ||
}, | ||
figure: { | ||
'picture > img': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
video: { | ||
marginTop: em(24, 14), | ||
marginBottom: em(24, 14), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
kbd: { | ||
fontSize: em(12, 14), | ||
lineHeight: round(16 / 12), | ||
marginTop: em(8, 12), | ||
borderRadius: rem(5), | ||
paddingTop: em(2, 14), | ||
paddingInlineEnd: em(5, 14), | ||
paddingBottom: em(2, 14), | ||
paddingInlineStart: em(5, 14), | ||
}, | ||
@@ -95,5 +106,5 @@ code: { | ||
paddingTop: em(8, 12), | ||
paddingRight: em(12, 12), | ||
paddingInlineEnd: em(12, 12), | ||
paddingBottom: em(8, 12), | ||
paddingLeft: em(12, 12), | ||
paddingInlineStart: em(12, 12), | ||
}, | ||
@@ -103,2 +114,3 @@ ol: { | ||
marginBottom: em(16, 14), | ||
paddingInlineStart: em(22, 14), | ||
}, | ||
@@ -108,2 +120,3 @@ ul: { | ||
marginBottom: em(16, 14), | ||
paddingInlineStart: em(22, 14), | ||
}, | ||
@@ -114,13 +127,7 @@ li: { | ||
}, | ||
ol: { | ||
paddingLeft: em(22, 14), | ||
}, | ||
'ol > li': { | ||
paddingLeft: em(6, 14), | ||
paddingInlineStart: em(6, 14), | ||
}, | ||
ul: { | ||
paddingLeft: em(22, 14), | ||
}, | ||
'ul > li': { | ||
paddingLeft: em(6, 14), | ||
paddingInlineStart: em(6, 14), | ||
}, | ||
@@ -131,12 +138,12 @@ '> ul > li p': { | ||
}, | ||
'> ul > li > *:first-child': { | ||
'> ul > li > p:first-child': { | ||
marginTop: em(16, 14), | ||
}, | ||
'> ul > li > *:last-child': { | ||
'> ul > li > p:last-child': { | ||
marginBottom: em(16, 14), | ||
}, | ||
'> ol > li > *:first-child': { | ||
'> ol > li > p:first-child': { | ||
marginTop: em(16, 14), | ||
}, | ||
'> ol > li > *:last-child': { | ||
'> ol > li > p:last-child': { | ||
marginBottom: em(16, 14), | ||
@@ -148,2 +155,13 @@ }, | ||
}, | ||
dl: { | ||
marginTop: em(16, 14), | ||
marginBottom: em(16, 14), | ||
}, | ||
dt: { | ||
marginTop: em(16, 14), | ||
}, | ||
dd: { | ||
marginTop: em(4, 14), | ||
paddingInlineStart: em(22, 14), | ||
}, | ||
hr: { | ||
@@ -170,24 +188,37 @@ marginTop: em(40, 14), | ||
'thead th': { | ||
paddingRight: em(12, 12), | ||
paddingInlineEnd: em(12, 12), | ||
paddingBottom: em(8, 12), | ||
paddingLeft: em(12, 12), | ||
paddingInlineStart: em(12, 12), | ||
}, | ||
'thead th:first-child': { | ||
paddingLeft: '0', | ||
paddingInlineStart: '0', | ||
}, | ||
'thead th:last-child': { | ||
paddingRight: '0', | ||
paddingInlineEnd: '0', | ||
}, | ||
'tbody td': { | ||
'tbody td, tfoot td': { | ||
paddingTop: em(8, 12), | ||
paddingRight: em(12, 12), | ||
paddingInlineEnd: em(12, 12), | ||
paddingBottom: em(8, 12), | ||
paddingLeft: em(12, 12), | ||
paddingInlineStart: em(12, 12), | ||
}, | ||
'tbody td:first-child': { | ||
paddingLeft: '0', | ||
'tbody td:first-child, tfoot td:first-child': { | ||
paddingInlineStart: '0', | ||
}, | ||
'tbody td:last-child': { | ||
paddingRight: '0', | ||
'tbody td:last-child, tfoot td:last-child': { | ||
paddingInlineEnd: '0', | ||
}, | ||
figure: { | ||
marginTop: em(24, 14), | ||
marginBottom: em(24, 14), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
fontSize: em(12, 14), | ||
lineHeight: round(16 / 12), | ||
marginTop: em(8, 12), | ||
}, | ||
}, | ||
@@ -222,3 +253,3 @@ { | ||
marginBottom: em(32, 20), | ||
paddingLeft: em(20, 20), | ||
paddingInlineStart: em(20, 20), | ||
}, | ||
@@ -252,18 +283,21 @@ h1: { | ||
}, | ||
video: { | ||
picture: { | ||
marginTop: em(32, 16), | ||
marginBottom: em(32, 16), | ||
}, | ||
figure: { | ||
'picture > img': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
video: { | ||
marginTop: em(32, 16), | ||
marginBottom: em(32, 16), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
kbd: { | ||
fontSize: em(14, 16), | ||
lineHeight: round(20 / 14), | ||
marginTop: em(12, 14), | ||
borderRadius: rem(5), | ||
paddingTop: em(3, 16), | ||
paddingInlineEnd: em(6, 16), | ||
paddingBottom: em(3, 16), | ||
paddingInlineStart: em(6, 16), | ||
}, | ||
@@ -286,5 +320,5 @@ code: { | ||
paddingTop: em(12, 14), | ||
paddingRight: em(16, 14), | ||
paddingInlineEnd: em(16, 14), | ||
paddingBottom: em(12, 14), | ||
paddingLeft: em(16, 14), | ||
paddingInlineStart: em(16, 14), | ||
}, | ||
@@ -294,2 +328,3 @@ ol: { | ||
marginBottom: em(20, 16), | ||
paddingInlineStart: em(26, 16), | ||
}, | ||
@@ -299,2 +334,3 @@ ul: { | ||
marginBottom: em(20, 16), | ||
paddingInlineStart: em(26, 16), | ||
}, | ||
@@ -305,13 +341,7 @@ li: { | ||
}, | ||
ol: { | ||
paddingLeft: em(26, 16), | ||
}, | ||
'ol > li': { | ||
paddingLeft: em(6, 16), | ||
paddingInlineStart: em(6, 16), | ||
}, | ||
ul: { | ||
paddingLeft: em(26, 16), | ||
}, | ||
'ul > li': { | ||
paddingLeft: em(6, 16), | ||
paddingInlineStart: em(6, 16), | ||
}, | ||
@@ -322,12 +352,12 @@ '> ul > li p': { | ||
}, | ||
'> ul > li > *:first-child': { | ||
'> ul > li > p:first-child': { | ||
marginTop: em(20, 16), | ||
}, | ||
'> ul > li > *:last-child': { | ||
'> ul > li > p:last-child': { | ||
marginBottom: em(20, 16), | ||
}, | ||
'> ol > li > *:first-child': { | ||
'> ol > li > p:first-child': { | ||
marginTop: em(20, 16), | ||
}, | ||
'> ol > li > *:last-child': { | ||
'> ol > li > p:last-child': { | ||
marginBottom: em(20, 16), | ||
@@ -339,2 +369,13 @@ }, | ||
}, | ||
dl: { | ||
marginTop: em(20, 16), | ||
marginBottom: em(20, 16), | ||
}, | ||
dt: { | ||
marginTop: em(20, 16), | ||
}, | ||
dd: { | ||
marginTop: em(8, 16), | ||
paddingInlineStart: em(26, 16), | ||
}, | ||
hr: { | ||
@@ -361,24 +402,37 @@ marginTop: em(48, 16), | ||
'thead th': { | ||
paddingRight: em(8, 14), | ||
paddingInlineEnd: em(8, 14), | ||
paddingBottom: em(8, 14), | ||
paddingLeft: em(8, 14), | ||
paddingInlineStart: em(8, 14), | ||
}, | ||
'thead th:first-child': { | ||
paddingLeft: '0', | ||
paddingInlineStart: '0', | ||
}, | ||
'thead th:last-child': { | ||
paddingRight: '0', | ||
paddingInlineEnd: '0', | ||
}, | ||
'tbody td': { | ||
'tbody td, tfoot td': { | ||
paddingTop: em(8, 14), | ||
paddingRight: em(8, 14), | ||
paddingInlineEnd: em(8, 14), | ||
paddingBottom: em(8, 14), | ||
paddingLeft: em(8, 14), | ||
paddingInlineStart: em(8, 14), | ||
}, | ||
'tbody td:first-child': { | ||
paddingLeft: '0', | ||
'tbody td:first-child, tfoot td:first-child': { | ||
paddingInlineStart: '0', | ||
}, | ||
'tbody td:last-child': { | ||
paddingRight: '0', | ||
'tbody td:last-child, tfoot td:last-child': { | ||
paddingInlineEnd: '0', | ||
}, | ||
figure: { | ||
marginTop: em(32, 16), | ||
marginBottom: em(32, 16), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
fontSize: em(14, 16), | ||
lineHeight: round(20 / 14), | ||
marginTop: em(12, 14), | ||
}, | ||
}, | ||
@@ -413,3 +467,3 @@ { | ||
marginBottom: em(40, 24), | ||
paddingLeft: em(24, 24), | ||
paddingInlineStart: em(24, 24), | ||
}, | ||
@@ -443,18 +497,21 @@ h1: { | ||
}, | ||
video: { | ||
picture: { | ||
marginTop: em(32, 18), | ||
marginBottom: em(32, 18), | ||
}, | ||
figure: { | ||
'picture > img': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
video: { | ||
marginTop: em(32, 18), | ||
marginBottom: em(32, 18), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
kbd: { | ||
fontSize: em(16, 18), | ||
lineHeight: round(24 / 16), | ||
marginTop: em(16, 16), | ||
borderRadius: rem(5), | ||
paddingTop: em(4, 18), | ||
paddingInlineEnd: em(8, 18), | ||
paddingBottom: em(4, 18), | ||
paddingInlineStart: em(8, 18), | ||
}, | ||
@@ -477,5 +534,5 @@ code: { | ||
paddingTop: em(16, 16), | ||
paddingRight: em(24, 16), | ||
paddingInlineEnd: em(24, 16), | ||
paddingBottom: em(16, 16), | ||
paddingLeft: em(24, 16), | ||
paddingInlineStart: em(24, 16), | ||
}, | ||
@@ -485,2 +542,3 @@ ol: { | ||
marginBottom: em(24, 18), | ||
paddingInlineStart: em(28, 18), | ||
}, | ||
@@ -490,2 +548,3 @@ ul: { | ||
marginBottom: em(24, 18), | ||
paddingInlineStart: em(28, 18), | ||
}, | ||
@@ -496,13 +555,7 @@ li: { | ||
}, | ||
ol: { | ||
paddingLeft: em(28, 18), | ||
}, | ||
'ol > li': { | ||
paddingLeft: em(8, 18), | ||
paddingInlineStart: em(8, 18), | ||
}, | ||
ul: { | ||
paddingLeft: em(28, 18), | ||
}, | ||
'ul > li': { | ||
paddingLeft: em(8, 18), | ||
paddingInlineStart: em(8, 18), | ||
}, | ||
@@ -513,12 +566,12 @@ '> ul > li p': { | ||
}, | ||
'> ul > li > *:first-child': { | ||
'> ul > li > p:first-child': { | ||
marginTop: em(24, 18), | ||
}, | ||
'> ul > li > *:last-child': { | ||
'> ul > li > p:last-child': { | ||
marginBottom: em(24, 18), | ||
}, | ||
'> ol > li > *:first-child': { | ||
'> ol > li > p:first-child': { | ||
marginTop: em(24, 18), | ||
}, | ||
'> ol > li > *:last-child': { | ||
'> ol > li > p:last-child': { | ||
marginBottom: em(24, 18), | ||
@@ -530,2 +583,13 @@ }, | ||
}, | ||
dl: { | ||
marginTop: em(24, 18), | ||
marginBottom: em(24, 18), | ||
}, | ||
dt: { | ||
marginTop: em(24, 18), | ||
}, | ||
dd: { | ||
marginTop: em(12, 18), | ||
paddingInlineStart: em(28, 18), | ||
}, | ||
hr: { | ||
@@ -552,24 +616,37 @@ marginTop: em(56, 18), | ||
'thead th': { | ||
paddingRight: em(12, 16), | ||
paddingInlineEnd: em(12, 16), | ||
paddingBottom: em(12, 16), | ||
paddingLeft: em(12, 16), | ||
paddingInlineStart: em(12, 16), | ||
}, | ||
'thead th:first-child': { | ||
paddingLeft: '0', | ||
paddingInlineStart: '0', | ||
}, | ||
'thead th:last-child': { | ||
paddingRight: '0', | ||
paddingInlineEnd: '0', | ||
}, | ||
'tbody td': { | ||
'tbody td, tfoot td': { | ||
paddingTop: em(12, 16), | ||
paddingRight: em(12, 16), | ||
paddingInlineEnd: em(12, 16), | ||
paddingBottom: em(12, 16), | ||
paddingLeft: em(12, 16), | ||
paddingInlineStart: em(12, 16), | ||
}, | ||
'tbody td:first-child': { | ||
paddingLeft: '0', | ||
'tbody td:first-child, tfoot td:first-child': { | ||
paddingInlineStart: '0', | ||
}, | ||
'tbody td:last-child': { | ||
paddingRight: '0', | ||
'tbody td:last-child, tfoot td:last-child': { | ||
paddingInlineEnd: '0', | ||
}, | ||
figure: { | ||
marginTop: em(32, 18), | ||
marginBottom: em(32, 18), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
fontSize: em(16, 18), | ||
lineHeight: round(24 / 16), | ||
marginTop: em(16, 16), | ||
}, | ||
}, | ||
@@ -604,3 +681,3 @@ { | ||
marginBottom: em(48, 30), | ||
paddingLeft: em(32, 30), | ||
paddingInlineStart: em(32, 30), | ||
}, | ||
@@ -634,18 +711,21 @@ h1: { | ||
}, | ||
video: { | ||
picture: { | ||
marginTop: em(40, 20), | ||
marginBottom: em(40, 20), | ||
}, | ||
figure: { | ||
'picture > img': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
video: { | ||
marginTop: em(40, 20), | ||
marginBottom: em(40, 20), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
kbd: { | ||
fontSize: em(18, 20), | ||
lineHeight: round(28 / 18), | ||
marginTop: em(18, 18), | ||
borderRadius: rem(5), | ||
paddingTop: em(5, 20), | ||
paddingInlineEnd: em(8, 20), | ||
paddingBottom: em(5, 20), | ||
paddingInlineStart: em(8, 20), | ||
}, | ||
@@ -668,5 +748,5 @@ code: { | ||
paddingTop: em(20, 18), | ||
paddingRight: em(24, 18), | ||
paddingInlineEnd: em(24, 18), | ||
paddingBottom: em(20, 18), | ||
paddingLeft: em(24, 18), | ||
paddingInlineStart: em(24, 18), | ||
}, | ||
@@ -676,2 +756,3 @@ ol: { | ||
marginBottom: em(24, 20), | ||
paddingInlineStart: em(32, 20), | ||
}, | ||
@@ -681,2 +762,3 @@ ul: { | ||
marginBottom: em(24, 20), | ||
paddingInlineStart: em(32, 20), | ||
}, | ||
@@ -687,13 +769,7 @@ li: { | ||
}, | ||
ol: { | ||
paddingLeft: em(32, 20), | ||
}, | ||
'ol > li': { | ||
paddingLeft: em(8, 20), | ||
paddingInlineStart: em(8, 20), | ||
}, | ||
ul: { | ||
paddingLeft: em(32, 20), | ||
}, | ||
'ul > li': { | ||
paddingLeft: em(8, 20), | ||
paddingInlineStart: em(8, 20), | ||
}, | ||
@@ -704,12 +780,12 @@ '> ul > li p': { | ||
}, | ||
'> ul > li > *:first-child': { | ||
'> ul > li > p:first-child': { | ||
marginTop: em(24, 20), | ||
}, | ||
'> ul > li > *:last-child': { | ||
'> ul > li > p:last-child': { | ||
marginBottom: em(24, 20), | ||
}, | ||
'> ol > li > *:first-child': { | ||
'> ol > li > p:first-child': { | ||
marginTop: em(24, 20), | ||
}, | ||
'> ol > li > *:last-child': { | ||
'> ol > li > p:last-child': { | ||
marginBottom: em(24, 20), | ||
@@ -721,2 +797,13 @@ }, | ||
}, | ||
dl: { | ||
marginTop: em(24, 20), | ||
marginBottom: em(24, 20), | ||
}, | ||
dt: { | ||
marginTop: em(24, 20), | ||
}, | ||
dd: { | ||
marginTop: em(12, 20), | ||
paddingInlineStart: em(32, 20), | ||
}, | ||
hr: { | ||
@@ -743,24 +830,37 @@ marginTop: em(56, 20), | ||
'thead th': { | ||
paddingRight: em(12, 18), | ||
paddingInlineEnd: em(12, 18), | ||
paddingBottom: em(16, 18), | ||
paddingLeft: em(12, 18), | ||
paddingInlineStart: em(12, 18), | ||
}, | ||
'thead th:first-child': { | ||
paddingLeft: '0', | ||
paddingInlineStart: '0', | ||
}, | ||
'thead th:last-child': { | ||
paddingRight: '0', | ||
paddingInlineEnd: '0', | ||
}, | ||
'tbody td': { | ||
'tbody td, tfoot td': { | ||
paddingTop: em(16, 18), | ||
paddingRight: em(12, 18), | ||
paddingInlineEnd: em(12, 18), | ||
paddingBottom: em(16, 18), | ||
paddingLeft: em(12, 18), | ||
paddingInlineStart: em(12, 18), | ||
}, | ||
'tbody td:first-child': { | ||
paddingLeft: '0', | ||
'tbody td:first-child, tfoot td:first-child': { | ||
paddingInlineStart: '0', | ||
}, | ||
'tbody td:last-child': { | ||
paddingRight: '0', | ||
'tbody td:last-child, tfoot td:last-child': { | ||
paddingInlineEnd: '0', | ||
}, | ||
figure: { | ||
marginTop: em(40, 20), | ||
marginBottom: em(40, 20), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
fontSize: em(18, 20), | ||
lineHeight: round(28 / 18), | ||
marginTop: em(18, 18), | ||
}, | ||
}, | ||
@@ -795,3 +895,3 @@ { | ||
marginBottom: em(64, 36), | ||
paddingLeft: em(40, 36), | ||
paddingInlineStart: em(40, 36), | ||
}, | ||
@@ -825,18 +925,21 @@ h1: { | ||
}, | ||
video: { | ||
picture: { | ||
marginTop: em(48, 24), | ||
marginBottom: em(48, 24), | ||
}, | ||
figure: { | ||
'picture > img': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
video: { | ||
marginTop: em(48, 24), | ||
marginBottom: em(48, 24), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
kbd: { | ||
fontSize: em(20, 24), | ||
lineHeight: round(32 / 20), | ||
marginTop: em(20, 20), | ||
borderRadius: rem(6), | ||
paddingTop: em(6, 24), | ||
paddingInlineEnd: em(8, 24), | ||
paddingBottom: em(6, 24), | ||
paddingInlineStart: em(8, 24), | ||
}, | ||
@@ -859,5 +962,5 @@ code: { | ||
paddingTop: em(24, 20), | ||
paddingRight: em(32, 20), | ||
paddingInlineEnd: em(32, 20), | ||
paddingBottom: em(24, 20), | ||
paddingLeft: em(32, 20), | ||
paddingInlineStart: em(32, 20), | ||
}, | ||
@@ -867,2 +970,3 @@ ol: { | ||
marginBottom: em(32, 24), | ||
paddingInlineStart: em(38, 24), | ||
}, | ||
@@ -872,2 +976,3 @@ ul: { | ||
marginBottom: em(32, 24), | ||
paddingInlineStart: em(38, 24), | ||
}, | ||
@@ -878,13 +983,7 @@ li: { | ||
}, | ||
ol: { | ||
paddingLeft: em(38, 24), | ||
}, | ||
'ol > li': { | ||
paddingLeft: em(10, 24), | ||
paddingInlineStart: em(10, 24), | ||
}, | ||
ul: { | ||
paddingLeft: em(38, 24), | ||
}, | ||
'ul > li': { | ||
paddingLeft: em(10, 24), | ||
paddingInlineStart: em(10, 24), | ||
}, | ||
@@ -895,12 +994,12 @@ '> ul > li p': { | ||
}, | ||
'> ul > li > *:first-child': { | ||
'> ul > li > p:first-child': { | ||
marginTop: em(32, 24), | ||
}, | ||
'> ul > li > *:last-child': { | ||
'> ul > li > p:last-child': { | ||
marginBottom: em(32, 24), | ||
}, | ||
'> ol > li > *:first-child': { | ||
'> ol > li > p:first-child': { | ||
marginTop: em(32, 24), | ||
}, | ||
'> ol > li > *:last-child': { | ||
'> ol > li > p:last-child': { | ||
marginBottom: em(32, 24), | ||
@@ -912,2 +1011,13 @@ }, | ||
}, | ||
dl: { | ||
marginTop: em(32, 24), | ||
marginBottom: em(32, 24), | ||
}, | ||
dt: { | ||
marginTop: em(32, 24), | ||
}, | ||
dd: { | ||
marginTop: em(12, 24), | ||
paddingInlineStart: em(38, 24), | ||
}, | ||
hr: { | ||
@@ -934,24 +1044,37 @@ marginTop: em(72, 24), | ||
'thead th': { | ||
paddingRight: em(12, 20), | ||
paddingInlineEnd: em(12, 20), | ||
paddingBottom: em(16, 20), | ||
paddingLeft: em(12, 20), | ||
paddingInlineStart: em(12, 20), | ||
}, | ||
'thead th:first-child': { | ||
paddingLeft: '0', | ||
paddingInlineStart: '0', | ||
}, | ||
'thead th:last-child': { | ||
paddingRight: '0', | ||
paddingInlineEnd: '0', | ||
}, | ||
'tbody td': { | ||
'tbody td, tfoot td': { | ||
paddingTop: em(16, 20), | ||
paddingRight: em(12, 20), | ||
paddingInlineEnd: em(12, 20), | ||
paddingBottom: em(16, 20), | ||
paddingLeft: em(12, 20), | ||
paddingInlineStart: em(12, 20), | ||
}, | ||
'tbody td:first-child': { | ||
paddingLeft: '0', | ||
'tbody td:first-child, tfoot td:first-child': { | ||
paddingInlineStart: '0', | ||
}, | ||
'tbody td:last-child': { | ||
paddingRight: '0', | ||
'tbody td:last-child, tfoot td:last-child': { | ||
paddingInlineEnd: '0', | ||
}, | ||
figure: { | ||
marginTop: em(48, 24), | ||
marginBottom: em(48, 24), | ||
}, | ||
'figure > *': { | ||
marginTop: '0', | ||
marginBottom: '0', | ||
}, | ||
figcaption: { | ||
fontSize: em(20, 24), | ||
lineHeight: round(32 / 20), | ||
marginTop: em(20, 20), | ||
}, | ||
}, | ||
@@ -969,24 +1092,2 @@ { | ||
// Invert (for dark mode) | ||
invert: { | ||
css: { | ||
'--tw-prose-body': 'var(--tw-prose-invert-body)', | ||
'--tw-prose-headings': 'var(--tw-prose-invert-headings)', | ||
'--tw-prose-lead': 'var(--tw-prose-invert-lead)', | ||
'--tw-prose-links': 'var(--tw-prose-invert-links)', | ||
'--tw-prose-bold': 'var(--tw-prose-invert-bold)', | ||
'--tw-prose-counters': 'var(--tw-prose-invert-counters)', | ||
'--tw-prose-bullets': 'var(--tw-prose-invert-bullets)', | ||
'--tw-prose-hr': 'var(--tw-prose-invert-hr)', | ||
'--tw-prose-quotes': 'var(--tw-prose-invert-quotes)', | ||
'--tw-prose-quote-borders': 'var(--tw-prose-invert-quote-borders)', | ||
'--tw-prose-captions': 'var(--tw-prose-invert-captions)', | ||
'--tw-prose-code': 'var(--tw-prose-invert-code)', | ||
'--tw-prose-pre-code': 'var(--tw-prose-invert-pre-code)', | ||
'--tw-prose-pre-bg': 'var(--tw-prose-invert-pre-bg)', | ||
'--tw-prose-th-borders': 'var(--tw-prose-invert-th-borders)', | ||
'--tw-prose-td-borders': 'var(--tw-prose-invert-td-borders)', | ||
}, | ||
}, | ||
// Gray color themes | ||
@@ -1007,2 +1108,4 @@ | ||
'--tw-prose-captions': colors.slate[500], | ||
'--tw-prose-kbd': colors.slate[900], | ||
'--tw-prose-kbd-shadows': hexToRgb(colors.slate[900]), | ||
'--tw-prose-code': colors.slate[900], | ||
@@ -1024,2 +1127,4 @@ '--tw-prose-pre-code': colors.slate[200], | ||
'--tw-prose-invert-captions': colors.slate[400], | ||
'--tw-prose-invert-kbd': colors.white, | ||
'--tw-prose-invert-kbd-shadows': hexToRgb(colors.white), | ||
'--tw-prose-invert-code': colors.white, | ||
@@ -1046,2 +1151,4 @@ '--tw-prose-invert-pre-code': colors.slate[300], | ||
'--tw-prose-captions': colors.gray[500], | ||
'--tw-prose-kbd': colors.gray[900], | ||
'--tw-prose-kbd-shadows': hexToRgb(colors.gray[900]), | ||
'--tw-prose-code': colors.gray[900], | ||
@@ -1063,2 +1170,4 @@ '--tw-prose-pre-code': colors.gray[200], | ||
'--tw-prose-invert-captions': colors.gray[400], | ||
'--tw-prose-invert-kbd': colors.white, | ||
'--tw-prose-invert-kbd-shadows': hexToRgb(colors.white), | ||
'--tw-prose-invert-code': colors.white, | ||
@@ -1085,2 +1194,4 @@ '--tw-prose-invert-pre-code': colors.gray[300], | ||
'--tw-prose-captions': colors.zinc[500], | ||
'--tw-prose-kbd': colors.zinc[900], | ||
'--tw-prose-kbd-shadows': hexToRgb(colors.zinc[900]), | ||
'--tw-prose-code': colors.zinc[900], | ||
@@ -1102,2 +1213,4 @@ '--tw-prose-pre-code': colors.zinc[200], | ||
'--tw-prose-invert-captions': colors.zinc[400], | ||
'--tw-prose-invert-kbd': colors.white, | ||
'--tw-prose-invert-kbd-shadows': hexToRgb(colors.white), | ||
'--tw-prose-invert-code': colors.white, | ||
@@ -1124,2 +1237,4 @@ '--tw-prose-invert-pre-code': colors.zinc[300], | ||
'--tw-prose-captions': colors.neutral[500], | ||
'--tw-prose-kbd': colors.neutral[900], | ||
'--tw-prose-kbd-shadows': hexToRgb(colors.neutral[900]), | ||
'--tw-prose-code': colors.neutral[900], | ||
@@ -1141,2 +1256,4 @@ '--tw-prose-pre-code': colors.neutral[200], | ||
'--tw-prose-invert-captions': colors.neutral[400], | ||
'--tw-prose-invert-kbd': colors.white, | ||
'--tw-prose-invert-kbd-shadows': hexToRgb(colors.white), | ||
'--tw-prose-invert-code': colors.white, | ||
@@ -1163,2 +1280,4 @@ '--tw-prose-invert-pre-code': colors.neutral[300], | ||
'--tw-prose-captions': colors.stone[500], | ||
'--tw-prose-kbd': colors.stone[900], | ||
'--tw-prose-kbd-shadows': hexToRgb(colors.stone[900]), | ||
'--tw-prose-code': colors.stone[900], | ||
@@ -1180,2 +1299,4 @@ '--tw-prose-pre-code': colors.stone[200], | ||
'--tw-prose-invert-captions': colors.stone[400], | ||
'--tw-prose-invert-kbd': colors.white, | ||
'--tw-prose-invert-kbd-shadows': hexToRgb(colors.white), | ||
'--tw-prose-invert-code': colors.white, | ||
@@ -1309,2 +1430,26 @@ '--tw-prose-invert-pre-code': colors.stone[300], | ||
}, | ||
// Invert (for dark mode) | ||
invert: { | ||
css: { | ||
'--tw-prose-body': 'var(--tw-prose-invert-body)', | ||
'--tw-prose-headings': 'var(--tw-prose-invert-headings)', | ||
'--tw-prose-lead': 'var(--tw-prose-invert-lead)', | ||
'--tw-prose-links': 'var(--tw-prose-invert-links)', | ||
'--tw-prose-bold': 'var(--tw-prose-invert-bold)', | ||
'--tw-prose-counters': 'var(--tw-prose-invert-counters)', | ||
'--tw-prose-bullets': 'var(--tw-prose-invert-bullets)', | ||
'--tw-prose-hr': 'var(--tw-prose-invert-hr)', | ||
'--tw-prose-quotes': 'var(--tw-prose-invert-quotes)', | ||
'--tw-prose-quote-borders': 'var(--tw-prose-invert-quote-borders)', | ||
'--tw-prose-captions': 'var(--tw-prose-invert-captions)', | ||
'--tw-prose-kbd': 'var(--tw-prose-invert-kbd)', | ||
'--tw-prose-kbd-shadows': 'var(--tw-prose-invert-kbd-shadows)', | ||
'--tw-prose-code': 'var(--tw-prose-invert-code)', | ||
'--tw-prose-pre-code': 'var(--tw-prose-invert-pre-code)', | ||
'--tw-prose-pre-bg': 'var(--tw-prose-invert-pre-bg)', | ||
'--tw-prose-th-borders': 'var(--tw-prose-invert-th-borders)', | ||
'--tw-prose-td-borders': 'var(--tw-prose-invert-td-borders)', | ||
}, | ||
}, | ||
} | ||
@@ -1318,2 +1463,3 @@ | ||
maxWidth: '65ch', | ||
p: {}, // Required to maintain correct order when merging | ||
'[class~="lead"]': { | ||
@@ -1331,2 +1477,11 @@ color: 'var(--tw-prose-lead)', | ||
}, | ||
'a strong': { | ||
color: 'inherit', | ||
}, | ||
'blockquote strong': { | ||
color: 'inherit', | ||
}, | ||
'thead th strong': { | ||
color: 'inherit', | ||
}, | ||
ol: { | ||
@@ -1372,2 +1527,6 @@ listStyleType: 'decimal', | ||
}, | ||
dt: { | ||
color: 'var(--tw-prose-headings)', | ||
fontWeight: '600', | ||
}, | ||
hr: { | ||
@@ -1381,4 +1540,4 @@ borderColor: 'var(--tw-prose-hr)', | ||
color: 'var(--tw-prose-quotes)', | ||
borderLeftWidth: '0.25rem', | ||
borderLeftColor: 'var(--tw-prose-quote-borders)', | ||
borderInlineStartWidth: '0.25rem', | ||
borderInlineStartColor: 'var(--tw-prose-quote-borders)', | ||
quotes: '"\\201C""\\201D""\\2018""\\2019"', | ||
@@ -1398,2 +1557,3 @@ }, | ||
fontWeight: '900', | ||
color: 'inherit', | ||
}, | ||
@@ -1406,2 +1566,3 @@ h2: { | ||
fontWeight: '800', | ||
color: 'inherit', | ||
}, | ||
@@ -1414,2 +1575,3 @@ h3: { | ||
fontWeight: '700', | ||
color: 'inherit', | ||
}, | ||
@@ -1422,8 +1584,16 @@ h4: { | ||
fontWeight: '700', | ||
color: 'inherit', | ||
}, | ||
// TODO: Figure out how to not need this, it's a merging issue | ||
'figure > *': {}, | ||
figcaption: { | ||
color: 'var(--tw-prose-captions)', | ||
img: {}, // Required to maintain correct order when merging | ||
picture: { | ||
display: 'block', | ||
}, | ||
video: {}, // Required to maintain correct order when merging | ||
kbd: { | ||
fontWeight: '500', | ||
fontFamily: 'inherit', | ||
color: 'var(--tw-prose-kbd)', | ||
boxShadow: | ||
'0 0 0 1px rgb(var(--tw-prose-kbd-shadows) / 10%), 0 3px 0 rgb(var(--tw-prose-kbd-shadows) / 10%)', | ||
}, | ||
code: { | ||
@@ -1440,4 +1610,22 @@ color: 'var(--tw-prose-code)', | ||
'a code': { | ||
color: 'var(--tw-prose-links)', | ||
color: 'inherit', | ||
}, | ||
'h1 code': { | ||
color: 'inherit', | ||
}, | ||
'h2 code': { | ||
color: 'inherit', | ||
}, | ||
'h3 code': { | ||
color: 'inherit', | ||
}, | ||
'h4 code': { | ||
color: 'inherit', | ||
}, | ||
'blockquote code': { | ||
color: 'inherit', | ||
}, | ||
'thead th code': { | ||
color: 'inherit', | ||
}, | ||
pre: { | ||
@@ -1469,3 +1657,2 @@ color: 'var(--tw-prose-pre-code)', | ||
tableLayout: 'auto', | ||
textAlign: 'left', | ||
marginTop: em(32, 16), | ||
@@ -1493,2 +1680,16 @@ marginBottom: em(32, 16), | ||
}, | ||
tfoot: { | ||
borderTopWidth: '1px', | ||
borderTopColor: 'var(--tw-prose-th-borders)', | ||
}, | ||
'tfoot td': { | ||
verticalAlign: 'top', | ||
}, | ||
'th, td': { | ||
textAlign: 'start', | ||
}, | ||
'figure > *': {}, // Required to maintain correct order when merging | ||
figcaption: { | ||
color: 'var(--tw-prose-captions)', | ||
}, | ||
}, | ||
@@ -1495,0 +1696,0 @@ defaultModifiers.gray.css, |
const isPlainObject = require('lodash.isplainobject') | ||
const parser = require('postcss-selector-parser') | ||
const parseSelector = parser() | ||
module.exports = { | ||
@@ -7,2 +10,54 @@ isUsableColor(color, values) { | ||
}, | ||
/** | ||
* @param {string} selector | ||
*/ | ||
commonTrailingPseudos(selector) { | ||
let ast = parseSelector.astSync(selector) | ||
/** @type {import('postcss-selector-parser').Pseudo[][]} */ | ||
let matrix = [] | ||
// Put the pseudo elements in reverse order in a sparse, column-major 2D array | ||
for (let [i, sel] of ast.nodes.entries()) { | ||
for (const [j, child] of [...sel.nodes].reverse().entries()) { | ||
// We only care about pseudo elements | ||
if (child.type !== 'pseudo' || !child.value.startsWith('::')) { | ||
break | ||
} | ||
matrix[j] = matrix[j] || [] | ||
matrix[j][i] = child | ||
} | ||
} | ||
let trailingPseudos = parser.selector() | ||
// At this point the pseudo elements are in a column-major 2D array | ||
// This means each row contains one "column" of pseudo elements from each selector | ||
// We can compare all the pseudo elements in a row to see if they are the same | ||
for (const pseudos of matrix) { | ||
// It's a sparse 2D array so there are going to be holes in the rows | ||
// We skip those | ||
if (!pseudos) { | ||
continue | ||
} | ||
let values = new Set(pseudos.map((p) => p.value)) | ||
// The pseudo elements are not the same | ||
if (values.size > 1) { | ||
break | ||
} | ||
pseudos.forEach((pseudo) => pseudo.remove()) | ||
trailingPseudos.prepend(pseudos[0]) | ||
} | ||
if (trailingPseudos.nodes.length) { | ||
return [trailingPseudos.toString(), ast.toString()] | ||
} | ||
return [null, selector] | ||
}, | ||
} |
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
No repository
Supply chain riskPackage does not have a linked source code repository. Without this field, a package will have no reference to the location of the source code use to generate the package.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
No repository
Supply chain riskPackage does not have a linked source code repository. Without this field, a package will have no reference to the location of the source code use to generate the package.
Found 1 instance in 1 package
110175
3158
399
5
+ Addedpostcss-selector-parser@6.0.10(transitive)
+ Addedtailwindcss@4.0.0-beta.9(transitive)
- Removed@alloc/quick-lru@5.2.0(transitive)
- Removed@isaacs/cliui@8.0.2(transitive)
- Removed@jridgewell/gen-mapping@0.3.8(transitive)
- Removed@jridgewell/resolve-uri@3.1.2(transitive)
- Removed@jridgewell/set-array@1.2.1(transitive)
- Removed@jridgewell/sourcemap-codec@1.5.0(transitive)
- Removed@jridgewell/trace-mapping@0.3.25(transitive)
- Removed@nodelib/fs.scandir@2.1.5(transitive)
- Removed@nodelib/fs.stat@2.0.5(transitive)
- Removed@nodelib/fs.walk@1.2.8(transitive)
- Removed@pkgjs/parseargs@0.11.0(transitive)
- Removedansi-regex@5.0.16.1.0(transitive)
- Removedansi-styles@4.3.06.2.1(transitive)
- Removedany-promise@1.3.0(transitive)
- Removedanymatch@3.1.3(transitive)
- Removedarg@5.0.2(transitive)
- Removedbalanced-match@1.0.2(transitive)
- Removedbinary-extensions@2.3.0(transitive)
- Removedbrace-expansion@2.0.1(transitive)
- Removedbraces@3.0.3(transitive)
- Removedcamelcase-css@2.0.1(transitive)
- Removedchokidar@3.6.0(transitive)
- Removedcolor-convert@2.0.1(transitive)
- Removedcolor-name@1.1.4(transitive)
- Removedcommander@4.1.1(transitive)
- Removedcross-spawn@7.0.6(transitive)
- Removeddidyoumean@1.2.2(transitive)
- Removeddlv@1.1.3(transitive)
- Removedeastasianwidth@0.2.0(transitive)
- Removedemoji-regex@8.0.09.2.2(transitive)
- Removedfast-glob@3.3.3(transitive)
- Removedfastq@1.18.0(transitive)
- Removedfill-range@7.1.1(transitive)
- Removedforeground-child@3.3.0(transitive)
- Removedfsevents@2.3.3(transitive)
- Removedfunction-bind@1.1.2(transitive)
- Removedglob@10.4.5(transitive)
- Removedglob-parent@5.1.26.0.2(transitive)
- Removedhasown@2.0.2(transitive)
- Removedis-binary-path@2.1.0(transitive)
- Removedis-core-module@2.16.1(transitive)
- Removedis-extglob@2.1.1(transitive)
- Removedis-fullwidth-code-point@3.0.0(transitive)
- Removedis-glob@4.0.3(transitive)
- Removedis-number@7.0.0(transitive)
- Removedisexe@2.0.0(transitive)
- Removedjackspeak@3.4.3(transitive)
- Removedjiti@1.21.7(transitive)
- Removedlilconfig@3.1.3(transitive)
- Removedlines-and-columns@1.2.4(transitive)
- Removedlru-cache@10.4.3(transitive)
- Removedmerge2@1.4.1(transitive)
- Removedmicromatch@4.0.8(transitive)
- Removedminimatch@9.0.5(transitive)
- Removedminipass@7.1.2(transitive)
- Removedmz@2.7.0(transitive)
- Removednanoid@3.3.8(transitive)
- Removednormalize-path@3.0.0(transitive)
- Removedobject-assign@4.1.1(transitive)
- Removedobject-hash@3.0.0(transitive)
- Removedpackage-json-from-dist@1.0.1(transitive)
- Removedpath-key@3.1.1(transitive)
- Removedpath-parse@1.0.7(transitive)
- Removedpath-scurry@1.11.1(transitive)
- Removedpicocolors@1.1.1(transitive)
- Removedpicomatch@2.3.1(transitive)
- Removedpify@2.3.0(transitive)
- Removedpirates@4.0.6(transitive)
- Removedpostcss@8.5.1(transitive)
- Removedpostcss-import@15.1.0(transitive)
- Removedpostcss-js@4.0.1(transitive)
- Removedpostcss-load-config@4.0.2(transitive)
- Removedpostcss-nested@6.2.0(transitive)
- Removedpostcss-selector-parser@6.1.2(transitive)
- Removedpostcss-value-parser@4.2.0(transitive)
- Removedqueue-microtask@1.2.3(transitive)
- Removedread-cache@1.0.0(transitive)
- Removedreaddirp@3.6.0(transitive)
- Removedresolve@1.22.10(transitive)
- Removedreusify@1.0.4(transitive)
- Removedrun-parallel@1.2.0(transitive)
- Removedshebang-command@2.0.0(transitive)
- Removedshebang-regex@3.0.0(transitive)
- Removedsignal-exit@4.1.0(transitive)
- Removedsource-map-js@1.2.1(transitive)
- Removedstring-width@4.2.35.1.2(transitive)
- Removedstrip-ansi@6.0.17.1.0(transitive)
- Removedsucrase@3.35.0(transitive)
- Removedsupports-preserve-symlinks-flag@1.0.0(transitive)
- Removedtailwindcss@3.4.17(transitive)
- Removedthenify@3.3.1(transitive)
- Removedthenify-all@1.6.0(transitive)
- Removedto-regex-range@5.0.1(transitive)
- Removedts-interface-checker@0.1.13(transitive)
- Removedwhich@2.0.2(transitive)
- Removedwrap-ansi@7.0.08.1.0(transitive)
- Removedyaml@2.7.0(transitive)