Best practices suggest avoiding overriding styles from vendor libraries to prevent unwanted side effects. Base library styles should not be altered β or as Harry Roberts describes, base styles should be treated as Immutable CSS.
See the interactive web app.
Installation
npm install --save immutable-css
Usage
immutableCss.processFiles(immutableSourceCss, customCss, options)
takes two stylesheet paths, and ensures the custom CSS doesn't override any selectors contained within the immutable source.
This is typically best when comparing vendor CSS (Bootstrap, Tachyons, Basscss, etc.) to your app's customizations.
var immutableCss = require('immutable-css')
immutableCss.processFiles('css/vendor.css', 'css/app.css')
// => [...]
immutableCss.processGlob(cssGlob, options)
takes a glob that matches CSS files and ensures that no stylesheet overrides selectors contained within another.
This is useful to ensure that CSS partials aren't mixing concerns by mutating selectors contained within another file.
var immutableCss = require('immutable-css')
immutableCss.processGlob('src/css/**/*.css', { verbose: true })
PostCSS
Using withImmutable CSS detects mutations among files by leveraging PostCSS sourcemaps. It is also best used as a PostCSS plugin in tandem with postcss-import
and postcss-reporter
.
var fs = require('fs')
var postcss = require('postcss')
var import = require('postcss-import')
var reporter = require('postcss-reporter')
var immutableCss = require('immutable-css')
var css = fs.readFileSync('styles.css', 'utf8')
var mutations = postcss([import(), immutableCss(), reporter()])
.process(css, { from: 'styles.css' })
Gulp
Using withvar gulp = require('gulp')
var postcss = require('gulp-postcss')
var import = require('postcss-import')
var reporter = require('postcss-reporter')
var immutableCss = require('immutable-css')
gulp.task('immutable', function () {
var processors = [
import,
immutableCss,
// If you want Immutable CSS to halt the gulp pipline if there are any warnings
// then set throwError to true
reporter({clearMessages: true, throwError: false})
]
gulp.src('assets/css/base.css')
.pipe(postcss(processors))
.pipe(gulp.dest('dist/css'))
})
Input
@import 'basscss';
.button {}
.left {}
.something-else {}
Output
β .button was mutated 2 times
[line 93, col 1]: /css/basscss.css
[line 3, col 1]: /css/custom.css
[immutable-css]
β .left was mutated 2 times
[line 291, col 1]: /css/basscss.css
[line 4, col 1]: /css/custom.css
[immutable-css]
Options
strict
(Boolean): Whether class mutations are allowed in the same file. Default:false
.ignoredClasses
(Array): List of classes to ignore for mutation violations.
Ex:['.some-mutable-class']
immutableClasses
(Array): List of classes to check against.
Ex:['.button', '.foobar']
immutablePrefixes
(Array): List of prefix regexes that are immutable.
Ex:[/\.u\-/, /\.util\-/]
callback
(Function): Callback that receives a mutations object.
Ex:function (mutations) { console.log(mutations) }
verbose
(Boolean): Whether mutations are logged (defaults to true with PostCSS).
Using the callback
Immutable CSS accepts an optional callback, which returns the mutations hash. The key is the mutated class name, the value is an array of mutating filenames.
postcss([
import(),
immutableCss({ ignoredClasses: ['.button'] }, function(mutations) {
console.log(mutations)
// => { '.foobar': [] }
})
]).process(css, { from: cssFile })
immutable-css-cli
Using thenpm i -g immutable-css-cli
immutable-css css/main.css
β .button was mutated 2 times
[line 93, col 1]: /css/_basscss.css
[line 11, col 1]: /css/_custom.css
[immutable-css]
β .left was mutated 2 times
[line 291, col 1]: /css/_basscss.css
[line 15, col 1]: /css/_custom.css
[immutable-css]
https://github.com/johnotander/immutable-css-cli
Dependencies
- https://github.com/postcss/postcss
- https://github.com/johnotander/get-css-classes
- https://github.com/css-modules/css-selector-tokenizer
Related Reading
- http://csswizardry.com/2015/03/immutable-css/
- http://csswizardry.com/2012/06/the-open-closed-principle-applied-to-css/
- http://www.jon.gold/2015/07/functional-css/
- http://eng.wealthfront.com/2013/08/functional-css-fcss.html
- http://www.basscss.com/docs/reference/principles/#immutable-utilities
License
MIT
Contributing
- Fork it
- Create your feature branch (
git checkout -b my-new-feature
) - Commit your changes (
git commit -am 'Add some feature'
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request
Crafted with <3 by @jxnblk & @4lpine.
This package was initially generated with yeoman and the p generator.