tailwindcss 3.1.8 → 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.
- package/README.md +6 -5
- package/lib/cli/build/deps.js +54 -0
- package/lib/cli/build/index.js +44 -0
- package/lib/cli/build/plugin.js +335 -0
- package/lib/cli/build/utils.js +78 -0
- package/lib/cli/build/watching.js +113 -0
- package/lib/cli/help/index.js +71 -0
- package/lib/cli/index.js +18 -0
- package/lib/cli/init/index.js +46 -0
- package/lib/cli/shared.js +12 -0
- package/lib/cli.js +11 -590
- package/lib/corePlugins.js +332 -108
- package/lib/css/preflight.css +5 -0
- package/lib/featureFlags.js +7 -4
- package/lib/index.js +6 -1
- package/lib/lib/content.js +167 -0
- package/lib/lib/defaultExtractor.js +15 -10
- package/lib/lib/detectNesting.js +2 -2
- package/lib/lib/evaluateTailwindFunctions.js +17 -1
- package/lib/lib/expandApplyAtRules.js +66 -37
- package/lib/lib/expandTailwindAtRules.js +10 -42
- package/lib/lib/findAtConfigPath.js +44 -0
- package/lib/lib/generateRules.js +180 -93
- package/lib/lib/normalizeTailwindDirectives.js +1 -1
- package/lib/lib/offsets.js +217 -0
- package/lib/lib/regex.js +1 -1
- package/lib/lib/setupContextUtils.js +339 -100
- package/lib/lib/setupTrackingContext.js +5 -39
- package/lib/lib/sharedState.js +2 -0
- package/lib/public/colors.js +1 -1
- package/lib/util/buildMediaQuery.js +6 -3
- package/lib/util/configurePlugins.js +1 -1
- package/lib/util/dataTypes.js +15 -19
- package/lib/util/formatVariantSelector.js +92 -8
- package/lib/util/getAllConfigs.js +14 -3
- package/lib/util/isValidArbitraryValue.js +1 -1
- package/lib/util/nameClass.js +3 -0
- package/lib/util/negateValue.js +15 -2
- package/lib/util/normalizeConfig.js +17 -3
- package/lib/util/normalizeScreens.js +100 -3
- package/lib/util/parseAnimationValue.js +1 -1
- package/lib/util/parseBoxShadowValue.js +1 -1
- package/lib/util/parseDependency.js +33 -54
- package/lib/util/parseGlob.js +34 -0
- package/lib/util/parseObjectStyles.js +1 -1
- package/lib/util/pluginUtils.js +86 -17
- package/lib/util/resolveConfig.js +2 -2
- package/lib/util/splitAtTopLevelOnly.js +31 -81
- package/lib/util/transformThemeValue.js +9 -2
- package/lib/util/validateConfig.js +1 -1
- package/lib/util/validateFormalSyntax.js +24 -0
- package/package.json +13 -11
- package/peers/.DS_Store +0 -0
- package/peers/.svgo.yml +75 -0
- package/peers/index.js +3332 -2032
- package/peers/orders/concentric-css.json +299 -0
- package/peers/orders/smacss.json +299 -0
- package/peers/orders/source.json +295 -0
- package/plugin.d.ts +3 -3
- package/scripts/release-channel.js +18 -0
- package/scripts/release-notes.js +21 -0
- package/src/.DS_Store +0 -0
- package/src/cli/build/deps.js +56 -0
- package/src/cli/build/index.js +45 -0
- package/src/cli/build/plugin.js +397 -0
- package/src/cli/build/utils.js +76 -0
- package/src/cli/build/watching.js +134 -0
- package/src/cli/help/index.js +70 -0
- package/src/cli/index.js +3 -0
- package/src/cli/init/index.js +50 -0
- package/src/cli/shared.js +5 -0
- package/src/cli.js +4 -696
- package/src/corePlugins.js +262 -39
- package/src/css/preflight.css +5 -0
- package/src/featureFlags.js +12 -2
- package/src/index.js +5 -0
- package/src/lib/content.js +205 -0
- package/src/lib/defaultExtractor.js +3 -0
- package/src/lib/evaluateTailwindFunctions.js +22 -1
- package/src/lib/expandApplyAtRules.js +70 -29
- package/src/lib/expandTailwindAtRules.js +8 -46
- package/src/lib/findAtConfigPath.js +48 -0
- package/src/lib/generateRules.js +223 -101
- package/src/lib/offsets.js +270 -0
- package/src/lib/setupContextUtils.js +376 -89
- package/src/lib/setupTrackingContext.js +4 -45
- package/src/lib/sharedState.js +2 -0
- package/src/util/buildMediaQuery.js +5 -3
- package/src/util/dataTypes.js +15 -17
- package/src/util/formatVariantSelector.js +113 -9
- package/src/util/getAllConfigs.js +14 -2
- package/src/util/nameClass.js +4 -0
- package/src/util/negateValue.js +10 -2
- package/src/util/normalizeConfig.js +22 -2
- package/src/util/normalizeScreens.js +99 -4
- package/src/util/parseBoxShadowValue.js +1 -1
- package/src/util/parseDependency.js +37 -42
- package/src/util/parseGlob.js +24 -0
- package/src/util/pluginUtils.js +90 -14
- package/src/util/resolveConfig.js +1 -1
- package/src/util/splitAtTopLevelOnly.js +23 -49
- package/src/util/transformThemeValue.js +9 -1
- package/src/util/validateFormalSyntax.js +34 -0
- package/stubs/defaultConfig.stub.js +19 -3
- package/tmp.css +11 -0
- package/tmp.dependency-graph.js +2 -0
- package/tmp.in.css +3 -0
- package/tmp.js +0 -0
- package/tmp.out.css +524 -0
- package/types/config.d.ts +47 -13
- package/types/generated/default-theme.d.ts +11 -0
- package/CHANGELOG.md +0 -2231
package/README.md
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
<p align="center">
|
|
2
|
-
<a href="https://tailwindcss.com
|
|
3
|
-
<
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
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
|
|
|
@@ -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
|
+
}
|