tailwindcss 0.0.0-insiders.ea139f2 → 0.0.0-insiders.ea4e1cd

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (236) hide show
  1. package/LICENSE +1 -2
  2. package/README.md +15 -7
  3. package/colors.d.ts +3 -0
  4. package/colors.js +2 -1
  5. package/defaultConfig.d.ts +3 -0
  6. package/defaultConfig.js +2 -1
  7. package/defaultTheme.d.ts +4 -0
  8. package/defaultTheme.js +2 -1
  9. package/lib/cli/build/deps.js +62 -0
  10. package/lib/cli/build/index.js +54 -0
  11. package/lib/cli/build/plugin.js +378 -0
  12. package/lib/cli/build/utils.js +88 -0
  13. package/lib/cli/build/watching.js +182 -0
  14. package/lib/cli/help/index.js +73 -0
  15. package/lib/cli/index.js +230 -0
  16. package/lib/cli/init/index.js +63 -0
  17. package/lib/cli-peer-dependencies.js +28 -7
  18. package/lib/cli.js +4 -703
  19. package/lib/corePluginList.js +12 -3
  20. package/lib/corePlugins.js +2373 -1863
  21. package/lib/css/preflight.css +10 -8
  22. package/lib/featureFlags.js +49 -26
  23. package/lib/index.js +1 -31
  24. package/lib/lib/cacheInvalidation.js +92 -0
  25. package/lib/lib/collapseAdjacentRules.js +30 -10
  26. package/lib/lib/collapseDuplicateDeclarations.js +60 -4
  27. package/lib/lib/content.js +181 -0
  28. package/lib/lib/defaultExtractor.js +243 -0
  29. package/lib/lib/detectNesting.js +21 -10
  30. package/lib/lib/evaluateTailwindFunctions.js +115 -50
  31. package/lib/lib/expandApplyAtRules.js +467 -161
  32. package/lib/lib/expandTailwindAtRules.js +160 -133
  33. package/lib/lib/findAtConfigPath.js +46 -0
  34. package/lib/lib/generateRules.js +553 -200
  35. package/lib/lib/getModuleDependencies.js +88 -37
  36. package/lib/lib/load-config.js +42 -0
  37. package/lib/lib/normalizeTailwindDirectives.js +46 -33
  38. package/lib/lib/offsets.js +306 -0
  39. package/lib/lib/partitionApplyAtRules.js +58 -0
  40. package/lib/lib/regex.js +74 -0
  41. package/lib/lib/remap-bitfield.js +89 -0
  42. package/lib/lib/resolveDefaultsAtRules.js +98 -58
  43. package/lib/lib/setupContextUtils.js +773 -321
  44. package/lib/lib/setupTrackingContext.js +70 -75
  45. package/lib/lib/sharedState.js +78 -10
  46. package/lib/lib/substituteScreenAtRules.js +14 -10
  47. package/lib/oxide/cli/build/deps.js +89 -0
  48. package/lib/oxide/cli/build/index.js +53 -0
  49. package/lib/oxide/cli/build/plugin.js +375 -0
  50. package/lib/oxide/cli/build/utils.js +87 -0
  51. package/lib/oxide/cli/build/watching.js +179 -0
  52. package/lib/oxide/cli/help/index.js +72 -0
  53. package/lib/oxide/cli/index.js +214 -0
  54. package/lib/oxide/cli/init/index.js +52 -0
  55. package/lib/oxide/cli.js +5 -0
  56. package/lib/oxide/postcss-plugin.js +2 -0
  57. package/lib/plugin.js +98 -0
  58. package/{nesting → lib/postcss-plugins/nesting}/README.md +2 -2
  59. package/lib/postcss-plugins/nesting/index.js +21 -0
  60. package/lib/postcss-plugins/nesting/plugin.js +89 -0
  61. package/lib/processTailwindFeatures.js +39 -26
  62. package/lib/public/colors.js +272 -246
  63. package/lib/public/create-plugin.js +9 -5
  64. package/lib/public/default-config.js +10 -6
  65. package/lib/public/default-theme.js +10 -6
  66. package/lib/public/load-config.js +12 -0
  67. package/lib/public/resolve-config.js +11 -6
  68. package/lib/util/applyImportantSelector.js +36 -0
  69. package/lib/util/bigSign.js +6 -1
  70. package/lib/util/buildMediaQuery.js +13 -6
  71. package/lib/util/cloneDeep.js +9 -6
  72. package/lib/util/cloneNodes.js +23 -3
  73. package/lib/util/color.js +70 -38
  74. package/lib/util/colorNames.js +752 -0
  75. package/lib/util/configurePlugins.js +7 -2
  76. package/lib/util/createPlugin.js +8 -6
  77. package/lib/util/createUtilityPlugin.js +16 -16
  78. package/lib/util/dataTypes.js +173 -108
  79. package/lib/util/defaults.js +14 -3
  80. package/lib/util/escapeClassName.js +13 -8
  81. package/lib/util/escapeCommas.js +7 -2
  82. package/lib/util/flattenColorPalette.js +11 -12
  83. package/lib/util/formatVariantSelector.js +228 -151
  84. package/lib/util/getAllConfigs.js +33 -12
  85. package/lib/util/hashConfig.js +9 -4
  86. package/lib/util/isKeyframeRule.js +7 -2
  87. package/lib/util/isPlainObject.js +7 -2
  88. package/lib/util/{isValidArbitraryValue.js → isSyntacticallyValidPropertyValue.js} +25 -15
  89. package/lib/util/log.js +27 -13
  90. package/lib/util/nameClass.js +27 -10
  91. package/lib/util/negateValue.js +25 -8
  92. package/lib/util/normalizeConfig.js +139 -65
  93. package/lib/util/normalizeScreens.js +131 -11
  94. package/lib/util/parseAnimationValue.js +44 -40
  95. package/lib/util/parseBoxShadowValue.js +34 -23
  96. package/lib/util/parseDependency.js +39 -55
  97. package/lib/util/parseGlob.js +36 -0
  98. package/lib/util/parseObjectStyles.js +15 -10
  99. package/lib/util/pluginUtils.js +159 -69
  100. package/lib/util/prefixSelector.js +30 -12
  101. package/lib/util/pseudoElements.js +229 -0
  102. package/lib/util/removeAlphaVariables.js +31 -0
  103. package/lib/util/resolveConfig.js +97 -75
  104. package/lib/util/resolveConfigPath.js +30 -12
  105. package/lib/util/responsive.js +11 -6
  106. package/lib/util/splitAtTopLevelOnly.js +51 -0
  107. package/lib/util/tap.js +6 -1
  108. package/lib/util/toColorValue.js +7 -3
  109. package/lib/util/toPath.js +26 -3
  110. package/lib/util/transformThemeValue.js +40 -30
  111. package/lib/util/validateConfig.js +37 -0
  112. package/lib/util/validateFormalSyntax.js +26 -0
  113. package/lib/util/withAlphaVariable.js +27 -15
  114. package/loadConfig.d.ts +4 -0
  115. package/loadConfig.js +2 -0
  116. package/nesting/index.js +2 -12
  117. package/package.json +66 -57
  118. package/peers/index.js +75964 -55560
  119. package/plugin.d.ts +11 -0
  120. package/plugin.js +2 -1
  121. package/resolveConfig.d.ts +12 -0
  122. package/resolveConfig.js +2 -1
  123. package/scripts/generate-types.js +105 -0
  124. package/scripts/release-channel.js +18 -0
  125. package/scripts/release-notes.js +21 -0
  126. package/scripts/swap-engines.js +40 -0
  127. package/scripts/type-utils.js +27 -0
  128. package/src/cli/build/deps.js +56 -0
  129. package/src/cli/build/index.js +49 -0
  130. package/src/cli/build/plugin.js +444 -0
  131. package/src/cli/build/utils.js +76 -0
  132. package/src/cli/build/watching.js +229 -0
  133. package/src/cli/help/index.js +70 -0
  134. package/src/cli/index.js +216 -0
  135. package/src/cli/init/index.js +79 -0
  136. package/src/cli-peer-dependencies.js +7 -1
  137. package/src/cli.js +4 -765
  138. package/src/corePluginList.js +1 -1
  139. package/src/corePlugins.js +786 -306
  140. package/src/css/preflight.css +10 -8
  141. package/src/featureFlags.js +21 -5
  142. package/src/index.js +1 -34
  143. package/src/lib/cacheInvalidation.js +52 -0
  144. package/src/lib/collapseAdjacentRules.js +21 -2
  145. package/src/lib/collapseDuplicateDeclarations.js +66 -1
  146. package/src/lib/content.js +208 -0
  147. package/src/lib/defaultExtractor.js +217 -0
  148. package/src/lib/detectNesting.js +9 -1
  149. package/src/lib/evaluateTailwindFunctions.js +79 -8
  150. package/src/lib/expandApplyAtRules.js +515 -153
  151. package/src/lib/expandTailwindAtRules.js +115 -86
  152. package/src/lib/findAtConfigPath.js +48 -0
  153. package/src/lib/generateRules.js +545 -147
  154. package/src/lib/getModuleDependencies.js +70 -30
  155. package/src/lib/load-config.ts +31 -0
  156. package/src/lib/normalizeTailwindDirectives.js +7 -1
  157. package/src/lib/offsets.js +373 -0
  158. package/src/lib/partitionApplyAtRules.js +52 -0
  159. package/src/lib/regex.js +74 -0
  160. package/src/lib/remap-bitfield.js +82 -0
  161. package/src/lib/resolveDefaultsAtRules.js +59 -17
  162. package/src/lib/setupContextUtils.js +701 -175
  163. package/src/lib/setupTrackingContext.js +51 -62
  164. package/src/lib/sharedState.js +58 -7
  165. package/src/oxide/cli/build/deps.ts +91 -0
  166. package/src/oxide/cli/build/index.ts +47 -0
  167. package/src/oxide/cli/build/plugin.ts +442 -0
  168. package/src/oxide/cli/build/utils.ts +74 -0
  169. package/src/oxide/cli/build/watching.ts +225 -0
  170. package/src/oxide/cli/help/index.ts +69 -0
  171. package/src/oxide/cli/index.ts +204 -0
  172. package/src/oxide/cli/init/index.ts +59 -0
  173. package/src/oxide/cli.ts +1 -0
  174. package/src/oxide/postcss-plugin.ts +1 -0
  175. package/src/plugin.js +107 -0
  176. package/src/postcss-plugins/nesting/README.md +42 -0
  177. package/src/postcss-plugins/nesting/index.js +13 -0
  178. package/src/postcss-plugins/nesting/plugin.js +80 -0
  179. package/src/processTailwindFeatures.js +12 -2
  180. package/src/public/colors.js +22 -0
  181. package/src/public/default-config.js +1 -1
  182. package/src/public/default-theme.js +2 -2
  183. package/src/public/load-config.js +2 -0
  184. package/src/util/applyImportantSelector.js +27 -0
  185. package/src/util/buildMediaQuery.js +5 -3
  186. package/src/util/cloneNodes.js +19 -2
  187. package/src/util/color.js +44 -12
  188. package/src/util/colorNames.js +150 -0
  189. package/src/util/dataTypes.js +51 -16
  190. package/src/util/defaults.js +6 -0
  191. package/src/util/formatVariantSelector.js +264 -144
  192. package/src/util/getAllConfigs.js +21 -2
  193. package/src/util/{isValidArbitraryValue.js → isSyntacticallyValidPropertyValue.js} +1 -1
  194. package/src/util/log.js +11 -7
  195. package/src/util/nameClass.js +4 -0
  196. package/src/util/negateValue.js +11 -3
  197. package/src/util/normalizeConfig.js +57 -5
  198. package/src/util/normalizeScreens.js +105 -7
  199. package/src/util/parseBoxShadowValue.js +4 -3
  200. package/src/util/parseDependency.js +37 -42
  201. package/src/util/parseGlob.js +24 -0
  202. package/src/util/pluginUtils.js +123 -24
  203. package/src/util/prefixSelector.js +30 -10
  204. package/src/util/pseudoElements.js +170 -0
  205. package/src/util/removeAlphaVariables.js +24 -0
  206. package/src/util/resolveConfig.js +74 -26
  207. package/src/util/resolveConfigPath.js +12 -1
  208. package/src/util/splitAtTopLevelOnly.js +52 -0
  209. package/src/util/toPath.js +23 -1
  210. package/src/util/transformThemeValue.js +13 -3
  211. package/src/util/validateConfig.js +26 -0
  212. package/src/util/validateFormalSyntax.js +34 -0
  213. package/src/util/withAlphaVariable.js +1 -1
  214. package/stubs/.gitignore +1 -0
  215. package/stubs/.prettierrc.json +6 -0
  216. package/stubs/{defaultConfig.stub.js → config.full.js} +206 -166
  217. package/stubs/postcss.config.js +6 -0
  218. package/stubs/tailwind.config.cjs +2 -0
  219. package/stubs/tailwind.config.js +2 -0
  220. package/stubs/tailwind.config.ts +3 -0
  221. package/types/config.d.ts +368 -0
  222. package/types/generated/.gitkeep +0 -0
  223. package/types/generated/colors.d.ts +298 -0
  224. package/types/generated/corePluginList.d.ts +1 -0
  225. package/types/generated/default-theme.d.ts +371 -0
  226. package/types/index.d.ts +7 -0
  227. package/CHANGELOG.md +0 -1843
  228. package/lib/constants.js +0 -37
  229. package/lib/lib/setupWatchingContext.js +0 -288
  230. package/nesting/plugin.js +0 -41
  231. package/scripts/install-integrations.js +0 -27
  232. package/scripts/rebuildFixtures.js +0 -68
  233. package/src/constants.js +0 -17
  234. package/src/lib/setupWatchingContext.js +0 -311
  235. /package/stubs/{simpleConfig.stub.js → config.simple.js} +0 -0
  236. /package/stubs/{defaultPostCssConfig.stub.js → postcss.config.cjs} +0 -0
@@ -1,13 +1,188 @@
1
1
  import postcss from 'postcss'
2
+ import parser from 'postcss-selector-parser'
3
+
2
4
  import { resolveMatches } from './generateRules'
3
- import bigSign from '../util/bigSign'
4
5
  import escapeClassName from '../util/escapeClassName'
6
+ import { applyImportantSelector } from '../util/applyImportantSelector'
7
+ import { movePseudos } from '../util/pseudoElements'
8
+
9
+ /** @typedef {Map<string, [any, import('postcss').Rule[]]>} ApplyCache */
10
+
11
+ function extractClasses(node) {
12
+ /** @type {Map<string, Set<string>>} */
13
+ let groups = new Map()
14
+
15
+ let container = postcss.root({ nodes: [node.clone()] })
16
+
17
+ container.walkRules((rule) => {
18
+ parser((selectors) => {
19
+ selectors.walkClasses((classSelector) => {
20
+ let parentSelector = classSelector.parent.toString()
21
+
22
+ let classes = groups.get(parentSelector)
23
+ if (!classes) {
24
+ groups.set(parentSelector, (classes = new Set()))
25
+ }
26
+
27
+ classes.add(classSelector.value)
28
+ })
29
+ }).processSync(rule.selector)
30
+ })
31
+
32
+ let normalizedGroups = Array.from(groups.values(), (classes) => Array.from(classes))
33
+ let classes = normalizedGroups.flat()
34
+
35
+ return Object.assign(classes, { groups: normalizedGroups })
36
+ }
37
+
38
+ let selectorExtractor = parser()
39
+
40
+ /**
41
+ * @param {string} ruleSelectors
42
+ */
43
+ function extractSelectors(ruleSelectors) {
44
+ return selectorExtractor.astSync(ruleSelectors)
45
+ }
46
+
47
+ function extractBaseCandidates(candidates, separator) {
48
+ let baseClasses = new Set()
49
+
50
+ for (let candidate of candidates) {
51
+ baseClasses.add(candidate.split(separator).pop())
52
+ }
53
+
54
+ return Array.from(baseClasses)
55
+ }
5
56
 
6
57
  function prefix(context, selector) {
7
58
  let prefix = context.tailwindConfig.prefix
8
59
  return typeof prefix === 'function' ? prefix(selector) : prefix + selector
9
60
  }
10
61
 
62
+ function* pathToRoot(node) {
63
+ yield node
64
+ while (node.parent) {
65
+ yield node.parent
66
+ node = node.parent
67
+ }
68
+ }
69
+
70
+ /**
71
+ * Only clone the node itself and not its children
72
+ *
73
+ * @param {*} node
74
+ * @param {*} overrides
75
+ * @returns
76
+ */
77
+ function shallowClone(node, overrides = {}) {
78
+ let children = node.nodes
79
+ node.nodes = []
80
+
81
+ let tmp = node.clone(overrides)
82
+
83
+ node.nodes = children
84
+
85
+ return tmp
86
+ }
87
+
88
+ /**
89
+ * Clone just the nodes all the way to the top that are required to represent
90
+ * this singular rule in the tree.
91
+ *
92
+ * For example, if we have CSS like this:
93
+ * ```css
94
+ * @media (min-width: 768px) {
95
+ * @supports (display: grid) {
96
+ * .foo {
97
+ * display: grid;
98
+ * grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
99
+ * }
100
+ * }
101
+ *
102
+ * @supports (backdrop-filter: blur(1px)) {
103
+ * .bar {
104
+ * backdrop-filter: blur(1px);
105
+ * }
106
+ * }
107
+ *
108
+ * .baz {
109
+ * color: orange;
110
+ * }
111
+ * }
112
+ * ```
113
+ *
114
+ * And we're cloning `.bar` it'll return a cloned version of what's required for just that single node:
115
+ *
116
+ * ```css
117
+ * @media (min-width: 768px) {
118
+ * @supports (backdrop-filter: blur(1px)) {
119
+ * .bar {
120
+ * backdrop-filter: blur(1px);
121
+ * }
122
+ * }
123
+ * }
124
+ * ```
125
+ *
126
+ * @param {import('postcss').Node} node
127
+ */
128
+ function nestedClone(node) {
129
+ for (let parent of pathToRoot(node)) {
130
+ if (node === parent) {
131
+ continue
132
+ }
133
+
134
+ if (parent.type === 'root') {
135
+ break
136
+ }
137
+
138
+ node = shallowClone(parent, {
139
+ nodes: [node],
140
+ })
141
+ }
142
+
143
+ return node
144
+ }
145
+
146
+ /**
147
+ * @param {import('postcss').Root} root
148
+ */
149
+ function buildLocalApplyCache(root, context) {
150
+ /** @type {ApplyCache} */
151
+ let cache = new Map()
152
+
153
+ root.walkRules((rule) => {
154
+ // Ignore rules generated by Tailwind
155
+ for (let node of pathToRoot(rule)) {
156
+ if (node.raws.tailwind?.layer !== undefined) {
157
+ return
158
+ }
159
+ }
160
+
161
+ // Clone what's required to represent this singular rule in the tree
162
+ let container = nestedClone(rule)
163
+ let sort = context.offsets.create('user')
164
+
165
+ for (let className of extractClasses(rule)) {
166
+ let list = cache.get(className) || []
167
+ cache.set(className, list)
168
+
169
+ list.push([
170
+ {
171
+ layer: 'user',
172
+ sort,
173
+ important: false,
174
+ },
175
+ container,
176
+ ])
177
+ }
178
+ })
179
+
180
+ return cache
181
+ }
182
+
183
+ /**
184
+ * @returns {ApplyCache}
185
+ */
11
186
  function buildApplyCache(applyCandidates, context) {
12
187
  for (let candidate of applyCandidates) {
13
188
  if (context.notClassCache.has(candidate) || context.applyClassCache.has(candidate)) {
@@ -35,6 +210,43 @@ function buildApplyCache(applyCandidates, context) {
35
210
  return context.applyClassCache
36
211
  }
37
212
 
213
+ /**
214
+ * Build a cache only when it's first used
215
+ *
216
+ * @param {() => ApplyCache} buildCacheFn
217
+ * @returns {ApplyCache}
218
+ */
219
+ function lazyCache(buildCacheFn) {
220
+ let cache = null
221
+
222
+ return {
223
+ get: (name) => {
224
+ cache = cache || buildCacheFn()
225
+
226
+ return cache.get(name)
227
+ },
228
+ has: (name) => {
229
+ cache = cache || buildCacheFn()
230
+
231
+ return cache.has(name)
232
+ },
233
+ }
234
+ }
235
+
236
+ /**
237
+ * Take a series of multiple caches and merge
238
+ * them so they act like one large cache
239
+ *
240
+ * @param {ApplyCache[]} caches
241
+ * @returns {ApplyCache}
242
+ */
243
+ function combineCaches(caches) {
244
+ return {
245
+ get: (name) => caches.flatMap((cache) => cache.get(name) || []),
246
+ has: (name) => caches.some((cache) => cache.has(name)),
247
+ }
248
+ }
249
+
38
250
  function extractApplyCandidates(params) {
39
251
  let candidates = params.split(/[\s\t\n]+/g)
40
252
 
@@ -45,48 +257,7 @@ function extractApplyCandidates(params) {
45
257
  return [candidates, false]
46
258
  }
47
259
 
48
- function partitionApplyParents(root) {
49
- let applyParents = new Set()
50
-
51
- root.walkAtRules('apply', (rule) => {
52
- applyParents.add(rule.parent)
53
- })
54
-
55
- for (let rule of applyParents) {
56
- let nodeGroups = []
57
- let lastGroup = []
58
-
59
- for (let node of rule.nodes) {
60
- if (node.type === 'atrule' && node.name === 'apply') {
61
- if (lastGroup.length > 0) {
62
- nodeGroups.push(lastGroup)
63
- lastGroup = []
64
- }
65
- nodeGroups.push([node])
66
- } else {
67
- lastGroup.push(node)
68
- }
69
- }
70
-
71
- if (lastGroup.length > 0) {
72
- nodeGroups.push(lastGroup)
73
- }
74
-
75
- if (nodeGroups.length === 1) {
76
- continue
77
- }
78
-
79
- for (let group of [...nodeGroups].reverse()) {
80
- let newParent = rule.clone({ nodes: [] })
81
- newParent.append(group)
82
- rule.after(newParent)
83
- }
84
-
85
- rule.remove()
86
- }
87
- }
88
-
89
- function processApply(root, context) {
260
+ function processApply(root, context, localCache) {
90
261
  let applyCandidates = new Set()
91
262
 
92
263
  // Collect all @apply rules and candidates
@@ -102,150 +273,341 @@ function processApply(root, context) {
102
273
  })
103
274
 
104
275
  // Start the @apply process if we have rules with @apply in them
105
- if (applies.length > 0) {
106
- // Fill up some caches!
107
- let applyClassCache = buildApplyCache(applyCandidates, context)
108
-
109
- /**
110
- * When we have an apply like this:
111
- *
112
- * .abc {
113
- * @apply hover:font-bold;
114
- * }
115
- *
116
- * What we essentially will do is resolve to this:
117
- *
118
- * .abc {
119
- * @apply .hover\:font-bold:hover {
120
- * font-weight: 500;
121
- * }
122
- * }
123
- *
124
- * Notice that the to-be-applied class is `.hover\:font-bold:hover` and that the utility candidate was `hover:font-bold`.
125
- * What happens in this function is that we prepend a `.` and escape the candidate.
126
- * This will result in `.hover\:font-bold`
127
- * Which means that we can replace `.hover\:font-bold` with `.abc` in `.hover\:font-bold:hover` resulting in `.abc:hover`
128
- */
129
- // TODO: Should we use postcss-selector-parser for this instead?
130
- function replaceSelector(selector, utilitySelectors, candidate) {
131
- let needle = `.${escapeClassName(candidate)}`
132
- let utilitySelectorsList = utilitySelectors.split(/\s*,\s*/g)
133
-
134
- return selector
135
- .split(/\s*,\s*/g)
136
- .map((s) => {
137
- let replaced = []
138
-
139
- for (let utilitySelector of utilitySelectorsList) {
140
- let replacedSelector = utilitySelector.replace(needle, s)
141
- if (replacedSelector === utilitySelector) {
142
- continue
143
- }
144
- replaced.push(replacedSelector)
276
+ if (applies.length === 0) {
277
+ return
278
+ }
279
+
280
+ // Fill up some caches!
281
+ let applyClassCache = combineCaches([localCache, buildApplyCache(applyCandidates, context)])
282
+
283
+ /**
284
+ * When we have an apply like this:
285
+ *
286
+ * .abc {
287
+ * @apply hover:font-bold;
288
+ * }
289
+ *
290
+ * What we essentially will do is resolve to this:
291
+ *
292
+ * .abc {
293
+ * @apply .hover\:font-bold:hover {
294
+ * font-weight: 500;
295
+ * }
296
+ * }
297
+ *
298
+ * Notice that the to-be-applied class is `.hover\:font-bold:hover` and that the utility candidate was `hover:font-bold`.
299
+ * What happens in this function is that we prepend a `.` and escape the candidate.
300
+ * This will result in `.hover\:font-bold`
301
+ * Which means that we can replace `.hover\:font-bold` with `.abc` in `.hover\:font-bold:hover` resulting in `.abc:hover`
302
+ *
303
+ * @param {string} selector
304
+ * @param {string} utilitySelectors
305
+ * @param {string} candidate
306
+ */
307
+ function replaceSelector(selector, utilitySelectors, candidate) {
308
+ let selectorList = extractSelectors(selector)
309
+ let utilitySelectorsList = extractSelectors(utilitySelectors)
310
+ let candidateList = extractSelectors(`.${escapeClassName(candidate)}`)
311
+ let candidateClass = candidateList.nodes[0].nodes[0]
312
+
313
+ selectorList.each((sel) => {
314
+ /** @type {Set<import('postcss-selector-parser').Selector>} */
315
+ let replaced = new Set()
316
+
317
+ utilitySelectorsList.each((utilitySelector) => {
318
+ let hasReplaced = false
319
+ utilitySelector = utilitySelector.clone()
320
+
321
+ utilitySelector.walkClasses((node) => {
322
+ if (node.value !== candidateClass.value) {
323
+ return
145
324
  }
146
- return replaced.join(', ')
147
- })
148
- .join(', ')
149
- }
150
325
 
151
- let perParentApplies = new Map()
326
+ // Don't replace multiple instances of the same class
327
+ // This is theoretically correct but only partially
328
+ // We'd need to generate every possible permutation of the replacement
329
+ // For example with `.foo + .foo { … }` and `section { @apply foo; }`
330
+ // We'd need to generate all of these:
331
+ // - `.foo + .foo`
332
+ // - `.foo + section`
333
+ // - `section + .foo`
334
+ // - `section + section`
335
+ if (hasReplaced) {
336
+ return
337
+ }
152
338
 
153
- // Collect all apply candidates and their rules
154
- for (let apply of applies) {
155
- let candidates = perParentApplies.get(apply.parent) || []
339
+ // Since you can only `@apply` class names this is sufficient
340
+ // We want to replace the matched class name with the selector the user is using
341
+ // Ex: Replace `.text-blue-500` with `.foo.bar:is(.something-cool)`
342
+ node.replaceWith(...sel.nodes.map((node) => node.clone()))
156
343
 
157
- perParentApplies.set(apply.parent, candidates)
344
+ // Record that we did something and we want to use this new selector
345
+ replaced.add(utilitySelector)
158
346
 
159
- let [applyCandidates, important] = extractApplyCandidates(apply.params)
347
+ hasReplaced = true
348
+ })
349
+ })
350
+
351
+ // Sort tag names before class names (but only sort each group (separated by a combinator)
352
+ // separately and not in total)
353
+ // This happens when replacing `.bar` in `.foo.bar` with a tag like `section`
354
+ for (let sel of replaced) {
355
+ let groups = [[]]
356
+ for (let node of sel.nodes) {
357
+ if (node.type === 'combinator') {
358
+ groups.push(node)
359
+ groups.push([])
360
+ } else {
361
+ let last = groups[groups.length - 1]
362
+ last.push(node)
363
+ }
364
+ }
160
365
 
161
- if (apply.parent.type === 'atrule') {
162
- if (apply.parent.name === 'screen') {
163
- const screenType = apply.parent.params
366
+ sel.nodes = []
367
+
368
+ for (let group of groups) {
369
+ if (Array.isArray(group)) {
370
+ group.sort((a, b) => {
371
+ if (a.type === 'tag' && b.type === 'class') {
372
+ return -1
373
+ } else if (a.type === 'class' && b.type === 'tag') {
374
+ return 1
375
+ } else if (a.type === 'class' && b.type === 'pseudo' && b.value.startsWith('::')) {
376
+ return -1
377
+ } else if (a.type === 'pseudo' && a.value.startsWith('::') && b.type === 'class') {
378
+ return 1
379
+ }
380
+
381
+ return 0
382
+ })
383
+ }
164
384
 
165
- throw apply.error(
166
- `@apply is not supported within nested at-rules like @screen. We suggest you write this as @apply ${applyCandidates
167
- .map((c) => `${screenType}:${c}`)
168
- .join(' ')} instead.`
169
- )
385
+ sel.nodes = sel.nodes.concat(group)
170
386
  }
387
+ }
388
+
389
+ sel.replaceWith(...replaced)
390
+ })
391
+
392
+ return selectorList.toString()
393
+ }
394
+
395
+ let perParentApplies = new Map()
396
+
397
+ // Collect all apply candidates and their rules
398
+ for (let apply of applies) {
399
+ let [candidates] = perParentApplies.get(apply.parent) || [[], apply.source]
400
+
401
+ perParentApplies.set(apply.parent, [candidates, apply.source])
402
+
403
+ let [applyCandidates, important] = extractApplyCandidates(apply.params)
404
+
405
+ if (apply.parent.type === 'atrule') {
406
+ if (apply.parent.name === 'screen') {
407
+ let screenType = apply.parent.params
171
408
 
172
409
  throw apply.error(
173
- `@apply is not supported within nested at-rules like @${apply.parent.name}. You can fix this by un-nesting @${apply.parent.name}.`
410
+ `@apply is not supported within nested at-rules like @screen. We suggest you write this as @apply ${applyCandidates
411
+ .map((c) => `${screenType}:${c}`)
412
+ .join(' ')} instead.`
174
413
  )
175
414
  }
176
415
 
177
- for (let applyCandidate of applyCandidates) {
178
- if (!applyClassCache.has(applyCandidate)) {
179
- if (applyCandidate === prefix(context, 'group')) {
180
- // TODO: Link to specific documentation page with error code.
181
- throw apply.error(`@apply should not be used with the '${applyCandidate}' utility`)
182
- }
416
+ throw apply.error(
417
+ `@apply is not supported within nested at-rules like @${apply.parent.name}. You can fix this by un-nesting @${apply.parent.name}.`
418
+ )
419
+ }
420
+
421
+ for (let applyCandidate of applyCandidates) {
422
+ if ([prefix(context, 'group'), prefix(context, 'peer')].includes(applyCandidate)) {
423
+ // TODO: Link to specific documentation page with error code.
424
+ throw apply.error(`@apply should not be used with the '${applyCandidate}' utility`)
425
+ }
426
+
427
+ if (!applyClassCache.has(applyCandidate)) {
428
+ throw apply.error(
429
+ `The \`${applyCandidate}\` class does not exist. If \`${applyCandidate}\` is a custom class, make sure it is defined within a \`@layer\` directive.`
430
+ )
431
+ }
432
+
433
+ let rules = applyClassCache.get(applyCandidate)
434
+
435
+ candidates.push([applyCandidate, important, rules])
436
+ }
437
+ }
438
+
439
+ for (let [parent, [candidates, atApplySource]] of perParentApplies) {
440
+ let siblings = []
183
441
 
184
- throw apply.error(
185
- `The \`${applyCandidate}\` class does not exist. If \`${applyCandidate}\` is a custom class, make sure it is defined within a \`@layer\` directive.`
442
+ for (let [applyCandidate, important, rules] of candidates) {
443
+ let potentialApplyCandidates = [
444
+ applyCandidate,
445
+ ...extractBaseCandidates([applyCandidate], context.tailwindConfig.separator),
446
+ ]
447
+
448
+ for (let [meta, node] of rules) {
449
+ let parentClasses = extractClasses(parent)
450
+ let nodeClasses = extractClasses(node)
451
+
452
+ // When we encounter a rule like `.dark .a, .b { … }` we only want to be left with `[.dark, .a]` if the base applyCandidate is `.a` or with `[.b]` if the base applyCandidate is `.b`
453
+ // So we've split them into groups
454
+ nodeClasses = nodeClasses.groups
455
+ .filter((classList) =>
456
+ classList.some((className) => potentialApplyCandidates.includes(className))
457
+ )
458
+ .flat()
459
+
460
+ // Add base utility classes from the @apply node to the list of
461
+ // classes to check whether it intersects and therefore results in a
462
+ // circular dependency or not.
463
+ //
464
+ // E.g.:
465
+ // .foo {
466
+ // @apply hover:a; // This applies "a" but with a modifier
467
+ // }
468
+ //
469
+ // We only have to do that with base classes of the `node`, not of the `parent`
470
+ // E.g.:
471
+ // .hover\:foo {
472
+ // @apply bar;
473
+ // }
474
+ // .bar {
475
+ // @apply foo;
476
+ // }
477
+ //
478
+ // This should not result in a circular dependency because we are
479
+ // just applying `.foo` and the rule above is `.hover\:foo` which is
480
+ // unrelated. However, if we were to apply `hover:foo` then we _did_
481
+ // have to include this one.
482
+ nodeClasses = nodeClasses.concat(
483
+ extractBaseCandidates(nodeClasses, context.tailwindConfig.separator)
484
+ )
485
+
486
+ let intersects = parentClasses.some((selector) => nodeClasses.includes(selector))
487
+ if (intersects) {
488
+ throw node.error(
489
+ `You cannot \`@apply\` the \`${applyCandidate}\` utility here because it creates a circular dependency.`
186
490
  )
187
491
  }
188
492
 
189
- let rules = applyClassCache.get(applyCandidate)
493
+ let root = postcss.root({ nodes: [node.clone()] })
190
494
 
191
- candidates.push([applyCandidate, important, rules])
192
- }
193
- }
495
+ // Make sure every node in the entire tree points back at the @apply rule that generated it
496
+ root.walk((node) => {
497
+ node.source = atApplySource
498
+ })
194
499
 
195
- for (const [parent, candidates] of perParentApplies) {
196
- let siblings = []
500
+ let canRewriteSelector =
501
+ node.type !== 'atrule' || (node.type === 'atrule' && node.name !== 'keyframes')
502
+
503
+ if (canRewriteSelector) {
504
+ root.walkRules((rule) => {
505
+ // Let's imagine you have the following structure:
506
+ //
507
+ // .foo {
508
+ // @apply bar;
509
+ // }
510
+ //
511
+ // @supports (a: b) {
512
+ // .bar {
513
+ // color: blue
514
+ // }
515
+ //
516
+ // .something-unrelated {}
517
+ // }
518
+ //
519
+ // In this case we want to apply `.bar` but it happens to be in
520
+ // an atrule node. We clone that node instead of the nested one
521
+ // because we still want that @supports rule to be there once we
522
+ // applied everything.
523
+ //
524
+ // However it happens to be that the `.something-unrelated` is
525
+ // also in that same shared @supports atrule. This is not good,
526
+ // and this should not be there. The good part is that this is
527
+ // a clone already and it can be safely removed. The question is
528
+ // how do we know we can remove it. Basically what we can do is
529
+ // match it against the applyCandidate that you want to apply. If
530
+ // it doesn't match the we can safely delete it.
531
+ //
532
+ // If we didn't do this, then the `replaceSelector` function
533
+ // would have replaced this with something that didn't exist and
534
+ // therefore it removed the selector altogether. In this specific
535
+ // case it would result in `{}` instead of `.something-unrelated {}`
536
+ if (!extractClasses(rule).some((candidate) => candidate === applyCandidate)) {
537
+ rule.remove()
538
+ return
539
+ }
540
+
541
+ // Strip the important selector from the parent selector if at the beginning
542
+ let importantSelector =
543
+ typeof context.tailwindConfig.important === 'string'
544
+ ? context.tailwindConfig.important
545
+ : null
546
+
547
+ // We only want to move the "important" selector if this is a Tailwind-generated utility
548
+ // We do *not* want to do this for user CSS that happens to be structured the same
549
+ let isGenerated = parent.raws.tailwind !== undefined
550
+
551
+ let parentSelector =
552
+ isGenerated && importantSelector && parent.selector.indexOf(importantSelector) === 0
553
+ ? parent.selector.slice(importantSelector.length)
554
+ : parent.selector
197
555
 
198
- for (let [applyCandidate, important, rules] of candidates) {
199
- for (let [meta, node] of rules) {
200
- let root = postcss.root({ nodes: [node.clone()] })
201
- let canRewriteSelector =
202
- node.type !== 'atrule' || (node.type === 'atrule' && node.name !== 'keyframes')
556
+ rule.selector = replaceSelector(parentSelector, rule.selector, applyCandidate)
203
557
 
204
- if (canRewriteSelector) {
205
- root.walkRules((rule) => {
206
- rule.selector = replaceSelector(parent.selector, rule.selector, applyCandidate)
558
+ // And then re-add it if it was removed
559
+ if (importantSelector && parentSelector !== parent.selector) {
560
+ rule.selector = applyImportantSelector(rule.selector, importantSelector)
561
+ }
207
562
 
208
- rule.walkDecls((d) => {
209
- d.important = meta.important || important
210
- })
563
+ rule.walkDecls((d) => {
564
+ d.important = meta.important || important
211
565
  })
212
- }
213
566
 
214
- // Insert it
215
- siblings.push([
216
- // Ensure that when we are sorting, that we take the layer order into account
217
- { ...meta, sort: meta.sort | context.layerOrder[meta.layer] },
218
- root.nodes[0],
219
- ])
567
+ // Move pseudo elements to the end of the selector (if necessary)
568
+ let selector = parser().astSync(rule.selector)
569
+ selector.each((sel) => movePseudos(sel))
570
+ rule.selector = selector.toString()
571
+ })
220
572
  }
221
- }
222
-
223
- // Inject the rules, sorted, correctly
224
- let nodes = siblings.sort(([a], [z]) => bigSign(a.sort - z.sort)).map((s) => s[1])
225
573
 
226
- // console.log(parent)
227
- // `parent` refers to the node at `.abc` in: .abc { @apply mt-2 }
228
- parent.after(nodes)
229
- }
574
+ // It could be that the node we were inserted was removed because the class didn't match
575
+ // If that was the *only* rule in the parent, then we have nothing add so we skip it
576
+ if (!root.nodes[0]) {
577
+ continue
578
+ }
230
579
 
231
- for (let apply of applies) {
232
- // If there are left-over declarations, just remove the @apply
233
- if (apply.parent.nodes.length > 1) {
234
- apply.remove()
235
- } else {
236
- // The node is empty, drop the full node
237
- apply.parent.remove()
580
+ // Insert it
581
+ siblings.push([meta.sort, root.nodes[0]])
238
582
  }
239
583
  }
240
584
 
241
- // Do it again, in case we have other `@apply` rules
242
- processApply(root, context)
585
+ // Inject the rules, sorted, correctly
586
+ let nodes = context.offsets.sort(siblings).map((s) => s[1])
587
+
588
+ // `parent` refers to the node at `.abc` in: .abc { @apply mt-2 }
589
+ parent.after(nodes)
590
+ }
591
+
592
+ for (let apply of applies) {
593
+ // If there are left-over declarations, just remove the @apply
594
+ if (apply.parent.nodes.length > 1) {
595
+ apply.remove()
596
+ } else {
597
+ // The node is empty, drop the full node
598
+ apply.parent.remove()
599
+ }
243
600
  }
601
+
602
+ // Do it again, in case we have other `@apply` rules
603
+ processApply(root, context, localCache)
244
604
  }
245
605
 
246
606
  export default function expandApplyAtRules(context) {
247
607
  return (root) => {
248
- partitionApplyParents(root)
249
- processApply(root, context)
608
+ // Build a cache of the user's CSS so we can use it to resolve classes used by @apply
609
+ let localCache = lazyCache(() => buildLocalApplyCache(root, context))
610
+
611
+ processApply(root, context, localCache)
250
612
  }
251
613
  }