DEV Community

Cover image for SvelteKit Theme Switch
Nico Bachner
Nico Bachner

Posted on • Updated on • Originally published at nicobachner.com

SvelteKit Theme Switch

When adding dark mode to my new SvelteKit project, I ran into a few issues when creating the theme switch. In this guide, I would like to share the solution I came up with.

Before we begin, I'd like to note that this guide uses TypeScript. TypeScript (TS) is JavaScript with types1, so if you are using regular JavaScript (JS), you can skip the type definitions.

Types

Note: this step is not necessary for those using JS instead of TS, and is also optional (but recommended) for those using TS

The first thing we'll do is to define our themes. We can do this in the global type definitions (src/global.d.ts). That way, we'll have access to the types throughout our project.

// src/global.d.ts
type Theme = 'system' | 'light' | 'dark'
Enter fullscreen mode Exit fullscreen mode

What we're doing here is declaring a global type called Theme, which we can access from anywhere in our project. This means that if we declare a variables type to be Theme, then we can only assign the values 'system', 'light', or 'dark' to it.

Apart from 'system', you can choose your theme values freely. You're also not limited to only two, so experiment away!

The 'system' value here is important. We want to greet the user with their preferred theme when they first visit the site. Therefore, we want the theme to correspond to their operating system's color scheme by default.

Svelte Store

Now that we've got type definitions out of the way, we can move on to the heart of the theme switch: the theme store.

The theme store is a Svelte Store. To create it, we use the writable function provided by Svelte.

// src/lib/stores.ts
import { writable } from 'svelte/store'

const theme = writable('system')

export { theme }
Enter fullscreen mode Exit fullscreen mode

Here, we're creating a Svelte Store called theme and assigning it the default value of 'system'. Again, it is important that 'system' is the default so that we respect the user's preferences.

Theme Switch Component

We can now use the Svelte Store we created in our theme switch component.

<!-- src/lib/components/ThemeSwitch.svelte -->
<script lang="ts">
  import { theme } from '$lib/stores'
</script>

<select bind:value="{$theme}">
  <option value="system">System</option>
  <option value="light">Light</option>
  <option value="dark">Dark</option>
</select>
Enter fullscreen mode Exit fullscreen mode

There's a lot going on here, so a quick walkthrough is in order.

We first import theme from '$lib/stores'. $lib/stores is a path alias for src/lib/stores.svelte, the file in which we created our theme Svelte Store.

We now want to modify the value of theme. We could do this by calling theme.set() (more info in the writable stores documentation). However, there is an easier way: using auto-subscriptions.

Since $theme is mutable2, we use the Svelte binding bind:value to get theme to track the changes to the value of the selected option. The browser does most of the heavy lifting in this case, since all we need to do is read the value attribute.

Style Switcher

We now have a Svelte Store that stores the theme value, and a theme switch component that updates the theme value. All that remains is the functionality for changing the theme based on the theme value.

The way I went about this is swapping stylesheets in the head of the generated document.

<!-- src/routes/__layout.svelte -->
<script lang="ts">
  import { theme } from '$lib/stores'
</script>

<svelte:head>
  <meta name="color-scheme" content={$theme == 'system' ? 'light dark' :
  $theme}/> <link rel="stylesheet" href={`/theme/${$theme}.css`} />
</svelte:head>

<slot />
Enter fullscreen mode Exit fullscreen mode

Here, we are dynamically loading a CSS stylesheet based on the current theme value. For example, on page load, the previous code will generate the following:

<head>
  <meta name="color-scheme" content="light dark" />
  <link rel="stylesheet" href="/theme/system.css" />
</head>
Enter fullscreen mode Exit fullscreen mode

And if the user then changes the theme to 'light', the head changes accordingly:

<head>
  <meta name="color-scheme" content="light dark" />
  <link rel="stylesheet" href="/theme/light.css" />
</head>
Enter fullscreen mode Exit fullscreen mode

Theme Styles

The only thing that remains is to define the styles of our project. We can do this anywhere in the static/ directory, as long as we remember to adjust the path in the stylesheet link accordingly.

Structure

If we follow the path convention I set up, we get the following structure:

static
└── theme
    ├── system.css
    ├── light.css
    └── dark.css
Enter fullscreen mode Exit fullscreen mode

Example

In light.css and dark.css (or whatever you choose to call you themes), we style our project accordingly. An example3:

/* light.css */
:root {
  --color-lightest: hsl(0deg, 0%, 100%);
  --color-lighter: hsl(0deg, 0%, 80%);
  --color-light: hsl(0deg, 0%, 60%);
  --color-strong: hsl(0deg, 0%, 40%);
  --color-stronger: hsl(0deg, 0%, 20%);
  --color-strongest: hsl(0deg, 0%, 0%);
}
Enter fullscreen mode Exit fullscreen mode

System Preferences

While light.css and dark.css are straightforward, the file system.css requires more attention. This is because we need to think about the user's system preferences. While the prefers-color-scheme media query makes accessing the user's preference a straightforward process, we need to keep in mind that the browser provides only two predefined choices, light and dark. Hence we need to style accordingly:

/* system.css */
@media (prefers-color-scheme: light) {
  :root {
    --color-lightest: hsl(0deg, 0%, 100%);
    --color-lighter: hsl(0deg, 0%, 80%);
    --color-light: hsl(0deg, 0%, 60%);
    --color-strong: hsl(0deg, 0%, 40%);
    --color-stronger: hsl(0deg, 0%, 20%);
    --color-strongest: hsl(0deg, 0%, 0%);
  }
}

@media (prefers-color-scheme: dark) {
  :root {
    --color-lightest: hsl(0deg, 0%, 0%);
    --color-lighter: hsl(0deg, 0%, 20%);
    --color-light: hsl(0deg, 0%, 40%);
    --color-strong: hsl(0deg, 0%, 60%);
    --color-stronger: hsl(0deg, 0%, 80%);
    --color-strongest: hsl(0deg, 0%, 100%);
  }
}
Enter fullscreen mode Exit fullscreen mode

Conclusion

That's it! You now have a working theme switch.

If you want to further improve your theme switch, you could store the selected value in localStorage. Then, when the user selects a particular theme, the same theme will also be loaded the next time they visit the page.


  1. Types in TypeScript explicitly declare the type of a variable. TypeScript also supports the definition of custom types, called type aliases. These can be manipulated similarly to JavaScript variables and imported from external files. 

  2. If a value is mutable, that means it can be changed by assigning a new value to it. In JavaScript, for example, let and var create mutable variables, whereas const creates immutable ones. 

  3. This guide uses CSS Custom Properties (a.k.a. CSS Variables) for theming, but the solution works with any theming method, as long as the styles are defined inside the files we are working with. 

Top comments (2)

Collapse
 
ambrt profile image
Rik • Edited

Did anyone solved the fact that Vite doesn't autoreload on changes in Css files (HMR)?

I followed this article and that's the only problem i encountered.
The problems happens because Vite doesn't bundle css files hence it doesn't watch them.

The workaround i tired is to make conditional import but it left basically mix of all themes
on website.

Collapse
 
nico_bachner profile image
Nico Bachner

The reason for this might be because the styles are located in the static/ directory. I think that Vite assumes all the files there are in fact static, as the directory name suggests.
I have tried many ways to include the styles in the src/ directory, but none seem to be as simple as my current solution.
If you do find a way to include the styles in src/ in an elegant manner, please do let me know as I would love to improve my solution in all ways possible.