谈谈PostCSS

前言

如今的前端,javascript的发展有目共睹,框架林立。同时,html也是齐头并进,推出了HTML5标准,而且获得了普及。这样的发展却惟独少了一个角色?javascript

CSS,就是这个看似不起眼的家伙,却在开发中发挥着和js同样重要的做用。css,是一种样式脚本,好像和编程语言有着必定的距离,咱们能够将之理解为一种描述方法。这彷佛致使css被轻视了。不过,css近几年来正在经历着一次巨变——CSS Module。我记得js的井喷期应该能够说是node带来的,它带来了Module的概念,使得JS能够被工程化开发项目。那么,今天的css,也将愈来愈美好。若是你喜欢个人文章,欢迎评论,欢迎Star~。欢迎关注个人github博客css

正文

既然做为一篇推广PostCSS的文章,咱们就应该先来了解一下这是什么,和咱们以前讲的CSS Module有啥关系?此处让我为大家娓娓道来。html

我想和大家说再见

预处理器

目前,在工程化开发中,使用最多的应该就是Less、Sass和Stylus。首先,仍是介绍一下它们吧。它们有个统一的名字——css预处理器。何为CSS预处理器?应该就是一种能够将你根据它的规则写出来的格式转成css的东西(仍是讲的通俗一点)。它们的出现能够说是恰逢其时,解决了css的一些缺憾:前端

  • 语法不够强大,不可以嵌套书写,不利于模块化开发
  • 没有变量和逻辑上的复用机制,致使在css的属性值中只能使用字面量形式,以及不断重复书写重复的样式,致使难以维护。

面对以上问题,css预处理器给出了很是可行的解决方案:java

  1. 变量:就像其余编程语言同样,免于多处修改。node

    • Sass:使用「$」对变量进行声明,变量名和变量值使用冒号进行分割
    • Less:使用「@」对变量进行声明
    • Stylus:中声明变量没有任何限定,结尾的分号无关紧要,但变量名和变量值之间必需要有『等号』。但须要注意的是,若是用“@”符号来声明变量,Stylus会进行编译,但不会赋值给变量。就是说,Stylus 不要使用『@』声明变量。Stylus 调用变量的方法和Less、Sass彻底相同。
  2. 做用域:有了变量,就必须得有做用域进行管理。就想js同样,它会从局部做用域开始往上查找变量。react

    • Sass:它的方式是三者中最差的,不存在全局变量的概念
    • Less:它的方式和js比较类似,逐级往上查找变量
    • Stylus:它的方式和Less比较类似,可是它和Sass同样更倾向于指令式查找
  3. 嵌套:对于css来讲,有嵌套的写法无疑是完美的,更像是父子层级之间明确关系webpack

    • 三者在这处的处理都是同样的,使用「&」表示父元素

有了这些方案,会使得咱们能够在保证DPY、可维护性、灵活性的前提下,编写css样式。git

回到话题中,之因此会出现向预处理器这样子的解决方案,归根结底仍是css标准发展的滞后性致使的。同时,咱们也应该考虑一下,真的只要预处理器就够了吗?每每在项目过大时,因为缺少模块的概念,全局变量的问题会持续困扰着你。每次定义选择器时,老是要顾及到其余文件中是否也使用了一样的命名。毕竟项目是团队的,而不是我的的。哪是否有方式能够解决这些问题呢?程序员

前人的方法

对于css命名冲突的问题,由来已久,能够说咱们前端开发人员,每天在苦思冥想,如何去优雅的解决这些问题。css并未像js同样出现了AMD、CMD和ES6 Module的模块化方案。

那么,回到问题,如何去解决呢?咱们的前人也有提出过不一样的方案:

  1. Object-Oriented CSS
  2. BEM
  3. SMACSS

方案能够说是层出不穷,不乏有团队内部的解决方案。可是大多数都是一个共同点——为选择器增长前缀。

这但是一个体力活,你可能须要手动的去编写长长的选择器,或许你可使用预编译的css语言。可是,它们彷佛并为解决本质的问题——为什么会形成这种缺憾。咱们不妨来看看,使用BEM规范写出来的例子:

<!-- 正确的。元素都位于 'search-form' 模块内 -->
<!-- 'search-form' 模块 -->
<form class="search-form">
    <!-- 在 'search-form' 模块内的 'input' 元素 -->
    <input class="search-form__input" />
    <!-- 在 'search-form' 模块内的 'button' 元素 -->
    <button class="search-form__button"></button>
</form>

<!-- 不正确的。元素位于 'search-form' 模块的上下文以外 -->
<!-- 'search-form' 模块 -->
<form class=""search-block>
</form>

<!-- 在 'search-form' 模块内的 'input' 元素 -->
<input class="search-form__input"/>

<!-- 在 'search-form' 模块内的 'button' 元素 -->
<button class="search-form__button"></button>

每次这样子写,估计是个程序员,都得加班吧,哈哈!

一种但愿

如今的网页开发,讲究的是组件化的思想,所以,急须要可行的css Module方式来完成网页组件的开发。自从2015年开始,国外就流行了CSS-in-JS(典型的表明,react的styled-components),还有一种就是CSS Module。

本篇谈及后者,须要对前者进行了解的话,自行Google便可

对于css,你们都知道,它是一门描述类语言,并不存在动态性。那么,要如何去造成module呢。咱们能够先来看一个react使用postcss的例子:

//example.css

.article {
    font-size: 14px;
}
.title {
    font-size: 20px;
}

以后,将这些命名打乱:

.zxcvb{
    font-size: 14px;
}
.zxcva{
    font-size: 20px;
}

将之命名对应的内容,放入到JSON文件中去:

{
    "article": "zxcvb",
    "title": "zxcva"
}

以后,在js文件中运用:

import style from 'style.json';

class Example extends Component{
    render() {
        return (
            <div classname={style.article}>
                <div classname={style.title}></div>
            </div>
        )
    }
}

这样子,就描绘出了一副css module的原型。固然,咱们不可能每次都须要手动去写这些东西。咱们须要自动化的插件帮助咱们完成这一个过程。以后,咱们应该先来了解一下postCSS。

我须要认识你

PostCSS是什么?或许,你会认为它是预处理器、或者后处理器等等。其实,它什么都不是。它能够理解为一种插件系统。使用它GitHub主页上的介绍:

PostCSS is a tool for transforming CSS with JS plugins. These plugins can support variables and mixins, transpile future CSS syntax, inline images, and more.

你能够在使用预处理器的状况下使用它,也能够在原生的css中使用它。它都是支持的,而且它具有着一个庞大的生态系统,例如你可能经常使用的Autoprefixer,就是PostCSS的一个很是受欢迎的插件,被Google, Shopify, Twitter, Bootstrap和CodePen等公司普遍使用。

固然,咱们也能够在CodePen中使用它:

CodePen

这里推荐你们看一下PostCSS的深刻系列

接下来,咱们来看一下PostCSS的配置:

这里咱们使用webpack+postcss+postcss-loader+cssnext+postcss-import的组合。

首先,咱们能够经过yarn来安装这些包:

yarn add --dev webpack extract-text-webpack-plugin css-loader file-loader postcss postcss-loader postcss-cssnext postcss-import

而后,咱们配置一下webpack.config.js:

const webpack = require('webpack');
const path = require('path');
const ExtractTextPlugin = require('extract-text-webpack-plugin');
module.exports = {
  context: path.resolve(__dirname, 'src'),
  entry: {
    app: './app.js';
  },
  module: {
    loaders: [
      {
        test: /\.css$/,
        use: ExtractTextPlugin.extract({
          use: [
            {
              loader: 'css-loader',
              options: { importLoaders: 1 },
            },
            'postcss-loader',
          ],
        }),
      },
    ],
  },
  output: {
    path: path.resolve(__dirname, 'dist/assets'),
  },
  plugins: [
    new ExtractTextPlugin('[name].bundle.css'),
  ],
};

而后在根目录下配置postcss.config.js

module.exports = {
  plugins: {
    'postcss-import': {},
    'postcss-cssnext': {
      browsers: ['last 2 versions', '> 5%'],
    },
  },
};

以后,就能够在开发中使用cssnext的特性了

/* Shared */
@import "shared/colors.css";
@import "shared/typography.css";
/* Components */
@import "components/Article.css";
/* shared/colors.css */
:root {
  --color-black: rgb(0,0,0);
  --color-blue: #32c7ff;
}

/* shared/typography.css */
:root {
  --font-text: "FF DIN", sans-serif;
  --font-weight: 300;
  --line-height: 1.5;
}

/* components/Article.css */
.article {
  font-size: 14px;
  & a {
    color: var(--color-blue);
  }
  & p {
    color: var(--color-black);
    font-family: var(--font-text);
    font-weight: var(--font-weight);
    line-height: var(--line-height);
  }
  @media (width > 600px) {
    max-width: 30em;
  }
}

最后使用webpack进行编译就能够了。

总结

PostCSS,国内尚未太流行起来,不过相信不久的未来也会逐渐的热门,而且国内的资源较少,不过最近新出了一本大漠老师们一块儿翻译的书——《深刻PostCSS Web设计》。有兴趣的人也能够去看一下,学习一些前言的东西。本篇也只是大概的写了一下PostCSS的东西,鉴于国内资源较少,因此参考了一下国外的博文教材,下面会有连接。

若是你对我写的有疑问,能够评论,如我写的有错误,欢迎指正。你喜欢个人博客,请给我关注Star~呦。你们一块儿总结一块儿进步。欢迎关注个人github博客

参考连接

PostCSS-modules: make CSS great again!

PostCSS Deep Dive: What You Need to Know

相关文章
相关标签/搜索