Stylelint
Why and How to Lint CSS
Andrey Sitnik
Evil Martians
data:image/s3,"s3://crabby-images/1503b/1503b82111c9e3ce602a90f2bc6a49cd32b95f78" alt=""
data:image/s3,"s3://crabby-images/8bcb4/8bcb4b49adc7dce0fa8d44724afc4041b23dc574" alt=""
data:image/s3,"s3://crabby-images/9e1ad/9e1ad7bfc1c61695723a428c361847eab3eebcae" alt=""
Our Projects
Russia
Our Open Source
data:image/s3,"s3://crabby-images/24fa7/24fa71c2746163163b7012b0089adb85ee1a5dff" alt=""
Linters History
data:image/s3,"s3://crabby-images/3fe72/3fe723422c8ab645aabd92441047cce2c948958c" alt=""
Source code
Parser
Analysis
Errors list
Linter
1978: First Linter, the Lint
data:image/s3,"s3://crabby-images/15052/1505293d78396a9af66a108eb499ebf37a1122e6" alt=""
1995: JavaScript
// true
1 == "1"
// easy to forget var
count = 1
data:image/s3,"s3://crabby-images/793e9/793e9647ff4a8c741217acd706f1f08bd2bcd0fb" alt=""
Linters
CoffeeScript
vs
data:image/s3,"s3://crabby-images/be0d9/be0d906006aec097f73ee354389a87824baaf7ff" alt=""
data:image/s3,"s3://crabby-images/e5b7d/e5b7def837dc12670096d1967bc9c161544c3880" alt=""
CoffeeScript :-(
JS Linters Evolution
data:image/s3,"s3://crabby-images/b0fda/b0fdabd647576718470fa4e6a849074ead47e29d" alt=""
data:image/s3,"s3://crabby-images/014e9/014e976b2487e6698aa4c870b0211d1e1903964b" alt=""
data:image/s3,"s3://crabby-images/dd1da/dd1da826fe3c001c32bc223b637e86a4c71ac990" alt=""
JSLint
JSHint
ESLint
→
→
ESLint: Whitelist
module.exports = {
'rules': {
'space-before-function-paren': [2],
'no-shadow-restricted-names': [2],
'computed-property-spacing': [2],
'no-empty-character-class': [2],
'no-irregular-whitespace': [2],
'no-unexpected-multiline': [2],
'no-multiple-empty-lines': [2],
'no-constant-condition': [2],
…
ESLint: Plugins
Source code
Parser
Plugin 1
Errors list
Plugin 2
ESLint: Fixing
if(foo) {
bar()}
if (foo) {
bar();
}
eslint --fix *.js
The Most Popular Dependencies
- mocha
- chai
- lodash
- grunt
- gulp
- eslint
- babel-preset-es2015
- request
- async
- should
CSS Linters
data:image/s3,"s3://crabby-images/6f56d/6f56d7bd9d677bf7b07e8f6a5733ff88788022fd" alt=""
Syntax Support
SCSS
CSSLint
SCSS Lint
CSSComb
CSS
Less
Features
Whitelist
CSSLint
SCSS Lint
CSSComb
Plugins
Fixing
Rules Count
CSSLint
SCSS Lint
CSSComb
38
26
62
data:image/s3,"s3://crabby-images/fd69f/fd69fa5320d2e6f67bc630f21345ddd65715b858" alt=""
Stylelint
data:image/s3,"s3://crabby-images/7c470/7c470eda06128e56437de5e47013b1a38f790643" alt=""
Source code
Parser
Plugin 1
Errors list
Plugin 2
ESLint Plugins Look Like …
data:image/s3,"s3://crabby-images/c096f/c096fcc655d5943c1cb3fa23e273aced3a7635dc" alt=""
One Framework to Rule Them All
cssnext
cssnano
Stylelint
PreCSS
CSS Modules
Changeable Parsers
CSS
SCSS
Less
SugarSS
Broken CSS
postcss-browser-reporter
data:image/s3,"s3://crabby-images/414b9/414b9cc6ffb69d2299c98f86ebd61a81dbce7c5d" alt=""
Stylelint Rule is a PostCSS Plugin
const ruleName = "comment-no-empty"
(root, result) => {
root.walkComments(comment => {
if ( comment.text && comment.text.length === 0 ) {
report({ … })
}
}
}
Rules White List
module.exports = {
'rules': {
'at-rule-name-case': 'lower',
'at-rule-semicolon-newline-after': 'always',
'block-closing-brace-newline-after': 'always',
'color-hex-case': 'lower',
'color-hex-length': 'short',
'color-hex-length': 'short',
'color-no-invalid-hex': true,
'indentation': 2,
…
Fixing
.foo {
color:black}
.foo {
color:black;
}
stylelint --fix *.css
SCSS
CSSLint
SCSS Lint
CSSComb
CSS
Less
Stylelint
Whitelist
CSSLint
SCSS Lint
CSSComb
Plugins
Fixing
Stylelint
CSSLint
SCSS Lint
CSSComb
38
26
62
Stylelint
176
Easy to Write Rules
Stylelint Users
data:image/s3,"s3://crabby-images/113e3/113e3cc6f6156270b49008fe6bccf3ca624a41e1" alt=""
The Most Popular in npm
data:image/s3,"s3://crabby-images/22245/222450f20b566741d21e39830b4c8396bf54eb91" alt=""
Why
data:image/s3,"s3://crabby-images/5777e/5777ef329ef47ee40c5f83c615ff48bf25f60475" alt=""
data:image/s3,"s3://crabby-images/3c8eb/3c8eba32838fdb42531c5771e55338258c789221" alt=""
Senior
Junior
Code Review
data:image/s3,"s3://crabby-images/43018/43018c512dfea3ad25406f3fecf6b5a7223cc78b" alt=""
data:image/s3,"s3://crabby-images/3c8eb/3c8eba32838fdb42531c5771e55338258c789221" alt=""
Senior
Junior
1. Less Time for Code Review
Stylelint
data:image/s3,"s3://crabby-images/36cca/36cca0baabf49e185eec5894954190ce1987e2d8" alt=""
data:image/s3,"s3://crabby-images/62e12/62e12c57be15df13e7990626844ef23d23476bd2" alt=""
2. Robot’s Criticism is Less Agressive
3. Sharing Best Practices
module.exports = {
'extend': 'stylelint-config-wordpress'
}
4. Preventing Errors
'color-no-invalid-hex',
'function-linear-gradient-no-nonstandard-direction',
'time-no-imperceptible',
'property-no-unknown',
'property-no-vendor-prefix',
'declaration-block-no-duplicate-properties',
'declaration-block-no-ignored-properties',
'declaration-block-no-shorthand-property-overrides',
'selector-class-pattern',
'selector-max-compound-selectors',
'selector-pseudo-element-no-unknown',
'selector-type-no-unknown',
'media-feature-no-missing-punctuation',
'no-unsupported-browser-features',
'no-unknown-animations',
'no-indistinguishable-colors',
'no-descending-specificity',
'no-browser-hacks'
#container .foo {
top: 10px;
}
…
.foo {
/* Expected selector .foo to come before #container .foo */
top: 0;
}
.foo {
opacity: 1;
/* Unsupported in IE 8 */
}
.foo {
display: inline;
width: 100px;
/* Unexpected width with display: inline */
}
.foo {
margin-top: 10px;
margin: 0 auto;
/* Unexpected shorthand margin after margin-top */
}
.foo {
color: black;
}
…
.bar {
background: #010101;
/* Unexpected indistinguishable colors #010101 and black */
}
How
data:image/s3,"s3://crabby-images/b34ea/b34ea861a16ff8e71691f6d0202025ed1a47bc6c" alt=""
1. Linter in Text Editor
data:image/s3,"s3://crabby-images/eb3e3/eb3e3fd8be8cc3f1d3bf1bca1c5278462540fff2" alt=""
2. lint-staged in Pre-Commit
"scripts": {
"lint-staged": "lint-staged",
},
"lint-staged": {
"*.css": "stylelint"
},
"pre-commit": ["lint-staged"]
3. Linting in CI
data:image/s3,"s3://crabby-images/27496/27496f64004fa7eefc6ab2ffe941d6d762003e48" alt=""
4. Start with Popular Config
module.exports = {
'extend': 'stylelint-config-standard'
}
5. Use Third-Party Plugins
module.exports = {
'plugins': [
'stylelint-scss'
],
'rules': {
'scss/selector-no-redundant-nesting-selector': true
}
}
dustinspecker/awesome-eslint
data:image/s3,"s3://crabby-images/fe554/fe55457fcf9764b469862fa07921644438de6bb1" alt=""
6. Write Custom Rules
import { utils } from "stylelint"
export const ruleName = namespace("NAME")
export const messages = utils.ruleMessages(ruleName, {
expected: "ERROR DESCRIPTION",
})
export default function () {
return (root, result) => {
/* BODY */
}
}
Repeating Colors to Variables
colors = { }
return (root, result) => {
root.walkDecls(decl => {
decl.value.match(/#[0-9a-f]{3,6}/, color => {
if ( colors[color] ) {
utils.report({ … })
} else {
colors[color] = true
}
})
})
}
Facebook Custom Rules
-
slow-css-properties
-
filters-with-svg-files
-
use-variables
-
mobile-flexbox
7. Happened Twice — Write a Rule
data:image/s3,"s3://crabby-images/cbc46/cbc46df645c5be93b3148ee4807b4490d06e4a40" alt=""
8. Lint Everything
gulp.task('default', ['lint',
'test',
'all-translated',
'spell-check',
'security-audit',
'file-size'])
postcss-sorting
div {
-moz-box-sizing: border-box;
width: 100%;
box-sizing: border-box;
position: absolute;
-webkit-box-sizing: border-box;
}
div {
position: absolute;
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
box-sizing: border-box;
width: 100%;
}
$ yaspeller /home/ai/Dev/postcss/README.md
✗ /home/ai/Dev/postcss/README.md 453 ms
-----
Typos: 1
1. transorming (suggest: transforming, transporting)
-----
Node Security Platform
> nsp check
(+) 1 vulnerabilities found
┌───────────────┬───────────────────────────────────────────────────────┐
│ │ ReDoS via long string of semicolons │
├───────────────┼───────────────────────────────────────────────────────┤
│ Name │ tough-cookie │
├───────────────┼───────────────────────────────────────────────────────┤
│ Installed │ 2.2.2 │
├───────────────┼───────────────────────────────────────────────────────┤
│ Vulnerable │ >=0.9.7 <=2.2.2 │
├───────────────┼───────────────────────────────────────────────────────┤
│ Patched │ >=2.3.0 │
├───────────────┼───────────────────────────────────────────────────────┤
│ Path │ my-test-project@undefined > honeybadger@1.1.2 > requ… │
├───────────────┼───────────────────────────────────────────────────────┤
│ More Info │ https://nodesecurity.io/advisories/130 │
└───────────────┴───────────────────────────────────────────────────────┘
Links
Stylelint: Why and How to Lint CSS
By Andrey Sitnik
Stylelint: Why and How to Lint CSS
- 15,954