postCSS是一个使用JS转换CSS的工具。
官网: https://postcss.nodejs.cn/

PostCSS是什么

PostCSS 是一个使用 JavaScript 插件转换 CSS 的工具。CSS编译器能够做到的事情,PostCSS 也可以做到,而且能够做得更好。
功能:

  • 检查(lint)你的 CSS
  • 支持 CSS Variables 和 Mixins
  • 编译尚未被浏览器广泛支持的先进的 CSS 语法,内联图片,等。

PostCSS 的核心 — 插件
它提供了一个把 CSS 解析成抽象语法树(AST )的工具,以及一堆 API,来帮助开发者更方便的构建 PostCSS 插件。开发者可以根据自己的需要,去挑选和自定义合适的 PostCSS 插件,来实现自己需要的功能。

例如,Post已存在的插件:

  • Autoprefixer 插件:
    自动为 CSS 属性添加私有前缀,以便于兼容不同的浏览器。
  • cssnano 插件:
    对CSS 代码进行压缩,以优化加载速度。

使用PostCSS插件可以按需为你的 CSS 流程增加功能。

postCSS的优点

  • 提高代码可读性
    将浏览器前缀添加到 CSS 规则中。 Autoprefixer 基于当前浏览器流行度和属性支持的数据来为css样式应用前缀。
  • 转换为通用的CSS
    让你将现代 CSS 转换为大多数浏览器可以理解的内容,并使用 cssdb 根据你的目标浏览器或运行时环境确定所需的 polyfill
  • 为css提供单独的命名空间,在不同的命名空间下css的类名可以一样
  • 规范的css
    使用stylelint(css代码检查器)强制执行一致的约定,避免样式表中的错误。

补充:polyfill

Polyfill是一段代码(通常是JavaScript),用来为旧的浏览器提供它原生不支持的较新的功能。这样就可以在旧的浏览器上使用这些功能而不会出现问题。

例如,旧版本的浏览器可能并不支持 ECMAScript 2015(也可以说是ES6)的特性,比如Promise和Object.assign。为了解决这个问题,开发者可以使用polyfill,它是在加载JavaScript代码之前的一段代码,这段代码会检查浏览器是否支持这些特性,若不支持则提供类似的功能。Polyfill允许开发者编写出最新的、最高级的代码,而不用担心兼容性问题。它会帮助你的网站或者Web应用在所有浏览器中保持一致的行为。

eg:手动编写Array.prototype.forEach
Array.prototype.forEach方法在现代浏览器中是内置的,但是在一些老版本的浏览器中可能并不存在。所以我们可以编写一个 Polyfill 来“填充”这个漏洞:

if (!Array.prototype.forEach) {
    Array.prototype.forEach = function(callback, thisArg) {
        if (this == null) {
            throw new TypeError('Array.prototype.forEach called on null or undefined');
        }
        var T, k;
        var O = Object(this);
        var len = O.length >>> 0;

        if (typeof callback !== "function") {
            throw new TypeError(callback + ' is not a function');
        }

        if (arguments.length > 1) {
            T = thisArg;
        }

        k = 0;

        while (k < len) {
            var kValue;
            if (k in O) {
                kValue = O[k];
                callback.call(T, kValue, k, O);
            }
            k++;
        }
    };
}

补充:Stylelint

Stylelint是强大的CSS检查器,可以帮助你避免错误并强制执行约定。
特性:

PostCSS架构

概述

PostCSS不是像Sass或Less一样的样式处理器,他没有自定义的语法和语义,它实际上不是一种语言,而是一种CSS语法转换的工具。PostCSS与CSS一起使用,任何有效的CSS都可以由PostCSS处理。

PostCSS是CSS语法转换的工具,允许自定义CSS的语法,这些语法可以被插件理解和转换。

工作流程

PostCSS解析方法

PostCSS是CSS的一个语法转换工具,对css代码进行解析。
常见的解析方法由两种:

  • 使用一个从字符串转换成AST(抽象语法树)的文件进行解析。
    Rework 分析器就是使用这种方法解析的,缺点是如果代码库过大,代码会变得难以阅读并且速度非常慢。

  • 拆分为词法分析
    流程是源字符串-->标记-->AST(抽象语法树)
    这是我们在PostCSS中的做法也是最流行的一种。 很多像 @babel/parser(Babel 背后的解析器)、CSSTree 这样的解析器都是这样写的。
    这种方法将标记化与解析步骤分离,可以编写非常快的分词器(但有时难以阅读代码)和易于阅读(但速度慢)的解析器。

PostCSS解析流程

PostCSS的工作流程:
在这里插入图片描述

  • 分词器
    lib/tokenize.js
    它接受 CSS 字符串并返回标记列表。
    css:

    .className { color: #FFF; }
    

    使用分词器分词之后的标记是:

    [
        ["word", ".className", 1, 1, 1, 10]
        ["space", " "]
        ["{", "{", 1, 12]
        ["space", " "]
        ["word", "color", 1, 14, 1, 18]
        [":", ":", 1, 19]
        ["space", " "]
        ["word", "#FFF" , 1, 21, 1, 23]
        [";", ";", 1, 24]
        ["space", " "]
        ["}", "}", 1, 26]
    ]
    

    单个令牌表示为列表,并且 space 令牌没有位置信息。

  • Parser解析器
    lib/parse.js、lib/parser.js

    Parse是传入语法分析的主要结构,解析器生成一个名为抽象语法树(AST)的结构,稍后可以通过插件对其进行转换。
    解析器对标记进行操作,而不是对源字符串进行操作。它主要使用 Tokenizer 提供的 nextToken 和 back 方法来获取单个或多个 token,然后构造 AST 的一部分,称为 Node

  • 处理器

    lib/processor.js
    处理器是一个非常简单的结构,用于初始化插件并运行语法转换

  • 插件

    插件可以是用户自定义的插件,也可以是PostCSS自带的插件,通过使用插件来为CSS添加特殊功能。

  • 字符串化器
    lib/stringify.jslib/stringifier.js

    Stringifier是一个基类,他将修改的AST转换为纯CSS字符串。Stringifier从提供的Node开始遍历AST,并调用相应的方法生成它的原始字符串表示形式。最重要的方法是Stringifier.stringify,他接受初始节点和分号指示符。

PostCSS使用

  • 安装:

    npm i postcss postcss-cli
    
  • postcss的配置文件
    配置文件是postcss.config.js,可以在里面进行一些配置操作。

  • 使用postcss进行css文件转化
    postcss 源css文件 -o 目标css文件

    postcss src/demo.css -o css/demo.css
    

    实时监听:只要源css文件一改变就进行转化postcss -w 源css文件 -o 目标css文件

PostCSS插件

PostCSS所有插件:https://postcss.nodejs.cn/docs/postcss-plugins

插件的使用

PostCSS 是一个插件驱策的工具,它本身并没有提供很多功能。要想增加额外的功能,就需要需要下载并安装相应的插件,然后在PostCSS的配置文件·postcss.config.js·中引用。

如,使用autoprefixer 插件(autoprefixer 插件的作用是根据不同的浏览器为样式属性添加前缀)

  • 下载插件
npm install autoprefixer --save-dev
  • 在PostCSS的配置文件中引入autoprefixer
    postcss.config.js
module.exports = {
  plugins: [
    require('autoprefixer')
  ],
};

在这之后,autoprefixer 就会自动在你的 CSS 规ules中添加供应商前缀了。这些前缀是根据 Can I Use 网站的数据自动决定的。

例如,下面的 CSS:

a {
  display: flex;
}

经过 autoprefixer 处理后,会变成:

a {
  display: -webkit-box;
  display: -webkit-flex;
  display: -ms-flexbox;
  display: flex;
}

这样一来,CSS 代码就可以在老版本的浏览器中也很好地运行了。

控制类插件

  • postcss-use插件

作用: 允许你直接在CSS代码中通过@use规则引入其他PostCSS插件,而无需在PostCSS的配置文件中进行配置。

使用:
下载

npm install postcss-use --save-dev

配置:
postcss.config.js

module.exports = {
    plugins: [
        require('postcss-use')()
    ]
};

引入其他插件

@use autoprefixer(browsers: ['last 2 versions']);
:fullscreen a {
  display: flex;
}

就会根据不同的浏览器为fullscreen添加前缀。

  • postcss-plugin-context插件

作用: 为不同点css代码应用不同的PostCSS插件。
使用:
如,使用postcss-plugin-context插件为不同css分别配置autoprefixer、cssnano:
下载

npm install autoprefixer cssnano postcss-plugin-context --save-dev

配置postcss.config.js

module.exports = {
  plugins: [
    require('postcss-plugin-context')({
      autoprefixer: require('autoprefixer'),
      cssnano: require('cssnano')
    }),
  ],
};

使用:

/* 为这部分 CSS 代码应用 autoprefixer 插件 */
@context autoprefixer {
  .box {
    display: flex;
  }
}

/* 为这部分 CSS 代码应用 cssnano 插件 */
@context cssnano {
  .title {
    font-size: 20px;
    margin-left: 10px;
    margin-right: 10px;
  }
}

包类插件

插件包需要下载才能使用,下载完之后在postcss的配置文件中进行配置,构建的时候会自动使用。

  • postcss-utilities :包括最常用的 mixin、快捷方式和辅助程序
  • atcss:根据特殊注释转换 CSS 的插件
  • cssnano:优化 CSS 大小以供生产使用的插件
  • oldie:可以转换css以兼容旧版Internet Explorer的插件
  • rucksack:包含通过新功能和快捷方式加速CSS开发的插件
  • level4:仅包含可让你编写CSS4而无需IE9回退的插件
  • short:添加并拓展了许多速记属性。
  • styleint:检查样式表的插件
  • postcss-hamster:用于垂直节奏、排版、模块化比例功能
  • postcss-preset-env:允许你将现代CSS转换为大多数浏览器可以理解的内容。
  • postcss -ui-theme:为你提供语法糖并允许你更改主题

未来的CSS语法相关插件

  • postcss-apply:支持自定义属性集引用
  • postcss-attribute-case-insensitive:支持不区分大小写的属性
  • postcss-bidirection:使用单一语法生成从左到右和从右到左的样式。
  • postcss-color-function:支持转换颜色的函数
  • postcss-color-gray:支持 gray() 功能
  • postcss-color-hex-alpha:#rrggbbaa 和 #rgba 表示法。
  • postcss-extend:递归地支持规则和占位符的规范,近似 @extend
  • postcss-pseudo-is:将is属性转换为更兼容的CSS
  • postcss-selector-not:将CSS4:not()转换为CSS3的 :not()

后备措施相关插件

  • postcss-color-rgba-fallback:将 rgba() 转换为十六进制。
  • postcss-epub:将 -epub- 前缀添加到相关属性中。
  • postcss-fallback:添加了fallback函数以避免重复声明
  • postcss-filter-gradient:为旧版IE添加了渐变滤镜
  • postcss-flexibility:为 Flexibility polyfill 添加 -js- 前缀。
  • postcss-hash-classname:将哈希字符串添加到你的css类名中。—— vue的scoped
  • postcss-mqwidth-to-class:将最小/最大宽度媒体查询转换为类。
  • postcss-opacity:为 IE8 添加了不透明度过滤器。
  • postcss-opacity-percentage:将 CSS4 基于百分比的 opacity 值转换为浮点值。
  • postcss-page-break:将 page-break- 后备添加到 break- 属性。
  • postcss-pseudoelements:将::转换成:钻则其一兼容IE8
  • postcss-replace-overflow-wrap :用 word-wrap 替换 overflow-wrap
  • postcss-vmin:为 IE9 中的 vmin 单元生成 vm 后备。
  • autoprefixer:使用 Can I Use 中的数据为你添加浏览器前缀。
  • postcss-pie:使 IE 具备了几个最有用的 CSS3 装饰功能。
  • cssgrace:为IE和其他旧浏览器提供各种辅助程序和转义CSS3。
  • pixrem:为 rem 单位生成像素回退。
  • webp-in-css:在 CSS 中使用 WebP 背景图片。
  • postcss-clamp:将clamp()变换为min/max的组合。
  • postcss-spring-easing:用生成的 linear() 函数替换 spring()

语言扩展相关插件

  • postcss-aspect-ratio:将元素的尺寸固定为纵横比。
  • postcss-atroot:将规则直接放置在根节点处。
  • postcss-click:允许使用 :click 伪类并在 JavaScript 中实现它。
  • postcss-conditionals:添加@if语法
  • postcss-css-variables:支持使用 W3C 类似语法的选择器变量和 at 规则。
  • postcss-define-property]:定义属性快捷方式。
  • postcss-for:添加@for循环
  • csstyle:将组建工作流程添加到样式中

颜色相关组件

  • post-ase-colors: 用从 ASE 调色板文件读取的值替换颜色名称。
  • postcss-brand-colors:在brand-colors模块中插入公司品牌颜色。
  • postcss-color-mix:将两种颜色混合在一起。
  • postcss-color-short:添加了简写颜色声明。
  • postcss-theme-colors:添加带有颜色组的深色和浅色主题。
  • postcss-rgba-hex: 将 rgba 值转换为 hex 类似物。
  • postcss-get-color:从图片中获取突出的颜色。
  • postcss-randomcolor:支持使用随机颜色的功能。

图片和字体相关的类

  • postcss-assets:简化 URL、插入图片尺寸和内联文件
  • postcss-assets-rebase:重新设定 url() 的资源基础。
  • postcss-copy-assets:将相对 url() 引用的资源复制到构建目录中。
  • postcss-data-packer:将嵌入的 Base64 数据移动到单独的文件中。
  • postcss-image-set:将 background-image 添加到 image-set() 的第一张图片。
  • [postcss-foft-classes]:为使用 Web 字体的块添加了保护类,以实现更好的字体加载策略。
  • postcss-placehold: 可以轻松放置占位符图片。
  • postcss-svg:将内联 SVG 插入 CSS 并允许管理它的颜色。
  • postcss-svgo:通过 SVGO 处理内联 SVG。
  • postcss-url:变基或内联 url()。
  • postcss-urlrebase:将 url() 重新设置为给定的根 URL。
  • postcss-write-svg:在 CSS 中编写内联 SVG。

网格(Grids)

  • postcss-grid:添加了语义网格系统。
  • postcss-grid-kiss:将 ASCII-art 网格转换为 CSS 网格布局。
  • postcss-grid-system:基于固定的列宽创建网格。
  • postcss-grid-fluid:创建流体网格。
  • postcss-layout:是一些常见 CSS 布局模式和网格系统的插件。

优化相关插件

  • postcss-import:内联@import规则引用的样式表。
  • postcss-calc:将calc()简化为值。
  • postcss-nested-import: 在嵌套规则块内内联 @import 规则引用的样式表。
  • postcss-partial-import:内联标准导入和类似 Sass 的部分。
  • postcss-remove-root:从样式表中删除 :root 的所有实例。
  • postcss-zindex:重新设置正 z-index 值的基准。
  • css-byebye: 删除了你不需要的 CSS 规则。
  • postcss-no-important:删除声明!important。

快捷方式

  • postcss-alias:为属性创建较短的别名。
  • postcss-border:在 border 属性中添加了所有边框的宽度和颜色的简写。
  • postcss-button:创建按钮。
  • postcss-center:居中元素。
  • postcss-circle:插入一个带有颜色的圆圈。
  • postcss-verthorz:添加垂直和水平间距声明。
  • font-magician:生成 CSS 中所需的所有 @font-face 规则

自定义PostCSS插件

大多数 PostCSS 插件都会做两件事:

  • 在 CSS 中查找某些内容(例如,will-change 属性)。
  • 更改找到的元素(例如,在 will-change 之前插入 transform: translateZ(0) 作为旧浏览器的填充)。

查找节点

PostCSS 将 CSS 解析为节点树(我们称之为 AST)。 这棵树可能包含:

  • Root: 树顶部的节点,代表 CSS 文件。
  • AtRule: 语句以 @ 开头,如 @charset “UTF-8” 或 @media (screen) {}。
  • Rule: 内部带有声明的选择器。 例如 input, button {}。
  • Declaration: 键值对,如 color: black;
  • Comment: 独立注释。 选择器内的注释、规则参数和值存储在节点的 raws 属性中。

可以在插件对象中添加方法来查找具有特定类型的所有节点:

module.exports = (opts = {}) => {
  return {
    postcssPlugin: 'PLUGIN NAME',
    Once (root) {
      每个文件调用一次,因为每个文件都有一个根
    },
    Declaration (decl,{Rule}) {
      // 所有声明节点
    }
  }
}
module.exports.postcss = true

所添加的方法都是插件事件,更多事件可以戳这里
插件时间的第二个参数是节点的创建者。

更改节点

当找到正确的节点时,需要更改它们或插入/删除周围的其他节点。
PostCSS 节点有一个类似 DOM 的 API 来转换 AST。 PostCSS 节点具有四处移动(如 Node#next 或 Node#parent)、查看子节点(如 Container#some)、删除节点或在内部添加新节点的方法。如果添加了新节点,复制 Node#source 以生成正确的源映射非常重要。

更改节点的操作需要在插件事件函数中进行编写:

Declaration (node, { Rule }) {
  let newRule = new Rule({ selector: 'a', source: node.source })
   node.root().append(newRule)
   newRule.append(node)
 }

eg:编译写个使字体变红的插件:

const plugin = () => {
  return {
    postcssPlugin: 'to-red',
    Rule (rule) {
      console.log(rule.toString())
    },
    Declaration (decl) {
      console.log(decl.toString())
      decl.value = 'red'
    }
  }
}
plugin.postcss = true

await postcss([plugin]).process('a { color: black }', { from })
// => a { color: black }
// => color: black
// => a { color: red }
// => color: red

PostCSS自定义语法规则

PostCSS可以转换任何语法的样式,而不仅限于CSS。通过编写自定义语法,你可以将样式转换为任何所需的格式。

自定义语法规则包含解析器和字符串生成器:

  • 解析器:将输入字符串解析到节点树。
    解析器包含两个步骤:分词标记数组、读取标记数组并构建树
  • 字符串转化器:通过节点树生成输出字符串。

语法 API 是一个非常简单的普通对象,具有 parse 和 stringify 函数,自定义语法规则格式:

module.exports = {
  parse: require('./parse'),
  stringify: require('./stringify')
}

解析器API,有两个参数:第一个参数是一个css字符串,第二个参数是一个函数接收带有 PostCSS 选项的对象,解析器API的返回值是一个函数,返回[Root]或[Document] 节点的函数。

const postcss = require('postcss')

module.exports = function parse (css, opts) {
  const root = postcss.root()
  // 处理css为dom节点,添加到root节点中
  //返回root节点
  return root
}

Stringifier API 比解析器 API 稍微复杂一些。 PostCSS 生成源映射,因此字符串生成器不能只返回字符串。 它必须将每个子字符串与其源节点链接起来
Stringifier 是一个接收 [Root] 或[Document] 节点构建器回调的函数。 然后它使用每个节点的字符串和节点实例调用构建器。

module.exports = function stringify (root, builder) {
  // Some magic
  const string = decl.prop + ':' + decl.value + ';'
  builder(string, decl)
  // Some science
};
Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐