summaryrefslogtreecommitdiff
path: root/scripts/frontend/startup_css/clean_css.js
blob: 67a0453e81671b18bfb8444719e343d8aaca20bc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
const { memoize, isString, isRegExp } = require('lodash');
const { parse } = require('postcss');
const { CSS_TO_REMOVE } = require('./constants');

const getSelectorRemoveTesters = memoize(() =>
  CSS_TO_REMOVE.map((x) => {
    if (isString(x)) {
      return (selector) => x === selector;
    }
    if (isRegExp(x)) {
      return (selector) => x.test(selector);
    }

    throw new Error(`Unexpected type in CSS_TO_REMOVE content "${x}". Expected String or RegExp.`);
  }),
);

const getRemoveTesters = memoize(() => {
  const selectorTesters = getSelectorRemoveTesters();

  // These are mostly carried over from the previous project
  // https://gitlab.com/gitlab-org/frontend/gitlab-css-statistics/-/blob/2aa00af25dba08fc71081c77206f45efe817ea4b/lib/gl_startup_extract.js
  return [
    (node) => node.type === 'comment',
    (node) =>
      node.type === 'atrule' &&
      (node.params === 'print' ||
        node.params === 'prefers-reduced-motion: reduce' ||
        node.name === 'keyframe' ||
        node.name === 'charset'),
    (node) => node.selector && node.selectors && !node.selectors.length,
    (node) => node.selector && selectorTesters.some((fn) => fn(node.selector)),
    (node) =>
      node.type === 'decl' &&
      (node.prop === 'transition' ||
        node.prop.indexOf('-webkit-') > -1 ||
        node.prop.indexOf('-ms-') > -1),
  ];
});

const getNodesToRemove = (nodes) => {
  const removeTesters = getRemoveTesters();
  const remNodes = [];

  nodes.forEach((node) => {
    if (removeTesters.some((fn) => fn(node))) {
      remNodes.push(node);
    } else if (node.nodes?.length) {
      remNodes.push(...getNodesToRemove(node.nodes));
    }
  });

  return remNodes;
};

const getEmptyNodesToRemove = (nodes) =>
  nodes
    .filter((node) => node.nodes)
    .reduce((acc, node) => {
      if (node.nodes.length) {
        acc.push(...getEmptyNodesToRemove(node.nodes));
      } else {
        acc.push(node);
      }

      return acc;
    }, []);

const cleanCSS = (css) => {
  const cssRoot = parse(css);

  getNodesToRemove(cssRoot.nodes).forEach((node) => {
    node.remove();
  });

  getEmptyNodesToRemove(cssRoot.nodes).forEach((node) => {
    node.remove();
  });

  return cssRoot.toResult().css;
};

module.exports = { cleanCSS };