tailwindcss 3.1.7 → 3.2.0

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 (112) hide show
  1. package/README.md +12 -8
  2. package/lib/cli/build/deps.js +54 -0
  3. package/lib/cli/build/index.js +44 -0
  4. package/lib/cli/build/plugin.js +335 -0
  5. package/lib/cli/build/utils.js +78 -0
  6. package/lib/cli/build/watching.js +113 -0
  7. package/lib/cli/help/index.js +71 -0
  8. package/lib/cli/index.js +18 -0
  9. package/lib/cli/init/index.js +46 -0
  10. package/lib/cli/shared.js +12 -0
  11. package/lib/cli.js +11 -590
  12. package/lib/corePlugins.js +332 -108
  13. package/lib/css/preflight.css +5 -0
  14. package/lib/featureFlags.js +7 -4
  15. package/lib/index.js +6 -1
  16. package/lib/lib/content.js +167 -0
  17. package/lib/lib/defaultExtractor.js +15 -10
  18. package/lib/lib/detectNesting.js +2 -2
  19. package/lib/lib/evaluateTailwindFunctions.js +17 -1
  20. package/lib/lib/expandApplyAtRules.js +71 -37
  21. package/lib/lib/expandTailwindAtRules.js +10 -42
  22. package/lib/lib/findAtConfigPath.js +44 -0
  23. package/lib/lib/generateRules.js +181 -96
  24. package/lib/lib/normalizeTailwindDirectives.js +1 -1
  25. package/lib/lib/offsets.js +217 -0
  26. package/lib/lib/regex.js +1 -1
  27. package/lib/lib/setupContextUtils.js +339 -100
  28. package/lib/lib/setupTrackingContext.js +5 -39
  29. package/lib/lib/sharedState.js +2 -0
  30. package/lib/public/colors.js +1 -1
  31. package/lib/util/buildMediaQuery.js +6 -3
  32. package/lib/util/configurePlugins.js +1 -1
  33. package/lib/util/dataTypes.js +15 -19
  34. package/lib/util/formatVariantSelector.js +92 -8
  35. package/lib/util/getAllConfigs.js +14 -3
  36. package/lib/util/isValidArbitraryValue.js +1 -1
  37. package/lib/util/nameClass.js +3 -0
  38. package/lib/util/negateValue.js +15 -2
  39. package/lib/util/normalizeConfig.js +17 -3
  40. package/lib/util/normalizeScreens.js +100 -3
  41. package/lib/util/parseAnimationValue.js +1 -1
  42. package/lib/util/parseBoxShadowValue.js +1 -1
  43. package/lib/util/parseDependency.js +33 -54
  44. package/lib/util/parseGlob.js +34 -0
  45. package/lib/util/parseObjectStyles.js +1 -1
  46. package/lib/util/pluginUtils.js +86 -17
  47. package/lib/util/resolveConfig.js +3 -3
  48. package/lib/util/splitAtTopLevelOnly.js +31 -81
  49. package/lib/util/transformThemeValue.js +9 -2
  50. package/lib/util/validateConfig.js +1 -1
  51. package/lib/util/validateFormalSyntax.js +24 -0
  52. package/package.json +14 -12
  53. package/peers/.DS_Store +0 -0
  54. package/peers/.svgo.yml +75 -0
  55. package/peers/index.js +3690 -2274
  56. package/peers/orders/concentric-css.json +299 -0
  57. package/peers/orders/smacss.json +299 -0
  58. package/peers/orders/source.json +295 -0
  59. package/plugin.d.ts +3 -3
  60. package/scripts/release-channel.js +18 -0
  61. package/scripts/release-notes.js +21 -0
  62. package/src/.DS_Store +0 -0
  63. package/src/cli/build/deps.js +56 -0
  64. package/src/cli/build/index.js +45 -0
  65. package/src/cli/build/plugin.js +397 -0
  66. package/src/cli/build/utils.js +76 -0
  67. package/src/cli/build/watching.js +134 -0
  68. package/src/cli/help/index.js +70 -0
  69. package/src/cli/index.js +3 -0
  70. package/src/cli/init/index.js +50 -0
  71. package/src/cli/shared.js +5 -0
  72. package/src/cli.js +4 -696
  73. package/src/corePlugins.js +262 -39
  74. package/src/css/preflight.css +5 -0
  75. package/src/featureFlags.js +12 -2
  76. package/src/index.js +5 -0
  77. package/src/lib/content.js +205 -0
  78. package/src/lib/defaultExtractor.js +3 -0
  79. package/src/lib/evaluateTailwindFunctions.js +22 -1
  80. package/src/lib/expandApplyAtRules.js +76 -29
  81. package/src/lib/expandTailwindAtRules.js +8 -46
  82. package/src/lib/findAtConfigPath.js +48 -0
  83. package/src/lib/generateRules.js +224 -105
  84. package/src/lib/offsets.js +270 -0
  85. package/src/lib/setupContextUtils.js +376 -89
  86. package/src/lib/setupTrackingContext.js +4 -45
  87. package/src/lib/sharedState.js +2 -0
  88. package/src/util/buildMediaQuery.js +5 -3
  89. package/src/util/dataTypes.js +15 -17
  90. package/src/util/formatVariantSelector.js +113 -9
  91. package/src/util/getAllConfigs.js +14 -2
  92. package/src/util/nameClass.js +4 -0
  93. package/src/util/negateValue.js +10 -2
  94. package/src/util/normalizeConfig.js +22 -2
  95. package/src/util/normalizeScreens.js +99 -4
  96. package/src/util/parseBoxShadowValue.js +1 -1
  97. package/src/util/parseDependency.js +37 -42
  98. package/src/util/parseGlob.js +24 -0
  99. package/src/util/pluginUtils.js +90 -14
  100. package/src/util/resolveConfig.js +2 -2
  101. package/src/util/splitAtTopLevelOnly.js +23 -49
  102. package/src/util/transformThemeValue.js +9 -1
  103. package/src/util/validateFormalSyntax.js +34 -0
  104. package/stubs/defaultConfig.stub.js +19 -3
  105. package/tmp.css +11 -0
  106. package/tmp.dependency-graph.js +2 -0
  107. package/tmp.in.css +3 -0
  108. package/tmp.js +0 -0
  109. package/tmp.out.css +524 -0
  110. package/types/config.d.ts +47 -13
  111. package/types/generated/default-theme.d.ts +11 -0
  112. package/CHANGELOG.md +0 -2222
package/README.md CHANGED
@@ -1,15 +1,19 @@
1
- <p>
2
- <a href="https://tailwindcss.com/#gh-light-mode-only" target="_blank">
3
- <img src="./.github/logo-light.svg" alt="Tailwind CSS" width="350" height="70">
4
- </a>
5
- <a href="https://tailwindcss.com/#gh-dark-mode-only" target="_blank">
6
- <img src="./.github/logo-dark.svg" alt="Tailwind CSS" width="350" height="70">
1
+ <p align="center">
2
+ <a href="https://tailwindcss.com" target="_blank">
3
+ <picture>
4
+ <source media="(prefers-color-scheme: dark)" srcset="./.github/logo-dark.svg">
5
+ <source media="(prefers-color-scheme: light)" srcset="./.github/logo-light.svg">
6
+ <img alt="Tailwind CSS" src="./.github/logo-light.svg" width="350" height="70" style="max-width: 100%;">
7
+ </picture>
7
8
  </a>
8
9
  </p>
9
10
 
10
- A utility-first CSS framework for rapidly building custom user interfaces.
11
+ <p align="center">
12
+ A utility-first CSS framework for rapidly building custom user interfaces.
13
+ </p>
14
+
11
15
 
12
- <p>
16
+ <p align="center">
13
17
  <a href="https://github.com/tailwindlabs/tailwindcss/actions"><img src="https://img.shields.io/github/workflow/status/tailwindlabs/tailwindcss/Node.js%20CI" alt="Build Status"></a>
14
18
  <a href="https://www.npmjs.com/package/tailwindcss"><img src="https://img.shields.io/npm/dt/tailwindcss.svg" alt="Total Downloads"></a>
15
19
  <a href="https://github.com/tailwindcss/tailwindcss/releases"><img src="https://img.shields.io/npm/v/tailwindcss.svg" alt="Latest Release"></a>
@@ -0,0 +1,54 @@
1
+ // @ts-check
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ function _export(target, all) {
7
+ for(var name in all)Object.defineProperty(target, name, {
8
+ enumerable: true,
9
+ get: all[name]
10
+ });
11
+ }
12
+ _export(exports, {
13
+ loadPostcss: ()=>loadPostcss,
14
+ loadPostcssImport: ()=>loadPostcssImport,
15
+ loadCssNano: ()=>loadCssNano,
16
+ loadAutoprefixer: ()=>loadAutoprefixer
17
+ });
18
+ const _indexJs = require("../../../peers/index.js");
19
+ function loadPostcss() {
20
+ // Try to load a local `postcss` version first
21
+ try {
22
+ return require("postcss");
23
+ } catch {}
24
+ return (0, _indexJs.lazyPostcss)();
25
+ }
26
+ function loadPostcssImport() {
27
+ // Try to load a local `postcss-import` version first
28
+ try {
29
+ return require("postcss-import");
30
+ } catch {}
31
+ return (0, _indexJs.lazyPostcssImport)();
32
+ }
33
+ function loadCssNano() {
34
+ let options = {
35
+ preset: [
36
+ "default",
37
+ {
38
+ cssDeclarationSorter: false
39
+ }
40
+ ]
41
+ };
42
+ // Try to load a local `cssnano` version first
43
+ try {
44
+ return require("cssnano");
45
+ } catch {}
46
+ return (0, _indexJs.lazyCssnano)()(options);
47
+ }
48
+ function loadAutoprefixer() {
49
+ // Try to load a local `autoprefixer` version first
50
+ try {
51
+ return require("autoprefixer");
52
+ } catch {}
53
+ return (0, _indexJs.lazyAutoprefixer)();
54
+ }
@@ -0,0 +1,44 @@
1
+ // @ts-check
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ Object.defineProperty(exports, "build", {
7
+ enumerable: true,
8
+ get: ()=>build
9
+ });
10
+ const _fs = /*#__PURE__*/ _interopRequireDefault(require("fs"));
11
+ const _path = /*#__PURE__*/ _interopRequireDefault(require("path"));
12
+ const _pluginJs = require("./plugin.js");
13
+ function _interopRequireDefault(obj) {
14
+ return obj && obj.__esModule ? obj : {
15
+ default: obj
16
+ };
17
+ }
18
+ async function build(args, configs) {
19
+ let input = args["--input"];
20
+ let shouldWatch = args["--watch"];
21
+ // TODO: Deprecate this in future versions
22
+ if (!input && args["_"][1]) {
23
+ console.error("[deprecation] Running tailwindcss without -i, please provide an input file.");
24
+ input = args["--input"] = args["_"][1];
25
+ }
26
+ if (input && input !== "-" && !_fs.default.existsSync(input = _path.default.resolve(input))) {
27
+ console.error(`Specified input file ${args["--input"]} does not exist.`);
28
+ process.exit(9);
29
+ }
30
+ if (args["--config"] && !_fs.default.existsSync(args["--config"] = _path.default.resolve(args["--config"]))) {
31
+ console.error(`Specified config file ${args["--config"]} does not exist.`);
32
+ process.exit(9);
33
+ }
34
+ // TODO: Reference the @config path here if exists
35
+ let configPath = args["--config"] ? args["--config"] : ((defaultPath)=>_fs.default.existsSync(defaultPath) ? defaultPath : null)(_path.default.resolve(`./${configs.tailwind}`));
36
+ let processor = await (0, _pluginJs.createProcessor)(args, configPath);
37
+ if (shouldWatch) {
38
+ /* Abort the watcher if stdin is closed to avoid zombie processes */ process.stdin.on("end", ()=>process.exit(0));
39
+ process.stdin.resume();
40
+ await processor.watch();
41
+ } else {
42
+ await processor.build();
43
+ }
44
+ }
@@ -0,0 +1,335 @@
1
+ // @ts-check
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ Object.defineProperty(exports, "createProcessor", {
7
+ enumerable: true,
8
+ get: ()=>createProcessor
9
+ });
10
+ const _path = /*#__PURE__*/ _interopRequireDefault(require("path"));
11
+ const _fs = /*#__PURE__*/ _interopRequireDefault(require("fs"));
12
+ const _postcssLoadConfig = /*#__PURE__*/ _interopRequireDefault(require("postcss-load-config"));
13
+ const _lilconfig = require("lilconfig");
14
+ const _plugins = /*#__PURE__*/ _interopRequireDefault(require("postcss-load-config/src/plugins" // Little bit scary, looking at private/internal API
15
+ ));
16
+ const _options = /*#__PURE__*/ _interopRequireDefault(require("postcss-load-config/src/options" // Little bit scary, looking at private/internal API
17
+ ));
18
+ const _processTailwindFeatures = /*#__PURE__*/ _interopRequireDefault(require("../../processTailwindFeatures"));
19
+ const _deps = require("./deps");
20
+ const _utils = require("./utils");
21
+ const _shared = require("../shared");
22
+ const _resolveConfigJs = /*#__PURE__*/ _interopRequireDefault(require("../../../resolveConfig.js"));
23
+ const _getModuleDependenciesJs = /*#__PURE__*/ _interopRequireDefault(require("../../lib/getModuleDependencies.js"));
24
+ const _contentJs = require("../../lib/content.js");
25
+ const _watchingJs = require("./watching.js");
26
+ const _fastGlob = /*#__PURE__*/ _interopRequireDefault(require("fast-glob"));
27
+ const _findAtConfigPathJs = require("../../lib/findAtConfigPath.js");
28
+ const _log = /*#__PURE__*/ _interopRequireDefault(require("../../util/log"));
29
+ function _interopRequireDefault(obj) {
30
+ return obj && obj.__esModule ? obj : {
31
+ default: obj
32
+ };
33
+ }
34
+ /**
35
+ *
36
+ * @param {string} [customPostCssPath ]
37
+ * @returns
38
+ */ async function loadPostCssPlugins(customPostCssPath) {
39
+ let config = customPostCssPath ? await (async ()=>{
40
+ let file = _path.default.resolve(customPostCssPath);
41
+ // Implementation, see: https://unpkg.com/browse/postcss-load-config@3.1.0/src/index.js
42
+ // @ts-ignore
43
+ let { config ={} } = await (0, _lilconfig.lilconfig)("postcss").load(file);
44
+ if (typeof config === "function") {
45
+ config = config();
46
+ } else {
47
+ config = Object.assign({}, config);
48
+ }
49
+ if (!config.plugins) {
50
+ config.plugins = [];
51
+ }
52
+ return {
53
+ file,
54
+ plugins: (0, _plugins.default)(config, file),
55
+ options: (0, _options.default)(config, file)
56
+ };
57
+ })() : await (0, _postcssLoadConfig.default)();
58
+ let configPlugins = config.plugins;
59
+ let configPluginTailwindIdx = configPlugins.findIndex((plugin)=>{
60
+ if (typeof plugin === "function" && plugin.name === "tailwindcss") {
61
+ return true;
62
+ }
63
+ if (typeof plugin === "object" && plugin !== null && plugin.postcssPlugin === "tailwindcss") {
64
+ return true;
65
+ }
66
+ return false;
67
+ });
68
+ let beforePlugins = configPluginTailwindIdx === -1 ? [] : configPlugins.slice(0, configPluginTailwindIdx);
69
+ let afterPlugins = configPluginTailwindIdx === -1 ? configPlugins : configPlugins.slice(configPluginTailwindIdx + 1);
70
+ return [
71
+ beforePlugins,
72
+ afterPlugins,
73
+ config.options
74
+ ];
75
+ }
76
+ function loadBuiltinPostcssPlugins() {
77
+ let postcss = (0, _deps.loadPostcss)();
78
+ let IMPORT_COMMENT = "__TAILWIND_RESTORE_IMPORT__: ";
79
+ return [
80
+ [
81
+ (root)=>{
82
+ root.walkAtRules("import", (rule)=>{
83
+ if (rule.params.slice(1).startsWith("tailwindcss/")) {
84
+ rule.after(postcss.comment({
85
+ text: IMPORT_COMMENT + rule.params
86
+ }));
87
+ rule.remove();
88
+ }
89
+ });
90
+ },
91
+ (0, _deps.loadPostcssImport)(),
92
+ (root)=>{
93
+ root.walkComments((rule)=>{
94
+ if (rule.text.startsWith(IMPORT_COMMENT)) {
95
+ rule.after(postcss.atRule({
96
+ name: "import",
97
+ params: rule.text.replace(IMPORT_COMMENT, "")
98
+ }));
99
+ rule.remove();
100
+ }
101
+ });
102
+ }
103
+ ],
104
+ [],
105
+ {}
106
+ ];
107
+ }
108
+ let state = {
109
+ /** @type {any} */ context: null,
110
+ /** @type {ReturnType<typeof createWatcher> | null} */ watcher: null,
111
+ /** @type {{content: string, extension: string}[]} */ changedContent: [],
112
+ configDependencies: new Set(),
113
+ contextDependencies: new Set(),
114
+ /** @type {import('../../lib/content.js').ContentPath[]} */ contentPaths: [],
115
+ refreshContentPaths () {
116
+ var ref;
117
+ this.contentPaths = (0, _contentJs.parseCandidateFiles)(this.context, (ref = this.context) === null || ref === void 0 ? void 0 : ref.tailwindConfig);
118
+ },
119
+ get config () {
120
+ return this.context.tailwindConfig;
121
+ },
122
+ get contentPatterns () {
123
+ return {
124
+ all: this.contentPaths.map((contentPath)=>contentPath.pattern),
125
+ dynamic: this.contentPaths.filter((contentPath)=>contentPath.glob !== undefined).map((contentPath)=>contentPath.pattern)
126
+ };
127
+ },
128
+ loadConfig (configPath, content) {
129
+ if (this.watcher && configPath) {
130
+ this.refreshConfigDependencies(configPath);
131
+ }
132
+ let config = configPath ? require(configPath) : {};
133
+ // @ts-ignore
134
+ config = (0, _resolveConfigJs.default)(config, {
135
+ content: {
136
+ files: []
137
+ }
138
+ });
139
+ // Override content files if `--content` has been passed explicitly
140
+ if ((content === null || content === void 0 ? void 0 : content.length) > 0) {
141
+ config.content.files = content;
142
+ }
143
+ return config;
144
+ },
145
+ refreshConfigDependencies (configPath) {
146
+ _shared.env.DEBUG && console.time("Module dependencies");
147
+ for (let file of this.configDependencies){
148
+ delete require.cache[require.resolve(file)];
149
+ }
150
+ if (configPath) {
151
+ let deps = (0, _getModuleDependenciesJs.default)(configPath).map(({ file })=>file);
152
+ for (let dependency of deps){
153
+ this.configDependencies.add(dependency);
154
+ }
155
+ }
156
+ _shared.env.DEBUG && console.timeEnd("Module dependencies");
157
+ },
158
+ readContentPaths () {
159
+ let content = [];
160
+ // Resolve globs from the content config
161
+ // TODO: When we make the postcss plugin async-capable this can become async
162
+ let files = _fastGlob.default.sync(this.contentPatterns.all);
163
+ for (let file of files){
164
+ content.push({
165
+ content: _fs.default.readFileSync(_path.default.resolve(file), "utf8"),
166
+ extension: _path.default.extname(file).slice(1)
167
+ });
168
+ }
169
+ // Resolve raw content in the tailwind config
170
+ let rawContent = this.config.content.files.filter((file)=>{
171
+ return file !== null && typeof file === "object";
172
+ });
173
+ for (let { raw: content1 , extension ="html" } of rawContent){
174
+ content1.push({
175
+ content: content1,
176
+ extension
177
+ });
178
+ }
179
+ return content;
180
+ },
181
+ getContext ({ createContext , cliConfigPath , root , result , content }) {
182
+ if (this.context) {
183
+ this.context.changedContent = this.changedContent.splice(0);
184
+ return this.context;
185
+ }
186
+ _shared.env.DEBUG && console.time("Searching for config");
187
+ var ref;
188
+ let configPath = (ref = (0, _findAtConfigPathJs.findAtConfigPath)(root, result)) !== null && ref !== void 0 ? ref : cliConfigPath;
189
+ _shared.env.DEBUG && console.timeEnd("Searching for config");
190
+ _shared.env.DEBUG && console.time("Loading config");
191
+ let config = this.loadConfig(configPath, content);
192
+ _shared.env.DEBUG && console.timeEnd("Loading config");
193
+ _shared.env.DEBUG && console.time("Creating context");
194
+ this.context = createContext(config, []);
195
+ Object.assign(this.context, {
196
+ userConfigPath: configPath
197
+ });
198
+ _shared.env.DEBUG && console.timeEnd("Creating context");
199
+ _shared.env.DEBUG && console.time("Resolving content paths");
200
+ this.refreshContentPaths();
201
+ _shared.env.DEBUG && console.timeEnd("Resolving content paths");
202
+ if (this.watcher) {
203
+ _shared.env.DEBUG && console.time("Watch new files");
204
+ this.watcher.refreshWatchedFiles();
205
+ _shared.env.DEBUG && console.timeEnd("Watch new files");
206
+ }
207
+ _shared.env.DEBUG && console.time("Reading content files");
208
+ for (let file of this.readContentPaths()){
209
+ this.context.changedContent.push(file);
210
+ }
211
+ _shared.env.DEBUG && console.timeEnd("Reading content files");
212
+ return this.context;
213
+ }
214
+ };
215
+ async function createProcessor(args, cliConfigPath) {
216
+ var ref;
217
+ let postcss = (0, _deps.loadPostcss)();
218
+ let input = args["--input"];
219
+ let output = args["--output"];
220
+ let includePostCss = args["--postcss"];
221
+ let customPostCssPath = typeof args["--postcss"] === "string" ? args["--postcss"] : undefined;
222
+ let [beforePlugins, afterPlugins, postcssOptions] = includePostCss ? await loadPostCssPlugins(customPostCssPath) : loadBuiltinPostcssPlugins();
223
+ if (args["--purge"]) {
224
+ _log.default.warn("purge-flag-deprecated", [
225
+ "The `--purge` flag has been deprecated.",
226
+ "Please use `--content` instead."
227
+ ]);
228
+ if (!args["--content"]) {
229
+ args["--content"] = args["--purge"];
230
+ }
231
+ }
232
+ var ref1;
233
+ let content = (ref1 = (ref = args["--content"]) === null || ref === void 0 ? void 0 : ref.split(/(?<!{[^}]+),/)) !== null && ref1 !== void 0 ? ref1 : [];
234
+ let tailwindPlugin = ()=>{
235
+ return {
236
+ postcssPlugin: "tailwindcss",
237
+ Once (root, { result }) {
238
+ _shared.env.DEBUG && console.time("Compiling CSS");
239
+ (0, _processTailwindFeatures.default)(({ createContext })=>{
240
+ console.error();
241
+ console.error("Rebuilding...");
242
+ return ()=>{
243
+ return state.getContext({
244
+ createContext,
245
+ cliConfigPath,
246
+ root,
247
+ result,
248
+ content
249
+ });
250
+ };
251
+ })(root, result);
252
+ _shared.env.DEBUG && console.timeEnd("Compiling CSS");
253
+ }
254
+ };
255
+ };
256
+ tailwindPlugin.postcss = true;
257
+ let plugins = [
258
+ ...beforePlugins,
259
+ tailwindPlugin,
260
+ !args["--minify"] && _utils.formatNodes,
261
+ ...afterPlugins,
262
+ !args["--no-autoprefixer"] && (0, _deps.loadAutoprefixer)(),
263
+ args["--minify"] && (0, _deps.loadCssNano)()
264
+ ].filter(Boolean);
265
+ /** @type {import('postcss').Processor} */ // @ts-ignore
266
+ let processor = postcss(plugins);
267
+ async function readInput() {
268
+ // Piping in data, let's drain the stdin
269
+ if (input === "-") {
270
+ return (0, _utils.drainStdin)();
271
+ }
272
+ // Input file has been provided
273
+ if (input) {
274
+ return _fs.default.promises.readFile(_path.default.resolve(input), "utf8");
275
+ }
276
+ // No input file provided, fallback to default atrules
277
+ return "@tailwind base; @tailwind components; @tailwind utilities";
278
+ }
279
+ async function build() {
280
+ let start = process.hrtime.bigint();
281
+ return readInput().then((css)=>processor.process(css, {
282
+ ...postcssOptions,
283
+ from: input,
284
+ to: output
285
+ })).then((result)=>{
286
+ if (!output) {
287
+ process.stdout.write(result.css);
288
+ return;
289
+ }
290
+ return Promise.all([
291
+ (0, _utils.outputFile)(output, result.css),
292
+ result.map && (0, _utils.outputFile)(output + ".map", result.map.toString())
293
+ ]);
294
+ }).then(()=>{
295
+ let end = process.hrtime.bigint();
296
+ console.error();
297
+ console.error("Done in", (end - start) / BigInt(1e6) + "ms.");
298
+ });
299
+ }
300
+ /**
301
+ * @param {{file: string, content(): Promise<string>, extension: string}[]} changes
302
+ */ async function parseChanges(changes) {
303
+ return Promise.all(changes.map(async (change)=>({
304
+ content: await change.content(),
305
+ extension: change.extension
306
+ })));
307
+ }
308
+ if (input !== undefined && input !== "-") {
309
+ state.contextDependencies.add(_path.default.resolve(input));
310
+ }
311
+ return {
312
+ build,
313
+ watch: async ()=>{
314
+ state.watcher = (0, _watchingJs.createWatcher)(args, {
315
+ state,
316
+ /**
317
+ * @param {{file: string, content(): Promise<string>, extension: string}[]} changes
318
+ */ async rebuild (changes) {
319
+ let needsNewContext = changes.some((change)=>{
320
+ return state.configDependencies.has(change.file) || state.contextDependencies.has(change.file);
321
+ });
322
+ if (needsNewContext) {
323
+ state.context = null;
324
+ } else {
325
+ for (let change of (await parseChanges(changes))){
326
+ state.changedContent.push(change);
327
+ }
328
+ }
329
+ return build();
330
+ }
331
+ });
332
+ await build();
333
+ }
334
+ };
335
+ }
@@ -0,0 +1,78 @@
1
+ // @ts-check
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ function _export(target, all) {
7
+ for(var name in all)Object.defineProperty(target, name, {
8
+ enumerable: true,
9
+ get: all[name]
10
+ });
11
+ }
12
+ _export(exports, {
13
+ indentRecursive: ()=>indentRecursive,
14
+ formatNodes: ()=>formatNodes,
15
+ readFileWithRetries: ()=>readFileWithRetries,
16
+ drainStdin: ()=>drainStdin,
17
+ outputFile: ()=>outputFile
18
+ });
19
+ const _fs = /*#__PURE__*/ _interopRequireDefault(require("fs"));
20
+ const _path = /*#__PURE__*/ _interopRequireDefault(require("path"));
21
+ function _interopRequireDefault(obj) {
22
+ return obj && obj.__esModule ? obj : {
23
+ default: obj
24
+ };
25
+ }
26
+ function indentRecursive(node, indent = 0) {
27
+ node.each && node.each((child, i)=>{
28
+ if (!child.raws.before || !child.raws.before.trim() || child.raws.before.includes("\n")) {
29
+ child.raws.before = `\n${node.type !== "rule" && i > 0 ? "\n" : ""}${" ".repeat(indent)}`;
30
+ }
31
+ child.raws.after = `\n${" ".repeat(indent)}`;
32
+ indentRecursive(child, indent + 1);
33
+ });
34
+ }
35
+ function formatNodes(root) {
36
+ indentRecursive(root);
37
+ if (root.first) {
38
+ root.first.raws.before = "";
39
+ }
40
+ }
41
+ async function readFileWithRetries(path, tries = 5) {
42
+ for(let n = 0; n <= tries; n++){
43
+ try {
44
+ return await _fs.default.promises.readFile(path, "utf8");
45
+ } catch (err) {
46
+ if (n !== tries) {
47
+ if (err.code === "ENOENT" || err.code === "EBUSY") {
48
+ await new Promise((resolve)=>setTimeout(resolve, 10));
49
+ continue;
50
+ }
51
+ }
52
+ throw err;
53
+ }
54
+ }
55
+ }
56
+ function drainStdin() {
57
+ return new Promise((resolve, reject)=>{
58
+ let result = "";
59
+ process.stdin.on("data", (chunk)=>{
60
+ result += chunk;
61
+ });
62
+ process.stdin.on("end", ()=>resolve(result));
63
+ process.stdin.on("error", (err)=>reject(err));
64
+ });
65
+ }
66
+ async function outputFile(file, newContents) {
67
+ try {
68
+ let currentContents = await _fs.default.promises.readFile(file, "utf8");
69
+ if (currentContents === newContents) {
70
+ return; // Skip writing the file
71
+ }
72
+ } catch {}
73
+ // Write the file
74
+ await _fs.default.promises.mkdir(_path.default.dirname(file), {
75
+ recursive: true
76
+ });
77
+ await _fs.default.promises.writeFile(file, newContents, "utf8");
78
+ }
@@ -0,0 +1,113 @@
1
+ // @ts-check
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ Object.defineProperty(exports, "createWatcher", {
7
+ enumerable: true,
8
+ get: ()=>createWatcher
9
+ });
10
+ const _chokidar = /*#__PURE__*/ _interopRequireDefault(require("chokidar"));
11
+ const _fs = /*#__PURE__*/ _interopRequireDefault(require("fs"));
12
+ const _micromatch = /*#__PURE__*/ _interopRequireDefault(require("micromatch"));
13
+ const _normalizePath = /*#__PURE__*/ _interopRequireDefault(require("normalize-path"));
14
+ const _path = /*#__PURE__*/ _interopRequireDefault(require("path"));
15
+ const _utilsJs = require("./utils.js");
16
+ function _interopRequireDefault(obj) {
17
+ return obj && obj.__esModule ? obj : {
18
+ default: obj
19
+ };
20
+ }
21
+ function createWatcher(args, { state , rebuild }) {
22
+ let shouldPoll = args["--poll"];
23
+ let shouldCoalesceWriteEvents = shouldPoll || process.platform === "win32";
24
+ // Polling interval in milliseconds
25
+ // Used only when polling or coalescing add/change events on Windows
26
+ let pollInterval = 10;
27
+ let watcher = _chokidar.default.watch([], {
28
+ // Force checking for atomic writes in all situations
29
+ // This causes chokidar to wait up to 100ms for a file to re-added after it's been unlinked
30
+ // This only works when watching directories though
31
+ atomic: true,
32
+ usePolling: shouldPoll,
33
+ interval: shouldPoll ? pollInterval : undefined,
34
+ ignoreInitial: true,
35
+ awaitWriteFinish: shouldCoalesceWriteEvents ? {
36
+ stabilityThreshold: 50,
37
+ pollInterval: pollInterval
38
+ } : false
39
+ });
40
+ let chain = Promise.resolve();
41
+ let pendingRebuilds = new Set();
42
+ let changedContent = [];
43
+ /**
44
+ *
45
+ * @param {*} file
46
+ * @param {(() => Promise<string>) | null} content
47
+ */ function recordChangedFile(file, content = null) {
48
+ file = _path.default.resolve(file);
49
+ content = content !== null && content !== void 0 ? content : async ()=>await _fs.default.promises.readFile(file, "utf8");
50
+ changedContent.push({
51
+ file,
52
+ content,
53
+ extension: _path.default.extname(file).slice(1)
54
+ });
55
+ chain = chain.then(()=>rebuild(changedContent));
56
+ return chain;
57
+ }
58
+ watcher.on("change", (file)=>recordChangedFile(file));
59
+ watcher.on("add", (file)=>recordChangedFile(file));
60
+ // Restore watching any files that are "removed"
61
+ // This can happen when a file is pseudo-atomically replaced (a copy is created, overwritten, the old one is unlinked, and the new one is renamed)
62
+ // TODO: An an optimization we should allow removal when the config changes
63
+ watcher.on("unlink", (file)=>{
64
+ file = (0, _normalizePath.default)(file);
65
+ // Only re-add the file if it's not covered by a dynamic pattern
66
+ if (!_micromatch.default.some([
67
+ file
68
+ ], state.contentPatterns.dynamic)) {
69
+ watcher.add(file);
70
+ }
71
+ });
72
+ // Some applications such as Visual Studio (but not VS Code)
73
+ // will only fire a rename event for atomic writes and not a change event
74
+ // This is very likely a chokidar bug but it's one we need to work around
75
+ // We treat this as a change event and rebuild the CSS
76
+ watcher.on("raw", (evt, filePath, meta)=>{
77
+ if (evt !== "rename") {
78
+ return;
79
+ }
80
+ let watchedPath = meta.watchedPath;
81
+ // Watched path might be the file itself
82
+ // Or the directory it is in
83
+ filePath = watchedPath.endsWith(filePath) ? watchedPath : _path.default.join(watchedPath, filePath);
84
+ // Skip this event since the files it is for does not match any of the registered content globs
85
+ if (!_micromatch.default.some([
86
+ filePath
87
+ ], state.contentPatterns.all)) {
88
+ return;
89
+ }
90
+ // Skip since we've already queued a rebuild for this file that hasn't happened yet
91
+ if (pendingRebuilds.has(filePath)) {
92
+ return;
93
+ }
94
+ pendingRebuilds.add(filePath);
95
+ chain = chain.then(async ()=>{
96
+ let content;
97
+ try {
98
+ content = await (0, _utilsJs.readFileWithRetries)(_path.default.resolve(filePath));
99
+ } finally{
100
+ pendingRebuilds.delete(filePath);
101
+ }
102
+ return recordChangedFile(filePath, ()=>content);
103
+ });
104
+ });
105
+ return {
106
+ fswatcher: watcher,
107
+ refreshWatchedFiles () {
108
+ watcher.add(Array.from(state.contextDependencies));
109
+ watcher.add(Array.from(state.configDependencies));
110
+ watcher.add(state.contentPatterns.all);
111
+ }
112
+ };
113
+ }