diff --git a/config/data/tag-aliases.js b/config/data/tag-aliases.js
index de29967a..0403716b 100644
--- a/config/data/tag-aliases.js
+++ b/config/data/tag-aliases.js
@@ -30,4 +30,5 @@ export default {
'tattoos': '#Tattoos',
tech: '#Tech',
technology: '#Tech',
+ 'web components': '#WebComponents'
}
diff --git a/package.json b/package.json
index d480a820..f5f3793f 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "coryd.dev",
- "version": "6.6.1",
+ "version": "6.6.2",
"description": "The source for my personal site. Built using 11ty and hosted on Netlify.",
"type": "module",
"scripts": {
diff --git a/src/assets/styles/components/theme-toggle.css b/src/assets/styles/components/theme-toggle.css
index fb248df5..354ccab8 100644
--- a/src/assets/styles/components/theme-toggle.css
+++ b/src/assets/styles/components/theme-toggle.css
@@ -1,4 +1,3 @@
-.theme__toggle,
.theme__toggle svg {
cursor: pointer;
}
diff --git a/src/posts/2024/building-a-theme-toggle-web-component.md b/src/posts/2024/building-a-theme-toggle-web-component.md
new file mode 100644
index 00000000..42fd13c8
--- /dev/null
+++ b/src/posts/2024/building-a-theme-toggle-web-component.md
@@ -0,0 +1,138 @@
+---
+date: '2024-02-27'
+title: 'Building a theme toggle web component'
+description: "I (very recently!) added a theme toggle to my site, right up there in the menu. It's a shiny sun or a purple moon depending on your preference. It was a liquid template with some JavaScript sprinkled in. I turned that into a web component."
+tags: ['CSS', 'javascript', 'web components', 'Eleventy']
+---
+I (very recently!) added a theme toggle to my site, right up there in the menu. It's a shiny sun or a purple moon depending on your preference. It was a liquid template with some JavaScript sprinkled in. I turned that into a web component.
+
+Much like my `now-playing` component, I start out by registering the component template:
+
+```javascript
+const themeToggleTemplate = document.createElement('template')
+
+themeToggleTemplate.innerHTML = `
+
+`
+
+themeToggleTemplate.id = "theme-toggle-template"
+
+if (!document.getElementById(themeToggleTemplate.id)) document.body.appendChild(themeToggleTemplate)
+```
+
+The template is generated with empty `light` and `dark` spans so that the clickable element (an icon or text) can be defined when it's implemented. Next, we register the tag name and create our `ThemeToggle` class:
+
+```javascript
+class ThemeToggle extends HTMLElement {
+ static register(tagName) {
+ if ("customElements" in window) customElements.define(tagName || "theme-toggle", ThemeToggle)
+ }
+```
+
+My verbose, `connectedCallback` handles appending the template and appending theme behavior. The button included in the template is selected and cached and a `setTheme` method is defined, allowing the logic contained therein to be reused on load (with a simple boolean argument) and when the `theme-toggle` is clicked. It checks where the user `prefersDarkScheme`, sets the cached `currentTheme` accordingly and adds the appropriate class to the document body. The event listener attached to the `theme-toggle` toggles the body class and runs the `setTheme` function, skipping the logic that runs on load.
+
+```javascript
+async connectedCallback() {
+ this.append(this.template)
+ const btn = this.querySelector('.theme__toggle')
+ const setTheme = (isOnLoad) => {
+ const prefersDarkScheme = window.matchMedia('(prefers-color-scheme: dark)').matches
+ const currentTheme = localStorage?.getItem('theme')
+ let theme
+ if (!currentTheme) localStorage?.setItem('theme', (prefersDarkScheme ? 'dark' : 'light'))
+ if (isOnLoad) {
+ if (currentTheme === 'dark') {
+ document.body.classList.add('theme__dark')
+ } else if (currentTheme === 'light') {
+ document.body.classList.add('theme__light')
+ } else if (prefersDarkScheme) {
+ document.body.classList.add('theme__dark')
+ } else if (!prefersDarkScheme) {
+ document.body.classList.add('theme__light')
+ }
+ }
+ if (prefersDarkScheme) {
+ theme = document.body.classList.contains('theme__light') ? 'light' : 'dark'
+ } else {
+ theme = document.body.classList.contains('theme__dark') ? 'dark' : 'light'
+ }
+ localStorage?.setItem('theme', theme)
+ }
+
+ setTheme(true);
+
+ btn.addEventListener('click', () => {
+ document.body.classList.toggle('theme__light')
+ document.body.classList.toggle('theme__dark')
+ setTheme()
+ })
+ }
+```
+
+The CSS for this is straightforward and contains a few vars specific to my implementation (related to icon color and SVG `stroke-width`):
+
+```css
+.theme__toggle svg {
+ cursor: pointer;
+}
+
+.theme__toggle:hover,
+.theme__toggle svg:hover {
+ stroke-width: var(--stroke-width-bold);
+}
+
+.theme__toggle > .light svg { stroke: var(--sun) !important; }
+.theme__toggle > .dark svg { stroke: var(--moon) !important; }
+
+.theme__toggle > .light ,
+.theme__toggle > .dark {
+ display: none;
+}
+
+.theme__dark .theme__toggle > .light {
+ display: inline;
+}
+
+.theme__dark .theme__toggle > .dark {
+ display: none;
+}
+
+.theme__light .theme__toggle > .light {
+ display: none;
+}
+
+.theme__light .theme__toggle > .dark {
+ display: inline;
+}
+```
+
+The final template that leverages the component looks like this:
+{% raw %}
+```liquid
+
+{% capture css %}
+ {% render "../../../assets/styles/components/theme-toggle.css" %}
+{% endcapture %}
+
+
+