Skip to content

Commit f5a3dea

Browse files
authoredMar 23, 2023
Add support for constants in calculations (#1922)
Closes #1900 See sass/sass#3258
1 parent 09a5f09 commit f5a3dea

11 files changed

+131
-24
lines changed
 

‎CHANGELOG.md

+12
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,15 @@
1+
## 1.60.0
2+
3+
* Add support for the `pi`, `e`, `infinity`, `-infinity`, and `NaN` constants in
4+
calculations. These will be interpreted as the corresponding numbers.
5+
6+
* Add support for unknown constants in calculations. These will be interpreted
7+
as unquoted strings.
8+
9+
* Serialize numbers with value `infinity`, `-infinity`, and `NaN` to `calc()`
10+
expressions rather than CSS-invalid identifiers. Numbers with complex units
11+
still can't be serialized.
12+
113
## 1.59.3
214

315
* Fix a performance regression introduced in 1.59.0.

‎lib/src/parse/stylesheet.dart

+12-5
Original file line numberDiff line numberDiff line change
@@ -2984,7 +2984,7 @@ abstract class StylesheetParser extends Parser {
29842984
/// Parses a single calculation value.
29852985
Expression _calculationValue() {
29862986
var next = scanner.peekChar();
2987-
if (next == $plus || next == $minus || next == $dot || isDigit(next)) {
2987+
if (next == $plus || next == $dot || isDigit(next)) {
29882988
return _number();
29892989
} else if (next == $dollar) {
29902990
return _variable();
@@ -3001,13 +3001,14 @@ abstract class StylesheetParser extends Parser {
30013001
whitespace();
30023002
scanner.expectChar($rparen);
30033003
return ParenthesizedExpression(value, scanner.spanFrom(start));
3004-
} else if (!lookingAtIdentifier()) {
3005-
scanner.error("Expected number, variable, function, or calculation.");
3006-
} else {
3004+
} else if (lookingAtIdentifier()) {
30073005
var start = scanner.state;
30083006
var ident = identifier();
30093007
if (scanner.scanChar($dot)) return namespacedExpression(ident, start);
3010-
if (scanner.peekChar() != $lparen) scanner.error('Expected "(" or ".".');
3008+
if (scanner.peekChar() != $lparen) {
3009+
return StringExpression(Interpolation([ident], scanner.spanFrom(start)),
3010+
quotes: false);
3011+
}
30113012

30123013
var lowerCase = ident.toLowerCase();
30133014
var calculation = _tryCalculation(lowerCase, start);
@@ -3019,6 +3020,12 @@ abstract class StylesheetParser extends Parser {
30193020
return FunctionExpression(
30203021
ident, _argumentInvocation(), scanner.spanFrom(start));
30213022
}
3023+
} else if (next == $minus) {
3024+
// This has to go after [lookingAtIdentifier] because a hyphen can start
3025+
// an identifier as well as a number.
3026+
return _number();
3027+
} else {
3028+
scanner.error("Expected number, variable, function, or calculation.");
30223029
}
30233030
}
30243031

‎lib/src/value/calculation.dart

+3-3
Original file line numberDiff line numberDiff line change
@@ -38,9 +38,9 @@ class SassCalculation extends Value {
3838
/// Creates a new calculation with the given [name] and [arguments]
3939
/// that will not be simplified.
4040
@internal
41-
static Value unsimplified(String name, Iterable<Object> arguments) {
42-
return SassCalculation._(name, List.unmodifiable(arguments));
43-
}
41+
static SassCalculation unsimplified(
42+
String name, Iterable<Object> arguments) =>
43+
SassCalculation._(name, List.unmodifiable(arguments));
4444

4545
/// Creates a `calc()` calculation with the given [argument].
4646
///

‎lib/src/visitor/async_evaluate.dart

+22-1
Original file line numberDiff line numberDiff line change
@@ -2379,7 +2379,28 @@ class _EvaluateVisitor
23792379
: result;
23802380
} else if (node is StringExpression) {
23812381
assert(!node.hasQuotes);
2382-
return CalculationInterpolation(await _performInterpolation(node.text));
2382+
var text = node.text.asPlain;
2383+
// If there's actual interpolation, create a CalculationInterpolation.
2384+
// Otherwise, create an UnquotedString. The main difference is that
2385+
// UnquotedStrings don't get extra defensive parentheses.
2386+
if (text == null) {
2387+
return CalculationInterpolation(await _performInterpolation(node.text));
2388+
}
2389+
2390+
switch (text.toLowerCase()) {
2391+
case 'pi':
2392+
return SassNumber(math.pi);
2393+
case 'e':
2394+
return SassNumber(math.e);
2395+
case 'infinity':
2396+
return SassNumber(double.infinity);
2397+
case '-infinity':
2398+
return SassNumber(double.negativeInfinity);
2399+
case 'nan':
2400+
return SassNumber(double.nan);
2401+
default:
2402+
return SassString(text, quotes: false);
2403+
}
23832404
} else if (node is BinaryOperationExpression) {
23842405
return await _addExceptionSpanAsync(
23852406
node,

‎lib/src/visitor/evaluate.dart

+23-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
// DO NOT EDIT. This file was generated from async_evaluate.dart.
66
// See tool/grind/synchronize.dart for details.
77
//
8-
// Checksum: 8a55729a9dc5dafe90954738907880052d930898
8+
// Checksum: 06d1dd221c149650242b3e09b3f507125606bf0f
99
//
1010
// ignore_for_file: unused_import
1111

@@ -2367,7 +2367,28 @@ class _EvaluateVisitor
23672367
: result;
23682368
} else if (node is StringExpression) {
23692369
assert(!node.hasQuotes);
2370-
return CalculationInterpolation(_performInterpolation(node.text));
2370+
var text = node.text.asPlain;
2371+
// If there's actual interpolation, create a CalculationInterpolation.
2372+
// Otherwise, create an UnquotedString. The main difference is that
2373+
// UnquotedStrings don't get extra defensive parentheses.
2374+
if (text == null) {
2375+
return CalculationInterpolation(_performInterpolation(node.text));
2376+
}
2377+
2378+
switch (text.toLowerCase()) {
2379+
case 'pi':
2380+
return SassNumber(math.pi);
2381+
case 'e':
2382+
return SassNumber(math.e);
2383+
case 'infinity':
2384+
return SassNumber(double.infinity);
2385+
case '-infinity':
2386+
return SassNumber(double.negativeInfinity);
2387+
case 'nan':
2388+
return SassNumber(double.nan);
2389+
default:
2390+
return SassString(text, quotes: false);
2391+
}
23712392
} else if (node is BinaryOperationExpression) {
23722393
return _addExceptionSpan(
23732394
node,

‎lib/src/visitor/serialize.dart

+40-2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import 'dart:math' as math;
66
import 'dart:typed_data';
77

88
import 'package:charcode/charcode.dart';
9+
import 'package:collection/collection.dart';
910
import 'package:source_maps/source_maps.dart';
1011
import 'package:string_scanner/string_scanner.dart';
1112

@@ -492,7 +493,35 @@ class _SerializeVisitor
492493
}
493494

494495
void _writeCalculationValue(Object value) {
495-
if (value is Value) {
496+
if (value is SassNumber && !value.value.isFinite) {
497+
if (value.numeratorUnits.length > 1 ||
498+
value.denominatorUnits.isNotEmpty) {
499+
if (!_inspect) {
500+
throw SassScriptException("$value isn't a valid CSS value.");
501+
}
502+
503+
_writeNumber(value.value);
504+
_buffer.write(value.unitString);
505+
return;
506+
}
507+
508+
if (value.value == double.infinity) {
509+
_buffer.write('infinity');
510+
} else if (value.value == double.negativeInfinity) {
511+
_buffer.write('-infinity');
512+
} else if (value.value.isNaN) {
513+
_buffer.write('NaN');
514+
}
515+
516+
var unit = value.numeratorUnits.firstOrNull;
517+
if (unit != null) {
518+
_writeOptionalSpace();
519+
_buffer.writeCharCode($asterisk);
520+
_writeOptionalSpace();
521+
_buffer.writeCharCode($1);
522+
_buffer.write(unit);
523+
}
524+
} else if (value is Value) {
496525
value.accept(this);
497526
} else if (value is CalculationInterpolation) {
498527
_buffer.write(value.value);
@@ -513,7 +542,11 @@ class _SerializeVisitor
513542
var right = value.right;
514543
var parenthesizeRight = right is CalculationInterpolation ||
515544
(right is CalculationOperation &&
516-
_parenthesizeCalculationRhs(value.operator, right.operator));
545+
_parenthesizeCalculationRhs(value.operator, right.operator)) ||
546+
(value.operator == CalculationOperator.dividedBy &&
547+
right is SassNumber &&
548+
!right.value.isFinite &&
549+
right.hasUnits);
517550
if (parenthesizeRight) _buffer.writeCharCode($lparen);
518551
_writeCalculationValue(right);
519552
if (parenthesizeRight) _buffer.writeCharCode($rparen);
@@ -760,6 +793,11 @@ class _SerializeVisitor
760793
return;
761794
}
762795

796+
if (!value.value.isFinite) {
797+
visitCalculation(SassCalculation.unsimplified('calc', [value]));
798+
return;
799+
}
800+
763801
_writeNumber(value.value);
764802

765803
if (!_inspect) {

‎pkg/sass_api/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
## 6.1.0
2+
3+
* No user-visible changes.
4+
15
## 6.0.3
26

37
* No user-visible changes.

‎pkg/sass_api/pubspec.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,15 +2,15 @@ name: sass_api
22
# Note: Every time we add a new Sass AST node, we need to bump the *major*
33
# version because it's a breaking change for anyone who's implementing the
44
# visitor interface(s).
5-
version: 6.0.3
5+
version: 6.1.0
66
description: Additional APIs for Dart Sass.
77
homepage: https://github.com/sass/dart-sass
88

99
environment:
1010
sdk: ">=2.17.0 <3.0.0"
1111

1212
dependencies:
13-
sass: 1.59.3
13+
sass: 1.60.0
1414

1515
dev_dependencies:
1616
dartdoc: ^5.0.0

‎pubspec.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
name: sass
2-
version: 1.59.3
2+
version: 1.60.0
33
description: A Sass implementation in Dart.
44
homepage: https://github.com/sass/dart-sass
55

‎test/cli/shared/repl.dart

+11-7
Original file line numberDiff line numberDiff line change
@@ -195,12 +195,14 @@ void sharedTests(Future<TestProcess> runSass(Iterable<String> arguments)) {
195195

196196
test("a runtime error", () async {
197197
var sass = await runSass(["--interactive"]);
198-
sass.stdin.writeln("max(2, 1 + blue)");
198+
sass.stdin.writeln("@use 'sass:math'");
199+
sass.stdin.writeln("math.max(2, 1 + blue)");
199200
await expectLater(
200201
sass.stdout,
201202
emitsInOrder([
202-
">> max(2, 1 + blue)",
203-
" ^^^^^^^^",
203+
">> @use 'sass:math'",
204+
">> math.max(2, 1 + blue)",
205+
" ^^^^^^^^",
204206
'Error: Undefined operation "1 + blue".'
205207
]));
206208
await sass.kill();
@@ -300,13 +302,15 @@ void sharedTests(Future<TestProcess> runSass(Iterable<String> arguments)) {
300302
group("and colorizes", () {
301303
test("an error in the source text", () async {
302304
var sass = await runSass(["--interactive", "--color"]);
303-
sass.stdin.writeln("max(2, 1 + blue)");
305+
sass.stdin.writeln("@use 'sass:math'");
306+
sass.stdin.writeln("math.max(2, 1 + blue)");
304307
await expectLater(
305308
sass.stdout,
306309
emitsInOrder([
307-
">> max(2, 1 + blue)",
308-
"\u001b[31m\u001b[1F\u001b[10C1 + blue",
309-
" ^^^^^^^^",
310+
">> @use 'sass:math'",
311+
">> math.max(2, 1 + blue)",
312+
"\u001b[31m\u001b[1F\u001b[15C1 + blue",
313+
" ^^^^^^^^",
310314
'\u001b[0mError: Undefined operation "1 + blue".'
311315
]));
312316
await sass.kill();

‎test/output_test.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ void main() {
9292
group("for floating-point numbers", () {
9393
test("Infinity", () {
9494
expect(compileString("a {b: 1e999}"),
95-
equalsIgnoringWhitespace("a { b: Infinity; }"));
95+
equalsIgnoringWhitespace("a { b: calc(infinity); }"));
9696
});
9797

9898
test(">= 1e21", () {

0 commit comments

Comments
 (0)
Please sign in to comment.