2023-03-01 20:46:05 +01:00
|
|
|
---
|
2023-03-09 17:19:39 +01:00
|
|
|
title: "Request for Comments: New JS API"
|
2023-03-01 20:46:05 +01:00
|
|
|
author: Natalie Weizenbaum
|
2023-03-03 16:39:31 +01:00
|
|
|
date: 2021-08-05 15:30:00 -8
|
2023-03-01 20:46:05 +01:00
|
|
|
---
|
|
|
|
|
|
|
|
I'm excited to officially unveil something that's been in the works for quite a
|
|
|
|
while now: a (proposal for a) brand new JavaScript API for Sass. This API has
|
|
|
|
been redesigned from the ground up based on lessons learned from both the Node
|
|
|
|
Sass API and various other historical Sass APIs in other languages through the
|
|
|
|
years, and it addresses many of the shortcomings of the existing API.
|
|
|
|
|
|
|
|
The API has four main components, all of which I'll cover in this post:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* [The core compilation API](#compilation)
|
|
|
|
* [The logger API](#loggers)
|
|
|
|
* [The importer API](#importers)
|
|
|
|
* [The function API](#functions)
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
As you read on, remember that this API is still just a proposal. We want to hear
|
|
|
|
from you, our users, whether it meets your needs and how we can improve it
|
|
|
|
before we lock it in to a full release. So go ahead and make your voices known
|
|
|
|
[on the issue tracker]!
|
|
|
|
|
|
|
|
[on the issue tracker]: https://github.com/sass/sass/issues/new
|
|
|
|
|
|
|
|
## Why a New API?
|
|
|
|
|
|
|
|
The existing JavaScript API is showing its age. It predates Dart Sass, having
|
|
|
|
been originally designed for the `node-sass` package, which wrapped the
|
|
|
|
now-[deprecated] [LibSass] implementation. (That's why we call it the "Node Sass
|
|
|
|
API"!) It grew organically and often messily along with LibSass, and ended up
|
|
|
|
with more than a few awkward legacy behaviors. Many of these behaviors are more
|
|
|
|
of a pain for implementation than anything else, but a few of them made life
|
|
|
|
quite difficult:
|
|
|
|
|
|
|
|
[deprecated]: /blog/libsass-is-deprecated
|
|
|
|
[LibSass]: /libsass
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* The importer API was built around file paths rather than URLs, and was tightly
|
|
|
|
coupled to the physical filesystem. This made it impossible to override *all*
|
2023-03-01 20:46:05 +01:00
|
|
|
file-based loads and present a fully virtual filesystem, and caused custom
|
|
|
|
Node importers to interact poorly with the new [module system].
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* The function API was built around mutable value objects, which runs counter to
|
2023-03-01 20:46:05 +01:00
|
|
|
Sass's immutable nature. It also provided no utility methods (such as looking
|
|
|
|
up a key in a map) to make it easier to implement idiomatic custom functions,
|
|
|
|
and didn't provide access to crucial information about values such as whether
|
|
|
|
strings were quoted.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* All of the asynchronous functions were callback-based rather than
|
2023-03-01 20:46:05 +01:00
|
|
|
promise-based.
|
|
|
|
|
2023-06-09 04:18:33 +02:00
|
|
|
[module system]: /blog/the-module-system-is-launched
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
The new API addresses these issues and more with a modern, idiomatic API that
|
|
|
|
will make working with Sass from JS a breeze.
|
|
|
|
|
|
|
|
## Compilation
|
|
|
|
|
|
|
|
At the heart of the API are four functions that do the actual Sass compilation,
|
|
|
|
two synchronous and two asynchronous. They're presented here in TypeScript
|
|
|
|
syntax to clarify exactly what they take and return, but you can always call
|
|
|
|
them from plain JS:
|
|
|
|
|
|
|
|
```ts
|
2023-03-09 17:19:39 +01:00
|
|
|
function compile(
|
|
|
|
path: string,
|
|
|
|
options?: Options<'sync'>
|
|
|
|
): CompileResult;
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
function compileString(
|
|
|
|
source: string,
|
2023-03-09 17:19:39 +01:00
|
|
|
options?: StringOptions<'sync'>
|
2023-03-01 20:46:05 +01:00
|
|
|
): CompileResult;
|
|
|
|
|
|
|
|
function compileAsync(
|
|
|
|
path: string,
|
2023-03-09 17:19:39 +01:00
|
|
|
options?: Options<'async'>
|
2023-03-01 20:46:05 +01:00
|
|
|
): Promise<CompileResult>;
|
|
|
|
|
|
|
|
function compileStringAsync(
|
|
|
|
source: string,
|
2023-03-09 17:19:39 +01:00
|
|
|
options?: StringOptions<'async'>
|
2023-03-01 20:46:05 +01:00
|
|
|
): Promise<CompileResult>;
|
|
|
|
```
|
|
|
|
|
|
|
|
The `compile()` and `compileAsync()` functions load a Sass file from a path on
|
|
|
|
disk, whereas `compileString()` and `compileStringAsync()` compile Sass source
|
|
|
|
code passed in as a string. All these take the following options:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* `alertAscii`: Whether errors and warnings should use only ASCII characters (as
|
2023-03-01 20:46:05 +01:00
|
|
|
opposed to, for example, Unicode box-drawing characters).
|
2023-03-09 17:19:39 +01:00
|
|
|
* `alertColor`: Whether errors and warnings should use terminal colors.
|
|
|
|
* `loadPaths`: A list of file paths to use to look up files to load, just like
|
2023-03-01 20:46:05 +01:00
|
|
|
`includePaths` in the old API.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `importers`: A list of [custom importers](#importers) to use to load Sass
|
2023-03-01 20:46:05 +01:00
|
|
|
source files.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `functions`: An object whose keys are Sass function signatures and whose
|
2023-03-01 20:46:05 +01:00
|
|
|
values are [custom functions](#functions).
|
2023-03-09 17:19:39 +01:00
|
|
|
* `quietDeps`: Whether to silence deprecation warnings in dependencies.
|
|
|
|
* `logger`: The [custom logger](#loggers) to use to emit warnings and debug
|
2023-03-01 20:46:05 +01:00
|
|
|
messages.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `sourceMap`: Whether to generate a source map during compilation.
|
|
|
|
* `style`: The output style, `'compressed'` or `'expanded'`.
|
|
|
|
* `verbose`: Whether to emit every deprecation warning encountered.
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
The `compileString()` and `compileStringAsync()` functions take a few additional
|
|
|
|
options:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* `syntax`: The syntax of the file, `'scss'` (the default), `'indented'`, or
|
2023-03-01 20:46:05 +01:00
|
|
|
`'css'`.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `url`: The [canonical URL](#canonicalizing) of the file.
|
|
|
|
* `importer`: The [custom importer](#importers) to treat as the file's source.
|
2023-03-01 20:46:05 +01:00
|
|
|
If this is passed, this importer will be used to resolve relative loads from
|
|
|
|
this stylesheet.
|
|
|
|
|
|
|
|
All these functions return an object with the following fields:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* `css`: The compiled CSS, as a string.
|
|
|
|
* `loadedUrls`: All the URLs loaded during the compilation, in no particular
|
2023-03-01 20:46:05 +01:00
|
|
|
order.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `sourceMap`: The source map for the file if `sourceMap: true` was passed, as
|
2023-03-01 20:46:05 +01:00
|
|
|
a decoded object.
|
|
|
|
|
|
|
|
As with the Node Sass API, the synchronous functions will be substantially
|
|
|
|
faster than their asynchronous counterparts. Unfortunately the new API will not
|
|
|
|
support the `fibers` option for speeding up asynchronous compilation, since [the
|
|
|
|
`fibers` package has been discontinued].
|
|
|
|
|
|
|
|
[the `fibers` package has been discontinued]: /blog/node-fibers-discontinued
|
|
|
|
|
|
|
|
## Loggers
|
|
|
|
|
|
|
|
The logger API gives you more fine-grained control over how and when warnings
|
|
|
|
and debug messages are emitted. Unlike other aspects of this proposal, a
|
2023-03-09 17:19:39 +01:00
|
|
|
`logger` option will also be added to the *old* API to allow you to control your
|
2023-03-01 20:46:05 +01:00
|
|
|
messages there without needing to upgrade to the new API immediately.
|
|
|
|
|
|
|
|
A logger implements the following interface:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
interface Logger {
|
|
|
|
warn?(
|
|
|
|
message: string,
|
|
|
|
options: {
|
|
|
|
deprecation: boolean;
|
|
|
|
span?: SourceSpan;
|
|
|
|
stack?: string;
|
2023-03-09 17:19:39 +01:00
|
|
|
}
|
2023-03-01 20:46:05 +01:00
|
|
|
): void;
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
debug?(
|
|
|
|
message: string,
|
|
|
|
options: {span: SourceSpan}
|
|
|
|
): void;
|
2023-03-01 20:46:05 +01:00
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
The `warn` function handles warnings, including both warnings from the compiler
|
|
|
|
itself and from `@warn` rules. It's passed:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* The warning message
|
|
|
|
* A flag indicating whether it's specifically a deprecation warning
|
|
|
|
* A span indicating where the warning was located, if it comes from a specific
|
2023-03-01 20:46:05 +01:00
|
|
|
location
|
2023-03-09 17:19:39 +01:00
|
|
|
* The Sass stack trace at the point at which the warning was encountered, if it
|
2023-03-01 20:46:05 +01:00
|
|
|
was encountered during execution
|
|
|
|
|
|
|
|
The `debug` function handles only `@debug` rules, and is just passed the message
|
|
|
|
and the rule's span. For more information on the `SourceSpan` type, see [the
|
|
|
|
Logger proposal].
|
|
|
|
|
|
|
|
[the Logger proposal]: https://github.com/sass/sass/tree/main/proposal/js-logger.d.ts
|
|
|
|
|
|
|
|
Sass will also provide a built-in logger, `Logger.silent`, that never emits any
|
|
|
|
messages. This will allow you to easily run Sass in "quiet mode" where no
|
|
|
|
warnings are ever surfaced.
|
|
|
|
|
|
|
|
## Importers
|
|
|
|
|
|
|
|
Rather than modeling importers as single-function callbacks, the new API models
|
|
|
|
them as objects that expose two methods: one that _canonicalizes_ a URL, and one
|
|
|
|
that _loads_ a canonical URL.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
// Importers for compileAsync() and compileStringAsync() are the same, except
|
|
|
|
// they may return Promises as well.
|
|
|
|
interface Importer {
|
2023-03-09 17:19:39 +01:00
|
|
|
canonicalize(
|
|
|
|
url: string,
|
|
|
|
options: {fromImport: boolean}
|
|
|
|
): URL | null;
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
load(canonicalUrl: URL): ImporterResult | null;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Note that even stylesheets that are loaded directly from the filesystem through
|
|
|
|
`compile()` or `loadPaths` are treated as though they're loaded by an importer.
|
|
|
|
This built-in filesystem importer canonicalizes all paths to `file:` URLs, and
|
|
|
|
loads those URLs from the physical filesystem.
|
|
|
|
|
|
|
|
### Canonicalizing
|
|
|
|
|
|
|
|
The first step determines the canonical URL for a stylesheet. Each stylesheet
|
|
|
|
has exactly one canonical URL that in turn refers to exactly one stylesheet. The
|
|
|
|
canonical URL must be absolute, including a scheme, but the specific structure
|
|
|
|
is up to the importer. In most cases, the stylesheet in question will exist on
|
|
|
|
disk and the importer will just return a `file:` URL for it.
|
|
|
|
|
|
|
|
The `canonicalize()` method takes a URL string that may be either relative or
|
|
|
|
absolute. If the importer recognizes that URL, it returns a corresponding
|
|
|
|
absolute URL (including a scheme). This is the _canonical URL_ for the
|
|
|
|
stylesheet in question. Although the input URL may omit a file extension or
|
|
|
|
an initial underscore, the canonical URL must be fully resolved.
|
|
|
|
|
|
|
|
For a stylesheet that's loaded from the filesystem, the canonical URL will be
|
|
|
|
the absolute `file:` URL of the physical file on disk. If it's generated
|
|
|
|
in-memory, the importer should choose a custom URL scheme to guarantee that
|
|
|
|
its canonical URLs don't conflict with any other importer's.
|
|
|
|
|
|
|
|
For example, if you're loading Sass files from a database, you might use the
|
|
|
|
scheme `db:`. The canonical URL for a stylesheet associated with key `styles`
|
|
|
|
in the database might be `db:styles`.
|
|
|
|
|
|
|
|
This function also takes a `fromImport` option that indicates whether the
|
|
|
|
importer is being invoked from an `@import` rule (as opposed to `@use`,
|
|
|
|
`@forward`, or `meta.load-css()`).
|
|
|
|
|
|
|
|
Having a canonical URL for each stylesheet allows Sass to ensure that the
|
|
|
|
same stylesheet isn't loaded multiple times in the new module system.
|
|
|
|
|
|
|
|
#### Canonicalizing Relative Loads
|
|
|
|
|
|
|
|
When a stylesheet tries to load a relative URL, such as `@use "variables"`, it's
|
|
|
|
not clear from the document itself whether that refers to a file that exists
|
|
|
|
relative to the stylesheet or to another importer or load path. Here's how the
|
|
|
|
importer API resolves that ambiguity:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* First, the relative URL is resolved relative to the canonical URL of the
|
2023-03-01 20:46:05 +01:00
|
|
|
stylesheet that contained the `@use` (or `@forward` or `@import`). For
|
|
|
|
example, if the canonical URL is `file:///path/to/my/_styles.scss`, then the
|
|
|
|
resolved URL will be `file:///path/to/my/variables`.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* This URL is then passed to the `canonicalize()` method of the importer that
|
2023-03-01 20:46:05 +01:00
|
|
|
loaded the old stylesheet. (That means it's important for your importers to
|
|
|
|
support absolute URLs!) If the importer recognizes it, it returns the
|
|
|
|
canonical value which is then passed to that importer's `load()`; otherwise,
|
|
|
|
it returns `null`.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* If the old stylesheet's importer didn't recognize the URL, it's passed to all
|
2023-03-01 20:46:05 +01:00
|
|
|
the `importers`' canonicalize functions in the order they appear in `options`,
|
|
|
|
then checked for in all the `loadPaths`. If none of those recognizes it, the
|
|
|
|
load fails.
|
|
|
|
|
|
|
|
It's important that local relative paths take precedence over other importers or
|
|
|
|
load paths, because otherwise your local stylesheets could get unexpectedly
|
|
|
|
broken by a dependency adding a file with a conflicting name.
|
|
|
|
|
|
|
|
### Loading
|
|
|
|
|
|
|
|
The second step actually loads the text of the stylesheet. The `load()`
|
|
|
|
method takes a canonical URL that was returned by `canonicalize()` and
|
|
|
|
returns the contents of the stylesheet at that URL. This is only called once
|
|
|
|
per compilation for each canonical URL; future loads of the same URL will
|
|
|
|
re-use either the existing module (for `@use` and `@forward`) or the parse
|
|
|
|
tree (for `@import`).
|
|
|
|
|
|
|
|
The `load()` method returns an object with the following fields:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* `css`: The text of the loaded stylesheet.
|
|
|
|
* `syntax`: The syntax of the file: `'scss'`, `'indented'`, or `'css'`.
|
|
|
|
* `sourceMapUrl`: An optional browser-accessible `URL` to include in source maps
|
2023-03-01 20:46:05 +01:00
|
|
|
when referring to this file.
|
|
|
|
|
|
|
|
### `FileImporter`
|
|
|
|
|
|
|
|
This proposal also adds a special type of importer known as a `FileImporter`.
|
|
|
|
This importer makes the common case of redirecting loads to somewhere on the
|
|
|
|
physical filesystem easier. It doesn't require the caller to implement
|
|
|
|
`load()`, since that's always going to be the same for files on disk.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
interface FileImporter {
|
|
|
|
findFileUrl(
|
|
|
|
url: string,
|
2023-03-09 17:19:39 +01:00
|
|
|
options: {fromImport: boolean}
|
2023-03-01 20:46:05 +01:00
|
|
|
): FileImporterResult | null;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
The `findFileUrl()` method takes a relative URL and returns an object with the
|
|
|
|
following fields:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* `url`: The absolute `file:` URL of the file to load. This URL doesn't need to
|
2023-03-01 20:46:05 +01:00
|
|
|
be fully canonicalized: the Sass compiler will take care of resolving
|
|
|
|
partials, file extensions, index files, and so on.
|
2023-03-09 17:19:39 +01:00
|
|
|
* `sourceMapUrl`: An optional browser-accessible `URL` to include in source maps
|
2023-03-01 20:46:05 +01:00
|
|
|
when referring to this file.
|
|
|
|
|
|
|
|
## Functions
|
|
|
|
|
|
|
|
The new function API's function type is very similar to the old API's:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
type CustomFunctionCallback = (args: Value[]) => Value;
|
|
|
|
```
|
|
|
|
|
|
|
|
The only differences are:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* Async functions return a `Promise<Value>` rather than calling a callback.
|
|
|
|
* The value types themselves are different.
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
The second point is pretty substantial, though! The new value types are much
|
|
|
|
more fleshed out than the old versions. Let's start with the parent class:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
abstract class Value {
|
|
|
|
/**
|
|
|
|
* Returns the values of `this` when interpreted as a list.
|
|
|
|
*
|
|
|
|
* - For a list, this returns its elements.
|
|
|
|
* - For a map, this returns each of its key/value pairs as a `SassList`.
|
|
|
|
* - For any other value, this returns a list that contains only that value.
|
|
|
|
*/
|
|
|
|
get asList(): List<Value>;
|
|
|
|
|
|
|
|
/** Whether `this` is a bracketed Sass list. */
|
|
|
|
get hasBrackets(): boolean;
|
|
|
|
|
|
|
|
/** Whether `this` is truthy (any value other than `null` or `false`). */
|
|
|
|
get isTruthy(): boolean;
|
|
|
|
|
|
|
|
/** Returns JS's null if this is `sassNull`, or `this` otherwise. */
|
|
|
|
get realNull(): null | Value;
|
|
|
|
|
|
|
|
/** If `this` is a list, return its separator. Otherwise, return `null`. */
|
|
|
|
get separator(): ListSeparator;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts the Sass index `sassIndex` to a JS index into the array returned
|
|
|
|
* by `asList`.
|
|
|
|
*
|
|
|
|
* Sass indices start counting at 1, and may be negative in order to index
|
|
|
|
* from the end of the list.
|
|
|
|
*/
|
|
|
|
sassIndexToListIndex(sassIndex: Value): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassBoolean`, and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertBoolean(name?: string): SassBoolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassColor`, and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertColor(name?: string): SassColor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassFunction`, and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of the parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertFunction(name?: string): SassFunction;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassMap` (or converts it to a `SassMap` if it's
|
|
|
|
* an empty list), and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of the parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertMap(name?: string): SassMap;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassNumber`, and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertNumber(name?: string): SassNumber;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it's a `SassString`, and throws an error otherwise.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertString(name?: string): SassString;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the value of `this` if it can be interpreted as a map.
|
|
|
|
*
|
|
|
|
* - If this is a map, returns its contents.
|
|
|
|
* - If this is an empty list, returns an empty map.
|
|
|
|
* - Otherwise, returns `null`.
|
|
|
|
*/
|
|
|
|
tryMap(): OrderedMap<Value, Value> | null;
|
|
|
|
|
|
|
|
/** Returns whether `this == other` in SassScript. */
|
|
|
|
equals(other: Value): boolean;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
There are a couple important things to note here:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* Because CSS doesn't have a strong syntactic differentiation between a single
|
2023-03-01 20:46:05 +01:00
|
|
|
element and a list containing one element, any Sass value may be treated as
|
|
|
|
though it's a list. The `Value` makes it easy to follow this convention by
|
|
|
|
making the `asList()`, `hasBrackets()`, and `separator()` getters available
|
|
|
|
for every `Value`.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* The list returned this was and the map returned by `asMap()` are immutable
|
2023-03-01 20:46:05 +01:00
|
|
|
types from the [`immutable` package]. This reflects Sass's built-in
|
|
|
|
immutability of all its types. Although these values can't be modified
|
|
|
|
directly, their APIs make it easy and efficient to create new values with
|
|
|
|
changes applied.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* Sass's list-indexing conventions are different than JavaScript's. The
|
2023-03-01 20:46:05 +01:00
|
|
|
`sassIndexToListIndex()` function makes it easy to convert from Sass index to
|
|
|
|
JS index.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* In Sass, any value may be used in a boolean context, with `false`
|
2023-03-01 20:46:05 +01:00
|
|
|
and `null` counting as "falsey" values. The `isTruthy` getter makes this
|
|
|
|
convention easy to follow.
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* The `assert*()` functions make it easy to ensure that you're being passed the
|
2023-03-01 20:46:05 +01:00
|
|
|
arguments you expect, and to throw an idiomatic error if you're not. They're
|
|
|
|
particularly useful for TypeScript users since they'll automatically narrow
|
|
|
|
the type of the `Value`.
|
|
|
|
|
|
|
|
[`immutable` package]: https://immutable-js.com/
|
|
|
|
|
|
|
|
Most Sass values have their own subclasses, but there are three singleton values
|
|
|
|
that are just available as constants: `sassTrue`, `sassFalse`, and `sassNull`
|
|
|
|
represent Sass's `true`, `false`, and `null` values respectively.
|
|
|
|
|
|
|
|
### Colors
|
|
|
|
|
|
|
|
The new API's `SassColor` class provides access to colors in RGB, HSL, and HWB
|
|
|
|
format. As with built-in Sass color functions, any attribute can be accessed on
|
|
|
|
any color regardless of how it was initially created.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassColor extends Value {
|
|
|
|
/** Creates an RGB color. */
|
|
|
|
static rgb(
|
|
|
|
red: number,
|
|
|
|
green: number,
|
|
|
|
blue: number,
|
2023-03-09 17:19:39 +01:00
|
|
|
alpha?: number
|
2023-03-01 20:46:05 +01:00
|
|
|
): SassColor;
|
|
|
|
|
|
|
|
/** Creates an HSL color. */
|
|
|
|
static hsl(
|
|
|
|
hue: number,
|
|
|
|
saturation: number,
|
|
|
|
lightness: number,
|
2023-03-09 17:19:39 +01:00
|
|
|
alpha?: number
|
2023-03-01 20:46:05 +01:00
|
|
|
): SassColor;
|
|
|
|
|
|
|
|
/** Creates an HWB color. */
|
|
|
|
static hwb(
|
|
|
|
hue: number,
|
|
|
|
whiteness: number,
|
|
|
|
blackness: number,
|
2023-03-09 17:19:39 +01:00
|
|
|
alpha?: number
|
2023-03-01 20:46:05 +01:00
|
|
|
): SassColor;
|
|
|
|
|
|
|
|
/** The color's red channel. */
|
|
|
|
get red(): number;
|
|
|
|
|
|
|
|
/** The color's green channel. */
|
|
|
|
get green(): number;
|
|
|
|
|
|
|
|
/** The color's blue channel. */
|
|
|
|
get blue(): number;
|
|
|
|
|
|
|
|
/** The color's hue. */
|
|
|
|
get hue(): number;
|
|
|
|
|
|
|
|
/** The color's saturation. */
|
|
|
|
get saturation(): number;
|
|
|
|
|
|
|
|
/** The color's lightness. */
|
|
|
|
get lightness(): number;
|
|
|
|
|
|
|
|
/** The color's whiteness. */
|
|
|
|
get whiteness(): number;
|
|
|
|
|
|
|
|
/** The color's blackeness. */
|
|
|
|
get blackness(): number;
|
|
|
|
|
|
|
|
/** The color's alpha channel. */
|
|
|
|
get alpha(): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a copy of `this` with the RGB channels updated to match `options`.
|
|
|
|
*/
|
|
|
|
changeRgb(options: {
|
|
|
|
red?: number;
|
|
|
|
green?: number;
|
|
|
|
blue?: number;
|
|
|
|
alpha?: number;
|
|
|
|
}): SassColor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a copy of `this` with the HSL values updated to match `options`.
|
|
|
|
*/
|
|
|
|
changeHsl(options: {
|
|
|
|
hue?: number;
|
|
|
|
saturation?: number;
|
|
|
|
lightness?: number;
|
|
|
|
alpha?: number;
|
|
|
|
}): SassColor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a copy of `this` with the HWB values updated to match `options`.
|
|
|
|
*/
|
|
|
|
changeHwb(options: {
|
|
|
|
hue?: number;
|
|
|
|
whiteness?: number;
|
|
|
|
blackness?: number;
|
|
|
|
alpha?: number;
|
|
|
|
}): SassColor;
|
|
|
|
|
|
|
|
/** Returns a copy of `this` with `alpha` as its alpha channel. */
|
|
|
|
changeAlpha(alpha: number): SassColor;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Numbers
|
|
|
|
|
|
|
|
The `SassNumber` class stores its numerator and denominator units as arrays
|
|
|
|
rather than strings. In addition, it provides methods for asserting that it has
|
|
|
|
specific units (`assertNoUnits()`, `assertUnit()`) and for converting it to
|
|
|
|
specific units (`convert()`, `convertToMatch()`, `convertValue()`,
|
|
|
|
`convertValueToMatch()`, `coerce()`, `coerceValue()`, `coerceValueToMatch()`).
|
|
|
|
|
|
|
|
Sass's numeric logic is also subtly different from JS, since Sass considers
|
|
|
|
numbers that differ by less than the 10th decimal digit to be identical. This
|
|
|
|
API provides a number of methods that help convert between this and JavaScript's
|
|
|
|
numeric logic.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassNumber extends Value {
|
|
|
|
/** Creates a Sass number with no units or a single numerator unit. */
|
|
|
|
constructor(value: number, unit?: string);
|
|
|
|
|
|
|
|
/** Creates a Sass number with multiple numerator and/or denominator units. */
|
|
|
|
static withUnits(
|
|
|
|
value: number,
|
|
|
|
options?: {
|
|
|
|
numeratorUnits?: string[] | List<string>;
|
|
|
|
denominatorUnits?: string[] | List<string>;
|
|
|
|
}
|
|
|
|
): SassNumber;
|
|
|
|
|
|
|
|
/** This number's value. */
|
|
|
|
get value(): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether `value` is an integer according to Sass's numeric logic.
|
|
|
|
*
|
|
|
|
* The integer value can be accessed using `asInt`.
|
|
|
|
*/
|
|
|
|
get isInt(): boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If `value` is an integer according to Sass's numeric logic, returns the
|
|
|
|
* corresponding JS integer, or `null` if `value` isn't an integer.
|
|
|
|
*/
|
|
|
|
get asInt(): number | null;
|
|
|
|
|
|
|
|
/** This number's numerator units. */
|
|
|
|
get numeratorUnits(): List<string>;
|
|
|
|
|
|
|
|
/** This number's denominator units. */
|
|
|
|
get denominatorUnits(): List<string>;
|
|
|
|
|
|
|
|
/** Whether `this` has numerator or denominator units. */
|
|
|
|
get hasUnits(): boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If `value` is an integer according to Sass's numeric logic, returns the
|
|
|
|
* corresponding JS integer, or throws an error if `value` isn't an integer.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of the parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertInt(name?: string): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If `value` is between `min` and `max` according to Sass's numeric logic,
|
|
|
|
* returns it clamped to that range. Otherwise, throws an error.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of the parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertInRange(min: number, max: number, name?: string): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it has no units. Otherwise, throws an error.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertNoUnits(name?: string): SassNumber;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `this` if it has `unit` as its single (numerator) unit. Otherwise,
|
|
|
|
* throws an error.
|
|
|
|
*
|
|
|
|
* The `name` parameter is used for error reporting. It should match the name
|
|
|
|
* of a parameter passed to the custom function (without the `$`).
|
|
|
|
*/
|
|
|
|
assertUnit(name?: stringunit: string): SassNumber;
|
|
|
|
|
|
|
|
/** Returns whether `this` has the single numerator unit `unit`. */
|
|
|
|
hasUnit(unit: string): boolean;
|
|
|
|
|
|
|
|
/** Returns whether this number's units are compatible with `unit`. */
|
|
|
|
compatibleWithUnit(unit: string): boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If this number's units are compatible with `newNumerators` and
|
|
|
|
* `newDenominators`, returns a new number with those units that's equal to
|
|
|
|
* `this`. Otherwise, throws an error.
|
|
|
|
*
|
|
|
|
* Note that unitless numbers are only compatible with other unitless numbers.
|
|
|
|
*/
|
|
|
|
convert(
|
|
|
|
newNumerators: string[] | List<string>,
|
|
|
|
newDenominators: string[] | List<string>
|
|
|
|
): SassNumber;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If this number's units are compatible with `other`'s, returns a new number
|
|
|
|
* with `other`'s units that's equal to `this`. Otherwise, throws an error.
|
|
|
|
*
|
|
|
|
* Note that unitless numbers are only compatible with other unitless numbers.
|
|
|
|
*/
|
|
|
|
convertToMatch(other: SassNumber): SassNumber;
|
|
|
|
|
|
|
|
/** Equivalent to `convert(newNumerators, newDenominators).value`. */
|
|
|
|
convertValue(
|
|
|
|
newNumerators: string[] | List<string>,
|
|
|
|
newDenominators: string[] | List<string>
|
|
|
|
): number;
|
|
|
|
|
|
|
|
/** Equivalent to `convertToMatch(other).value`. */
|
|
|
|
convertValueToMatch(other: SassNumber): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Like `convert()`, but if `this` is unitless returns a copy of it with the
|
|
|
|
* same value and the given units.
|
|
|
|
*/
|
|
|
|
coerce(
|
|
|
|
newNumerators: string[] | List<string>,
|
|
|
|
newDenominators: string[] | List<string>
|
|
|
|
): SassNumber;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Like `convertToMatch()`, but if `this` is unitless returns a copy of it
|
|
|
|
* with the same value and `other`'s units.
|
|
|
|
*/
|
|
|
|
coerceToMatch(other: SassNumber): SassNumber;
|
|
|
|
|
|
|
|
/** Equivalent to `coerce(newNumerators, newDenominators).value`. */
|
|
|
|
coerceValue(
|
|
|
|
newNumerators: string[] | List<string>,
|
|
|
|
newDenominators: string[] | List<string>
|
|
|
|
): number;
|
|
|
|
|
|
|
|
/** Equivalent to `coerceToMatch(other).value`. */
|
|
|
|
coerceValueToMatch(other: SassNumber): number;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Strings
|
|
|
|
|
|
|
|
The `SassString` class provides access to information about whether or not the
|
|
|
|
string is quoted. As with lists, JS's notion of indexes differs from Sass's, so
|
|
|
|
it also provides the `sassIndexToStringIndex()` method to convert a JS index
|
|
|
|
into a Sass index.
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassString extends Value {
|
|
|
|
/** Creates a string with the given `text`. */
|
|
|
|
constructor(
|
|
|
|
text: string,
|
|
|
|
options?: {
|
|
|
|
/** @default true */
|
|
|
|
quotes: boolean;
|
2023-03-09 17:19:39 +01:00
|
|
|
}
|
2023-03-01 20:46:05 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
/** Creates an empty string`. */
|
|
|
|
static empty(options?: {
|
|
|
|
/** @default true */
|
|
|
|
quotes: boolean;
|
|
|
|
}): SassString;
|
|
|
|
|
|
|
|
/** The contents of `this`. */
|
|
|
|
get text(): string;
|
|
|
|
|
|
|
|
/** Whether `this` has quotes. */
|
|
|
|
get hasQuotes(): boolean;
|
|
|
|
|
|
|
|
/** The number of Unicode code points in `text`. */
|
|
|
|
get sassLength(): number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts the Sass index `sassIndex` to a JS index into `text`.
|
|
|
|
*
|
|
|
|
* Sass indices start counting at 1, and may be negative in order to index
|
|
|
|
* from the end of the list. In addition, Sass indexes strings by Unicode code
|
|
|
|
* point, while JS indexes them by UTF-16 code unit.
|
|
|
|
*/
|
|
|
|
sassIndexToStringIndex(sassIndex: Value): number;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Lists
|
|
|
|
|
|
|
|
As mentioned above, most list functions are on the `Value` superclass to make it
|
|
|
|
easy to follow the Sass convention of treating all values as lists. However, the
|
|
|
|
`SassList` class can still be constructed to make new lists:
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassList extends Value {
|
|
|
|
/** Creates a Sass list with the given `contents`. */
|
|
|
|
constructor(
|
|
|
|
contents: Value[] | List<Value>,
|
|
|
|
options?: {
|
|
|
|
/** @default ',' */
|
|
|
|
separator?: ListSeparator;
|
|
|
|
/** @default false */
|
|
|
|
brackets?: boolean;
|
2023-03-09 17:19:39 +01:00
|
|
|
}
|
2023-03-01 20:46:05 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
/** Creates an empty Sass list. */
|
|
|
|
static empty(options?: {
|
|
|
|
/** @default null */
|
|
|
|
separator?: ListSeparator;
|
|
|
|
/** @default false */
|
|
|
|
brackets?: boolean;
|
|
|
|
}): SassList;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Maps
|
|
|
|
|
|
|
|
The `SassMap` class simply exposes its contents as an `OrderedMap` from the
|
|
|
|
[`immutable` package].
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassMap extends Value {
|
|
|
|
/** Creates a Sass map with the given `contents`. */
|
|
|
|
constructor(contents: OrderedMap<Value, Value>);
|
|
|
|
|
|
|
|
/** Creates an empty Sass map. */
|
|
|
|
static empty(): SassMap;
|
|
|
|
|
|
|
|
/** Returns this map's contents. */
|
|
|
|
get contents(): OrderedMap<Value, Value>;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### Functions
|
|
|
|
|
|
|
|
The `SassFunction` class is fairly restrictive: it just allows a new first-class
|
|
|
|
function to be created with a synchronous callback. These functions can't be
|
|
|
|
invoked by custom functions—but they still provide more power than the old API!
|
|
|
|
|
|
|
|
```ts
|
|
|
|
class SassFunction extends Value {
|
|
|
|
/**
|
|
|
|
* Creates a Sass function value with the given `signature` that calls
|
|
|
|
* `callback` when it's invoked.
|
|
|
|
*/
|
2023-03-09 17:19:39 +01:00
|
|
|
constructor(
|
|
|
|
signature: string,
|
|
|
|
callback: CustomFunctionCallback
|
|
|
|
);
|
2023-03-01 20:46:05 +01:00
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## More Information
|
|
|
|
|
|
|
|
If you want to know more about these proposals and see their most up-to-date
|
|
|
|
forms, they're available on GitHub to view in full:
|
|
|
|
|
2023-03-09 17:19:39 +01:00
|
|
|
* [Compile API proposal](https://github.com/sass/sass/tree/main/proposal/new-js-api.d.ts)
|
|
|
|
* [Logger proposal](https://github.com/sass/sass/blob/main/proposal/js-logger.d.ts)
|
|
|
|
* [Importer proposal](https://github.com/sass/sass/blob/main/proposal/new-js-importer.d.ts)
|
|
|
|
* [Functions and values proposal](https://github.com/sass/sass/blob/main/proposal/new-function-and-values-api.d.ts)
|
2023-03-01 20:46:05 +01:00
|
|
|
|
|
|
|
We're eager for feedback, so please [let us know what you think]! The proposals
|
|
|
|
in question will be open for at least a month after this blog post goes live,
|
|
|
|
and possibly more depending on how lively the discussion around them is.
|
|
|
|
|
|
|
|
[let us know what you think]: https://github.com/sass/sass/issues/new
|