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
@@ -9,7 +9,7 @@
9
9
  box-sizing: border-box; /* 1 */
10
10
  border-width: 0; /* 2 */
11
11
  border-style: solid; /* 2 */
12
- border-color: currentColor; /* 2 */
12
+ border-color: theme('borderColor.DEFAULT', currentColor); /* 2 */
13
13
  }
14
14
 
15
15
  ::before,
@@ -22,6 +22,8 @@
22
22
  2. Prevent adjustments of font size after orientation changes in iOS.
23
23
  3. Use a more readable tab size.
24
24
  4. Use the user's configured `sans` font-family by default.
25
+ 5. Use the user's configured `sans` font-feature-settings by default.
26
+ 6. Use the user's configured `sans` font-variation-settings by default.
25
27
  */
26
28
 
27
29
  html {
@@ -30,6 +32,8 @@ html {
30
32
  -moz-tab-size: 4; /* 3 */
31
33
  tab-size: 4; /* 3 */
32
34
  font-family: theme('fontFamily.sans', ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"); /* 4 */
35
+ font-feature-settings: theme('fontFamily.sans[1].fontFeatureSettings', normal); /* 5 */
36
+ font-variation-settings: theme('fontFamily.sans[1].fontVariationSettings', normal); /* 6 */
33
37
  }
34
38
 
35
39
  /*
@@ -58,7 +62,7 @@ hr {
58
62
  Add the correct text decoration in Chrome, Edge, and Safari.
59
63
  */
60
64
 
61
- abbr[title] {
65
+ abbr:where([title]) {
62
66
  text-decoration: underline dotted;
63
67
  }
64
68
 
@@ -144,7 +148,7 @@ sup {
144
148
  table {
145
149
  text-indent: 0; /* 1 */
146
150
  border-color: inherit; /* 2 */
147
- border-collapse: collapse; /* 3 */
151
+ border-collapse: collapse; /* 3 */
148
152
  }
149
153
 
150
154
  /*
@@ -160,6 +164,7 @@ select,
160
164
  textarea {
161
165
  font-family: inherit; /* 1 */
162
166
  font-size: 100%; /* 1 */
167
+ font-weight: inherit; /* 1 */
163
168
  line-height: inherit; /* 1 */
164
169
  color: inherit; /* 1 */
165
170
  margin: 0; /* 2 */
@@ -310,7 +315,7 @@ textarea {
310
315
 
311
316
  input::placeholder,
312
317
  textarea::placeholder {
313
- opacity: 1; /* 1 */
318
+ opacity: 1; /* 1 */
314
319
  color: theme('colors.gray.400', #9ca3af); /* 2 */
315
320
  }
316
321
 
@@ -358,10 +363,7 @@ video {
358
363
  height: auto;
359
364
  }
360
365
 
361
- /*
362
- Ensure the default browser behavior of the `hidden` attribute.
363
- */
364
-
366
+ /* Make elements with the HTML hidden attribute stay hidden by default */
365
367
  [hidden] {
366
368
  display: none;
367
369
  }
@@ -1,13 +1,29 @@
1
- import chalk from 'chalk'
1
+ import colors from 'picocolors'
2
2
  import log from './util/log'
3
3
 
4
4
  let defaults = {
5
- optimizeUniversalDefaults: true,
5
+ optimizeUniversalDefaults: false,
6
+ generalizedModifiers: true,
7
+ get disableColorOpacityUtilitiesByDefault() {
8
+ return __OXIDE__
9
+ },
10
+ get relativeContentPathsByDefault() {
11
+ return __OXIDE__
12
+ },
6
13
  }
7
14
 
8
15
  let featureFlags = {
9
- future: [],
10
- experimental: ['optimizeUniversalDefaults'],
16
+ future: [
17
+ 'hoverOnlyWhenSupported',
18
+ 'respectDefaultRingColorOpacity',
19
+ 'disableColorOpacityUtilitiesByDefault',
20
+ 'relativeContentPathsByDefault',
21
+ ],
22
+ experimental: [
23
+ 'optimizeUniversalDefaults',
24
+ 'generalizedModifiers',
25
+ // 'variantGrouping',
26
+ ],
11
27
  }
12
28
 
13
29
  export function flagEnabled(config, flag) {
@@ -41,7 +57,7 @@ export function issueFlagNotices(config) {
41
57
 
42
58
  if (experimentalFlagsEnabled(config).length > 0) {
43
59
  let changes = experimentalFlagsEnabled(config)
44
- .map((s) => chalk.yellow(s))
60
+ .map((s) => colors.yellow(s))
45
61
  .join(', ')
46
62
 
47
63
  log.warn('experimental-flags-enabled', [
package/src/index.js CHANGED
@@ -1,34 +1 @@
1
- import setupTrackingContext from './lib/setupTrackingContext'
2
- import setupWatchingContext from './lib/setupWatchingContext'
3
- import processTailwindFeatures from './processTailwindFeatures'
4
- import { env } from './lib/sharedState'
5
-
6
- module.exports = function tailwindcss(configOrPath) {
7
- return {
8
- postcssPlugin: 'tailwindcss',
9
- plugins: [
10
- env.DEBUG &&
11
- function (root) {
12
- console.log('\n')
13
- console.time('JIT TOTAL')
14
- return root
15
- },
16
- function (root, result) {
17
- let setupContext =
18
- env.TAILWIND_MODE === 'watch'
19
- ? setupWatchingContext(configOrPath)
20
- : setupTrackingContext(configOrPath)
21
-
22
- processTailwindFeatures(setupContext)(root, result)
23
- },
24
- env.DEBUG &&
25
- function (root) {
26
- console.timeEnd('JIT TOTAL')
27
- console.log('\n')
28
- return root
29
- },
30
- ].filter(Boolean),
31
- }
32
- }
33
-
34
- module.exports.postcss = true
1
+ module.exports = require('./plugin')
@@ -0,0 +1,52 @@
1
+ import crypto from 'crypto'
2
+ import * as sharedState from './sharedState'
3
+
4
+ /**
5
+ * Calculate the hash of a string.
6
+ *
7
+ * This doesn't need to be cryptographically secure or
8
+ * anything like that since it's used only to detect
9
+ * when the CSS changes to invalidate the context.
10
+ *
11
+ * This is wrapped in a try/catch because it's really dependent
12
+ * on how Node itself is build and the environment and OpenSSL
13
+ * version / build that is installed on the user's machine.
14
+ *
15
+ * Based on the environment this can just outright fail.
16
+ *
17
+ * See https://github.com/nodejs/node/issues/40455
18
+ *
19
+ * @param {string} str
20
+ */
21
+ function getHash(str) {
22
+ try {
23
+ return crypto.createHash('md5').update(str, 'utf-8').digest('binary')
24
+ } catch (err) {
25
+ return ''
26
+ }
27
+ }
28
+
29
+ /**
30
+ * Determine if the CSS tree is different from the
31
+ * previous version for the given `sourcePath`.
32
+ *
33
+ * @param {string} sourcePath
34
+ * @param {import('postcss').Node} root
35
+ */
36
+ export function hasContentChanged(sourcePath, root) {
37
+ let css = root.toString()
38
+
39
+ // We only care about files with @tailwind directives
40
+ // Other files use an existing context
41
+ if (!css.includes('@tailwind')) {
42
+ return false
43
+ }
44
+
45
+ let existingHash = sharedState.sourceHashMap.get(sourcePath)
46
+ let rootHash = getHash(css)
47
+ let didChange = existingHash !== rootHash
48
+
49
+ sharedState.sourceHashMap.set(sourcePath, rootHash)
50
+
51
+ return didChange
52
+ }
@@ -5,7 +5,7 @@ let comparisonMap = {
5
5
  let types = new Set(Object.keys(comparisonMap))
6
6
 
7
7
  export default function collapseAdjacentRules() {
8
- return (root) => {
8
+ function collapseRulesIn(root) {
9
9
  let currentRule = null
10
10
  root.each((node) => {
11
11
  if (!types.has(node.type)) {
@@ -29,11 +29,30 @@ export default function collapseAdjacentRules() {
29
29
  (currentRule[property] ?? '').replace(/\s+/g, ' ')
30
30
  )
31
31
  ) {
32
- currentRule.append(node.nodes)
32
+ // An AtRule may not have children (for example if we encounter duplicate @import url(…) rules)
33
+ if (node.nodes) {
34
+ currentRule.append(node.nodes)
35
+ }
36
+
33
37
  node.remove()
34
38
  } else {
35
39
  currentRule = node
36
40
  }
37
41
  })
42
+
43
+ // After we've collapsed adjacent rules & at-rules, we need to collapse
44
+ // adjacent rules & at-rules that are children of at-rules.
45
+ // We do not care about nesting rules because Tailwind CSS
46
+ // explicitly does not handle rule nesting on its own as
47
+ // the user is expected to use a nesting plugin
48
+ root.each((node) => {
49
+ if (node.type === 'atrule') {
50
+ collapseRulesIn(node)
51
+ }
52
+ })
53
+ }
54
+
55
+ return (root) => {
56
+ collapseRulesIn(root)
38
57
  }
39
58
  }
@@ -3,6 +3,7 @@ export default function collapseDuplicateDeclarations() {
3
3
  root.walkRules((node) => {
4
4
  let seen = new Map()
5
5
  let droppable = new Set([])
6
+ let byProperty = new Map()
6
7
 
7
8
  node.walkDecls((decl) => {
8
9
  // This could happen if we have nested selectors. In that case the
@@ -14,15 +15,79 @@ export default function collapseDuplicateDeclarations() {
14
15
  }
15
16
 
16
17
  if (seen.has(decl.prop)) {
17
- droppable.add(seen.get(decl.prop))
18
+ // Exact same value as what we have seen so far
19
+ if (seen.get(decl.prop).value === decl.value) {
20
+ // Keep the last one, drop the one we've seen so far
21
+ droppable.add(seen.get(decl.prop))
22
+ // Override the existing one with the new value. This is necessary
23
+ // so that if we happen to have more than one declaration with the
24
+ // same value, that we keep removing the previous one. Otherwise we
25
+ // will only remove the *first* one.
26
+ seen.set(decl.prop, decl)
27
+ return
28
+ }
29
+
30
+ // Not the same value, so we need to check if we can merge it so
31
+ // let's collect it first.
32
+ if (!byProperty.has(decl.prop)) {
33
+ byProperty.set(decl.prop, new Set())
34
+ }
35
+
36
+ byProperty.get(decl.prop).add(seen.get(decl.prop))
37
+ byProperty.get(decl.prop).add(decl)
18
38
  }
19
39
 
20
40
  seen.set(decl.prop, decl)
21
41
  })
22
42
 
43
+ // Drop all the duplicate declarations with the exact same value we've
44
+ // already seen so far.
23
45
  for (let decl of droppable) {
24
46
  decl.remove()
25
47
  }
48
+
49
+ // Analyze the declarations based on its unit, drop all the declarations
50
+ // with the same unit but the last one in the list.
51
+ for (let declarations of byProperty.values()) {
52
+ let byUnit = new Map()
53
+
54
+ for (let decl of declarations) {
55
+ let unit = resolveUnit(decl.value)
56
+ if (unit === null) {
57
+ // We don't have a unit, so should never try and collapse this
58
+ // value. This is because we can't know how to do it in a correct
59
+ // way (e.g.: overrides for older browsers)
60
+ continue
61
+ }
62
+
63
+ if (!byUnit.has(unit)) {
64
+ byUnit.set(unit, new Set())
65
+ }
66
+
67
+ byUnit.get(unit).add(decl)
68
+ }
69
+
70
+ for (let declarations of byUnit.values()) {
71
+ // Get all but the last one
72
+ let removableDeclarations = Array.from(declarations).slice(0, -1)
73
+
74
+ for (let decl of removableDeclarations) {
75
+ decl.remove()
76
+ }
77
+ }
78
+ }
26
79
  })
27
80
  }
28
81
  }
82
+
83
+ let UNITLESS_NUMBER = Symbol('unitless-number')
84
+
85
+ function resolveUnit(input) {
86
+ let result = /^-?\d*.?\d+([\w%]+)?$/g.exec(input)
87
+
88
+ if (result) {
89
+ return result[1] ?? UNITLESS_NUMBER
90
+ }
91
+
92
+ return null
93
+ }
@@ -0,0 +1,208 @@
1
+ // @ts-check
2
+
3
+ import fs from 'fs'
4
+ import path from 'path'
5
+ import isGlob from 'is-glob'
6
+ import fastGlob from 'fast-glob'
7
+ import normalizePath from 'normalize-path'
8
+ import { parseGlob } from '../util/parseGlob'
9
+ import { env } from './sharedState'
10
+
11
+ /** @typedef {import('../../types/config.js').RawFile} RawFile */
12
+ /** @typedef {import('../../types/config.js').FilePath} FilePath */
13
+
14
+ /**
15
+ * @typedef {object} ContentPath
16
+ * @property {string} original
17
+ * @property {string} base
18
+ * @property {string | null} glob
19
+ * @property {boolean} ignore
20
+ * @property {string} pattern
21
+ */
22
+
23
+ /**
24
+ * Turn a list of content paths (absolute or not; glob or not) into a list of
25
+ * absolute file paths that exist on the filesystem
26
+ *
27
+ * If there are symlinks in the path then multiple paths will be returned
28
+ * one for the symlink and one for the actual file
29
+ *
30
+ * @param {*} context
31
+ * @param {import('tailwindcss').Config} tailwindConfig
32
+ * @returns {ContentPath[]}
33
+ */
34
+ export function parseCandidateFiles(context, tailwindConfig) {
35
+ let files = tailwindConfig.content.files
36
+
37
+ // Normalize the file globs
38
+ files = files.filter((filePath) => typeof filePath === 'string')
39
+ files = files.map(normalizePath)
40
+
41
+ // Split into included and excluded globs
42
+ let tasks = fastGlob.generateTasks(files)
43
+
44
+ /** @type {ContentPath[]} */
45
+ let included = []
46
+
47
+ /** @type {ContentPath[]} */
48
+ let excluded = []
49
+
50
+ for (const task of tasks) {
51
+ included.push(...task.positive.map((filePath) => parseFilePath(filePath, false)))
52
+ excluded.push(...task.negative.map((filePath) => parseFilePath(filePath, true)))
53
+ }
54
+
55
+ let paths = [...included, ...excluded]
56
+
57
+ // Resolve paths relative to the config file or cwd
58
+ paths = resolveRelativePaths(context, paths)
59
+
60
+ // Resolve symlinks if possible
61
+ paths = paths.flatMap(resolvePathSymlinks)
62
+
63
+ // Update cached patterns
64
+ paths = paths.map(resolveGlobPattern)
65
+
66
+ return paths
67
+ }
68
+
69
+ /**
70
+ *
71
+ * @param {string} filePath
72
+ * @param {boolean} ignore
73
+ * @returns {ContentPath}
74
+ */
75
+ function parseFilePath(filePath, ignore) {
76
+ let contentPath = {
77
+ original: filePath,
78
+ base: filePath,
79
+ ignore,
80
+ pattern: filePath,
81
+ glob: null,
82
+ }
83
+
84
+ if (isGlob(filePath)) {
85
+ Object.assign(contentPath, parseGlob(filePath))
86
+ }
87
+
88
+ return contentPath
89
+ }
90
+
91
+ /**
92
+ *
93
+ * @param {ContentPath} contentPath
94
+ * @returns {ContentPath}
95
+ */
96
+ function resolveGlobPattern(contentPath) {
97
+ // This is required for Windows support to properly pick up Glob paths.
98
+ // Afaik, this technically shouldn't be needed but there's probably
99
+ // some internal, direct path matching with a normalized path in
100
+ // a package which can't handle mixed directory separators
101
+ let base = normalizePath(contentPath.base)
102
+
103
+ // If the user's file path contains any special characters (like parens) for instance fast-glob
104
+ // is like "OOOH SHINY" and treats them as such. So we have to escape the base path to fix this
105
+ base = fastGlob.escapePath(base)
106
+
107
+ contentPath.pattern = contentPath.glob ? `${base}/${contentPath.glob}` : base
108
+ contentPath.pattern = contentPath.ignore ? `!${contentPath.pattern}` : contentPath.pattern
109
+
110
+ return contentPath
111
+ }
112
+
113
+ /**
114
+ * Resolve each path relative to the config file (when possible) if the experimental flag is enabled
115
+ * Otherwise, resolve relative to the current working directory
116
+ *
117
+ * @param {any} context
118
+ * @param {ContentPath[]} contentPaths
119
+ * @returns {ContentPath[]}
120
+ */
121
+ function resolveRelativePaths(context, contentPaths) {
122
+ let resolveFrom = []
123
+
124
+ // Resolve base paths relative to the config file (when possible) if the experimental flag is enabled
125
+ if (context.userConfigPath && context.tailwindConfig.content.relative) {
126
+ resolveFrom = [path.dirname(context.userConfigPath)]
127
+ }
128
+
129
+ return contentPaths.map((contentPath) => {
130
+ contentPath.base = path.resolve(...resolveFrom, contentPath.base)
131
+
132
+ return contentPath
133
+ })
134
+ }
135
+
136
+ /**
137
+ * Resolve the symlink for the base directory / file in each path
138
+ * These are added as additional dependencies to watch for changes because
139
+ * some tools (like webpack) will only watch the actual file or directory
140
+ * but not the symlink itself even in projects that use monorepos.
141
+ *
142
+ * @param {ContentPath} contentPath
143
+ * @returns {ContentPath[]}
144
+ */
145
+ function resolvePathSymlinks(contentPath) {
146
+ let paths = [contentPath]
147
+
148
+ try {
149
+ let resolvedPath = fs.realpathSync(contentPath.base)
150
+ if (resolvedPath !== contentPath.base) {
151
+ paths.push({
152
+ ...contentPath,
153
+ base: resolvedPath,
154
+ })
155
+ }
156
+ } catch {
157
+ // TODO: log this?
158
+ }
159
+
160
+ return paths
161
+ }
162
+
163
+ /**
164
+ * @param {any} context
165
+ * @param {ContentPath[]} candidateFiles
166
+ * @param {Map<string, number>} fileModifiedMap
167
+ * @returns {[{ content: string, extension: string }[], Map<string, number>]}
168
+ */
169
+ export function resolvedChangedContent(context, candidateFiles, fileModifiedMap) {
170
+ let changedContent = context.tailwindConfig.content.files
171
+ .filter((item) => typeof item.raw === 'string')
172
+ .map(({ raw, extension = 'html' }) => ({ content: raw, extension }))
173
+
174
+ let [changedFiles, mTimesToCommit] = resolveChangedFiles(candidateFiles, fileModifiedMap)
175
+
176
+ for (let changedFile of changedFiles) {
177
+ let extension = path.extname(changedFile).slice(1)
178
+ changedContent.push({ file: changedFile, extension })
179
+ }
180
+
181
+ return [changedContent, mTimesToCommit]
182
+ }
183
+
184
+ /**
185
+ *
186
+ * @param {ContentPath[]} candidateFiles
187
+ * @param {Map<string, number>} fileModifiedMap
188
+ * @returns {[Set<string>, Map<string, number>]}
189
+ */
190
+ function resolveChangedFiles(candidateFiles, fileModifiedMap) {
191
+ let paths = candidateFiles.map((contentPath) => contentPath.pattern)
192
+ let mTimesToCommit = new Map()
193
+
194
+ let changedFiles = new Set()
195
+ env.DEBUG && console.time('Finding changed files')
196
+ let files = fastGlob.sync(paths, { absolute: true })
197
+ for (let file of files) {
198
+ let prevModified = fileModifiedMap.get(file) || -Infinity
199
+ let modified = fs.statSync(file).mtimeMs
200
+
201
+ if (modified > prevModified) {
202
+ changedFiles.add(file)
203
+ mTimesToCommit.set(file, modified)
204
+ }
205
+ }
206
+ env.DEBUG && console.timeEnd('Finding changed files')
207
+ return [changedFiles, mTimesToCommit]
208
+ }