diff --git a/CHANGELOG.md b/CHANGELOG.md index 539705f1e..fab2d1e1d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,7 @@ +## 1.80.5-dev + +* No user-visible changes. + ## 1.80.4 * No user-visible changes. diff --git a/lib/src/ast/sass/statement/stylesheet.dart b/lib/src/ast/sass/statement/stylesheet.dart index 55aad5419..a030a71aa 100644 --- a/lib/src/ast/sass/statement/stylesheet.dart +++ b/lib/src/ast/sass/statement/stylesheet.dart @@ -53,6 +53,11 @@ final class Stylesheet extends ParentStatement> { @internal final List parseTimeWarnings; + /// The set of (normalized) global variable names defined by this stylesheet + /// to the spans where they're defined. + @internal + final Map globalVariables; + Stylesheet(Iterable children, FileSpan span) : this.internal(children, span, []); @@ -62,8 +67,11 @@ final class Stylesheet extends ParentStatement> { @internal Stylesheet.internal(Iterable children, this.span, List parseTimeWarnings, - {this.plainCss = false}) + {this.plainCss = false, Map? globalVariables}) : parseTimeWarnings = UnmodifiableListView(parseTimeWarnings), + globalVariables = globalVariables == null + ? const {} + : Map.unmodifiable(globalVariables), super(List.unmodifiable(children)) { loop: for (var child in this.children) { diff --git a/lib/src/parse/stylesheet.dart b/lib/src/parse/stylesheet.dart index cab1bf972..94f551d2d 100644 --- a/lib/src/parse/stylesheet.dart +++ b/lib/src/parse/stylesheet.dart @@ -61,13 +61,13 @@ abstract class StylesheetParser extends Parser { var _inExpression = false; /// A map from all variable names that are assigned with `!global` in the - /// current stylesheet to the nodes where they're defined. + /// current stylesheet to the spans where they're defined. /// /// These are collected at parse time because they affect the variables /// exposed by the module generated for this stylesheet, *even if they aren't /// evaluated*. This allows us to ensure that the stylesheet always exposes /// the same set of variable names no matter how it's evaluated. - final _globalVariables = {}; + final _globalVariables = {}; /// Warnings discovered while parsing that should be emitted during /// evaluation once a proper logger is available. @@ -100,15 +100,8 @@ abstract class StylesheetParser extends Parser { }); scanner.expectDone(); - /// Ensure that all global variable assignments produce a variable in this - /// stylesheet, even if they aren't evaluated. See sass/language#50. - statements.addAll(_globalVariables.values.map((declaration) => - VariableDeclaration(declaration.name, - NullExpression(declaration.expression.span), declaration.span, - guarded: true))); - return Stylesheet.internal(statements, scanner.spanFrom(start), warnings, - plainCss: plainCss); + plainCss: plainCss, globalVariables: _globalVariables); }); } @@ -288,7 +281,7 @@ abstract class StylesheetParser extends Parser { guarded: guarded, global: global, comment: precedingComment); - if (global) _globalVariables.putIfAbsent(name, () => declaration); + if (global) _globalVariables.putIfAbsent(name, () => declaration.span); return declaration; } diff --git a/lib/src/visitor/async_evaluate.dart b/lib/src/visitor/async_evaluate.dart index 69d9b0d88..4dc806e67 100644 --- a/lib/src/visitor/async_evaluate.dart +++ b/lib/src/visitor/async_evaluate.dart @@ -1005,6 +1005,14 @@ final class _EvaluateVisitor for (var child in node.children) { await child.accept(this); } + + // Make sure all global variables declared in a module always appear in the + // module's definition, even if their assignments aren't reached. + for (var (name, span) in node.globalVariables.pairs) { + visitVariableDeclaration( + VariableDeclaration(name, NullExpression(span), span, guarded: true)); + } + return null; } diff --git a/lib/src/visitor/evaluate.dart b/lib/src/visitor/evaluate.dart index 5e29377df..e06601361 100644 --- a/lib/src/visitor/evaluate.dart +++ b/lib/src/visitor/evaluate.dart @@ -5,7 +5,7 @@ // DO NOT EDIT. This file was generated from async_evaluate.dart. // See tool/grind/synchronize.dart for details. // -// Checksum: e7260fedcd4f374ba517a93d038c3c53586c9622 +// Checksum: 396c8f169d95c601598b8c3be1f4b948ca22effa // // ignore_for_file: unused_import @@ -1005,6 +1005,14 @@ final class _EvaluateVisitor for (var child in node.children) { child.accept(this); } + + // Make sure all global variables declared in a module always appear in the + // module's definition, even if their assignments aren't reached. + for (var (name, span) in node.globalVariables.pairs) { + visitVariableDeclaration( + VariableDeclaration(name, NullExpression(span), span, guarded: true)); + } + return null; } diff --git a/pkg/sass-parser/CHANGELOG.md b/pkg/sass-parser/CHANGELOG.md index 3c61db629..00ebfbc62 100644 --- a/pkg/sass-parser/CHANGELOG.md +++ b/pkg/sass-parser/CHANGELOG.md @@ -1,3 +1,7 @@ +## 0.4.2-dev + +* Add support for parsing variable declarations. + ## 0.4.1 * Add `BooleanExpression` and `NumberExpression`. diff --git a/pkg/sass-parser/lib/index.ts b/pkg/sass-parser/lib/index.ts index 6b878c390..f79ef0f6b 100644 --- a/pkg/sass-parser/lib/index.ts +++ b/pkg/sass-parser/lib/index.ts @@ -96,6 +96,11 @@ export { StatementType, StatementWithChildren, } from './src/statement'; +export { + VariableDeclaration, + VariableDeclarationProps, + VariableDeclarationRaws, +} from './src/statement/variable-declaration'; /** Options that can be passed to the Sass parsers to control their behavior. */ export type SassParserOptions = Pick; diff --git a/pkg/sass-parser/lib/src/sass-internal.ts b/pkg/sass-parser/lib/src/sass-internal.ts index 8c7093daa..1b31d6101 100644 --- a/pkg/sass-parser/lib/src/sass-internal.ts +++ b/pkg/sass-parser/lib/src/sass-internal.ts @@ -180,6 +180,14 @@ declare namespace SassInternal { readonly configuration: ConfiguredVariable[]; } + class VariableDeclaration extends Statement { + readonly namespace: string | null; + readonly name: string; + readonly expression: Expression; + readonly isGuarded: boolean; + readonly isGlobal: boolean; + } + class ConfiguredVariable extends SassNode { readonly name: string; readonly expression: Expression; @@ -238,6 +246,7 @@ export type Stylesheet = SassInternal.Stylesheet; export type StyleRule = SassInternal.StyleRule; export type SupportsRule = SassInternal.SupportsRule; export type UseRule = SassInternal.UseRule; +export type VariableDeclaration = SassInternal.VariableDeclaration; export type ConfiguredVariable = SassInternal.ConfiguredVariable; export type Interpolation = SassInternal.Interpolation; export type Expression = SassInternal.Expression; @@ -260,6 +269,7 @@ export interface StatementVisitorObject { visitStyleRule(node: StyleRule): T; visitSupportsRule(node: SupportsRule): T; visitUseRule(node: UseRule): T; + visitVariableDeclaration(node: VariableDeclaration): T; } export interface ExpressionVisitorObject { diff --git a/pkg/sass-parser/lib/src/statement/__snapshots__/variable-declaration.test.ts.snap b/pkg/sass-parser/lib/src/statement/__snapshots__/variable-declaration.test.ts.snap new file mode 100644 index 000000000..b2b5e0501 --- /dev/null +++ b/pkg/sass-parser/lib/src/statement/__snapshots__/variable-declaration.test.ts.snap @@ -0,0 +1,22 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`a variable declaration toJSON 1`] = ` +{ + "expression": <"bar">, + "global": false, + "guarded": false, + "inputs": [ + { + "css": "baz.$foo: "bar"", + "hasBOM": false, + "id": "", + }, + ], + "namespace": "baz", + "raws": {}, + "sassType": "variable-declaration", + "source": <1:1-1:16 in 0>, + "type": "decl", + "variableName": "foo", +} +`; diff --git a/pkg/sass-parser/lib/src/statement/declaration-internal.d.ts b/pkg/sass-parser/lib/src/statement/declaration-internal.d.ts new file mode 100644 index 000000000..c10d2fa6e --- /dev/null +++ b/pkg/sass-parser/lib/src/statement/declaration-internal.d.ts @@ -0,0 +1,77 @@ +// Copyright 2024 Google Inc. Use of this source code is governed by an +// MIT-style license that can be found in the LICENSE file or at +// https://opensource.org/licenses/MIT. + +import * as postcss from 'postcss'; + +import {Rule} from './rule'; +import {Root} from './root'; +import {AtRule, ChildNode, Comment, Declaration, NewNode} from '.'; + +/** + * A fake intermediate class to convince TypeScript to use Sass types for + * various upstream methods. + * + * @hidden + */ +export class _Declaration extends postcss.Declaration { + // Override the PostCSS container types to constrain them to Sass types only. + // Unfortunately, there's no way to abstract this out, because anything + // mixin-like returns an intersection type which doesn't actually override + // parent methods. See microsoft/TypeScript#59394. + + after(newNode: NewNode): this; + append(...nodes: NewNode[]): this; + assign(overrides: Partial): this; + before(newNode: NewNode): this; + cloneAfter(overrides?: Partial): this; + cloneBefore(overrides?: Partial): this; + each( + callback: (node: ChildNode, index: number) => false | void + ): false | undefined; + every( + condition: (node: ChildNode, index: number, nodes: ChildNode[]) => boolean + ): boolean; + insertAfter(oldNode: postcss.ChildNode | number, newNode: NewNode): this; + insertBefore(oldNode: postcss.ChildNode | number, newNode: NewNode): this; + next(): ChildNode | undefined; + prepend(...nodes: NewNode[]): this; + prev(): ChildNode | undefined; + replaceWith(...nodes: NewNode[]): this; + root(): Root; + some( + condition: (node: ChildNode, index: number, nodes: ChildNode[]) => boolean + ): boolean; + walk( + callback: (node: ChildNode, index: number) => false | void + ): false | undefined; + walkAtRules( + nameFilter: RegExp | string, + callback: (atRule: AtRule, index: number) => false | void + ): false | undefined; + walkAtRules( + callback: (atRule: AtRule, index: number) => false | void + ): false | undefined; + walkComments( + callback: (comment: Comment, indexed: number) => false | void + ): false | undefined; + walkComments( + callback: (comment: Comment, indexed: number) => false | void + ): false | undefined; + walkDecls( + propFilter: RegExp | string, + callback: (decl: Declaration, index: number) => false | void + ): false | undefined; + walkDecls( + callback: (decl: Declaration, index: number) => false | void + ): false | undefined; + walkRules( + selectorFilter: RegExp | string, + callback: (rule: Rule, index: number) => false | void + ): false | undefined; + walkRules( + callback: (rule: Rule, index: number) => false | void + ): false | undefined; + get first(): ChildNode | undefined; + get last(): ChildNode | undefined; +} diff --git a/pkg/sass-parser/lib/src/statement/declaration-internal.js b/pkg/sass-parser/lib/src/statement/declaration-internal.js new file mode 100644 index 000000000..8472c1ec9 --- /dev/null +++ b/pkg/sass-parser/lib/src/statement/declaration-internal.js @@ -0,0 +1,5 @@ +// Copyright 2024 Google Inc. Use of this source code is governed by an +// MIT-style license that can be found in the LICENSE file or at +// https://opensource.org/licenses/MIT. + +exports._Declaration = require('postcss').Declaration; diff --git a/pkg/sass-parser/lib/src/statement/index.ts b/pkg/sass-parser/lib/src/statement/index.ts index 10a0aae11..deb6422b1 100644 --- a/pkg/sass-parser/lib/src/statement/index.ts +++ b/pkg/sass-parser/lib/src/statement/index.ts @@ -18,6 +18,10 @@ import {ForRule, ForRuleProps} from './for-rule'; import {Root} from './root'; import {Rule, RuleProps} from './rule'; import {UseRule, UseRuleProps} from './use-rule'; +import { + VariableDeclaration, + VariableDeclarationProps, +} from './variable-declaration'; // TODO: Replace this with the corresponding Sass types once they're // implemented. @@ -28,7 +32,7 @@ export {Declaration} from 'postcss'; * * @category Statement */ -export type AnyStatement = Comment | Root | Rule | GenericAtRule; +export type AnyStatement = Comment | Root | Rule | AtRule | VariableDeclaration; /** * Sass statement types. @@ -49,7 +53,8 @@ export type StatementType = | 'for-rule' | 'error-rule' | 'use-rule' - | 'sass-comment'; + | 'sass-comment' + | 'variable-declaration'; /** * All Sass statements that are also at-rules. @@ -78,7 +83,7 @@ export type Comment = CssComment | SassComment; * * @category Statement */ -export type ChildNode = Rule | AtRule | Comment; +export type ChildNode = Rule | AtRule | Comment | VariableDeclaration; /** * The properties that can be used to construct {@link ChildNode}s. @@ -97,7 +102,8 @@ export type ChildProps = | GenericAtRuleProps | RuleProps | SassCommentChildProps - | UseRuleProps; + | UseRuleProps + | VariableDeclarationProps; /** * The Sass eqivalent of PostCSS's `ContainerProps`. @@ -185,6 +191,7 @@ const visitor = sassInternal.createStatementVisitor({ return rule; }, visitUseRule: inner => new UseRule(undefined, inner), + visitVariableDeclaration: inner => new VariableDeclaration(undefined, inner), }); /** Appends parsed versions of `internal`'s children to `container`. */ @@ -301,6 +308,8 @@ export function normalize( result.push(new SassComment(node)); } else if ('useUrl' in node) { result.push(new UseRule(node)); + } else if ('variableName' in node) { + result.push(new VariableDeclaration(node)); } else { result.push(...postcssNormalizeAndConvertToSass(self, node, sample)); } diff --git a/pkg/sass-parser/lib/src/statement/variable-declaration.test.ts b/pkg/sass-parser/lib/src/statement/variable-declaration.test.ts new file mode 100644 index 000000000..cfb943296 --- /dev/null +++ b/pkg/sass-parser/lib/src/statement/variable-declaration.test.ts @@ -0,0 +1,619 @@ +// Copyright 2024 Google Inc. Use of this source code is governed by an +// MIT-style license that can be found in the LICENSE file or at +// https://opensource.org/licenses/MIT. + +import {StringExpression, VariableDeclaration, sass, scss} from '../..'; +import * as utils from '../../../test/utils'; + +describe('a variable declaration', () => { + let node: VariableDeclaration; + beforeEach( + () => + void (node = new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + })) + ); + + describe('with no namespace and no flags', () => { + function describeNode( + description: string, + create: () => VariableDeclaration + ): void { + describe(description, () => { + beforeEach(() => (node = create())); + + it('has a type', () => expect(node.type.toString()).toBe('decl')); + + it('has a sassType', () => + expect(node.sassType.toString()).toBe('variable-declaration')); + + it('has no namespace', () => expect(node.namespace).toBeUndefined()); + + it('has a name', () => expect(node.variableName).toBe('foo')); + + it('has an expression', () => + expect(node).toHaveStringExpression('expression', 'bar')); + + it('has a value', () => expect(node.value).toBe('bar')); + + it('is not guarded', () => expect(node.guarded).toBe(false)); + + it('is not global', () => expect(node.global).toBe(false)); + }); + } + + describeNode( + 'parsed as SCSS', + () => scss.parse('$foo: bar').nodes[0] as VariableDeclaration + ); + + describeNode( + 'parsed as Sass', + () => sass.parse('$foo: bar').nodes[0] as VariableDeclaration + ); + + describe('constructed manually', () => { + describeNode( + 'with an Expression', + () => + new VariableDeclaration({ + variableName: 'foo', + expression: new StringExpression({text: 'bar'}), + }) + ); + + describeNode( + 'with child props', + () => + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + }) + ); + + describeNode( + 'with a value', + () => + new VariableDeclaration({ + variableName: 'foo', + value: 'bar', + }) + ); + }); + + describeNode('constructed from ChildProps', () => + utils.fromChildProps({variableName: 'foo', expression: {text: 'bar'}}) + ); + }); + + describe('with a namespace', () => { + function describeNode( + description: string, + create: () => VariableDeclaration + ): void { + describe(description, () => { + beforeEach(() => (node = create())); + + it('has a type', () => expect(node.type.toString()).toBe('decl')); + + it('has a sassType', () => + expect(node.sassType.toString()).toBe('variable-declaration')); + + it('has a namespace', () => expect(node.namespace).toBe('baz')); + + it('has a name', () => expect(node.variableName).toBe('foo')); + + it('has an expression', () => + expect(node).toHaveStringExpression('expression', 'bar')); + + it('has a value', () => expect(node.value).toBe('"bar"')); + + it('is not guarded', () => expect(node.guarded).toBe(false)); + + it('is not global', () => expect(node.global).toBe(false)); + }); + } + + describeNode( + 'parsed as SCSS', + () => scss.parse('baz.$foo: "bar"').nodes[0] as VariableDeclaration + ); + + describeNode( + 'parsed as Sass', + () => sass.parse('baz.$foo: "bar"').nodes[0] as VariableDeclaration + ); + + describeNode( + 'constructed manually', + () => + new VariableDeclaration({ + namespace: 'baz', + variableName: 'foo', + expression: new StringExpression({text: 'bar', quotes: true}), + }) + ); + + describeNode('constructed from ChildProps', () => + utils.fromChildProps({ + namespace: 'baz', + variableName: 'foo', + expression: {text: 'bar', quotes: true}, + }) + ); + }); + + describe('guarded', () => { + function describeNode( + description: string, + create: () => VariableDeclaration + ): void { + describe(description, () => { + beforeEach(() => (node = create())); + + it('has a type', () => expect(node.type.toString()).toBe('decl')); + + it('has a sassType', () => + expect(node.sassType.toString()).toBe('variable-declaration')); + + it('has no namespace', () => expect(node.namespace).toBeUndefined()); + + it('has a name', () => expect(node.variableName).toBe('foo')); + + it('has an expression', () => + expect(node).toHaveStringExpression('expression', 'bar')); + + it('has a value', () => expect(node.value).toBe('"bar"')); + + it('is guarded', () => expect(node.guarded).toBe(true)); + + it('is not global', () => expect(node.global).toBe(false)); + }); + } + + describeNode( + 'parsed as SCSS', + () => scss.parse('$foo: "bar" !default').nodes[0] as VariableDeclaration + ); + + describeNode( + 'parsed as Sass', + () => sass.parse('$foo: "bar" !default').nodes[0] as VariableDeclaration + ); + + describeNode( + 'constructed manually', + () => + new VariableDeclaration({ + variableName: 'foo', + expression: new StringExpression({text: 'bar', quotes: true}), + guarded: true, + }) + ); + + describeNode('constructed from ChildProps', () => + utils.fromChildProps({ + variableName: 'foo', + expression: {text: 'bar', quotes: true}, + guarded: true, + }) + ); + }); + + describe('global', () => { + function describeNode( + description: string, + create: () => VariableDeclaration + ): void { + describe(description, () => { + beforeEach(() => (node = create())); + + it('has a type', () => expect(node.type.toString()).toBe('decl')); + + it('has a sassType', () => + expect(node.sassType.toString()).toBe('variable-declaration')); + + it('has no namespace', () => expect(node.namespace).toBeUndefined()); + + it('has a name', () => expect(node.variableName).toBe('foo')); + + it('has an expression', () => + expect(node).toHaveStringExpression('expression', 'bar')); + + it('has a value', () => expect(node.value).toBe('"bar"')); + + it('is not guarded', () => expect(node.guarded).toBe(false)); + + it('is global', () => expect(node.global).toBe(true)); + }); + } + + describeNode( + 'parsed as SCSS', + () => scss.parse('$foo: "bar" !global').nodes[0] as VariableDeclaration + ); + + describeNode( + 'parsed as Sass', + () => sass.parse('$foo: "bar" !global').nodes[0] as VariableDeclaration + ); + + describeNode( + 'constructed manually', + () => + new VariableDeclaration({ + variableName: 'foo', + expression: new StringExpression({text: 'bar', quotes: true}), + global: true, + }) + ); + + describeNode('constructed from ChildProps', () => + utils.fromChildProps({ + variableName: 'foo', + expression: {text: 'bar', quotes: true}, + global: true, + }) + ); + }); + + it('throws an error when assigned a new prop', () => + expect(() => (node.prop = 'bar')).toThrow()); + + it('assigned a new namespace', () => { + node.namespace = 'baz'; + expect(node.namespace).toBe('baz'); + expect(node.prop).toBe('baz.$foo'); + }); + + it('assigned a new variableName', () => { + node.variableName = 'baz'; + expect(node.variableName).toBe('baz'); + expect(node.prop).toBe('$baz'); + }); + + it('assigned a new expression', () => { + const old = node.expression; + node.expression = {text: 'baz'}; + expect(old.parent).toBeUndefined(); + expect(node).toHaveStringExpression('expression', 'baz'); + }); + + it('assigned a new expression', () => { + const old = node.expression; + node.expression = {text: 'baz'}; + expect(old.parent).toBeUndefined(); + expect(node).toHaveStringExpression('expression', 'baz'); + }); + + it('assigned a value', () => { + node.value = 'Helvetica, sans-serif'; + expect(node).toHaveStringExpression('expression', 'Helvetica, sans-serif'); + }); + + it('is a variable', () => expect(node.variable).toBe(true)); + + describe('stringifies', () => { + describe('to SCSS', () => { + describe('with default raws', () => { + it('with no flags', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + }).toString() + ).toBe('$foo: bar')); + + describe('with a namespace', () => { + it("that's an identifier", () => + expect( + new VariableDeclaration({ + namespace: 'baz', + variableName: 'foo', + expression: {text: 'bar'}, + }).toString() + ).toBe('baz.$foo: bar')); + + it("that's not an identifier", () => + expect( + new VariableDeclaration({ + namespace: 'b z', + variableName: 'foo', + expression: {text: 'bar'}, + }).toString() + ).toBe('b\\20z.$foo: bar')); + }); + + it("with a name that's not an identifier", () => + expect( + new VariableDeclaration({ + variableName: 'f o', + expression: {text: 'bar'}, + }).toString() + ).toBe('$f\\20o: bar')); + + it('global', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + global: true, + }).toString() + ).toBe('$foo: bar !global')); + + it('guarded', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + guarded: true, + }).toString() + ).toBe('$foo: bar !default')); + + it('with both flags', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + global: true, + guarded: true, + }).toString() + ).toBe('$foo: bar !default !global')); + }); + + describe('with a namespace raw', () => { + it('that matches', () => + expect( + new VariableDeclaration({ + namespace: 'baz', + variableName: 'foo', + expression: {text: 'bar'}, + raws: {namespace: {raw: 'b\\41z', value: 'baz'}}, + }).toString() + ).toBe('b\\41z.$foo: bar')); + + it("that doesn't match", () => + expect( + new VariableDeclaration({ + namespace: 'baz', + variableName: 'foo', + expression: {text: 'bar'}, + raws: {namespace: {raw: 'z\\41p', value: 'zap'}}, + }).toString() + ).toBe('baz.$foo: bar')); + }); + + describe('with a variableName raw', () => { + it('that matches', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + raws: {variableName: {raw: 'f\\f3o', value: 'foo'}}, + }).toString() + ).toBe('$f\\f3o: bar')); + + it("that doesn't match", () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + raws: {namespace: {raw: 'z\\41p', value: 'zap'}}, + }).toString() + ).toBe('$foo: bar')); + }); + + it('with between', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + raws: {between: '/**/:'}, + }).toString() + ).toBe('$foo/**/:bar')); + + describe('with a flags raw', () => { + it('that matches both', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + guarded: true, + raws: { + flags: { + raw: '/**/!default', + value: {guarded: true, global: false}, + }, + }, + }).toString() + ).toBe('$foo: bar/**/!default')); + + it('that matches only one', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + guarded: true, + raws: { + flags: { + raw: '/**/!default !global', + value: {guarded: true, global: true}, + }, + }, + }).toString() + ).toBe('$foo: bar !default')); + + it('that matches neither', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + guarded: true, + raws: { + flags: { + raw: '/**/!global', + value: {guarded: false, global: true}, + }, + }, + }).toString() + ).toBe('$foo: bar !default')); + }); + + describe('with an afterValue raw', () => { + it('without flags', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + raws: {afterValue: '/**/'}, + }).toString() + ).toBe('$foo: bar/**/')); + + it('with flags', () => + expect( + new VariableDeclaration({ + variableName: 'foo', + expression: {text: 'bar'}, + global: true, + raws: {afterValue: '/**/'}, + }).toString() + ).toBe('$foo: bar !global/**/')); + }); + }); + }); + + describe('clone', () => { + let original: VariableDeclaration; + beforeEach(() => { + original = scss.parse('baz.$foo: bar !default') + .nodes[0] as VariableDeclaration; + // TODO: remove this once raws are properly parsed + original.raws.between = ' :'; + }); + + describe('with no overrides', () => { + let clone: VariableDeclaration; + beforeEach(() => void (clone = original.clone())); + + describe('has the same properties:', () => { + it('prop', () => expect(clone.prop).toBe('baz.$foo')); + + it('namespace', () => expect(clone.namespace).toBe('baz')); + + it('variableName', () => expect(clone.variableName).toBe('foo')); + + it('expression', () => + expect(clone).toHaveStringExpression('expression', 'bar')); + + it('global', () => expect(clone.global).toBe(false)); + + it('guarded', () => expect(clone.guarded).toBe(true)); + }); + + describe('creates a new', () => { + it('self', () => expect(clone).not.toBe(original)); + + for (const attr of ['expression', 'raws'] as const) { + it(attr, () => expect(clone[attr]).not.toBe(original[attr])); + } + }); + }); + + describe('overrides', () => { + describe('raws', () => { + it('defined', () => + expect(original.clone({raws: {afterValue: ' '}}).raws).toEqual({ + afterValue: ' ', + })); + + it('undefined', () => + expect(original.clone({raws: undefined}).raws).toEqual({ + between: ' :', + })); + }); + + describe('namespace', () => { + describe('defined', () => { + let clone: VariableDeclaration; + beforeEach(() => { + clone = original.clone({namespace: 'zap'}); + }); + + it('changes namespace', () => expect(clone.namespace).toBe('zap')); + + it('changes prop', () => expect(clone.prop).toBe('zap.$foo')); + }); + + describe('undefined', () => { + let clone: VariableDeclaration; + beforeEach(() => { + clone = original.clone({namespace: undefined}); + }); + + it('removes namespace', () => + expect(clone.namespace).toBeUndefined()); + + it('changes prop', () => expect(clone.prop).toBe('$foo')); + }); + }); + + describe('variableName', () => { + describe('defined', () => { + let clone: VariableDeclaration; + beforeEach(() => { + clone = original.clone({variableName: 'zap'}); + }); + + it('changes variableName', () => + expect(clone.variableName).toBe('zap')); + + it('changes prop', () => expect(clone.prop).toBe('baz.$zap')); + }); + + describe('undefined', () => { + let clone: VariableDeclaration; + beforeEach(() => { + clone = original.clone({variableName: undefined}); + }); + + it('preserves variableName', () => + expect(clone.variableName).toBe('foo')); + + it('preserves prop', () => expect(clone.prop).toBe('baz.$foo')); + }); + }); + + describe('expression', () => { + it('defined changes expression', () => + expect( + original.clone({expression: {text: 'zap'}}) + ).toHaveStringExpression('expression', 'zap')); + + it('undefined preserves expression', () => + expect( + original.clone({expression: undefined}) + ).toHaveStringExpression('expression', 'bar')); + }); + + describe('guarded', () => { + it('defined changes guarded', () => + expect(original.clone({guarded: false}).guarded).toBe(false)); + + it('undefined preserves guarded', () => + expect(original.clone({guarded: undefined}).guarded).toBe(true)); + }); + + describe('global', () => { + it('defined changes global', () => + expect(original.clone({global: true}).global).toBe(true)); + + it('undefined preserves global', () => + expect(original.clone({global: undefined}).global).toBe(false)); + }); + }); + }); + + it('toJSON', () => + expect(scss.parse('baz.$foo: "bar"').nodes[0]).toMatchSnapshot()); +}); diff --git a/pkg/sass-parser/lib/src/statement/variable-declaration.ts b/pkg/sass-parser/lib/src/statement/variable-declaration.ts new file mode 100644 index 000000000..780e2a4cf --- /dev/null +++ b/pkg/sass-parser/lib/src/statement/variable-declaration.ts @@ -0,0 +1,228 @@ +// Copyright 2024 Google Inc. Use of this source code is governed by an +// MIT-style license that can be found in the LICENSE file or at +// https://opensource.org/licenses/MIT. + +import * as postcss from 'postcss'; +import type {DeclarationRaws} from 'postcss/lib/declaration'; + +import {Expression, ExpressionProps} from '../expression'; +import {convertExpression} from '../expression/convert'; +import {fromProps} from '../expression/from-props'; +import {LazySource} from '../lazy-source'; +import {RawWithValue} from '../raw-with-value'; +import * as sassInternal from '../sass-internal'; +import * as utils from '../utils'; +import {Statement, StatementWithChildren} from '.'; +import {_Declaration} from './declaration-internal'; + +/** + * The set of raws supported by {@link VariableDeclaration}. + * + * @category Statement + */ +export interface VariableDeclarationRaws + extends Omit { + /** + * The variable's namespace. + * + * This may be different than {@link VariableDeclarationRaws.namespace} if the + * name contains escape codes or underscores. + */ + namespace?: RawWithValue; + + /** + * The variable's name, not including the `$`. + * + * This may be different than {@link VariableDeclarationRaws.variableName} if + * the name contains escape codes or underscores. + */ + variableName?: RawWithValue; + + /** The whitespace and colon between the variable name and value. */ + between?: string; + + /** The `!default` and/or `!global` flags, including preceding whitespace. */ + flags?: RawWithValue<{guarded: boolean; global: boolean}>; + + /** + * The space symbols between the end of the variable declaration and the + * semicolon afterwards. Always empty for a variable that isn't followed by a + * semicolon. + */ + afterValue?: string; +} + +/** + * The initializer properties for {@link VariableDeclaration}. + * + * @category Statement + */ +export type VariableDeclarationProps = { + raws?: VariableDeclarationRaws; + namespace?: string; + variableName: string; + guarded?: boolean; + global?: boolean; +} & ({expression: Expression | ExpressionProps} | {value: string}); + +/** + * A Sass variable declaration. Extends [`postcss.Declaration`]. + * + * [`postcss.AtRule`]: https://postcss.org/api/#declaration + * + * @category Statement + */ +export class VariableDeclaration + extends _Declaration> + implements Statement +{ + readonly sassType = 'variable-declaration' as const; + declare parent: StatementWithChildren | undefined; + declare raws: VariableDeclarationRaws; + + /** + * The variable name, not including `$`. + * + * This is the parsed value, with escapes resolved to the characters they + * represent. + */ + declare namespace: string | undefined; + + /** + * The variable name, not including `$`. + * + * This is the parsed and normalized value, with underscores converted to + * hyphens and escapes resolved to the characters they represent. + */ + declare variableName: string; + + /** The variable's value. */ + get expression(): Expression { + return this._expression; + } + set expression(value: Expression | ExpressionProps) { + if (this._expression) this._expression.parent = undefined; + if (!('sassType' in value)) value = fromProps(value); + if (value) value.parent = this; + this._expression = value; + } + private _expression!: Expression; + + /** Whether the variable has a `!default` flag. */ + declare guarded: boolean; + + /** Whether the variable has a `!global` flag. */ + declare global: boolean; + + get prop(): string { + return ( + (this.namespace + ? (this.raws.namespace?.value === this.namespace + ? this.raws.namespace.raw + : sassInternal.toCssIdentifier(this.namespace)) + '.' + : '') + + '$' + + (this.raws.variableName?.value === this.variableName + ? this.raws.variableName.raw + : sassInternal.toCssIdentifier(this.variableName)) + ); + } + set prop(value: string) { + throw new Error("VariableDeclaration.prop can't be overwritten."); + } + + get value(): string { + return this.expression.toString(); + } + set value(value: string) { + this.expression = {text: value}; + } + + get important(): boolean { + // TODO: Return whether `this.expression` is a nested series of unbracketed + // list expressions that ends in the unquoted string `!important` (or an + // unquoted string ending in " !important", which can occur if `value` is + // set // manually). + throw new Error('Not yet implemented'); + } + set important(value: boolean) { + // TODO: If value !== this.important, either set this to a space-separated + // list whose second value is `!important` or remove the existing + // `!important` from wherever it's defined. Or if that's too complex, just + // bake this to a string expression and edit that. + throw new Error('Not yet implemented'); + } + + get variable(): boolean { + return true; + } + + constructor(defaults: VariableDeclarationProps); + /** @hidden */ + constructor(_: undefined, inner: sassInternal.VariableDeclaration); + constructor( + defaults?: VariableDeclarationProps, + inner?: sassInternal.VariableDeclaration + ) { + super(defaults as unknown as postcss.DeclarationProps); + this.raws ??= {}; + + if (inner) { + this.source = new LazySource(inner); + this.namespace = inner.namespace ? inner.namespace : undefined; + this.variableName = inner.name; + this.expression = convertExpression(inner.expression); + this.guarded = inner.isGuarded; + this.global = inner.isGlobal; + } else { + this.guarded ??= false; + this.global ??= false; + } + } + + clone(overrides?: Partial): this { + return utils.cloneNode( + this, + overrides, + [ + 'raws', + {name: 'namespace', explicitUndefined: true}, + 'variableName', + 'expression', + 'guarded', + 'global', + ], + ['value'] + ); + } + + toJSON(): object; + /** @hidden */ + toJSON(_: string, inputs: Map): object; + toJSON(_?: string, inputs?: Map): object { + return utils.toJSON( + this, + ['namespace', 'variableName', 'expression', 'guarded', 'global'], + inputs + ); + } + + /** @hidden */ + toString(): string { + return ( + this.prop + + (this.raws.between ?? ': ') + + this.expression + + (this.raws.flags?.value?.guarded === this.guarded && + this.raws.flags?.value?.global === this.global + ? this.raws.flags.raw + : (this.guarded ? ' !default' : '') + (this.global ? ' !global' : '')) + + (this.raws.afterValue ?? '') + ); + } + + /** @hidden */ + get nonStatementChildren(): ReadonlyArray { + return [this.expression]; + } +} diff --git a/pkg/sass-parser/package.json b/pkg/sass-parser/package.json index cb179370a..209440ef5 100644 --- a/pkg/sass-parser/package.json +++ b/pkg/sass-parser/package.json @@ -1,6 +1,6 @@ { "name": "sass-parser", - "version": "0.4.1", + "version": "0.4.2-dev", "description": "A PostCSS-compatible wrapper of the official Sass parser", "repository": "sass/sass", "author": "Google Inc.", diff --git a/pkg/sass_api/CHANGELOG.md b/pkg/sass_api/CHANGELOG.md index 310abb6eb..268767744 100644 --- a/pkg/sass_api/CHANGELOG.md +++ b/pkg/sass_api/CHANGELOG.md @@ -1,3 +1,7 @@ +## 14.1.1-dev + +* No user-visible changes. + ## 14.1.0 * Add `Expression.isCalculationSafe`, which returns true when this expression diff --git a/pkg/sass_api/pubspec.yaml b/pkg/sass_api/pubspec.yaml index 93418b057..5d118d0ae 100644 --- a/pkg/sass_api/pubspec.yaml +++ b/pkg/sass_api/pubspec.yaml @@ -2,7 +2,7 @@ name: sass_api # Note: Every time we add a new Sass AST node, we need to bump the *major* # version because it's a breaking change for anyone who's implementing the # visitor interface(s). -version: 14.1.0 +version: 14.1.1-dev description: Additional APIs for Dart Sass. homepage: https://github.com/sass/dart-sass @@ -10,7 +10,7 @@ environment: sdk: ">=3.0.0 <4.0.0" dependencies: - sass: 1.80.4 + sass: 1.80.5 dev_dependencies: dartdoc: ^8.0.14 diff --git a/pubspec.yaml b/pubspec.yaml index 53fcaa858..f7817a98b 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -1,5 +1,5 @@ name: sass -version: 1.80.4 +version: 1.80.5-dev description: A Sass implementation in Dart. homepage: https://github.com/sass/dart-sass