conventions.js 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Gengkun He @ahabhgk
  4. */
  5. "use strict";
  6. /** @typedef {import("../../declarations/WebpackOptions").CssGeneratorExportsConvention} CssGeneratorExportsConvention */
  7. /**
  8. * @param {string} input input
  9. * @param {CssGeneratorExportsConvention | undefined} convention convention
  10. * @returns {string[]} results
  11. */
  12. module.exports.cssExportConvention = (input, convention) => {
  13. const set = new Set();
  14. if (typeof convention === "function") {
  15. set.add(convention(input));
  16. } else {
  17. switch (convention) {
  18. case "camel-case": {
  19. set.add(input);
  20. set.add(module.exports.camelCase(input));
  21. break;
  22. }
  23. case "camel-case-only": {
  24. set.add(module.exports.camelCase(input));
  25. break;
  26. }
  27. case "dashes": {
  28. set.add(input);
  29. set.add(module.exports.dashesCamelCase(input));
  30. break;
  31. }
  32. case "dashes-only": {
  33. set.add(module.exports.dashesCamelCase(input));
  34. break;
  35. }
  36. case "as-is": {
  37. set.add(input);
  38. break;
  39. }
  40. }
  41. }
  42. return Array.from(set);
  43. };
  44. // Copy from css-loader
  45. /**
  46. * @param {string} input input
  47. * @returns {string} result
  48. */
  49. module.exports.dashesCamelCase = input =>
  50. input.replace(/-+(\w)/g, (match, firstLetter) => firstLetter.toUpperCase());
  51. // Copy from css-loader
  52. /**
  53. * @param {string} input input
  54. * @returns {string} result
  55. */
  56. module.exports.camelCase = input => {
  57. let result = input.trim();
  58. if (result.length === 0) {
  59. return "";
  60. }
  61. if (result.length === 1) {
  62. return result.toLowerCase();
  63. }
  64. const hasUpperCase = result !== result.toLowerCase();
  65. if (hasUpperCase) {
  66. result = preserveCamelCase(result);
  67. }
  68. return result
  69. .replace(/^[_.\- ]+/, "")
  70. .toLowerCase()
  71. .replace(/[_.\- ]+([\p{Alpha}\p{N}_]|$)/gu, (_, p1) => p1.toUpperCase())
  72. .replace(/\d+([\p{Alpha}\p{N}_]|$)/gu, m => m.toUpperCase());
  73. };
  74. // Copy from css-loader
  75. /**
  76. * @param {string} string string
  77. * @returns {string} result
  78. */
  79. const preserveCamelCase = string => {
  80. let result = string;
  81. let isLastCharLower = false;
  82. let isLastCharUpper = false;
  83. let isLastLastCharUpper = false;
  84. for (let i = 0; i < result.length; i++) {
  85. const character = result[i];
  86. if (isLastCharLower && /[\p{Lu}]/u.test(character)) {
  87. result = `${result.slice(0, i)}-${result.slice(i)}`;
  88. isLastCharLower = false;
  89. isLastLastCharUpper = isLastCharUpper;
  90. isLastCharUpper = true;
  91. i += 1;
  92. } else if (
  93. isLastCharUpper &&
  94. isLastLastCharUpper &&
  95. /[\p{Ll}]/u.test(character)
  96. ) {
  97. result = `${result.slice(0, i - 1)}-${result.slice(i - 1)}`;
  98. isLastLastCharUpper = isLastCharUpper;
  99. isLastCharUpper = false;
  100. isLastCharLower = true;
  101. } else {
  102. isLastCharLower =
  103. character.toLowerCase() === character &&
  104. character.toUpperCase() !== character;
  105. isLastLastCharUpper = isLastCharUpper;
  106. isLastCharUpper =
  107. character.toUpperCase() === character &&
  108. character.toLowerCase() !== character;
  109. }
  110. }
  111. return result;
  112. };