index.js 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. const feature_unit = {
  2. 'width': 'px',
  3. 'height': 'px',
  4. 'device-width': 'px',
  5. 'device-height': 'px',
  6. 'aspect-ratio': '',
  7. 'device-aspect-ratio': '',
  8. 'color': '',
  9. 'color-index': '',
  10. 'monochrome': '',
  11. 'resolution': 'dpi'
  12. };
  13. // Supported min-/max- attributes
  14. const feature_name = Object.keys(feature_unit);
  15. const step = .001; // smallest even number that won’t break complex queries (1in = 96px)
  16. const power = {
  17. '>': 1,
  18. '<': -1
  19. };
  20. const minmax = {
  21. '>': 'min',
  22. '<': 'max'
  23. };
  24. function create_query(name, gtlt, eq, value) {
  25. return value.replace(/([-\d\.]+)(.*)/, function (_match, number, unit) {
  26. const initialNumber = parseFloat(number);
  27. if (parseFloat(number) || eq) {
  28. // if eq is true, then number remains same
  29. if (!eq) {
  30. // change integer pixels value only on integer pixel
  31. if (unit === 'px' && initialNumber === parseInt(number, 10)) {
  32. number = initialNumber + power[gtlt];
  33. } else {
  34. number = Number(Math.round(parseFloat(number) + step * power[gtlt] + 'e6')+'e-6');
  35. }
  36. }
  37. } else {
  38. number = power[gtlt] + feature_unit[name];
  39. }
  40. return '(' + minmax[gtlt] + '-' + name + ': ' + number + unit + ')';
  41. });
  42. }
  43. function transform(rule) {
  44. /**
  45. * 转换 <mf-name> <|>= <mf-value>
  46. * $1 $2 $3
  47. * (width >= 300px) => (min-width: 300px)
  48. * (width <= 900px) => (max-width: 900px)
  49. */
  50. if (!rule.params.includes('<') && !rule.params.includes('>')) {
  51. return
  52. }
  53. // The value doesn't support negative values
  54. // But -0 is always equivalent to 0 in CSS, and so is also accepted as a valid <mq-boolean> value.
  55. rule.params = rule.params.replace(/\(\s*([a-z-]+?)\s*([<>])(=?)\s*((?:-?\d*\.?(?:\s*\/?\s*)?\d+[a-z]*)?)\s*\)/gi, function($0, $1, $2, $3, $4) {
  56. if (feature_name.indexOf($1) > -1) {
  57. return create_query($1, $2, $3, $4);
  58. }
  59. // If it is not the specified attribute, don't replace
  60. return $0;
  61. })
  62. /**
  63. * 转换 <mf-value> <|<= <mf-name> <|<= <mf-value>
  64. * 转换 <mf-value> >|>= <mf-name> >|>= <mf-value>
  65. * $1 $2$3 $4 $5$6 $7
  66. * (500px <= width <= 1200px) => (min-width: 500px) and (max-width: 1200px)
  67. * (500px < width <= 1200px) => (min-width: 501px) and (max-width: 1200px)
  68. * (900px >= width >= 300px) => (min-width: 300px) and (max-width: 900px)
  69. */
  70. rule.params = rule.params.replace(/\(\s*((?:-?\d*\.?(?:\s*\/?\s*)?\d+[a-z]*)?)\s*(<|>)(=?)\s*([a-z-]+)\s*(<|>)(=?)\s*((?:-?\d*\.?(?:\s*\/?\s*)?\d+[a-z]*)?)\s*\)/gi, function($0, $1, $2, $3, $4, $5, $6, $7) {
  71. if (feature_name.indexOf($4) > -1) {
  72. if ($2 === '<' && $5 === '<' || $2 === '>' && $5 === '>') {
  73. const min = ($2 === '<') ? $1 : $7;
  74. const max = ($2 === '<') ? $7 : $1;
  75. // output differently depended on expression direction
  76. // <mf-value> <|<= <mf-name> <|<= <mf-value>
  77. // or
  78. // <mf-value> >|>= <mf-name> >|>= <mf-value>
  79. let equals_for_min = $3;
  80. let equals_for_max = $6;
  81. if ($2 === '>') {
  82. equals_for_min = $6;
  83. equals_for_max = $3;
  84. }
  85. return create_query($4, '>', equals_for_min, min) + ' and ' + create_query($4, '<', equals_for_max, max);
  86. }
  87. }
  88. // If it is not the specified attribute, don't replace
  89. return $0;
  90. });
  91. }
  92. module.exports = () => ({
  93. postcssPlugin: 'postcss-media-minmax',
  94. AtRule: {
  95. media: (atRule) => {
  96. transform(atRule);
  97. },
  98. 'custom-media': (atRule) => {
  99. transform(atRule);
  100. },
  101. },
  102. });
  103. module.exports.postcss = true