Color Scheme

A website’s color scheme should be stored in CSS custom properties. It is common practice to define custom properties on the :root pseudo-class for global reference:

:root {
    --color-text: black;
    --color-base: white;
}

With the code example above the text color can be referenced with var(--color-text):

p {
    color: var(--color-text);
}

Browser preference

Users express their color scheme preference by choosing light mode, dark mode or ‘automatic’ in their browser settings. The prefers-color-scheme CSS @media query feature indicates if a light or dark color scheme is preferred1. This @media query feature can be utilized to overwrite the CSS variables that store the color scheme colors2:

@media (prefers-color-scheme:dark) {
    :root {
        --color-text: white;
        --color-base: black;
    }
}

Favicons

The same media feature can be used inside HTML to show a light or dark favicon based on the color scheme preference:

<head>
    <link
        rel="icon"
        href="favicon-black.ico"
        media="(prefers-color-scheme: light)"
    >
    <link
        rel="icon"
        href="favicon-white.ico"
        media="(prefers-color-scheme: dark)"
    >
</head>

The black favicon is shown when light mode is selected, while the white favicon shown in dark mode. At the time of writing (September 2024) Firefox does not update the favicon when toggling between light and dark mode.

An alternative method is to use an SVG as favicon and style it using CSS:

<svg>
  <style>
    path {
      fill: black;
    }

    @media (prefers-color-scheme: dark) {
      path {
        fill: white;
      }
    }
  </style>
  <path d="..." />
</svg>

Footnotes

  1. This is explained in detail in the MDN web docs.

  2. This also works inside <style> tags in SVG files.