A conditional/dynamic importer for node-sass. It provides the ability to @import
Sass files dynamically based on
their (environment) extension prefix, similar to React Native's
platform-specific extensions behaviour.
It reads in a list of environments
extension prefixes, which it'll attempt to use over the default file.
The example use case for this importer is as follows, say you have the following folder structure:
scss
├── custom
│ ├── style.custom.scss
│ ├── style.development.scss
│ ├── style.production.scss
│ └── style.scss
└── main.scss
And you want to import a different version of style.scss
based on a given build environment/variable.
This is not currently possible easily because Sass does not allow dynamic @import
s
using interpolation or in if
statements.
This importer allows you to simply pass in your current environment into the importer, and it checks for whether the environment-specific override file exists before importing it.
The environments
will be a list of environments ordered by the priority with which they should be used.
If none of the environment file overrides are available, then it falls back to the original file.
environments
: An array of environment extensions to look up. e.g.// process.env.NODE_ENV = 'production'; // Look for [`${file}.production.scss`, `${file}.fallback.scss`] [process.env.NODE_ENV, 'fallback']
This module hooks into node-sass's importer api.
var sass = require('node-sass');
var conditionalImporter = require('node-sass-conditional-importer');
sass.render({
file: scssFilename,
importer: [
conditionalImporter({
environments: [
// Search for `*.custom.scss` files first,
// Followed `*.(development|production).scss` files.
'custom',
process.env.NODE_ENV,
],
}),
// .. other importers
],
}, function(err, result) { /*...*/ });
Webpack / sass-loader
import conditionalImporter from 'node-sass-conditional-importer';
// Webpack config
export default {
module: {
loaders: [{
test: /\.scss$/,
loaders: ['style', 'css', 'sass']
}],
},
sassLoader: {
importer: conditionalImporter({
environments: [
// Import based on the NODE_ENV environment variable.
process.env.NODE_ENV,
],
})
}
};
import conditionalImporter from 'node-sass-conditional-importer';
// Webpack config
export default {
module: {
rules: [
{
test: /\.scss$/,
use: [
'style-loader',
{
loader: 'css-loader',
options: {
importLoaders: 1
},
},
{
loader: 'sass-loader',
options: {
importer: conditionalImporter({
environments: [
// Import based on the NODE_ENV environment variable.
process.env.NODE_ENV,
],
}),
},
},
],
},
],
},
};
Should you care to resolve paths using some kind of custom logic, for example,
resolving ~/
relative to the project root or some other arbitrary directory,
you can do it using the following:
main.scss
:
@import '~/dynamic.scss';
body {
background: $background;
}
custom/dynamic.myenvironment.scss
:
$background: red;
var path = require('path');
var sass = require('node-sass');
var conditionalImporter = require('node-sass-conditional-importer');
sass.render({
file: './main.scss',
importer: [
conditionalImporter({
environments: ['myenvironment'],
resolver: function(dir, url) {
return url.startsWith('~/')
? path.resolve(dir, 'custom', url.substr(2))
: path.resolve(dir, url);
},
})
],
}, function(err, result) { console.log(err || result.css.toString()) });
-
With a folder structure like:
scss ├── custom │ ├── style.custom.scss │ ├── style.development.scss │ ├── style.production.scss │ └── style.scss └── main.scss
A file like
style.production.scss
may not be used to importstyle.scss
as it'll result in an import loop.The recommended solution is to create a shared include file like
_style--shared.scss
and import that instead.
This importer is inspired by node-sass-json-importer.
node-sass-conditional-importer is MIT licensed, as found in the LICENSE file.