Dark Mode with pure CSS


date: 2021-05-24 title: Dark mode switch with pure CSS description: Who knew it was possible to do a dark mode switch with pure CSS? I’ll show you how to do mine, but feel free to adapt it as you want! cover: https://images.unsplash.com/photo-1536613105185-09ea1249a2cb?ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&ixlib=rb-1.2.1&auto=format&fit=crop&q=80 tags:

  • tuto
  • css toc: true —

So here’s how to do a CSS-only dark mode for your website ! I’ll show you how to do mine, but feel free to adapt it as you want!

Why would I do that?

  • Writing JavaScript sucks
  • Consuming JavaScript as a client sucks
  • Accessibility: some people browse internet without internet, or with extensions like NoScript
  • It’s actually simple to do this with pure CSS, so why not?

How can CSS handle two possible states?

You probably wonder how can CSS change the color theme of a whole page. Because, knowing which color theme to display require a “switch”, something that can be “activated”… Is JavaScript the only solution?

No, these things exists in bare HTML too! There are many tags that are interactive. Here, we’ll use an input tag, with a checkbox. If the checkbox is checked, the Dark Mode will be active, and the Light Mode otherwise.

<input type="checkbox" id="dark-mode-switch" />

The CSS can access to the checkbox state with the :checked pseudo-class (there are many more!).

input#dark-mode-switch {
  /* Light Theme */
  background: #fff;
}

input#dark-mode-switch:checked {
  /* Dark Theme */
  background: #222220;
}

Changing the theme

There is a problem in the previous code snippet.

input#dark-mode-switch:checked {
  /* ↑ here */
  background: #222220;
}

We are selecting the switch itself, but it doesn’t matter to us, we want to change the whole body! And with CSS, it isn’t possible to select a parent tag…

But, it is possible to select a neighbour or a child tag, with CSS combinator! The A ~ B CSS combinator requires selector B to be an element with the same parent node as the selector A.

input#dark-mode-switch:checked ~ .general-theme {
  /* dark theme */
  background: #222220;
  color: #ddd;
}

.general-theme {
  /* light theme */
  background: #fff;
  color: #222;
}

Then, we’ll place the switch right after the body tag, and we’ll insert the whole site inside a div right after, just like this:

<head>
  <!-- your head --->
</head>
<body>
  <input type="checkbox" id="dark-mode-switch" />
  <div class="general-theme">
    <div class="my-website">
      <!-- your website content --->
    </div>
  </div>
</body>

Now, we need something to activate that checkbox: a label!

<label for="dark-mode-switch" class="ewen-toggle"> Toggle! </label>

You can place it wherever you want. But it’ll probably look ugly, so unless you know some CSS, if you want a fancy switch, just checkout the last chapter of this article for the code of this website’s switch ;)

Storing the theme

If you’ve gone this far and copy-pasted the code to your website, you might have noticed that when navigating, the theme isn’t stored. When the visitor goes from a Dark Mode activated page to another page, the theme goes back to Light Mode. How can we prevent that?

Unluckily, this isn’t just “displaying things” anymore. If you want your theme to be persistent across your pages, you have no other choice than using cookies or local storage, which require javascript.

var switcher = document.getElementById("dark-mode-switch");

// Click on dark mode icon. Store user preference through sessions
switcher.addEventListener("change", function () {
  // If dark mode is selected
  if (this.checked) {
    // set the a variable in user's browser
    localStorage.setItem("darkMode", "true");
  } else {
    // timeout of 100ms to avoid clipping
    setTimeout(function () {
      localStorage.setItem("darkMode", "false");
    }, 100);
  }
});

Now that the variable is set, we need to load user’s preference each time they visit the website. We’ll get the value from local storage, and simply check the checkbox we talked about earlier!

// Checks Storage. Keep user preference on page reload
if (localStorage.getItem("darkMode") === "true") {
  // Checking the checkbox
  switcher.checked = true;
}

I don’t consider it as a defeat: No-JS users that want dark mode will simply read your articles with Dark Mode activated, and navigate with light mode. Also, we’ve used CSS where it is possible and JS where it’s necessary, so everything is in the right place!

Setting defaults

It’s better to respect your visitor’s default settings for their first visit: it’s probably what they’ll want. You can set the default’s theme to their preferred color with the following code.

if (localStorage.getItem("darkMode") === null) {
  if (
    window.matchMedia &&
    window.matchMedia("(prefers-color-scheme: dark)").matches
  ) {
    // dark mode
    switcher.checked = true;
    localStorage.setItem("darkMode", "true");
  } else {
    switcher.checked = false;
    localStorage.setItem("darkMode", "false");
  }
}

And that’s it, enjoy! 😁

Appendix: a pretty switch

Just copy-paste this.

<div class="sidebar-toggler">
  <div class="ewen-toggle-wrapper">
    <label for="dark-mode-switch" class="ewen-toggle">
      <div class="ewen-toggle-track">
        <div class="ewen-toggle-track-moon">
          <img
            src=""
            role="presentation"
            style="pointer-events: none"
            width="16"
            height="16"
          />
        </div>
        <div class="ewen-toggle-track-sun">
          <img
            src=""
            role="presentation"
            style="pointer-events: none"
            width="16"
            height="16"
          />
        </div>
      </div>
      <div class="ewen-toggle-thumb"></div>
      <input
        class="ewen-toggle-screenreader-only"
        type="checkbox"
        aria-label="Switch between Dark and Light mode"
      />
    </label>
  </div>
</div>
.ewen-toggle {
  touch-action: pan-x;

  display: inline-block;
  position: relative;
  cursor: pointer;
  border: 0;
  padding: 0;

  -webkit-touch-callout: none;
  user-select: none;

  -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
  -webkit-tap-highlight-color: transparent;
}

.ewen-toggle-screenreader-only {
  border: 0;
  clip: rect(0 0 0 0);
  height: 1px;
  margin: -1px;
  overflow: hidden;
  padding: 0;
  position: absolute;
  width: 1px;
}

.ewen-toggle-track {
  width: 50px;
  height: 24px;
  padding: 0;
  border-radius: 30px;
  background-color: #000;
}

#dark-mode-switch:checked ~ .general-theme .ewen-toggle-track {
  width: 50px;
  height: 24px;
  padding: 0;
  border-radius: 30px;
  background-color: #000;
}

.ewen-toggle-track-moon {
  position: absolute;
  width: 17px;
  height: 17px;
  left: 5px;
  top: 0px;
  bottom: 0px;
  margin-top: auto;
  margin-bottom: auto;
  line-height: 0;
  opacity: 0;
}

#dark-mode-switch:checked
  ~ .general-theme
  .ewen-toggle
  .ewen-toggle-track-moon {
  opacity: 1;
  background-color: #000 !important;
}

.ewen-toggle-track-sun {
  position: absolute;
  width: 17px;
  height: 17px;
  right: 5px;
  top: 0px;
  bottom: 0px;
  margin-top: auto;
  margin-bottom: auto;
  line-height: 0;
  opacity: 1;
  transition: opacity 0.25s ease;
}

#dark-mode-switch:checked ~ .general-theme .ewen-toggle-track-sun {
  opacity: 0;
}

.ewen-toggle-thumb {
  position: absolute;
  top: 1px;
  left: 1px;
  width: 22px;
  height: 22px;
  border-radius: 50%;
  background-color: #fafafa;
  box-sizing: border-box;
  transform: translateX(0);
}

#dark-mode-switch:checked ~ .general-theme .ewen-toggle-thumb {
  transform: translateX(26px);
  background-color: #fafafa;
}

.ewen-toggle-wrapper {
  justify-content: right;
  align-items: center;
}

The pretty button is forked from https://github.com/gaearon/overreacted.io, credits to Dan Abramov.