DEV Community

Nick Benksim
Nick Benksim

Posted on • Originally published at csscodelab.com

Using Custom Properties for Dynamic Theme Changes

The Ultimate Guide to Dynamic Theme Switching with CSS Custom Properties

Grab your coffee, pull up a chair, and let’s talk about a feature that almost every client asks for these days: theme switching. Whether it is a sleek dark mode for night owls, a high-contrast mode for accessibility, or a flamboyant "cyberpunk neon" theme for a marketing campaign, modern websites are expected to change their skin on the fly.

If you are still writing separate stylesheets or duplicating thousands of lines of code just to change a few background colors, stop right there. Today, we are going to master the art of dynamic theme switching using CSS Custom Properties. It is clean, incredibly fast, and requires only a tiny pinch of JavaScript to handle the state. Let's dive in!

How We Suffered Before (The Dark Ages of Theme Switching)

Remember how we used to build dark mode five or ten years ago? It was an absolute horror show. We had two main workarounds, and both of them made us want to question our career choices.

First, there was the Sass Multi-Class Nightmare. We would compile massive CSS files where every selector was prepended with a theme class, like .dark-theme .card .card-title. Not only did this bloat our bundle sizes, but it also made maintenance a living hell. If you want to reminisce about how we managed complex CSS before modern features took over, check out our thoughts on Why Use CSS Nesting Instead of SASS and LESS.

Second, there was the JavaScript Inline-Style Swap. We would write heavy JS scripts that queried every single element on the page and manually swapped inline styles. It caused layout thrashing, horrible lag, and created the dreaded "flash of bright light" when a dark-mode user refreshed the page. It was clumsy, slow, and totally un-semantic.

The Modern Way: Dynamic Runtime Custom Properties

Today, CSS variables do all the heavy lifting for us. Because custom properties are evaluated at runtime and inherit down the cascade, we can redefine them at the root level, and the entire page instantly updates. No layout recalculations, no stylesheet swapping, and absolutely zero CSS duplication.

By defining our design tokens at the :root level and leveraging HTML data-* attributes, we can switch entire visual schemes by changing a single attribute on the <html> element. This approach is highly modular and forms the backbone of modern CSS design systems. In fact, understanding this architecture is why Variables (CSS Variables) Are the Foundation of Scalable Design in any production-ready application.

Let's look at how we can implement a multi-theme system (Light, Dark, and Cyberpunk) with a butter-smooth transition effect.

Ready-to-Use Code Snippet

Here is a complete, lightweight, and responsive implementation of a dynamic theme switcher. Paste this into your project to see the magic happen instantly.

<!DOCTYPE html>
<html lang="en" data-theme="light">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Dynamic Theme Switcher</title>
  <style>
    /* 1. Define Design Tokens for Light Theme (Default) */
    :root {
      --bg-color: #f4f4f9;
      --card-bg: #ffffff;
      --text-color: #1a1a24;
      --accent-color: #3b82f6;
      --border-color: #e2e8f0;
      --transition-speed: 0.3s;
    }

    /* 2. Redefine Tokens for Dark Theme */
    [data-theme="dark"] {
      --bg-color: #0f172a;
      --card-bg: #1e293b;
      --text-color: #f8fafc;
      --accent-color: #10b981;
      --border-color: #334155;
    }

    /* 3. Redefine Tokens for Cyberpunk Theme */
    [data-theme="cyberpunk"] {
      --bg-color: #120136;
      --card-bg: #03001e;
      --text-color: #00f0ff;
      --accent-color: #ff007f;
      --border-color: #ff007f;
    }

    /* Global styles applying our tokens */
    body {
      background-color: var(--bg-color);
      color: var(--text-color);
      font-family: 'Segoe UI', system-ui, sans-serif;
      margin: 0;
      padding: 2rem;
      display: flex;
      justify-content: center;
      align-items: center;
      min-height: 100vh;
      /* Smooth transitions for theme switching */
      transition: background-color var(--transition-speed) ease, 
                  color var(--transition-speed) ease;
    }

    .card {
      background-color: var(--card-bg);
      border: 2px solid var(--border-color);
      border-radius: 12px;
      padding: 2rem;
      max-width: 400px;
      box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
      transition: background-color var(--transition-speed) ease, 
                  border-color var(--transition-speed) ease;
    }

    h1 {
      margin-top: 0;
      color: var(--accent-color);
    }

    .btn-group {
      display: flex;
      gap: 0.5rem;
      margin-top: 1.5rem;
    }

    button {
      background: var(--accent-color);
      color: var(--bg-color);
      border: none;
      padding: 0.6rem 1.2rem;
      border-radius: 6px;
      cursor: pointer;
      font-weight: bold;
      transition: opacity 0.2s ease;
    }

    button:hover {
      opacity: 0.9;
    }
  </style>
</head>
<body>

  <div class="card">
    <h1>Theme Switcher</h1>
    <p>Experience modern theme switching using CSS Custom Properties. Super light, incredibly fast, and clean.</p>
    
    <div class="btn-group">
      <button onclick="setTheme('light')">Light</button>
      <button onclick="setTheme('dark')">Dark</button>
      <button onclick="setTheme('cyberpunk')">Neon</button>
    </div>
  </div>

  <script>
    function setTheme(themeName) {
      document.documentElement.setAttribute('data-theme', themeName);
      localStorage.setItem('theme', themeName);
    }

    // Restore user preference on load
    const savedTheme = localStorage.getItem('theme') || 'light';
    setTheme(savedTheme);
  </script>

</body>
</html>

Common Beginner Mistakes to Avoid

Even though custom properties make theme switching a breeze, there are a couple of pitfalls that trip up developers new to this workflow:

  • The "Flash of Light Theme" (FOUC): If you apply the theme inside your main JS bundle after the HTML has finished rendering, your dark-mode users will get a blinding flash of light theme on every page load. Always place the tiny theme-checking script inline at the very top of your <head> or immediately after the opening <html> tag so it executes before the page renders.
  • Animating too many properties: While adding transition: all 0.3s to your elements is tempting to animate color changes, it is terrible for performance. It will force the browser to recalculate layouts, rendering paths, and shadows on every hover or scroll. Only transition specific properties like background-color, color, and border-color.
  • Forgetting system preferences: Don't force users to click a toggle if their operating system already tells you what they prefer. Always check (prefers-color-scheme: dark) using media queries or JavaScript to set your default fallback state.

That is all there is to it! With just a handful of design tokens and a data-attribute switch, you can scale your styling system infinitely without writing duplicate stylesheets ever again.

πŸ”₯ We publish more advanced CSS tricks, ready-to-use snippets, and tutorials in our Telegram channel. Subscribe so you don't miss out!

Top comments (0)