CSS media features to improve accessibility
- CSS
- Usability
When people talk about accessibility and CSS, they often mean properties that affect the accessibility tree and assistive technologies such as screen readers and voice control software. But there's another ally in the battle for accessible interfaces — media features.
Media features are conditions for the @media
CSS at-rule. They indicate specific characteristics of the device or browser. For example, screen orientation (orientation
) or display mode (display-mode
).
In this article, I'll talk about a few media features: prefers-reduced-motion
, prefers-color-scheme
, inverted-colors
, forced-colors
, ms-high-contrast
, prefers-contrast
, and prefers-reduced-transparency
.
These media features track operating system settings which changed by users who aren't satisfied with the default behavior of the system. For example, people with disabilities and those who are uncomfortable with the default design. Real world examples:
- Users prone to epileptic seizures turn off animation because it can trigger a photosensitive seizures
- Some people with astigmatism choose a dark theme and reduce contrast to avoid eye strain.
Considering user preferences will make the site's interface more flexible and personalized. This will not only help improve its accessibility but can also increase conversion rates. It's always nice to use products that cares about your preferences.
Most customizations apply only to the operating system. Many of them, like animations and contrast levels, don't change website interfaces. It all depends on whether the developers have taken them into account. This is where media features come in handy.
You can also track user settings with JavaScript, but I don't want to make this post bigger. I'll just focus on CSS features.
System settings
Let's take a look at the settings which can be considered in web interfaces now or in the future.
Animation
Animation settings allow you to change the speed of animations or turn them off completely. These settings don't affect websites unless they contain specific CSS at-rule styles.
Users who may adjust animation settings include:
- People with vestibular disorders or those prone to seizures
- People with cognitive disabilities and neurodivergent users, particularly those with attention deficit disorder (ADD).
Most operating systems include the ″Reduce motion″ or ″Minimize animation″ setting to accommodate these users needs.
Color scheme
Users can also change the color scheme settings and select the colors that will dominate the system. These are either light or dark shades. The setting does not affect sites if they do not support color schemes.
The following categories of users utilize color scheme setting:
- People with visual impairments. For example, a person with low vision, eye strain, and light sensitivity
- Those with cognitive disabilities and neurodivergent folks. For instance, individuals with ADD
- Everyone else due to aesthetic preferences, habits, or lighting conditions.
Color schemes can be selected in all popular operating systems. macOS and iOS have an additional automatic theme. If selected, it applies a light theme during the day and a dark theme at night.
Inverted colors
Inverted colors mode replaces system colors with their opposites, like a negative. It's a part of the filtered display mode category of settings.
Colors change not only in the system but also in browser tabs. So users might choose this mode instead of a dark theme. The screenshot shows how the inversion mode works. Yellow shades have turned blue, green has become magenta, and white has turned black.
Who uses inverted colors mode:
- People with visual impairments. For example, individuals with glaucoma or eye strain
- People with migraines and headaches
- Other users due to personal preferences, habits, or lighting conditions.
Most operating systems have inverted colors setting. On iOS, there are even two types of inversion: Smart and classic. In smart invert mode, pictures and videos are not inverted, while in classic one, all content is inverted.
Forced colors
Forced colors mode limits the number of colors to improve text readability by changing the contrast between text and background. High-contrast colors are mainly used. This mode changes colors both in the system and on websites.
Who uses forced colors mode:
- Users with visual impairments
- People with migraines and headaches
- People with photosensitive epilepsy
- Users who need to reduce visual noise to improve concentration.
Currently, this mode can only be selected on Windows. On Windows 10 and earlier versions, it's called Windows high contrast mode (WHCM). On Windows 11, it's called contrast themes.
The high contrast mode has several pre-set color schemes:
- High contrast black
- High contrast white
- High contrast one (1) and two (2).
The technology for replacing the color palette depends on the browser. It differs in Chromium-based browsers, Firefox (Quantum engine), Internet Explorer (Trident), and in older versions of Edge (EdgeHTML). For example, in Vivaldi (Chromium), the white background of the site becomes black, gray links and borders become bright yellow, and regular text becomes white instead of dark gray.
In Firefox, the originally white background will remain the same, gray links and borders will become bright blue, and regular text will be black instead of dark gray.
You can change the default behavior in Firefox. The setting is located in the Language and Appearance section, specifically in the ″Colors″ subsection. Open the modal window with the ″Manage Colors…″ button and select the ″Use system colors″ checkbox.
On Windows 11, the set of contrast themes has changed: Aquatic, desert, dusk, and night sky have been added.
If the pre-set themes don't suit you, you can customize them yourself. This includes the option to reduce contrast.
Contrast
Users can separately increase or decrease the level of contrast in the system without changing the screen brightness.
The contrast settings is used by:
- People with visual impairments, for example, those with glaucoma
- People who have migraines and headaches
- Those with old or low-quality displays
- Others who need higher contrast due to lighting conditions.
macOS and iOS have the increased contrast mode. It increases the difference between shades of gray and makes element borders more distinct.
Increasing contrast affects the appearance of both the system and web interfaces. Unlike system windows, only the contrast level changes on websites.
Let's look at what happens to a modal window in macOS with default mode and with increased contrast mode. Both windows show an image of a screaming cat called ″cute-cat.jpeg″. The first window has no frame, and the background of the control panel with buttons and additional settings is light gray. The second window looks visually different from the first. It has a black frame, the background of the control panel has become white, and all buttons and other elements in the panel have separate black frames.
Transparency
Users can enable or disable background transparency. Those who increase contrast often choose an opaque background.
A transparent background can increase cognitive load and reduce text readability. Therefore, this setting is used by:
- People with visual impairments. For example, those with astigmatism or low vision
- Users with cognitive disabilities and neurodivergent people. For instance, dyslexic users or those with ADD
- People with migraines and headaches.
Transparency can be precisely adjusted on Windows and in macOS. These settings not only affect the transparency in the system interface but also on websites and in browsers. This screenshot shows the navigation background of the first window is semi-transparent, while the second one has an opaque and solid background.
A few words about media types
The @media
at-rule in CSS has several media types. They describe the device on which the document is displayed.
all
: All devices. It's set automatically if no other type is specifiedscreen
: Devices with screens. For example, phones and laptopsprint
: Devices with preview and print functions. This includes printersspeech
: Devices with speech synthesis. For example, screen readers and voice assistants.
The speech
media type may be interesting from the point of view of accessibility. However, it is not currently supported by any browser. It used to be supported by the Opera browser on the Presto engine but support ceased after switching to Blink.
In the future, it may be useful for special styles for screen readers, such as to apply CSS properties for speech synthesis devices to some elements on a page.
Media features
Now we move on to media features that will help make web interfaces more accessible.
Some of them are not yet well supported by browsers. Things may change in the future with the development of CSS. In any case, it's useful to know about them.
prefers-reduced-motion
It tracks whether animation settings are selected to reduce its intensity. This is useful for any animation on the website. The animation can be slowed down or completely disabled.
Values for prefers-reduced-motion
:
no-preference
: Default animation settingsreduce
: Modified animation settings.
Browser support for prefers-reduced-motion
is relatively good: 97% (as of 2024).
In this code example we stop animation completely by using animation: none
.
@media (prefers-reduced-motion: reduce) {
.danger-animation {
animation: none;
}
}
Another example: The website uses scroll-behavior
for smooth scrolling to blocks on a page. If the page has a lot of content, such animation might cause motion sickness. In this case, you can animate the scrolling when animation is not configured in the system settings.
@media (prefers-reduced-motion: no-preference) {
html {
scroll-behavior: smooth;
}
}
Parallax effects often causes users to feel nauseous and dizzy. To prevent anyone from feeling unwell due to an image with position: sticky
, you can replace the property value with relative
.
.parallax-scrolling-image {
position: sticky;
}
@media (prefer-reduced-motion: reduce) {
.parallax-scrolling-image {
position: relative;
}
}
Animation can also be an important part of a website. Therefore, it's better to base decisions on the content. You can always slow down the animation so that it's not dangerous for users or doesn't distract them.
I wrote more about this media feature and animation requirements in a post about people with vestibular conditions and seizures.
Test the preference-reduced-motion feature
In Chrome and other Chromium-based browsers, you can quickly emulate reduced animations in the developer tools. Click on the three-dot button ″Customize and control DevTools″, then expand ″More tools″ and open the ″Rendering″ section. Finally, select the option ″Emulate CSS media feature prefers-reduced-motion″.
Firefox doesn't allow emulating prefers-reduced-motion
through its inspector, but there's a workaround. Enter about:config
in the browser's search bar, find ui.prefersReducedMotion
through the search, and set it to 1
. This will completely turn animation off. Setting it to 0
will return you to normal playback.
You can also change the animation settings manually.
Windows 11: Settings → Personalization → Visual effects → Animation effects.
MacOS: System Preferences → Accessibility → Display → Reduce Motion.
iOS: Settings → Accessibility → Motion → Reduce Motion.
Android: Settings → Accessibility → Display → Remove animations.
prefers-color-scheme
It determines the selected color theme.
Values for prefers-color-scheme
:
light
for light themedark
for dark theme.
prefers-color-scheme
has high browser support — 96% (as of the second half of 2024).
Developers can control all styles when working with dark themes on a website. It's particularly important to pay attention to the colors of backgrounds, texts, interactive elements in different states, icons and images, as well as other decorative elements.
For example, images in dark theme can be made less contrasting using filter, and you can also change the values of background-color
and color
. In the example, I use pure black and white for code readability. In the real world, it's better to avoid such high-contrast colors. White text on a black background is usually difficult to read for a large number of users.
.body {
color: black;
background-color: white;
}
@media (prefers-color-scheme: dark) {
.body {
color: white;
background-color: black;
}
.image {
filter: brightness(0.8) contrast(1.2);
}
}
Test the prefers-color-scheme feature
In Chromium-based browsers, you can enable the emulation of prefers-color-scheme
through the developer tools. Click on the three-dot button ″Customize and control DevTools″, then click on ″More tools″ and go to the ″Rendering″ section. There, enable the option ″Emulate CSS media feature prefers-color-scheme″.
In Firefox, the themes are located in the first tab with the inspector. The toggles are positioned near the styles block. The toggle with a moon icon enables the dark theme emulation, while the sun icon enables the light theme emulation.
In Safari, dark theme emulation is enabled in the web inspector. The toggle is labelled ″Force Dark Appearance″ and is located in the panel below all the tabs.
You can manually switch the theme in your operating system.
Windows 11: Settings → Personalization → Colors → Dark in the app mode section.
macOS: System Preferences → General → Dark or Auto in the appearance section.
iOS: Settings → Display and brightness → Dark in the appearance section.
Android: Settings → Display → Dark theme.
inverted-colors
It tracks the inverted colors mode.
The inverted-colors
values:
none
: Mode is not selected, the system displays default colorsinverted
: Inverted colors mode is selected.
Global browser support for inverted-colors
is 18% (second half of 2024). Currently, it's only supported in Safari.
For the inverted colors mode, additional styles are usually not required, except for canceling the inversion of videos and images. This can be done using the filter
CSS property.
@media (inverted-colors: inverted) {
img,
video {
filter: invert(100%);
}
}
Test the inverted-colors feature
There is only manually option for testing styles for the inverted colors mode.
macOS: System preferences → Accessibility → Display → Invert colors.
iOS: Settings → Accessibility → Display and text size → Smart invert or Classic invert.
forced-colors
It tracks forced colors mode. For example, high contrast mode in Windows. Works well with another media feature prefers-color-scheme
.
Possiblr values for forced-colors
:
none
: Forced colors mode isn't selected and the color palette isn't limitedactive
: Forced colors mode is enabled.
Browser support of forced-colors
is quite high — 93% (second half of 2024). So use it instead of the deprecated ms-high-contrast
in newer versions of Edge.
When the browser learns about the selection of forced colors mode, it limits the color palette to a small set of colors. Some values are canceled altogether.
The following properties have their static colors (set by developers) replaced with dynamic (system) colors:
color
background-color
border-color
outline-color
column-rule-color
text-decoration-color
text-emphasis-color
webkit-tap-highlight-color
fill
stroke
flood-color
stop-color
lighting-color
.
For this list of properties, other values are forcibly set:
color-scheme
—light dark
;background-image
withouturl
gets the valuenone
if it's an interactive element. The exception is buttons in Internet Explorer and legacy versions of Edgebox-shadow
—none
text-shadow
—none
scrollbar-color
—auto
.
In the case of high contrast mode, it's better to turn off the internal designer 👩🎨. People need the mode not for aesthetics, but to improve content readability. So the media feature for forced colors should be used wisely and only when it's really necessary. In most cases, the browser itself handles this task.
The main rule for working with styles in forced colors mode is to use dynamic colors instead of static ones. They can be set using special CSS system color keywords. For example, Window
for the window background, ButtonText
for button text color, and WindowText
for text color. This way, we let the system know which color to replace our static color with.
Another important feature of this mode is that browsers decide on applying system styles based on the semantics of elements. When a website has div-based layout, the browser will apply regular text styles to all elements. This also applies to elements with ARIA roles. For example, <div role="link">
is not a real link, so the system colors for link text won't be applied to this element.
If text is placed over an image, a black backplate will appear behind it. Its styles cannot be controlled. The first screenshot shows the original Xbox site interface in Vivaldi on Windows 10. The second shows it in black high contrast mode. A black backplate has appeared under the text.
If you keep semantics in mind and want to optimize the website for high contrast mode, you only need to pay attention to a few points. These are SVGs, images with transparent backgrounds, checkboxes and radio buttons, as well as elements with the box-shadow
CSS property. These are the ones that most often need fine-tuning.
In the image, I've gathered elements from different websites that users of high contrast mode might have difficulty with. Buttons and links with SVG icons blend into the background or become less contrasting. A black logo or dark image with a transparent background can also blend into the background. In the case of a checkbox or radio button, it's unclear whether they are selected or not. The icons of social media remained blue, gray, and red and look low-contrast against the black background. Yandex logo is blended into the background, and he black circle from the toggle button and the letters of the same color. Finally, the black icon with a cross for opening the menu disappeared against the background of the same color.
In this example, I improved buttons with icons in high contrast mode using the buttonText
keyword. This way, buttons will paint based on the system color for them.
@media (forced-colors: active) {
.button__svg {
fill: buttonText;
}
}
You can check how this works in Windows using the demo with a button with inline SVG from CodePen.
In this code we set styles for the focus indicator using both box-shadow
and outline
CSS properties. In high contrast mode, the outline
properties will apply instead of the box-shadow
one canceled by the system. In default color mode, only box-shadow
will be visible.
.button {
border: 1px solid transparent;
}
.button:focus {
box-shadow: 0 0 4px 1px darkslateblue;
outline: 2px solid transparent;
}
For solid color SVG icons, it's better to use the currentColor
keyword for fill
and stroke
. This way, they will inherit the system color of the parent element.
Let's imagine we have a link with an inline SVG icon. In default color mode, the link color is indigo, in high contrast mode it will be overridden by the system. The SVG with the class .link__svg
also inherits the same color in default mode, and in high contrast mode it eill be the system color.
.link {
color: indigo;
}
.link__svg {
fill: currentColor;
}
Let's take the demo with a Twitter link as an example. If we enable black high contrast mode, the link text and icon will become the system yellow color.
Bug in SVG
In older versions of Chromium-based browsers, SVG icons may not change color. This happened due to changes in the CSS specification.
If you really want to fix this issue, use a temporary hack with the forced-color-adjust: auto
property for your SVG. In this case, the child element will inherit the color of the parent when high contrast mode is enabled.
.link {
color: indigo;
}
.link__svg {
fill: currentColor;
forced-color-adjust: auto;
}
You can check how the hack works in my demo with a fix for the Twitter link from CodePen.
Another way to deal with the bug is forced-color-adjust: preserve-parent-color
. In this case, the element inherits the parent's value if no other methods are used, such as inherit
or currentColor
. In default colors mode, it behaves like the none
value.
You can also completely cancel the replacement of static colors with dynamic ones using forced-color-adjust: none
. In most situations, it's better not to use it. It could be useful if it's important to preserve colors. A good example for that is color palettes.
Test the forced-colors feature
In Chromium-based browsers, open the developer tools. Find the button with three dots named ″Customize and control DevTools″, expand the ″More tools″ dropdown and go to the ″Rendering″ section. Then enable the option ″Emulate CSS media feature forced-colors″.
Firefox doesn't allow emulating forced colors mode through its inspector, but there's a workaround. Enter about:config
in the browser's search bar, find ui.forcedColors
through the search, and set the value to true
or 1
. Don't forget to revert it back afterwards.
You can also enable forced colors mode manually in Windows 11: Settings → Personalization → Colors → Contrast themes.
ms-high-contrast (deprecated)
It precisely tracks whether the user has selected high contrast mode. Corresponds to high contrast themes in Windows versions up to 11.
The media feature is non-standard and already deprecated. It's better to use the modern alternative forced-colors
instead. It may be needed when support for Internet Explorer and Edge version 18 and lower is required.
Available values for ms-high-contrast
:
black-on-white
: For mode with black text on white backgroundwhite-on-black
: For mode with white text on black backgroundactive
: For other color combinations.
When you work with styles for ms-high-contrast
, consider the same things as for forced-colors
. The only peculiarity is that this media feature removes background-image
from interactive elements. Here's how you can change this behaviour:
@media (-ms-high-contrast: black-on-white) {
.interactive-element {
background-image: url("bg-image.png");
}
}
prefers-contrast
It determines the selection of settings that decrease or increase the contrast difference between colors.
Currently, prefers-contrast
doesn't necessarily track a high level of contrast, as in high contrast mode. However, it's worth noting that in the future, this media feature may be more closely related to it.
prefers-contrast
can have the following values:
no-preference
: No settings are selectedmore
: Contrast is increasedless
: Contrast is decreasedcustom
: Forced colors mode is selected in the system. For example, Windows high contrast mode.
Global browser support for prefers-contrast
is quite high — 93% (second half of 2024).
When considering the contrast mode, you can add any styles that will improve the user experience. In this example, we change the border of an element to ensure the styles are visible in high contrast:
.image {
border: 3px dashed lightpink;
}
@media (prefers-contrast: more) {
.image {
border: 3px solid black;
}
}
In this code all media expressions define high contrast mode.
@media (forced-colors: active) {
/* CSS properties */
}
@media (prefers-contrast: forced) {
/* CSS properties */
}
@media (prefers-contrast: custom) {
/* CSS properties */
}
You can see forced-colors
in action right now in Safari using my demo with a cute otter on CodePen.
Test the prefers-contrast feature
Now you can quickly emulate prefers-contrast
in Chromium-based browsers. Open the developer tools and find the button with three dots ″Customize and control DevTools″. Then expand the ″More tools″ dropdown and go to the ″Rendering″ section. After that, enable the option ″Emulate CSS media feature prefers-contrast″.
Firefox doesn't allow emulating prefers-contrast
in the inspector, but there's an alternative. Enter about:config
in the browser's search bar and find the ui.prefersСontrast
setting through the search. Now set its value to true
or 1
, just don't forget to revert it back afterwards.
You can test the media feature in systems through the contrast increase setting.
macOS: System preferences → Accessibility → Display → Increase contrast.
iOS: Settings → Accessibility → Display and text size → Increase contrast.
prefers-reduced-transparency
It tracks the disabling or reduction of transparency
in the system.
The list of all values for prefers-reduced-transparency
:
no-preference
: A user hasn't changed transparency settingsreduce
: Transparency is reduced or turned off.
Currently, browsers support of prefers-reduced-transparency
is quite well — 71% (second half of 2024).
Let's take a look at a practical example. In the code below, I overrode opacity
for elements with a transparent background:
.transparency-bg {
opacity: 0.5;
}
@media (prefers-reduced-transparency: reduce) {
.transparency-bg {
opacity: 1;
}
}
Test the prefers-reduced-transparency feature
You can quickly emulate the transparency settings in the Chromium developer tools. Find the button with three dots named ″Customize and control DevTools″ in the top panel. Expand the ″More tools″ dropdown and go to the ″Rendering″ section. Then enable the option ″Emulate CSS media feature prefers-reduced-transparency″.
Firefox doesn't allow emulating prefers-reduced-transparency
directly through the inspector, but there's an alternative. Enter about:config
in the browser's search bar and find the ui.prefersReducedTransparency
setting through the search. Set its value to true
or 1
, and but don't forget to revert it back afterwards.
Of course, you can turn transparency off directly in your operating system.
Windows 11: Settings → Personalization → Colors → Transparency effects.
macOS: System preferences → Accessibility → Display → Reduce transparency.
iOS: Settings → Accessibility → Display and text size → Reduce transparency
What I didn't mention
In this post, I didn't cover only one media feature for user preferences — prefers-reduced-data
. It tracks the volume of data reception chosen by a user. Although it's related to performance, it's also useful for accessibility. For more details, read ″Creating websites with prefers-reduced-data
″.
Further reading
- Media Queries Level 5
- Operating System and Browser Accessibility Display Modes
- thoughtbot.com, dark mode, and other user preferences
- Respecting Users’ Motion Preferences
- Creating Accessible UI Animations
- A Complete Guide to Dark Mode on the Web
- I no longer understand
prefers-contrast
- Styling for Windows high contrast with new standards for forced colors
- CurrentColor SVG in forced colors modes
prefers-reduced-motion
(Doka Guide)prefers-reduced-transparency
(Doka Guide)forced-colors
(Doka Guide)prefers-color-scheme
(Doka Guide).
Thanks to Vasiliy Dudin for his help with editing and translating this text.