Skip to content

Commit 38b0c7b

Browse files
committed
[Color 4] Add initial implementation of Color Level 4
This is still missing some planned deprecations and doubtlessly has bugs in the new features, but it's enough to provide a baseline to begin iterating on in smaller chunks. See #1805 See sass/sass#2831
1 parent 790eb8a commit 38b0c7b

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+4674
-658
lines changed

CHANGELOG.md

+137
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,140 @@
1+
## 1.57.0
2+
3+
* Add support for CSS Color Level 4 [color spaces]. Each color value now tracks
4+
its color space along with the values of each channel in that color space.
5+
There are two general principles to keep in mind when dealing with new color
6+
spaces:
7+
8+
1. With the exception of legacy color spaces (`rgb`, `hsl`, and `hwb`), colors
9+
will always be emitted in the color space they were defined in unless
10+
they're explicitly converted.
11+
12+
2. The `color.to-space()` function is the only way to convert a color to
13+
another color space. Some built-in functions may do operations in a
14+
different color space, but they'll always convert back to the original space
15+
afterwards.
16+
17+
* `rgb` colors can now have non-integer channels and channels outside the normal
18+
gamut of 0-255. These colors are always emitted using the `rgb()` syntax so
19+
that modern browsers that are being displayed on wide-gamut devices can
20+
display the most accurate color possible.
21+
22+
* Add support for all the new color syntax defined in Color Level 4, including:
23+
24+
* `oklab()`, `oklch()`, `lab()`, and `lch()` functions;
25+
* a top-level `hwb()` function that matches the space-separated CSS syntax;
26+
* and a `color()` function that supports the `srgb`, `srgb-linear`,
27+
`display-p3`, `a98-rgb`, `prophoto-rgb`, `rec2020`, `xyz`, `xyz-d50`, and
28+
`xyz-d65` color spaces.
29+
30+
* Add new functions for working with color spaces:
31+
32+
* `color.to-space($color, $space)` converts `$color` to the given `$space`. In
33+
most cases this conversion is lossless—the color may end up out-of-gamut for
34+
the destination color space, but browsers will generally display it as best
35+
they can regardless. However, the `hsl` and `hwb` spaces can't represent
36+
out-of-gamut colors and so will be clamped.
37+
38+
* `color.channel($color, $channel, $space: null)` returns the value of the
39+
given `$channel` in `$color`, after converting it to `$space` if necessary.
40+
It should be used instead of the old channel-specific functions such as
41+
`color.red()` and `color.hue()`.
42+
43+
* `color.same($color1, $color2)` returns whether two colors represent the same
44+
color even across color spaces. It differs from `$color1 == $color2` because
45+
`==` never consider colors in different (non-legacy) spaces as equal.
46+
47+
* `color.is-in-gamut($color, $space: null)` returns whether `$color` is
48+
in-gamut for its color space (or `$space` if it's passed).
49+
50+
* `color.to-gamut($color, $space: null)` returns `$color` constrained to its
51+
space's gamut (or to `$space`'s gamut, if passed). This is generally not
52+
recommended since even older browsers will display out-of-gamut colors as
53+
best they can, but it may be necessary in some cases.
54+
55+
* `color.space($color)`: Returns the name of `$color`'s color space.
56+
57+
* `color.is-legacy($color)`: Returns whether `$color` is in a legacy color
58+
space (`rgb`, `hsl`, or `hwb`).
59+
60+
* `color.is-powerless($color, $channel, $space: null)`: Returns whether the
61+
given `$channel` of `$color` is powerless in `$space` (or its own color
62+
space). A channel is "powerless" if its value doesn't affect the way the
63+
color is displayed, such as hue for a color with 0 chroma.
64+
65+
* `color.is-missing($color, $channel)`: Returns whether `$channel`'s value is
66+
missing in `$color`. Missing channels can be explicitly specified using the
67+
special value `none` and can appear automatically when `color.to-space()`
68+
returns a color with a powerless channel. Missing channels are usually
69+
treated as 0, except when interpolating between two colors and in
70+
`color.mix()` where they're treated as the same value as the other color.
71+
72+
* Update existing functions to support color spaces:
73+
74+
* `hsl()` and `color.hwb()` no longer forbid out-of-bounds values. Instead,
75+
they follow the CSS spec by clamping them to within the allowed range.
76+
77+
* `color.change()`, `color.adjust()`, and `color.scale()` now support all
78+
channels of all color spaces. However, if you want to modify a channel
79+
that's not in `$color`'s own color space, you have to explicitly specify the
80+
space with the `$space` parameter. (For backwards-compatibility, this
81+
doesn't apply to legacy channels of legacy colors—for example, you can still
82+
adjust an `rgb` color's saturation without passing `$space: hsl`).
83+
84+
* `color.mix()` and `color.invert()` now support the standard CSS algorithm
85+
for interpolating between two colors (the same one that's used for gradients
86+
and animations). To use this, pass the color space to use for interpolation
87+
to the `$method` parameter. For polar color spaces like `hsl` and `oklch`,
88+
this parameter also allows you to specify how hue interpolation is handled.
89+
90+
* `color.complement()` now supports a `$space` parameter that indicates which
91+
color space should be used to take the complement.
92+
93+
* `color.grayscale()` now operates in the `oklch` space for non-legacy colors.
94+
95+
* `color.ie-hex-str()` now automatically converts its color to the `rgb` space
96+
and gamut-maps it so that it can continue to take colors from any color
97+
space.
98+
99+
[color spaces]: https://developer.mozilla.org/en-US/docs/Web/CSS/color_value
100+
101+
### Dart API
102+
103+
* Added a `ColorSpace` class which represents the various color spaces defined
104+
in the CSS spec.
105+
106+
* Added `SassColor.space` which returns a color's color space.
107+
108+
* Added `SassColor.channels` and `.channelsOrNull` which returns a list
109+
of channel values, with missing channels converted to 0 or exposed as null,
110+
respectively.
111+
112+
* Added `SassColor.isLegacy`, `.isInGamut`, `.channel()`, `.isChannelMissing()`,
113+
`.isChannelPowerless()`, `.toSpace()`, `.toGamut()`, `.changeChannels()`, and
114+
`.interpolate()` which do the same thing as the Sass functions of the
115+
corresponding names.
116+
117+
* `SassColor.rgb()` now allows out-of-bounds and non-integer arguments.
118+
119+
* `SassColor.hsl()` and `.hwb()` now allow out-of-bounds arguments.
120+
121+
* Added `SassColor.hwb()`, `.srgb()`, `.srgbLinear()`, `.displayP3()`,
122+
`.a98Rgb()`, `.prophotoRgb()`, `.rec2020()`, `.xyzD50()`, `.xyzD65()`,
123+
`.lab()`, `.lch()`, `.oklab()`, `.oklch()`, and `.forSpace()` constructors.
124+
125+
* Deprecated `SassColor.red`, `.green`, `.blue`, `.hue`, `.saturation`,
126+
`.lightness`, `.whiteness`, and `.blackness` in favor of
127+
`SassColor.channel()`.
128+
129+
* Deprecated `SassColor.changeRgb()`, `.changeHsl()`, and `.changeHwb()` in
130+
favor of `SassColor.changeChannels()`.
131+
132+
* Added `SassNumber.convertValueToUnit()` as a shorthand for
133+
`SassNumber.convertValue()` with a single numerator.
134+
135+
* Added `InterpolationMethod` and `HueInterpolationMethod` which collectively
136+
represent the method to use to interpolate two colors.
137+
1138
## 1.56.1
2139

3140
### Embedded Sass

lib/sass.dart

+6-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,12 @@ export 'src/importer.dart';
2929
export 'src/logger.dart';
3030
export 'src/syntax.dart';
3131
export 'src/value.dart'
32-
hide ColorFormat, SassApiColor, SassApiValue, SpanColorFormat;
32+
hide
33+
ColorChannel,
34+
ColorFormat,
35+
LinearChannel,
36+
SassApiColorSpace,
37+
SpanColorFormat;
3338
export 'src/visitor/serialize.dart' show OutputStyle;
3439
export 'src/evaluation_context.dart' show warn;
3540

0 commit comments

Comments
 (0)