index.es.mjs 15 KB

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