Native HTML light and dark color scheme switching

HTML CSS

It’s getting dark early in Berlin in the winter. It’s not even close to evening, but my OS and all apps have already switched to dark mode. Well, not all of them, unfortunately. And that’s the thing: dark mode has become a quality-of-life feature for many users, and I often try to avoid using apps or websites that haven’t implemented it, especially in the evening. They literally hurt my eyes!

When it comes to color scheme implementations, they range from rather useless ones that require a page reload to more sensible ones that query the prefers-color-scheme media feature and apply changes in CSS on the fly:

body {
	background-color: #ffffff;
	color: #000000;

	@media (prefers-color-scheme: dark) {
		background-color: #000000;
		color: #ffffff;
	}
}

⭐ I’ll be using native CSS nesting in all demos throughout this article. It works in all modern browsers and makes code a bit more compact, especially when it comes to media queries. But if you’re not familiar with CSS nesting, you can use this handy Lightning CSS playground to figure out how it looks without nesting.

This approach is already a good start. But it only covers the simplest case and doesn’t allow users to choose a different color scheme for this specific website. Just like light color schemes hurt my eyes in the evening, many people are not comfortable with dark schemes or with particular ones that aren’t good for them. So, it’s all about user choice.

Currently, there’s no way to directly override a user’s OS preference if you want to offer a scheme selector on your page. Fortunately, in the CSS Media Queries Level 5 spec, there’s a PreferenceManager interface that will solve this problem. Meanwhile, the most popular solution these days is to use JavaScript to set an attribute like <html data-scheme="dark"> reflecting the forced scheme and use it in CSS:

body {
	background-color: #ffffff;
	color: #000000;

	[data-scheme='dark'] & {
		background-color: #000000;
		color: #ffffff;
	}
}

This approach always seemed hacky to me. However, if you know HTML well enough, there are a few much more convenient native options available.

Setting color-scheme

If you ever fall into the dark scheme rabbit hole, the first thing you’ll learn is the color-scheme CSS property. It’s essential for setting the scene for everything else. Most importantly, it turns on the browser’s default dark scheme support. Which, unfortunately, browsers can’t enable by default for backward compatibility reasons.

:root {
	color-scheme: light dark;
}

The light dark value means that we’re choosing to support both light and dark schemes in our code. The property will be inherited down the document tree, and the browser will enable some default styling for built-in primitives when needed. Thank you, browser!

Two browser windows with the same page: one in the light scheme, the other in the dark scheme. The text on the page says: link is not a button! The link word is a link, the button word is a button. All page elements, including scrollbars, are perfectly aligned with the scheme.
Default browser dark styles enabled by the color-scheme: light dark property.

When switching your schemes, it’s important to switch the value of this property, too: set color-scheme: light on the root along with light styles and the other way around for the dark ones. Remember this, it’ll come in handy later.

And somewhere along these lines, you’ll probably read that you can also set this mode right in HTML using the <meta> element. Why would you do that? Oh well, who knows? Maybe you don’t use CSS or whatever. So silly, right?

<meta name="color-scheme" content="light dark">

It turns out that it’s not just a flag for the browser but a tool you can use to force color schemes using JavaScript! But only if you use the fairly new light-dark() CSS function.

Switching color-scheme

Remember the earlier example with media queries and ‌prefers-color-scheme?

body {
	background-color: #ffffff;
	color: #000000;

	@media (prefers-color-scheme: dark) {
		background-color: #000000;
		color: #ffffff;
	}
}

You can also express the same idea like this:

body {
	background-color: light-dark(#ffffff, #000000);
	color: light-dark(#000000, #ffffff);
}

⭐ By the way, I use --color-back and --color-text variables in my demos. It makes it easier to set colors even in a small demo, let alone a bigger project. But to make things easier to read, I chose to set colors directly in code samples.

Here’s the three-position switch I often use. Along with “light” and “dark” options that force a certain scheme, there’s also the “auto” option that gives the control over the color scheme back to the OS, selected by default.

<section aria-label="Color scheme switcher">
	<button aria-pressed="false" value="light">
		Light
	</button>
	<button aria-pressed="true" value="light dark">
		Auto
	</button>
	<button aria-pressed="false" value="dark">
		Dark
	</button>
</section>

And to make it all work, a simple script that takes the button’s value and sets it to the <meta name="color-scheme">:

const colorScheme = document.querySelector('meta[name=color-scheme]');
const switchButtons = document.querySelectorAll('button');

switchButtons.forEach(button => {
	button.addEventListener('click', () => {
		const currentButton = button;

		switchButtons.forEach(
			button => button.setAttribute(
				'aria-pressed', button === currentButton
			)
		);

		colorScheme.content = button.value;
	});
});

As you can see, once we set content="dark" the browser switches to the last value in the light-dark() function and the other way around with the light one. This HTML’s color-scheme turned out not so silly after all!

To make it work properly, you’ll need to decide where to store your global color-scheme value, so the script can force it. In this example, I chose to use the HTML one, so I removed the color-scheme property from CSS. But you can also keep it in CSS and force it via JavaScript like so:

<html style="color-scheme: dark">
Open in the new tab

⭐ For a good UX, you’ll also need a way to store user preference somewhere in localStorage, so the users won’t have to switch it next time they visit or when opening a new tab. I’m sure you can figure it out on your own!

One of the downsides of this approach is the browser support: the light-dark() CSS function has been available in all modern browsers since May 2024, which makes it “newly available” on the Baseline scale. It will become “widely available” only around November 2026 or 30 months later. You can transpile it for older browsers using Lightning CSS or PostCSS plugin, but make sure you check the output and test it in an older browser. It might be a bit tricky at times.

As for the other major downside, the light-dark() function accepts only colors for now. Trust me, I tried to use it with other values, and it didn’t work. So, what’s the problem? You might need to change not just colors but images or font properties that work better with the dark scheme.

In this case, there’s another HTML solution for you!

Linking a CSS scheme

You all know the most common way of linking CSS to a page: it’s a stylesheet linked from a file.

<link rel="stylesheet" href="index.css">

But did you know that you can also set the media attribute to conditionally load and apply CSS based on user preferences? Sure you can! But first, you’d need to split your files into light and dark styles:

<link
	rel="stylesheet"
	href="light.css"
	media="(prefers-color-scheme: light)"
>
<link
	rel="stylesheet"
	href="dark.css"
	media="(prefers-color-scheme: dark)"
>

It doesn’t look too convenient this way. A much better approach would be to split your styles into three parts! Yes, I’m serious. Hear me out!

  1. The main file, called index.css, will contain all your styles and use CSS variables for anything you need to change depending on a color scheme.
  2. The light.css file will contain only CSS variables with values set to everything that makes sense for the light scheme.
  3. The dark.css, you guessed it, will have everything dark.

Remember to set the appropriate color-scheme property values in each color scheme file to help the browser: color-scheme: light and color-scheme: dark, respectfully.

Interestingly enough, the file that doesn’t fit user preferences will still be loaded by the browser but with lower priority. I went into much greater detail about this in my other “Condi­tionally adaptive CSS” article if you’re curious.

So, this three-file CSS architecture now has all the inconveniences and no benefits compared to the previously discussed solution with media queries. What now? Don’t you worry, it was just the first step. Now, to the switching.

Switching CSS schemes

When you’re linking your much simpler single-file styles, there’s something else you’re implicitly setting: the media attribute. If it’s not explicitly set, it means that you want your styles to apply to all media types. So, it defaults to all:

<link rel="stylesheet" href="index.css" media="all">

There are techniques using media="print" for lazy-loading CSS, but since we’ve learned that the media attribute can take not only media types but complex media queries, you can negate it as not all and it won’t be applied to any media.

<link rel="stylesheet" href="index.css" media="not all">

But why would you need not to load your styles? Now you see where it’s going. We can use it to switch between color schemes and force user preferences.

Let’s take the same three-position switch from the previous example but change the auto button’s value to auto to better match what it does. Our JavaScript function becomes a bit bigger, but the idea stays the same. The only difference this time is that we’re changing the media attribute’s value:

const styleLight = document.querySelector('link[rel=stylesheet][media*=prefers-color-scheme][media*=light]');
const styleDark = document.querySelector('link[rel=stylesheet][media*=prefers-color-scheme][media*=dark]');

function switchScheme(scheme) {
	let lightMedia;
	let darkMedia;

	if (scheme === 'auto') {
		lightMedia = '(prefers-color-scheme: light)';
		darkMedia = '(prefers-color-scheme: dark)';
	} else {
		lightMedia = (scheme === 'light') ? 'all' : 'not all';
		darkMedia = (scheme === 'dark') ? 'all' : 'not all';
	}

	styleLight.media = lightMedia;
	styleDark.media = darkMedia;
}

In short, if we force the dark scheme, the dark.css gets media="all" instead of the prefers-color-scheme and the light.css one gets ‌media="not all", and the other way around for the light scheme. Once the user chooses the “auto” option, we stop forcing and restoring all previous prefers-color-scheme media values.

Open in the new tab

Given that scheme files containing only variables are relatively small, and browsers download all CSS files anyway (only the priority differs), the switching happens seamlessly. You can check this method in action on this website. And if you happen to use Safari or Vivaldi browsers, you might notice something else changing while you switch the schemes.

One more thing

There’s the theme_color key in the web manifest that sets the installed app’s chrome color. You can also set it via HTML and even use the media attribute to apply different theme colors depending on the color scheme:

<meta
	name="theme-color"
	content="#c1f07c"
	media="(prefers-color-scheme: light)"
>
<meta
	name="theme-color"
	content="#9874d3"
	media="(prefers-color-scheme: dark)"
>

I bet you saw that coming: you can also force the theme color while switching the color scheme using all and not all values via the same script. How cool is that?

Two Safari windows: one in the light scheme, the other in the dark scheme. The light one has a lime top panel, the dark one has a violet top panel.
Different top panel colors using the theme-color meta element.

Homework

As all of us did, I learned most of what I know from other people’s posts and articles. I want to say a special thank you to Thomas Steiner for his early articles on color scheme switching: Prefers-color-scheme: Hello darkness, my old friend and Improved dark mode default styling with the color-scheme. Please also have a look at Darin Senneff’s Progressively-enhanced dark mode and Sara Joy’s Come to the light-dark() side articles for a different perspective on the matter. And let’s dream about a better future for color scheme switching while reading Bramus’ What if you had real control over light mode / dark mode on a per-site basis? article.


Initially published in the HTMHell advent calendar in December 2024. Thank you to Manuel Matuzovic, Saptak Sengupta, and Karl Stolley for proofreading and feedback.