Skip to content

Commit 4c3bd0e

Browse files
authored
Deprecate explicitly passing null as an alpha value (#2049)
This allows us to reserve null to indicate a missing alpha channel instead in future versions.
1 parent 8802c69 commit 4c3bd0e

File tree

12 files changed

+95
-28
lines changed

12 files changed

+95
-28
lines changed

CHANGELOG.md

+19
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,27 @@
22

33
### Dart API
44

5+
* Deprecate explicitly passing `null` as the alpha channel for
6+
`SassColor.rgb()`, `SassColor.hsl()`, and `SassColor.hwb()`. Omitting the
7+
`alpha` channel is still allowed. In future releases, `null` will be used to
8+
indicate a [missing component]. This deprecation is named `null-alpha`.
9+
10+
[missing component]: https://developer.mozilla.org/en-US/docs/Web/CSS/color_value#missing_color_components
11+
512
* Include protocol buffer definitions when uploading the `sass` package to pub.
613

14+
### JS API
15+
16+
* Deprecate explicitly passing `null` as the alpha channel for `new
17+
SassColor()`. Omitting the `alpha` channel or passing `undefined` for it is
18+
still allowed. In future releases, `null` will be used to indicate a [missing
19+
component]. This deprecation is named `null-alpha`.
20+
21+
[missing component]: https://developer.mozilla.org/en-US/docs/Web/CSS/color_value#missing_color_components
22+
23+
(Note that this was already prohibited by the TypeScript types, but in
24+
practice prior to this `null` was treated as `1`.)
25+
726
## 1.64.2
827

928
* No user-visible changes.

lib/src/deprecation.dart

+5
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
import 'package:collection/collection.dart';
66
import 'package:pub_semver/pub_semver.dart';
77

8+
import 'io.dart';
89
import 'util/nullable.dart';
910

1011
/// A deprecated feature in the language.
@@ -59,6 +60,10 @@ enum Deprecation {
5960
description:
6061
'Using !default or !global multiple times for one variable.'),
6162

63+
nullAlpha('null-alpha',
64+
deprecatedIn: '1.62.3',
65+
description: 'Passing null as alpha in the ${isJS ? 'JS' : 'Dart'} API.'),
66+
6267
/// Deprecation for `@import` rules.
6368
import.future('import', description: '@import rules.'),
6469

lib/src/functions/color.dart

+6-3
Original file line numberDiff line numberDiff line change
@@ -600,7 +600,8 @@ Value _rgb(String name, List<Value> arguments) {
600600
fuzzyRound(_percentageOrUnitless(green, 255, "green")),
601601
fuzzyRound(_percentageOrUnitless(blue, 255, "blue")),
602602
alpha.andThen((alpha) =>
603-
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")),
603+
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")) ??
604+
1,
604605
ColorFormat.rgbFunction);
605606
}
606607

@@ -644,7 +645,8 @@ Value _hsl(String name, List<Value> arguments) {
644645
saturation.value.clamp(0, 100),
645646
lightness.value.clamp(0, 100),
646647
alpha.andThen((alpha) =>
647-
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")),
648+
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")) ??
649+
1,
648650
ColorFormat.hslFunction);
649651
}
650652

@@ -693,7 +695,8 @@ Value _hwb(List<Value> arguments) {
693695
whiteness.valueInRange(0, 100, "whiteness"),
694696
blackness.valueInRange(0, 100, "blackness"),
695697
alpha.andThen((alpha) =>
696-
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")));
698+
_percentageOrUnitless(alpha.assertNumber("alpha"), 1, "alpha")) ??
699+
1);
697700
}
698701

699702
Object /* SassString | List<Value> */ _parseChannels(

lib/src/io/interface.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ bool get isMacOS => throw '';
2020
bool get hasTerminal => throw '';
2121

2222
/// Whether we're running as JS (browser or Node.js).
23-
bool get isJS => throw '';
23+
const bool isJS = false;
2424

2525
/// Whether we're running as Node.js (not browser or Dart VM).
2626
bool get isNode => throw '';

lib/src/io/js.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -227,7 +227,7 @@ bool get isWindows => process?.platform == 'win32';
227227

228228
bool get isMacOS => process?.platform == 'darwin';
229229

230-
bool get isJS => true;
230+
const bool isJS = true;
231231

232232
/// The fs module object, used to check whether this has been loaded as Node.
233233
///

lib/src/io/vm.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ bool get isMacOS => io.Platform.isMacOS;
2222

2323
bool get hasTerminal => io.stdout.hasTerminal;
2424

25-
bool get isJS => false;
25+
const bool isJS = false;
2626

2727
bool get isNode => false;
2828

lib/src/js/value/color.dart

+12-5
Original file line numberDiff line numberDiff line change
@@ -8,19 +8,20 @@ import '../../util/nullable.dart';
88
import '../../util/number.dart';
99
import '../../value.dart';
1010
import '../reflection.dart';
11+
import '../utils.dart';
1112

1213
/// The JavaScript `SassColor` class.
1314
final JSClass colorClass = () {
1415
var jsClass = createJSClass('sass.SassColor', (Object self, _Channels color) {
1516
if (color.red != null) {
1617
return SassColor.rgb(fuzzyRound(color.red!), fuzzyRound(color.green!),
17-
fuzzyRound(color.blue!), color.alpha);
18+
fuzzyRound(color.blue!), _handleNullAlpha(color.alpha));
1819
} else if (color.saturation != null) {
19-
return SassColor.hsl(
20-
color.hue!, color.saturation!, color.lightness!, color.alpha);
20+
return SassColor.hsl(color.hue!, color.saturation!, color.lightness!,
21+
_handleNullAlpha(color.alpha));
2122
} else {
22-
return SassColor.hwb(
23-
color.hue!, color.whiteness!, color.blackness!, color.alpha);
23+
return SassColor.hwb(color.hue!, color.whiteness!, color.blackness!,
24+
_handleNullAlpha(color.alpha));
2425
}
2526
});
2627

@@ -68,6 +69,12 @@ final JSClass colorClass = () {
6869
return jsClass;
6970
}();
7071

72+
/// Converts an undefined [alpha] to 1.
73+
///
74+
/// This ensures that an explicitly null alpha will produce a deprecation
75+
/// warning when passed to the Dart API.
76+
num? _handleNullAlpha(num? alpha) => isUndefined(alpha) ? 1 : alpha;
77+
7178
@JS()
7279
@anonymous
7380
class _Channels {

lib/src/parse/stylesheet.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -2232,7 +2232,7 @@ abstract class StylesheetParser extends Parser {
22322232
red,
22332233
green,
22342234
blue,
2235-
alpha,
2235+
alpha ?? 1,
22362236
// Don't emit four- or eight-digit hex colors as hex, since that's not
22372237
// yet well-supported in browsers.
22382238
alpha == null ? SpanColorFormat(scanner.spanFrom(start)) : null);

lib/src/value/color.dart

+42-13
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,10 @@ import 'dart:math' as math;
77
import 'package:meta/meta.dart';
88
import 'package:source_span/source_span.dart';
99

10+
import '../deprecation.dart';
11+
import '../evaluation_context.dart';
1012
import '../exception.dart';
13+
import '../io.dart';
1114
import '../util/number.dart';
1215
import '../value.dart';
1316
import '../visitor/interface/value.dart';
@@ -98,52 +101,63 @@ class SassColor extends Value {
98101

99102
/// Creates an RGB color.
100103
///
104+
/// Passing `null` to [alpha] is deprecated, and will change behavior in
105+
/// future versions of Dart Sass to represent a [missing component] instead of
106+
/// being equivalent to `1`. Callers who want to create opaque colors should
107+
/// explicitly pass `1` or not pass [alpha] at all.
108+
///
109+
/// [missing component]: https://developer.mozilla.org/en-US/docs/Web/CSS/color_value#missing_color_components
110+
///
101111
/// Throws a [RangeError] if [red], [green], and [blue] aren't between `0` and
102112
/// `255`, or if [alpha] isn't between `0` and `1`.
103-
SassColor.rgb(int red, int green, int blue, [num? alpha])
104-
: this.rgbInternal(red, green, blue, alpha);
113+
SassColor.rgb(int red, int green, int blue, [num? alpha = 1])
114+
: this.rgbInternal(red, green, blue, _handleNullAlpha(alpha));
105115

106116
/// Like [SassColor.rgb], but also takes a [format] parameter.
107117
///
108118
/// @nodoc
109119
@internal
110120
SassColor.rgbInternal(this._red, this._green, this._blue,
111-
[num? alpha, this.format])
112-
: _alpha = alpha == null
113-
? 1
114-
: fuzzyAssertRange(alpha.toDouble(), 0, 1, "alpha") {
121+
[num alpha = 1, this.format])
122+
: _alpha = fuzzyAssertRange(alpha.toDouble(), 0, 1, "alpha") {
115123
RangeError.checkValueInInterval(red, 0, 255, "red");
116124
RangeError.checkValueInInterval(green, 0, 255, "green");
117125
RangeError.checkValueInInterval(blue, 0, 255, "blue");
118126
}
119127

120128
/// Creates an HSL color.
121129
///
130+
/// Passing `null` to [alpha] is deprecated, and will change behavior in
131+
/// future versions of Dart Sass to represent a [missing component] instead of
132+
/// being equivalent to `1`. Callers who want to create opaque colors should
133+
/// explicitly pass `1` or not pass [alpha] at all.
134+
///
135+
/// [missing component]: https://developer.mozilla.org/en-US/docs/Web/CSS/color_value#missing_color_components
136+
///
122137
/// Throws a [RangeError] if [saturation] or [lightness] aren't between `0`
123138
/// and `100`, or if [alpha] isn't between `0` and `1`.
124-
SassColor.hsl(num hue, num saturation, num lightness, [num? alpha])
125-
: this.hslInternal(hue, saturation, lightness, alpha);
139+
SassColor.hsl(num hue, num saturation, num lightness, [num? alpha = 1])
140+
: this.hslInternal(hue, saturation, lightness, _handleNullAlpha(alpha));
126141

127142
/// Like [SassColor.hsl], but also takes a [format] parameter.
128143
///
129144
/// @nodoc
130145
@internal
131146
SassColor.hslInternal(num hue, num saturation, num lightness,
132-
[num? alpha, this.format])
147+
[num alpha = 1, this.format])
133148
: _hue = hue % 360,
134149
_saturation =
135150
fuzzyAssertRange(saturation.toDouble(), 0, 100, "saturation"),
136151
_lightness =
137152
fuzzyAssertRange(lightness.toDouble(), 0, 100, "lightness"),
138-
_alpha = alpha == null
139-
? 1
140-
: fuzzyAssertRange(alpha.toDouble(), 0, 1, "alpha");
153+
_alpha = fuzzyAssertRange(alpha.toDouble(), 0, 1, "alpha");
141154

142155
/// Creates an HWB color.
143156
///
144157
/// Throws a [RangeError] if [whiteness] or [blackness] aren't between `0` and
145158
/// `100`, or if [alpha] isn't between `0` and `1`.
146-
factory SassColor.hwb(num hue, num whiteness, num blackness, [num? alpha]) {
159+
factory SassColor.hwb(num hue, num whiteness, num blackness,
160+
[num? alpha = 1]) {
147161
// From https://www.w3.org/TR/css-color-4/#hwb-to-rgb
148162
var scaledHue = hue % 360 / 360;
149163
var scaledWhiteness =
@@ -171,6 +185,21 @@ class SassColor extends Value {
171185
toRgb(scaledHue - 1 / 3), alpha);
172186
}
173187

188+
/// Prints a deprecation warning if [alpha] is explicitly `null`.
189+
static num _handleNullAlpha(num? alpha) {
190+
if (alpha != null) return alpha;
191+
192+
warnForDeprecation(
193+
'Passing null for alpha in the ${isJS ? 'JS' : 'Dart'} API is '
194+
'deprecated.\n'
195+
'To preserve current behavior, pass 1${isJS ? ' or undefined' : ''} '
196+
'instead.'
197+
'\n'
198+
'More info: https://sass-lang.com/d/null-alpha',
199+
Deprecation.nullAlpha);
200+
return 1;
201+
}
202+
174203
SassColor._(this._red, this._green, this._blue, this._hue, this._saturation,
175204
this._lightness, this._alpha)
176205
: format = null;

pkg/sass_api/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,10 @@
77

88
* All uses of classes from the `tuple` package have been replaced by record
99
types.
10+
11+
## 7.2.2
12+
13+
* No user-visible changes.
1014

1115
## 7.2.1
1216

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: 8.0.0-dev
5+
version: 8.0.0
66
description: Additional APIs for Dart Sass.
77
homepage: https://github.com/sass/dart-sass
88

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

1212
dependencies:
13-
sass: 1.64.2
13+
sass: 1.64.3
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.64.3-dev
2+
version: 1.64.3
33
description: A Sass implementation in Dart.
44
homepage: https://github.com/sass/dart-sass
55

0 commit comments

Comments
 (0)