hsl()
Baseline Widely available *
This feature is well established and works across many devices and browser versions. It’s been available across browsers since January 2020.
* Some parts of this feature may have varying levels of support.
Note:
The hsla()
functional notation is an alias for hsl()
. They are exactly equivalent. It is recommended to use hsl()
.
The hsl()
functional notation expresses a color in the sRGB color space according to its hue, saturation, and lightness components. An optional alpha component represents the color's transparency.
Try it
Defining complementary colors with hsl()
can be done by adding or subtracting 180 degrees from the hue value, as they are positioned on the same diameter of the color wheel. For example, if the hue angle of a color is 10deg
, its complementary has 190deg
as its hue angle.
Syntax
/* Absolute values */
hsl(120deg 75% 25%)
hsl(120 75 25) /* deg and % units are optional */
hsl(120deg 75% 25% / 60%)
hsl(none 75% 25%)
/* Relative values */
hsl(from green h s l / 0.5)
hsl(from #0000FF h s calc(l + 20))
hsl(from rgb(200 0 0) calc(h + 30) s calc(l + 30))
/* Legacy 'hsla()' alias */
hsla(120deg 75% 25% / 60%)
/* Legacy format */
hsl(120, 75%, 25%)
hsl(120deg, 75%, 25%, 0.8)
Note: hsl()
/hsla()
can also be written in a legacy form in which all values are separated with commas, for example hsl(120, 75%, 25%)
or hsla(120deg, 75%, 25%, 0.8)
. The none
value is not permitted in the comma-separated legacy syntax, the deg
on the hue value is optional, and the %
units are required for the saturation and lightness values.
Values
Below are descriptions of the allowed values for both absolute and relative colors.
Absolute value syntax
hsl(H S L[ / A])
The parameters are as follows:
H
-
A
<number>
, an<angle>
, or the keywordnone
(equivalent to0deg
in this case) representing the color's<hue>
angle. S
-
A
<percentage>
or the keywordnone
(equivalent to0%
in this case). This value represents the color's saturation. Here100%
is completely saturated, while0%
is completely unsaturated (gray). L
-
A
<percentage>
or the keywordnone
(equivalent to0%
in this case). This value represents the color's lightness. Here100%
is white,0%
is black, and50%
is "normal". A
Optional-
An
<alpha-value>
representing the alpha channel value of the color, where the number0
corresponds to0%
(fully transparent) and1
corresponds to100%
(fully opaque). Additionally, the keywordnone
can be used to explicitly specify no alpha channel. If theA
channel value is not explicitly specified, it defaults to 100%. If included, the value is preceded by a slash (/
).
Note:
See Missing color components for more information on the effect of none
.
Note:
Absolute hsl()
colors are serialized to rgb()
values. The values of the red, green, and blue components may be rounded in serialization.
Relative value syntax
hsl(from <color> H S L[ / A])
The parameters are as follows:
from <color>
-
The keyword
from
is always included when defining a relative color, followed by a<color>
value representing the origin color. This is the original color that the relative color is based on. The origin color can be any valid<color>
syntax, including another relative color. H
-
A
<number>
, an<angle>
, or the keywordnone
(equivalent to0deg
in this case) representing the output color's<hue>
angle. S
-
A
<percentage>
or the keywordnone
(equivalent to0%
in this case). This represents the saturation of the output color. Here100%
is completely saturated, while0%
is completely unsaturated (gray). L
-
A
<percentage>
or the keywordnone
(equivalent to0%
in this case). This represents the lightness of the output color. Here100%
is white,0%
is black, and50%
is "normal". A
Optional-
An
<alpha-value>
representing the alpha channel value of the output color, where the number0
corresponds to0%
(fully transparent) and1
corresponds to100%
(fully opaque). Additionally, the keywordnone
can be used to explicitly specify no alpha channel. If theA
channel value is not explicitly specified, it defaults to the alpha channel value of the origin color. If included, the value is preceded by a slash (/
).
Note:
To fully enable the representation of the full spectrum of visible colors, the output of relative hsl()
color functions is serialized to color(srgb)
. That means that querying the output color value via the HTMLElement.style
property or the CSSStyleDeclaration.getPropertyValue()
method returns the output color as a color(srgb ...)
value.
Defining relative color output channel components
When using relative color syntax inside an hsl()
function, the browser converts the origin color into an equivalent HSL color (if it is not already specified as such). The color is defined as three distinct color channel values — h
(hue), s
(saturation), and l
(lightness) — plus an alpha channel value (alpha
). These channel values are made available inside the function to be used when defining the output color channel values:
- The
h
value is resolved to a<number>
between0
and360
, inclusive, that represents the origin color's<hue>
degree value. - The
s
andl
values are each resolved to a<number>
between0
and100
, inclusive, where100
is equivalent to100%
. - The
alpha
value is resolved to a<number>
between0
and1
, inclusive.
When defining a relative color, the different channels of the output color can be expressed in several different ways. Below, we'll study some examples to illustrate these.
In the first two examples below, we are using relative color syntax. However, the first one outputs the same color as the origin color and the second one outputs a color not based on the origin color at all. They don't really create relative colors! You'd be unlikely to ever use these in a real codebase, and would probably just use an absolute color value instead. We included these examples as a starting point for learning about relative hsl()
syntax.
Let's start with an origin color of rgb(255 0 0)
(equivalent to hsl(0 100% 50%)
). The following function outputs the same color as the origin color — it uses the origin color's h
, s
, and l
channel values (0
, 100%
, and 50%
) as the output channel values:
hsl(from rgb(255 0 0) h s l)
This function's output color is the sRGB color()
equivalent of hsl(0 100% 50%)
: color(srgb 1 0 0)
.
The next function uses absolute values for the output color's channel values, outputting a completely different color not based on the origin color:
hsl(from rgb(255 0 0) 240 60% 70%)
In the above case, the output color is the sRGB color()
equivalent of hsl(240 60% 70%)
: color(srgb 0.52 0.52 0.88)
.
The following function creates a relative color based on the origin color:
hsl(from rgb(255 0 0) h 30% 60%)
This example:
- Converts the origin color (
rgb(255 0 0)
) into anhsl()
equivalent (hsl(0 100% 50%)
). - Sets the
H
channel value for the output color to those of the origin colorhsl()
equivalent'sH
channel value —0
. - Sets the output color's
S
andL
channel values to new values not based on the origin color:30%
and60%
, respectively.
The final output color is the equivalent of hsl(0 30% 60%)
in the sRGB color space — color(srgb 0.72 0.48 0.48)
.
Note: As mentioned above, if the output color is using a different color model to the origin color, the origin color is converted to the same model as the output color in the background so that it can be represented in a way that is compatible (i.e. using the same channels).
In the examples we've seen so far in this section, the alpha channels have not been explicitly specified for either the origin or output colors. When the output color alpha channel is not specified, it defaults to the same value as the origin color alpha channel. When the origin color alpha channel is not specified (and it is not a relative color), it defaults to 1
. Therefore, the origin and output alpha channel values are 1
for the above examples.
Let's look at some examples that specify origin and output alpha channel values. The first one specifies the output alpha channel value as being the same as the origin alpha channel value, whereas the second one specifies a different output alpha channel value, unrelated to the origin alpha channel value.
hsl(from rgb(255 0 0 / 0.8) h s l / alpha)
/* Computed output color: color(srgb 1 0 0 / 0.8) */
hsl(from rgb(255 0 0 / 0.8) h s l / 0.5)
/* Computed output color: color(srgb 1 0 0 / 0.5) */
In the following example, the rgb()
origin color is again converted into an hsl()
representation — hsl(0 100% 50% / 0.8)
. calc()
calculations are applied to the H
, S
, L
, and A
values, and the final output color is the equivalent of hsl(60 80% 30% / 0.7)
in the sRGB color space: color(srgb 0.72 0.72 0.08 / 0.7)
.
hsl(from rgb(255 0 0 / 0.8) calc(h + 60) calc(s - 20) calc(l - 10) / calc(alpha - 0.1))
Note:
Because the origin color channel values are resolved to <number>
values, you have to add numbers to them when using them in calculations, even in cases where a channel would normally accept <percentage>
, <angle>
, or other value types. Adding a <percentage>
to a <number>
, for example, doesn't work.
Formal syntax
<hsl()> =
<legacy-hsl-syntax> |
<modern-hsl-syntax>
<legacy-hsl-syntax> =
hsl( <hue> , <percentage> , <percentage> , <alpha-value>? )
<modern-hsl-syntax> =
hsl( [ <hue> | none ] [ <percentage> | <number> | none ] [ <percentage> | <number> | none ] [ / [ <alpha-value> | none ] ]? )
<hue> =
<number> |
<angle>
<alpha-value> =
<number> |
<percentage>
Examples
Using hsl() with conic-gradient()
The hsl()
function works well with conic-gradient()
as both deal with angles.
CSS
div {
width: 100px;
height: 100px;
background: conic-gradient(
hsl(360 100% 50%),
hsl(315 100% 50%),
hsl(270 100% 50%),
hsl(225 100% 50%),
hsl(180 100% 50%),
hsl(135 100% 50%),
hsl(90 100% 50%),
hsl(45 100% 50%),
hsl(0 100% 50%)
);
clip-path: circle(closest-side);
}
Result
Using relative colors with hsl()
This example styles three <div>
elements with different background colors. The middle one is given the unmodified --base-color
, while the left and right ones are given lightened and darkened variants of that --base-color
.
These variants are defined using relative colors — the --base-color
custom property is passed into an hsl()
function, and the output color has its lightness channel modified to achieve the desired effect via a calc()
function, while the hue and saturation are left unchanged. The lightened color has 20% added to the lightness channel, and the darkened color has 20% subtracted from it.
CSS
:root {
--base-color: orange;
}
/* As per the spec, s and l values should resolve to a number between 0-100
However, Chrome 121+ incorrectly resolves them to numbers between 0-1
hence currently using calculations like l + 0.2 instead of l + 20 */
#one {
background-color: hsl(from var(--base-color) h s calc(l + 0.2));
}
#two {
background-color: var(--base-color);
}
#three {
background-color: hsl(from var(--base-color) h s calc(l - 0.2));
}
/* Use @supports to add in support for old syntax that requires % units to
be specified in lightness calculations. This is required for
Safari 16.4+ */
@supports (color: hsl(from red h s calc(l - 20%))) {
#one {
background-color: hsl(from var(--base-color) h s calc(l + 20%));
}
#three {
background-color: hsl(from var(--base-color) h s calc(l - 20%));
}
}
Result
The output is as follows:
Legacy syntax: comma-separated values
For legacy reasons, the hsl()
function accepts a form in which all values are separated using commas.
HTML
<div class="space-separated"></div>
<div class="comma-separated"></div>
CSS
div {
width: 100px;
height: 50px;
margin: 1rem;
}
div.space-separated {
background-color: hsl(0 100% 50% / 50%);
}
div.comma-separated {
background-color: hsl(0, 100%, 50%, 0.5);
}
Result
Legacy versus modern syntax
The example demonstrates how the hsla()
syntax is an alias for hsl()
; both are supported using both modern and legacy (comma-separated) syntaxes.
HTML
<div class="modern">HSL</div>
<div class="legacy">HSL</div>
<div class="modernWithAlpha">HSL</div>
<div class="modernHSLA">HSLA</div>
<div class="legacyHSLA">HSLA</div>
CSS
div {
width: 100px;
min-height: 50px;
font-family: sans-serif;
display: flex;
align-items: center;
justify-content: center;
}
body {
display: flex;
gap: 20px;
}
div.modern {
background-color: hsl(90 80% 50%);
}
div.legacy {
background-color: hsl(90, 80%, 50%);
}
div.modernWithAlpha {
background-color: hsl(90 80% 50% / 50%);
}
div.modernHSLA {
background-color: hsla(90 80% 50% / 50%);
}
div.legacyHSLA {
background-color: hsla(90, 80%, 50%, 0.5);
}
Result
Specifications
Specification |
---|
CSS Color Module Level 5 # relative-HSL |
CSS Color Module Level 4 # the-hsl-notation |
Browser compatibility
See also
<hue>
data typelch()
andhwb()
color functions- Hue interpolation in
color-mix()
- List of all color notations
- sRGB color picker and conversion tool
- Using relative colors
- CSS colors module
- Color picker tool by Lea Verou