mirror of
https://github.com/tailwindlabs/tailwindcss.git
synced 2025-12-08 21:36:08 +00:00
* make `jit` mode the default when no mode is specified
* unify JIT and AOT codepaths
* ensure `Object.entries` on undefined doesn't break
It could be that sometimes you don't have values in your config (e.g.: `presets: []`), this in turn will break some plugins where we assume we have a value.
* drop AOT specific tests
These tests are all covered by JIT mode already and were AOT specific.
* simplify tests, and add a few
Some of the tests were written for AOT specifically, some were missing. We also updated the way we write those tests, essentially making Tailwind a blackbox, by testing against the final output.
Now that JIT mode is the default, this is super fast because we only generate what is used, instead of partially testing in a 3MB file or building it all, then purging.
* add some todo's to make sure we warn in a few cases
* make `darkMode: 'media'`, the default
This also includes moving dark mode tests to its own dedicated file.
* remove PostCSS 7 compat mode
* update CLI to be JIT-first
* fix integration tests
This is not a _real_ fix, but it does solve the broken test for now.
* warn when using @responsive or @variants
* remove the JIT preview warning
* remove AOT-only code paths
* remove all `mode: 'jit'` blocks
Also remove `variants: {}` since they are not useful in `JIT` mode
anymore.
* drop unused dependencies
* rename `purge` to `content`
* remove static CDN builds
* mark `--purge` as deprecated in the CLI
This will still work, but a warning will be printed and it won't show up
in the `--help` output.
* cleanup nesting plugin
We don't have to duplicate it anymore since there is no PostCSS 7
version anymore.
* make sure integration tests run in band
* cleanup folder structure
* make sure nesting folder is available
* simplify resolving of purge/content information
103 lines
2.3 KiB
JavaScript
103 lines
2.3 KiB
JavaScript
import postcss from 'postcss'
|
|
import selectorParser from 'postcss-selector-parser'
|
|
|
|
function minimumImpactSelector(nodes) {
|
|
let pseudos = nodes.filter((n) => n.type === 'pseudo')
|
|
let [bestNode] = nodes
|
|
|
|
for (let [type, getNode = (n) => n] of [
|
|
['class'],
|
|
[
|
|
'id',
|
|
(n) =>
|
|
selectorParser.attribute({
|
|
attribute: 'id',
|
|
operator: '=',
|
|
value: n.value,
|
|
quoteMark: '"',
|
|
}),
|
|
],
|
|
['attribute'],
|
|
]) {
|
|
let match = nodes.find((n) => n.type === type)
|
|
|
|
if (match) {
|
|
bestNode = getNode(match)
|
|
break
|
|
}
|
|
}
|
|
|
|
return [bestNode, ...pseudos].join('').trim()
|
|
}
|
|
|
|
let elementSelectorParser = selectorParser((selectors) => {
|
|
return selectors.map((s) => {
|
|
let nodes = s
|
|
.split((n) => n.type === 'combinator')
|
|
.pop()
|
|
.filter((n) => n.type !== 'pseudo' || n.value.startsWith('::'))
|
|
|
|
return minimumImpactSelector(nodes)
|
|
})
|
|
})
|
|
|
|
let cache = new Map()
|
|
|
|
function extractElementSelector(selector) {
|
|
if (!cache.has(selector)) {
|
|
cache.set(selector, elementSelectorParser.transformSync(selector))
|
|
}
|
|
|
|
return cache.get(selector)
|
|
}
|
|
|
|
export default function resolveDefaultsAtRules() {
|
|
return (root) => {
|
|
let variableNodeMap = new Map()
|
|
let universals = new Set()
|
|
|
|
root.walkAtRules('defaults', (rule) => {
|
|
if (rule.nodes && rule.nodes.length > 0) {
|
|
universals.add(rule)
|
|
return
|
|
}
|
|
|
|
let variable = rule.params
|
|
if (!variableNodeMap.has(variable)) {
|
|
variableNodeMap.set(variable, new Set())
|
|
}
|
|
|
|
variableNodeMap.get(variable).add(rule.parent)
|
|
|
|
rule.remove()
|
|
})
|
|
|
|
for (let universal of universals) {
|
|
let selectors = new Set()
|
|
|
|
let rules = variableNodeMap.get(universal.params) ?? []
|
|
|
|
for (let rule of rules) {
|
|
for (let selector of extractElementSelector(rule.selector)) {
|
|
selectors.add(selector)
|
|
}
|
|
}
|
|
|
|
if (selectors.size === 0) {
|
|
universal.remove()
|
|
continue
|
|
}
|
|
|
|
let universalRule = postcss.rule()
|
|
|
|
// TODO: Fix this, this is a hotfix
|
|
// universalRule.selectors = [...selectors]
|
|
universalRule.selectors = ['*', '::before', '::after']
|
|
|
|
universalRule.append(universal.nodes)
|
|
universal.before(universalRule)
|
|
universal.remove()
|
|
}
|
|
}
|
|
}
|