Bootstrap, in the current latest version, v5.3.0, has started supporting color modes. This means that in your Bootstrap layouts you can choose from its default light or dark mode, or even create a new one.
In the Bootstrap docs, they have implemented a dropdown color mode toggle that can work as a starting point for anyone who wants something similar. But there’s no actual switcher component included in Bootstrap.
data:image/s3,"s3://crabby-images/22871/228716accc35efba798f72370c9f353f37af3002" alt="Bootstrap's color mode toggler"
data:image/s3,"s3://crabby-images/31819/318196b2a27963b18950777352c1e609916fb210" alt="Bootstrap's color mode toggler"
data:image/s3,"s3://crabby-images/81757/81757b013f011d8e200710b3589bbd24759f626a" alt="Bootstrap's color mode toggler"
What if we want to build our own custom color mode switcher while respecting Bootstrap’s built-in dark mode styles? This is absolutely possible. Recently, we built a theme switcher, so let’s make it work with Bootstrap.
What we built in the past
As a reminder, here’s the light/dark toggle switch component that we built in the previous tutorial:
Bootstrap color mode switcher
To turn it into a Bootstrap-based toggle, we’ll make the following modifications:
- First, we’ll apply the dark mode using the
data-bs-theme="dark"
HTML attribute instead of thetheme-dark
class.
- Next, we’ll remove all our dark mode CSS variables as Bootstrap has built-in dark mode styles.
1 |
/*No need for them*/
|
2 |
|
3 |
:root { |
4 |
--white: #fff; |
5 |
--black: black; |
6 |
--text-color: var(--black); |
7 |
--bg-color: var(--white); |
8 |
}
|
9 |
|
10 |
.theme-dark { |
11 |
color-scheme: dark; |
12 |
--text-color: var(--white); |
13 |
--bg-color: var(--black); |
14 |
}
|
- Finally, to avoid conflicts with the previous demo, we’ll add the
bs
prefix before all our local storage items. For example, we’ll replace thedark-mode
key with thebs-dark-mode
one.
Updated JavaScript
Here’s the new required JavaScript code:
1 |
const html = document.documentElement; |
2 |
const switches = document.querySelector(".switches"); |
3 |
const inputs = switches.querySelectorAll("input"); |
4 |
|
5 |
if (localStorage.getItem("bs-dark-mode")) { |
6 |
html.setAttribute("data-bs-theme", "dark"); |
7 |
}
|
8 |
|
9 |
if (localStorage.getItem("bs-selected-radio")) { |
10 |
switches.querySelector( |
11 |
`#${localStorage.getItem("bs-selected-radio")}` |
12 |
).checked = "true"; |
13 |
}
|
14 |
|
15 |
const setTheme = (theme) => { |
16 |
if (theme === "dark") { |
17 |
html.setAttribute("data-bs-theme", "dark"); |
18 |
localStorage.setItem("bs-dark-mode", "true"); |
19 |
} else { |
20 |
html.removeAttribute("data-bs-theme"); |
21 |
localStorage.removeItem("bs-dark-mode"); |
22 |
}
|
23 |
};
|
24 |
|
25 |
const handleMediaChange = (e) => { |
26 |
if (switches.querySelector('[type="radio"]:checked').id === "auto") { |
27 |
setTheme(e.matches ? "dark" : "light"); |
28 |
}
|
29 |
};
|
30 |
|
31 |
const handleInputChange = (e) => { |
32 |
const themeMode = e.target.id; |
33 |
if ( |
34 |
themeMode === "dark" || |
35 |
(themeMode === "auto" && |
36 |
window.matchMedia("(prefers-color-scheme: dark)").matches) |
37 |
) { |
38 |
setTheme("dark"); |
39 |
} else { |
40 |
setTheme("light"); |
41 |
}
|
42 |
localStorage.setItem("bs-selected-radio", themeMode); |
43 |
};
|
44 |
|
45 |
window
|
46 |
.matchMedia("(prefers-color-scheme: dark)") |
47 |
.addEventListener("change", handleMediaChange); |
48 |
|
49 |
inputs.forEach((input) => input.addEventListener("input", handleInputChange)); |
And the resulting demo:
Override Bootstrap’s dark mode variables
The Bootstrap built-in dark mode is great, but it would be nicer if we knew how to customize these styles.
For this demonstration, I created a new Bootstrap project on GitHub. This time, I installed and included it via npm. Also, I used the Prepros app for easier compilation of Sass files.
As another reminder, a few years ago, I went through how to customize Bootstrap’s Sass files. The concept remains the same.
By default, Bootstrap stores all its dark mode-specific Sass variables in the _variables-dark.scss
file.
data:image/s3,"s3://crabby-images/a44c3/a44c3243dca27eb231f79c6c45df33c1ef34dca7" alt="Bootstrap's dark mode variables"
data:image/s3,"s3://crabby-images/dab7f/dab7fa9a4c966175f7f2f1931afa7c6857962491" alt="Bootstrap's dark mode variables"
data:image/s3,"s3://crabby-images/1c6e0/1c6e0cbaa3b26f91144aa049a0c7f64874d8792a" alt="Bootstrap's dark mode variables"
In our case, let’s customize the default foreground and background page colors that Bootstrap uses in dark mode. To do so, we’ll navigate to our custom main.scss
Sass file, target the associated variables, and modify their values, like this:
data:image/s3,"s3://crabby-images/1d2bc/1d2bc0967ef54f507c49c359c12c28e1d9e189cc" alt="The project structure"
data:image/s3,"s3://crabby-images/c98a9/c98a94297f7f4a3e3df6823ac7c7ba52f02967e5" alt="The project structure"
data:image/s3,"s3://crabby-images/e528b/e528b8a4a1d3cc68f8e09ab5ebdac199ab28e50e" alt="The project structure"
data:image/s3,"s3://crabby-images/7e83c/7e83c13602a423172aeb6877c4a68205dd85645d" alt="Overriding the target Sass variables"
data:image/s3,"s3://crabby-images/34c88/34c88f2e445eb2693ecaa9755db995aa248205e4" alt="Overriding the target Sass variables"
data:image/s3,"s3://crabby-images/61551/61551c5c2231d0785237447006abf2677ebf0e15" alt="Overriding the target Sass variables"
This results in an appearance like this one:
data:image/s3,"s3://crabby-images/b13ca/b13cae4431b24c65166165e90a700b379028a326" alt=""
data:image/s3,"s3://crabby-images/6c812/6c812237a6dbb22b91d982c771fe0d4814e6768f" alt=""
data:image/s3,"s3://crabby-images/a60bb/a60bba6dce5ec53353e3ccec524d0573f2faa826" alt=""
We can also use Bootstrap’s color-mode
mixin to apply additional styles in dark mode like this:
1 |
@include color-mode(dark) { |
2 |
body { |
3 |
border: 1px solid red; |
4 |
}
|
5 |
}
|
This outputs to:
1 |
[data-bs-theme=dark] body { |
2 |
border: 1px solid red; |
3 |
}
|
Conclusion
That’s all, folks! Shortly, Bootstrap might provide an official toggle switch component for overriding color schemes. But for now, you can take advantage of the one we built here!
As always, thanks a lot for reading!