index.js 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. /**
  2. * font variant convertion map
  3. *
  4. * @type {Object}
  5. */
  6. const fontVariantProperties = {
  7. "font-variant-ligatures": {
  8. "common-ligatures": "\"liga\", \"clig\"",
  9. "no-common-ligatures": "\"liga\", \"clig off\"",
  10. "discretionary-ligatures": "\"dlig\"",
  11. "no-discretionary-ligatures": "\"dlig\" off",
  12. "historical-ligatures": "\"hlig\"",
  13. "no-historical-ligatures": "\"hlig\" off",
  14. contextual: "\"calt\"",
  15. "no-contextual": "\"calt\" off"
  16. },
  17. "font-variant-position": {
  18. sub: "\"subs\"",
  19. "super": "\"sups\"",
  20. normal: "\"subs\" off, \"sups\" off"
  21. },
  22. "font-variant-caps": {
  23. "small-caps": "\"smcp\"",
  24. "all-small-caps": "\"smcp\", \"c2sc\"",
  25. "petite-caps": "\"pcap\"",
  26. "all-petite-caps": "\"pcap\", \"c2pc\"",
  27. unicase: "\"unic\"",
  28. "titling-caps": "\"titl\""
  29. },
  30. "font-variant-numeric": {
  31. "lining-nums": "\"lnum\"",
  32. "oldstyle-nums": "\"onum\"",
  33. "proportional-nums": "\"pnum\"",
  34. "tabular-nums": "\"tnum\"",
  35. "diagonal-fractions": "\"frac\"",
  36. "stacked-fractions": "\"afrc\"",
  37. ordinal: "\"ordn\"",
  38. "slashed-zero": "\"zero\""
  39. },
  40. "font-kerning": {
  41. normal: "\"kern\"",
  42. none: "\"kern\" off"
  43. },
  44. "font-variant": {
  45. normal: "normal",
  46. inherit: "inherit"
  47. }
  48. }
  49. // The `font-variant` property is a shorthand for all the others.
  50. for (const prop in fontVariantProperties) {
  51. const keys = fontVariantProperties[prop]
  52. for (const key in keys) {
  53. if (!(key in fontVariantProperties["font-variant"])) {
  54. fontVariantProperties["font-variant"][key] = keys[key]
  55. }
  56. }
  57. }
  58. // Find font-feature-settings declaration before given declaration,
  59. // create if does not exist
  60. function getFontFeatureSettingsPrevTo(decl) {
  61. let fontFeatureSettings = null;
  62. decl.parent.walkDecls((decl) => {
  63. if (decl.prop === "font-feature-settings") {
  64. fontFeatureSettings = decl;
  65. }
  66. })
  67. if (fontFeatureSettings === null) {
  68. fontFeatureSettings = decl.clone()
  69. fontFeatureSettings.prop = "font-feature-settings"
  70. fontFeatureSettings.value = ""
  71. decl.parent.insertBefore(decl, fontFeatureSettings)
  72. }
  73. return fontFeatureSettings
  74. }
  75. function walkRule(rule) {
  76. let fontFeatureSettings = null
  77. // read custom media queries
  78. rule.walkDecls((decl) => {
  79. if (!fontVariantProperties[decl.prop]) {
  80. return null
  81. }
  82. let newValue = decl.value
  83. if (decl.prop === "font-variant") {
  84. newValue = decl.value.split(/\s+/g).map((val) => {
  85. return fontVariantProperties["font-variant"][val]
  86. }).join(", ")
  87. }
  88. else if (fontVariantProperties[decl.prop][decl.value]) {
  89. newValue = fontVariantProperties[decl.prop][decl.value]
  90. }
  91. if (fontFeatureSettings === null) {
  92. fontFeatureSettings = getFontFeatureSettingsPrevTo(decl);
  93. }
  94. if (fontFeatureSettings.value && fontFeatureSettings.value !== newValue) {
  95. fontFeatureSettings.value += ", " + newValue;
  96. }
  97. else {
  98. fontFeatureSettings.value = newValue;
  99. }
  100. })
  101. }
  102. /**
  103. * Expose the font-variant plugin.
  104. */
  105. module.exports = () => {
  106. return {
  107. postcssPlugin: "postcss-font-variant",
  108. Once(root) {
  109. root.walkRules(walkRule)
  110. }
  111. }
  112. }
  113. module.exports.postcss = true