index.cjs.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. 'use strict';
  2. var fs = require('fs');
  3. var path = require('path');
  4. var postcss = require('postcss');
  5. function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
  6. function _interopNamespace(e) {
  7. if (e && e.__esModule) return e;
  8. var n = Object.create(null);
  9. if (e) {
  10. Object.keys(e).forEach(function (k) {
  11. if (k !== 'default') {
  12. var d = Object.getOwnPropertyDescriptor(e, k);
  13. Object.defineProperty(n, k, d.get ? d : {
  14. enumerable: true,
  15. get: function () {
  16. return e[k];
  17. }
  18. });
  19. }
  20. });
  21. }
  22. n['default'] = e;
  23. return Object.freeze(n);
  24. }
  25. var fs__default = /*#__PURE__*/_interopDefaultLegacy(fs);
  26. var path__default = /*#__PURE__*/_interopDefaultLegacy(path);
  27. function parse(string, splitByAnd) {
  28. const array = [];
  29. let buffer = '';
  30. let split = false;
  31. let func = 0;
  32. let i = -1;
  33. while (++i < string.length) {
  34. const char = string[i];
  35. if (char === '(') {
  36. func += 1;
  37. } else if (char === ')') {
  38. if (func > 0) {
  39. func -= 1;
  40. }
  41. } else if (func === 0) {
  42. if (splitByAnd && andRegExp.test(buffer + char)) {
  43. split = true;
  44. } else if (!splitByAnd && char === ',') {
  45. split = true;
  46. }
  47. }
  48. if (split) {
  49. array.push(splitByAnd ? new MediaExpression(buffer + char) : new MediaQuery(buffer));
  50. buffer = '';
  51. split = false;
  52. } else {
  53. buffer += char;
  54. }
  55. }
  56. if (buffer !== '') {
  57. array.push(splitByAnd ? new MediaExpression(buffer) : new MediaQuery(buffer));
  58. }
  59. return array;
  60. }
  61. class MediaQueryList {
  62. constructor(string) {
  63. this.nodes = parse(string);
  64. }
  65. invert() {
  66. this.nodes.forEach(node => {
  67. node.invert();
  68. });
  69. return this;
  70. }
  71. clone() {
  72. return new MediaQueryList(String(this));
  73. }
  74. toString() {
  75. return this.nodes.join(',');
  76. }
  77. }
  78. class MediaQuery {
  79. constructor(string) {
  80. const [, before, media, after] = string.match(spaceWrapRegExp);
  81. const [, modifier = '', afterModifier = ' ', type = '', beforeAnd = '', and = '', beforeExpression = '', expression1 = '', expression2 = ''] = media.match(mediaRegExp) || [];
  82. const raws = {
  83. before,
  84. after,
  85. afterModifier,
  86. originalModifier: modifier || '',
  87. beforeAnd,
  88. and,
  89. beforeExpression
  90. };
  91. const nodes = parse(expression1 || expression2, true);
  92. Object.assign(this, {
  93. modifier,
  94. type,
  95. raws,
  96. nodes
  97. });
  98. }
  99. clone(overrides) {
  100. const instance = new MediaQuery(String(this));
  101. Object.assign(instance, overrides);
  102. return instance;
  103. }
  104. invert() {
  105. this.modifier = this.modifier ? '' : this.raws.originalModifier;
  106. return this;
  107. }
  108. toString() {
  109. const {
  110. raws
  111. } = this;
  112. return `${raws.before}${this.modifier}${this.modifier ? `${raws.afterModifier}` : ''}${this.type}${raws.beforeAnd}${raws.and}${raws.beforeExpression}${this.nodes.join('')}${this.raws.after}`;
  113. }
  114. }
  115. class MediaExpression {
  116. constructor(string) {
  117. const [, value, after = '', and = '', afterAnd = ''] = string.match(andRegExp) || [null, string];
  118. const raws = {
  119. after,
  120. and,
  121. afterAnd
  122. };
  123. Object.assign(this, {
  124. value,
  125. raws
  126. });
  127. }
  128. clone(overrides) {
  129. const instance = new MediaExpression(String(this));
  130. Object.assign(instance, overrides);
  131. return instance;
  132. }
  133. toString() {
  134. const {
  135. raws
  136. } = this;
  137. return `${this.value}${raws.after}${raws.and}${raws.afterAnd}`;
  138. }
  139. }
  140. const modifierRE = '(not|only)';
  141. const typeRE = '(all|print|screen|speech)';
  142. const noExpressionRE = '([\\W\\w]*)';
  143. const expressionRE = '([\\W\\w]+)';
  144. const noSpaceRE = '(\\s*)';
  145. const spaceRE = '(\\s+)';
  146. const andRE = '(?:(\\s+)(and))';
  147. const andRegExp = new RegExp(`^${expressionRE}(?:${andRE}${spaceRE})$`, 'i');
  148. const spaceWrapRegExp = new RegExp(`^${noSpaceRE}${noExpressionRE}${noSpaceRE}$`);
  149. const mediaRegExp = new RegExp(`^(?:${modifierRE}${spaceRE})?(?:${typeRE}(?:${andRE}${spaceRE}${expressionRE})?|${expressionRE})$`, 'i');
  150. var mediaASTFromString = (string => new MediaQueryList(string));
  151. var getCustomMediaFromRoot = ((root, opts) => {
  152. // initialize custom selectors
  153. const customMedias = {}; // for each custom selector atrule that is a child of the css root
  154. root.nodes.slice().forEach(node => {
  155. if (isCustomMedia(node)) {
  156. // extract the name and selectors from the params of the custom selector
  157. const [, name, selectors] = node.params.match(customMediaParamsRegExp); // write the parsed selectors to the custom selector
  158. customMedias[name] = mediaASTFromString(selectors); // conditionally remove the custom selector atrule
  159. if (!Object(opts).preserve) {
  160. node.remove();
  161. }
  162. }
  163. });
  164. return customMedias;
  165. }); // match the custom selector name
  166. const customMediaNameRegExp = /^custom-media$/i; // match the custom selector params
  167. const customMediaParamsRegExp = /^(--[A-z][\w-]*)\s+([\W\w]+)\s*$/; // whether the atrule is a custom selector
  168. const isCustomMedia = node => node.type === 'atrule' && customMediaNameRegExp.test(node.name) && customMediaParamsRegExp.test(node.params);
  169. /* Get Custom Media from CSS File
  170. /* ========================================================================== */
  171. async function getCustomMediaFromCSSFile(from) {
  172. const css = await readFile(from);
  173. const root = postcss.parse(css, {
  174. from
  175. });
  176. return getCustomMediaFromRoot(root, {
  177. preserve: true
  178. });
  179. }
  180. /* Get Custom Media from Object
  181. /* ========================================================================== */
  182. function getCustomMediaFromObject(object) {
  183. const customMedia = Object.assign({}, Object(object).customMedia, Object(object)['custom-media']);
  184. for (const key in customMedia) {
  185. customMedia[key] = mediaASTFromString(customMedia[key]);
  186. }
  187. return customMedia;
  188. }
  189. /* Get Custom Media from JSON file
  190. /* ========================================================================== */
  191. async function getCustomMediaFromJSONFile(from) {
  192. const object = await readJSON(from);
  193. return getCustomMediaFromObject(object);
  194. }
  195. /* Get Custom Media from JS file
  196. /* ========================================================================== */
  197. async function getCustomMediaFromJSFile(from) {
  198. const object = await Promise.resolve().then(function () { return /*#__PURE__*/_interopNamespace(require(from)); });
  199. return getCustomMediaFromObject(object);
  200. }
  201. /* Get Custom Media from Sources
  202. /* ========================================================================== */
  203. function getCustomMediaFromSources(sources) {
  204. return sources.map(source => {
  205. if (source instanceof Promise) {
  206. return source;
  207. } else if (source instanceof Function) {
  208. return source();
  209. } // read the source as an object
  210. const opts = source === Object(source) ? source : {
  211. from: String(source)
  212. }; // skip objects with custom media
  213. if (Object(opts).customMedia || Object(opts)['custom-media']) {
  214. return opts;
  215. } // source pathname
  216. const from = path__default['default'].resolve(String(opts.from || '')); // type of file being read from
  217. const type = (opts.type || path__default['default'].extname(from).slice(1)).toLowerCase();
  218. return {
  219. type,
  220. from
  221. };
  222. }).reduce(async (customMedia, source) => {
  223. const {
  224. type,
  225. from
  226. } = await source;
  227. if (type === 'css' || type === 'pcss') {
  228. return Object.assign(await customMedia, await getCustomMediaFromCSSFile(from));
  229. }
  230. if (type === 'js') {
  231. return Object.assign(await customMedia, await getCustomMediaFromJSFile(from));
  232. }
  233. if (type === 'json') {
  234. return Object.assign(await customMedia, await getCustomMediaFromJSONFile(from));
  235. }
  236. return Object.assign(await customMedia, getCustomMediaFromObject(await source));
  237. }, {});
  238. }
  239. /* Helper utilities
  240. /* ========================================================================== */
  241. const readFile = from => new Promise((resolve, reject) => {
  242. fs__default['default'].readFile(from, 'utf8', (error, result) => {
  243. if (error) {
  244. reject(error);
  245. } else {
  246. resolve(result);
  247. }
  248. });
  249. });
  250. const readJSON = async from => JSON.parse(await readFile(from));
  251. // return transformed medias, replacing custom pseudo medias with custom medias
  252. function transformMediaList(mediaList, customMedias) {
  253. let index = mediaList.nodes.length - 1;
  254. while (index >= 0) {
  255. const transformedMedias = transformMedia(mediaList.nodes[index], customMedias);
  256. if (transformedMedias.length) {
  257. mediaList.nodes.splice(index, 1, ...transformedMedias);
  258. }
  259. --index;
  260. }
  261. return mediaList;
  262. } // return custom pseudo medias replaced with custom medias
  263. function transformMedia(media, customMedias) {
  264. const transpiledMedias = [];
  265. for (const index in media.nodes) {
  266. const {
  267. value,
  268. nodes
  269. } = media.nodes[index];
  270. const key = value.replace(customPseudoRegExp, '$1');
  271. if (key in customMedias) {
  272. for (const replacementMedia of customMedias[key].nodes) {
  273. // use the first available modifier unless they cancel each other out
  274. const modifier = media.modifier !== replacementMedia.modifier ? media.modifier || replacementMedia.modifier : '';
  275. const mediaClone = media.clone({
  276. modifier,
  277. // conditionally use the raws from the first available modifier
  278. raws: !modifier || media.modifier ? { ...media.raws
  279. } : { ...replacementMedia.raws
  280. },
  281. type: media.type || replacementMedia.type
  282. }); // conditionally include more replacement raws when the type is present
  283. if (mediaClone.type === replacementMedia.type) {
  284. Object.assign(mediaClone.raws, {
  285. and: replacementMedia.raws.and,
  286. beforeAnd: replacementMedia.raws.beforeAnd,
  287. beforeExpression: replacementMedia.raws.beforeExpression
  288. });
  289. }
  290. mediaClone.nodes.splice(index, 1, ...replacementMedia.clone().nodes.map(node => {
  291. // use raws and spacing from the current usage
  292. if (media.nodes[index].raws.and) {
  293. node.raws = { ...media.nodes[index].raws
  294. };
  295. }
  296. node.spaces = { ...media.nodes[index].spaces
  297. };
  298. return node;
  299. })); // remove the currently transformed key to prevent recursion
  300. const nextCustomMedia = getCustomMediasWithoutKey(customMedias, key);
  301. const retranspiledMedias = transformMedia(mediaClone, nextCustomMedia);
  302. if (retranspiledMedias.length) {
  303. transpiledMedias.push(...retranspiledMedias);
  304. } else {
  305. transpiledMedias.push(mediaClone);
  306. }
  307. }
  308. return transpiledMedias;
  309. } else if (nodes && nodes.length) {
  310. transformMediaList(media.nodes[index], customMedias);
  311. }
  312. }
  313. return transpiledMedias;
  314. }
  315. const customPseudoRegExp = /\((--[A-z][\w-]*)\)/;
  316. const getCustomMediasWithoutKey = (customMedias, key) => {
  317. const nextCustomMedias = Object.assign({}, customMedias);
  318. delete nextCustomMedias[key];
  319. return nextCustomMedias;
  320. };
  321. var transformAtrules = ((root, customMedia, opts) => {
  322. root.walkAtRules(mediaAtRuleRegExp, atrule => {
  323. if (customPseudoRegExp$1.test(atrule.params)) {
  324. const mediaAST = mediaASTFromString(atrule.params);
  325. const params = String(transformMediaList(mediaAST, customMedia));
  326. if (opts.preserve) {
  327. atrule.cloneBefore({
  328. params
  329. });
  330. } else {
  331. atrule.params = params;
  332. }
  333. }
  334. });
  335. });
  336. const mediaAtRuleRegExp = /^media$/i;
  337. const customPseudoRegExp$1 = /\(--[A-z][\w-]*\)/;
  338. /* Write Custom Media from CSS File
  339. /* ========================================================================== */
  340. async function writeCustomMediaToCssFile(to, customMedia) {
  341. const cssContent = Object.keys(customMedia).reduce((cssLines, name) => {
  342. cssLines.push(`@custom-media ${name} ${customMedia[name]};`);
  343. return cssLines;
  344. }, []).join('\n');
  345. const css = `${cssContent}\n`;
  346. await writeFile(to, css);
  347. }
  348. /* Write Custom Media from JSON file
  349. /* ========================================================================== */
  350. async function writeCustomMediaToJsonFile(to, customMedia) {
  351. const jsonContent = JSON.stringify({
  352. 'custom-media': customMedia
  353. }, null, ' ');
  354. const json = `${jsonContent}\n`;
  355. await writeFile(to, json);
  356. }
  357. /* Write Custom Media from Common JS file
  358. /* ========================================================================== */
  359. async function writeCustomMediaToCjsFile(to, customMedia) {
  360. const jsContents = Object.keys(customMedia).reduce((jsLines, name) => {
  361. jsLines.push(`\t\t'${escapeForJS(name)}': '${escapeForJS(customMedia[name])}'`);
  362. return jsLines;
  363. }, []).join(',\n');
  364. const js = `module.exports = {\n\tcustomMedia: {\n${jsContents}\n\t}\n};\n`;
  365. await writeFile(to, js);
  366. }
  367. /* Write Custom Media from Module JS file
  368. /* ========================================================================== */
  369. async function writeCustomMediaToMjsFile(to, customMedia) {
  370. const mjsContents = Object.keys(customMedia).reduce((mjsLines, name) => {
  371. mjsLines.push(`\t'${escapeForJS(name)}': '${escapeForJS(customMedia[name])}'`);
  372. return mjsLines;
  373. }, []).join(',\n');
  374. const mjs = `export const customMedia = {\n${mjsContents}\n};\n`;
  375. await writeFile(to, mjs);
  376. }
  377. /* Write Custom Media to Exports
  378. /* ========================================================================== */
  379. function writeCustomMediaToExports(customMedia, destinations) {
  380. return Promise.all(destinations.map(async destination => {
  381. if (destination instanceof Function) {
  382. await destination(defaultCustomMediaToJSON(customMedia));
  383. } else {
  384. // read the destination as an object
  385. const opts = destination === Object(destination) ? destination : {
  386. to: String(destination)
  387. }; // transformer for custom media into a JSON-compatible object
  388. const toJSON = opts.toJSON || defaultCustomMediaToJSON;
  389. if ('customMedia' in opts) {
  390. // write directly to an object as customMedia
  391. opts.customMedia = toJSON(customMedia);
  392. } else if ('custom-media' in opts) {
  393. // write directly to an object as custom-media
  394. opts['custom-media'] = toJSON(customMedia);
  395. } else {
  396. // destination pathname
  397. const to = String(opts.to || ''); // type of file being written to
  398. const type = (opts.type || path__default['default'].extname(to).slice(1)).toLowerCase(); // transformed custom media
  399. const customMediaJSON = toJSON(customMedia);
  400. if (type === 'css') {
  401. await writeCustomMediaToCssFile(to, customMediaJSON);
  402. }
  403. if (type === 'js') {
  404. await writeCustomMediaToCjsFile(to, customMediaJSON);
  405. }
  406. if (type === 'json') {
  407. await writeCustomMediaToJsonFile(to, customMediaJSON);
  408. }
  409. if (type === 'mjs') {
  410. await writeCustomMediaToMjsFile(to, customMediaJSON);
  411. }
  412. }
  413. }
  414. }));
  415. }
  416. /* Helper utilities
  417. /* ========================================================================== */
  418. const defaultCustomMediaToJSON = customMedia => {
  419. return Object.keys(customMedia).reduce((customMediaJSON, key) => {
  420. customMediaJSON[key] = String(customMedia[key]);
  421. return customMediaJSON;
  422. }, {});
  423. };
  424. const writeFile = (to, text) => new Promise((resolve, reject) => {
  425. fs__default['default'].writeFile(to, text, error => {
  426. if (error) {
  427. reject(error);
  428. } else {
  429. resolve();
  430. }
  431. });
  432. });
  433. const escapeForJS = string => string.replace(/\\([\s\S])|(')/g, '\\$1$2').replace(/\n/g, '\\n').replace(/\r/g, '\\r');
  434. const creator = opts => {
  435. // whether to preserve custom media and at-rules using them
  436. const preserve = 'preserve' in Object(opts) ? Boolean(opts.preserve) : false; // sources to import custom media from
  437. const importFrom = [].concat(Object(opts).importFrom || []); // destinations to export custom media to
  438. const exportTo = [].concat(Object(opts).exportTo || []); // promise any custom media are imported
  439. const customMediaPromise = getCustomMediaFromSources(importFrom);
  440. return {
  441. postcssPlugin: 'postcss-custom-media',
  442. Once: async root => {
  443. const customMedia = Object.assign(await customMediaPromise, getCustomMediaFromRoot(root, {
  444. preserve
  445. }));
  446. await writeCustomMediaToExports(customMedia, exportTo);
  447. transformAtrules(root, customMedia, {
  448. preserve
  449. });
  450. }
  451. };
  452. };
  453. creator.postcss = true;
  454. module.exports = creator;
  455. //# sourceMappingURL=index.cjs.js.map