Parent Wrapper Classes in Sass
I ran into a small head-scratcher of a problem the other day when I was migrating an old project's Sass stylesheets to use @use
and @forward
instead of @import
. I had a stylesheet that used @import
statements inside of a class:
.custom-class { @import "partials/spacing"; @import "partials/typography";}
This sort of thing worked using @import
statements, because those could be thrown just about anywhere in a Sass project, but it won't work with @use
because @use
statements pretty much have to be at the top-level of the Sass file they are called in. I'll walk through what the solution is below, but first let's get into a little bit of a contrived example project space to demonstrate the issue.
Example project setup
We'll start with our entry point file, styles/styles.scss
:
/* styles/styles.scss */.custom-class { @import "partials/spacing"; @import "partials/typography";}
We saw this above. It's a somewhat contrived example, but not very far off from what I was actually working with. It loads two partial files.
First, styles/partials/_spacing.scss
:
/* styles/partials/_spacing.scss */@use "../functions/pxToRem";
$size--100: 2px;$size--200: 4px;$size--300: 8px;
$space-map: ( 0: 0, 100: pxToRem.pxToRem($size--100), 200: pxToRem.pxToRem($size--200), 300: pxToRem.pxToRem($size--300),) !default;
@function Space($value) { @return map.get($space-map, $value);}
@each $index, $size in $space-map { .margin-#{$index} { margin: #{$size}; }}
Second, styles/partials/_typography.scss
:
/* styles/partials/_typography.scss */$font-size--100: 11px;$line-height--100: 1.7;$font-size--200: 13px;$line-height--200: 1.64;$font-size--300: 16px;$line-height--300: 1.5;$font-size--400: 18px;$line-height--400: 1.48;
$font-size-base: $font-size--400;
@mixin f100 { font-size: $font-size--100; line-height: $line-height--100;}
@mixin f200 { font-size: $font-size--200; line-height: $line-height--200;}
@mixin f300 { font-size: $font-size--300; line-height: $line-height--300;}
@mixin f400 { font-size: $font-size--400; line-height: $line-height--400;}
.f100 { @include f100;}
.f200 { @include f200;}
.f300 { @include f300;}
.f400 { @include f400;}
These two partials define some utility classes (ex. margin-200
or f400
) that we want to ultimately compile within the custom-class
class. If we run our compile command, we see that this works:
/* compiled css */.custom-styles .f100 { font-size: 11px; line-height: 1.7;}.custom-styles .f200 { font-size: 13px; line-height: 1.64;}.custom-styles .f300 { font-size: 16px; line-height: 1.5;}.custom-styles .f400 { font-size: 18px; line-height: 1.48;}.custom-styles .margin-0 { margin: 0;}.custom-styles .margin-100 { margin: 0.1111111111rem;}.custom-styles .margin-200 { margin: 0.2222222222rem;}.custom-styles .margin-300 { margin: 0.4444444444rem;}.custom-styles .f100 { font-size: 11px; line-height: 1.7;}.custom-styles .f200 { font-size: 13px; line-height: 1.64;}.custom-styles .f300 { font-size: 16px; line-height: 1.5;}.custom-styles .f400 { font-size: 18px; line-height: 1.48;}
The problem is that we get this warning when we run that compile step:
Deprecation Warning [import]: Sass @import rules are deprecated and will be removed in Dart Sass 3.0.0.
More info and automated migrator: https://sass-lang.com/d/import
╷3 │ @import "partials/typography"; │ ^^^^^^^^^^^^^^^^^^^^^ ╵ styles/styles.scss 3:11 root stylesheet
This should look familiar if you read my @use and @forward in Sass post. If you didn't and aren't sure what this warning means, please go back and read that post.
Use meta.load-css
instead
So how do we fix it? Well we could try to swap @import
for @use
:
/* styles/styles.scss */.custom-class { @use "partials/spacing"; @use "partials/typography";}
But we won't get very far. We get this error when we run the compile command:
Error: This at-rule is not allowed here. ╷2 │ @use "partials/spacing"; │ ^^^^^^^^^^^^^^^^^^^^^^^ ╵ styles/styles.scss 2:3 root stylesheet
Instead, we have to use the sass:meta
built-in module:
/* styles/styles.scss */@use "sass:meta";
.custom-class { @include meta.load-css("partials/spacing"); @include meta.load-css("partials/typography");}
meta.load-css
lets you import a Sass file inside of a selector, in this case from a relative path, and includes the CSS as the contents of a mixin, scoped to that parent selector.
With this change made, when I run the compile command, the compiled CSS is exactly what we want, and there are no deprecation warnings:
/* compiled css */.custom-class .f100 { font-size: 11px; line-height: 1.7;}.custom-class .f200 { font-size: 13px; line-height: 1.64;}.custom-class .f300 { font-size: 16px; line-height: 1.5;}.custom-class .f400 { font-size: 18px; line-height: 1.48;}.custom-class .margin-0 { margin: 0;}.custom-class .margin-100 { margin: 0.1111111111rem;}.custom-class .margin-200 { margin: 0.2222222222rem;}.custom-class .margin-300 { margin: 0.4444444444rem;}.custom-class .f100 { font-size: 11px; line-height: 1.7;}.custom-class .f200 { font-size: 13px; line-height: 1.64;}.custom-class .f300 { font-size: 16px; line-height: 1.5;}.custom-class .f400 { font-size: 18px; line-height: 1.48;}