Using index files when working with Sass and @use rules

Posted on  • tagged with 

With @import at-rules slowly being phased out of the main implementation of Sass (dart-sass) and eventually deprecated, its time to learn how to use @use rules and the neat features that come along with it.

The main implementation of Sass is dart-sass and for the sake of this article, I will be using the dart-sass implementation as it gets new features before any of the other implementations. Before jumping into the details of @use and index files. The @import at-rules in Sass are slowly being phased out and will eventually be deprecated in the next few years.

The Sass team discourages the continued use of the @import rule. Sass will gradually phase it out over the next few years, and eventually remove it from the language entirely. Prefer the @use rule instead.

Only Dart Sass currently supports @use. Users of other implementations must use the @import rule instead.

On GitHub, I helped drive the transition from @import to @use at-rules for mdn/mdn-minimalist which is the base Sass that powers MDN Web Docs. This conversion required me to study the Sass docs for transitioning from @import to @use and the details in this article are things I think are important when getting started with using @use and or working towards replacing your @import at-rules.

A new way to import

Before the @use at-rule was introduced to Sass, we relied on importing things (mixins, variables, stylesheets) with the @import at-rule. Unfortunately, the @import rules introduced many serious issues that can be described on the Sass docs. A few are:

Due to all of these flaws, the Sass team introduced a brand new and improved way to import your Sass stylesheets (mixins, functions, and variables) with @use. And honestly, it is so darn good. The old @import usage made everything globally accessible and now with @use the default namespace is the last component of the URL unless otherwise specified.

Using @import

Before @use was introduced in dart-sass, if we had a directory of partial files containing some mixins like this:

Filename: ./mixins/_sr-only.scss
@mixin screen-reader-only() {
...
}
Filename: ./mixins/_invert.scss
@mixin invert($c) {
filter: invert(#{$c});
}
Filename: ./mixins/_square.scss
@mixin square($x, $y, $c) {
width: $x;
height: $y;
background: $c;
}

We would have to import each file by using numerous @import at-rules and one by one import the partial files to our main file for usage and eventually compilation.

Filename: style.scss
/* Import the mixins */
@import "./mixins/sr-only";
@import "./mixins/invert";
@import "./mixins/square";

.hidden {
@include sr-only();
}

.demo {
@include invert(0.30);
@include square(20px, 20px, #f06);
}

If you only had a few mixins or files to import, this implementation wouldn't be that all bad. But if the mixins directory had 50 or more partial files, then writing each @import line-by-line would start to be overwhelming in large codebases.

The power of @use at-rules

With the introduction of @use at-rules, we have the ability to use index files. Using these index files allow us to use @forward rules in a _index.scss file so that when we load the URL of the directory ./mixins/, all of the partial files will be loaded with it. Giving us access to an entire directory of sass files with a single @use at-rule. Below is an example of using @forward to load stylesheets inside an index file.

Filename: ./mixins/_index.scss
@forward "./mixins/sr-only";
@forward "./mixins/invert";
@forward "./mixins/square";

Now instead of writing three separate @import or @use rules to load the URLs of the mixins we need. Simply load the URL representing the directory of partial files with a defined index file. In our case, the mixins directory can be loaded to make all of the forwarded stylesheets available with a single @use rule. Not only is this usage super clean and maintainable but saves quite a bit of time when many partial files need to be loaded.

Note: The @use at-rules must be placed at the top of the file, before any other content.

Filename: style.scss
@use "./mixins/";

.hidden {
@include mixins.sr-only():
}

.demo {
@include mixins.invert(0.30);
@include mixins.invert(20px, 20px, #f06);
}

Voila! One thing to note is that the "generic" @use at-rule usage will define a default namespace for the loaded content as the final component of the URL. So in our case, the namespace for this @use rule will be mixins. And accessing the loaded members within the directory will simply require you to reference the namespace and use dot notation like mixins.foo().

Sometimes you don't want to have a namespace attached to the loaded folder. Sass gives us the flexibility to define a custom namespace with @use "<URL>" as <namespace>;

@use "./mixins/" as m;

.hidden {
@include m.sr-only():
}

.demo {
@include m.invert(0.30);
@include m.invert(20px, 20px, #f06);
}

Or completely disregard a namespace by not defining one with @use "<URL>" as *;. This makes it so you can reference loaded members without a namespace and without using dot notation, just as you would if they were defined in the same file.

Only do this if you know there won't be any naming conflicts with other loaded members.

@use "./mixins/" as *;

.hidden {
@include sr-only():
}

.demo {
@include invert(0.30);
@include invert(20px, 20px, #f06);
}

There is more to discuss but this should cover the basics of @use at-rules. I'm hoping the information and examples in this article can provide Sass users a quick walkthrough of @use and index files. While also helping people in converting their projects from @import to @use since Sass will be phasing out @import over the next few years.

Table of contents

Share