Skip to content

Commit

Permalink
Add build check for undeclared css variables
Browse files Browse the repository at this point in the history
  • Loading branch information
xypnox committed Nov 21, 2023
1 parent a54969c commit e0a5b6a
Show file tree
Hide file tree
Showing 16 changed files with 301 additions and 104 deletions.
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -19,3 +19,5 @@ pnpm-debug.log*

# macOS-specific files
.DS_Store

cssVariables.json
5 changes: 5 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,11 @@ All commands are run from the root of the project, from a terminal:

---

# Checks

- `css variables`
We have a postcss check that throws warnings when a css variable is encountered that has either not been declared in the style tag or is not present in the global theme variables.

# Dependencies

- solid-js
Expand Down
34 changes: 34 additions & 0 deletions checkCssVars.cjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
const postcss = require('postcss');

const cssVariables = require('./cssVariables.json');

console.log({ cssVariables })

const plugin = postcss.plugin('postcss-validate-vars', () => {
return (root, result) => {
// Check for undefined variables
const localCssVariables = [];

root.walkDecls(decl => {
if (decl.prop.startsWith('--')) {
localCssVariables.push(decl.prop);
}
});

root.walkDecls(decl => {
if (decl.value.includes('var(')) {
const matches = decl.value.match(/var\(([^)]+)\)/);
if (matches) {
const variable = matches[1];
if (!cssVariables.includes(variable) && !localCssVariables.includes(variable)) {
const file = decl.source.input.file || 'unknown';
const line = decl.source.start.line;
result.warn(`${file} - ${line} : Undefined variable ${variable} used in ${decl.toString()}`);
}
}
}
});
};
});

module.exports = plugin;
9 changes: 6 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
{
"name": "com",
"type": "module",
"version": "0.0.1",
"scripts": {
"dev": "astro dev",
"dev": "pnpm run convert && pnpm run astro:dev",
"start": "astro dev",
"build": "astro build",
"convert": "tsc -p tsconfig-dev.json && node dist/buildChecks/cssJsonExport.js",
"build": "pnpm run convert && pnpm run astro:build",
"astro:build": "astro build",
"astro:dev": "astro dev",
"preview": "astro preview",
"astro": "astro"
},
Expand All @@ -24,6 +26,7 @@
"solid-styled-components": "^0.28.5"
},
"devDependencies": {
"@types/node": "^20.9.3",
"typescript": "^5.2.2"
}
}
30 changes: 21 additions & 9 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions postcss.config.cjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
module.exports = {
plugins: [
require('./checkCssVars.cjs')
],
};
5 changes: 5 additions & 0 deletions src/buildChecks/cssJsonExport.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import { themeCssVars } from '../theme'

import { writeFileSync } from 'fs';

writeFileSync('./cssVariables.json', JSON.stringify(Object.keys(themeCssVars).map(k => `--${k}`)));
1 change: 1 addition & 0 deletions src/components/nav/Navbar.astro
Original file line number Diff line number Diff line change
Expand Up @@ -134,4 +134,5 @@ import '../../scripts/menu'
font-size: var(--font-size-lg);
box-shadow: 0px 8px 16px #00000080;
}
}
</style>
176 changes: 176 additions & 0 deletions src/content/blog/dark-modes/index.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,176 @@
---
title: "Dark Modes"
description: "A discussion about the dark modes"
date: 2023-11-22

# coverImage: "./test-social3.webp"
# coverAlt: "Some text for describing the cover image"
categories: ["sample"]
tags: ["design", "dev"]

hidden: true
---

The difficulties faced in making a dark mode.

I have a multitude of thoughts about theming, styling, modes etc, this post is about the difficulties usually encountered in implementing a dark mode.

# Is it needed?

Dark mode is cool and all but does it make sense in your situation?

The point of having modes and the ability to switch between them is directly dependent on the content that is being presented. Not everything needs to be switchable, between light and the dark.

Whether one should have a dark mode depends on what one wants to show, i.e. it depends on the content being shown.

The best place for a dark mode are commonly used applications that serve a variety of people, who may prefer one color scheme over the other. One such group is the developers.

Applications are opened more frequently and people spend more time with their interface than normal websites. Moreover, the usability trumps any branding specific design implementations. Websites, specifically websites of companies, brands etc are more targeted towards conveying a distinct and unique identity, where uniqueness can sometimes trump usability (in the sense of not following the standard layouts).

If you are a company/startup targeting a developer userbase or similar people that might prefer the dark mode, think whether it is necessary to have a light mode. How about just having the dark mode?

Moreover, there are now extensions that will convert a well designed light mode to a dark mode. And the users that prefer dark mode would be usually using such extensions.

---

The reasons to implement a dark mode can be:

- The interface is complex or has to be viewed frequently (ex: Editor, Reader, Dashboard etc.)
- The people using the interface spend a long time on it and may depend on it for their work.
- ~~There is not much to do and there is a lot of free time at hand~~

---

# The culprits

Once you have reached the decision that a dark mode is indeed what you want, you can take a step back and think about what you are converting dark mode into. That is, the content.

One of the most common methods of converting a light mode to a dark mode is to simply switch the background and text colors. While this serves for basic text only content, there are various elements that don't directly translate into a usable dark mode.

## The code block

What color mode should the code block be in?

Should light mode and dark mode both have different color schemes? What if your light mode already has a dark code highlighting? Does this mean we add a light code highlighting theme for dark mode?

## Contrasty designer

There are designs for software (websites, products) that use different modes (dark/light) in the same interface to distinguish a sidebar/navigation/sections of a page.

Ex: Sidebar with black back and white text beside a product dashboard, a website with dark and light sections in the landing page.

While this is effective to some degree. It is neither dark nor light. And the contrast is needless in most cases. Why would you use a different mode when a border or a slightly different shade of background could have achieved the same? Is it because of a "brand identity"? But even then, what kind of brand is both dark and light and that at the same time?

## Shady Shadows

Shadows become extremely hard to see in dark modes. More so if the background is completely black.

## Only prefers color scheme

1. Perhaps if you have managed to make the dark theme, tying it only to the prefers color scheme option feels kinda bad. Having the ability is cool, but the inability for the user to select the theme they want instead of forcing them to switch the theme for the entire system is kinda bad.


## The ugly gray

Yes I call all dark mode themes that are monochrome and use a grey background ugly. And I will die on this hill. Especially the ones of the old material design (pre v3) days.

https://m2.material.io/design/color/dark-theme.html#properties

https://m3.material.io/styles/color/choosing-a-scheme

And I think these gray themes are what have given the dark mode the aura of unsaturated monochromatic soulless interfaces. As if colors are meant to be reserved for the light mode.

---

# Some good things

## Colors that were lost

Some colors look bad on white background if used as text. Yellow is a prime example. But in the world of dark mode, yellow literally shines!

// Add example yellow calry website

## CSS variables for theming

When I dwelled in the world of react and styled components, I used their theme providers which functioned as:

```tsx
const Button = styled.button`
/* Color the border and text with theme.main */
color: ${props => props.theme.main};
border: 2px solid ${props => props.theme.main};
`;

// Define what props.theme will look like
const theme = {
main: "mediumseagreen"
};

component(
<div>
<ThemeProvider theme={theme}>
<Button>Themed</Button>
</ThemeProvider>
</div>
);
```

But when the themes changed, all the css was recalculated and generated again, which was suboptimal. Moreover this strategy stops working once you move out of the react ecosystem. The better way of theming is to use css variables.

We define the variables ones for each theme:

```css
:root {
--primary: mediumseagreen;
}
.dark {
--primary: skyblue;
}
```

And then we use those variables in the css styles:

```css
button {
color: var(--primary);
}
```

We can add simple code to add a .dark class to our body/html element to toggle between the themes. None of the styles are recomputed, and this can be used across frameworks or even without any framework. Moreover you can always override the theme variables in smaller contexts:

```html
<div class="special">
<button>Hello</button>
</div>

<style>
.special {
--primary: red;
}
</style>
```

## A step further via objects

But these do not have type safety. When you write css and later refactor any of the theme variables, the older ones are left dangling. While fallbacks are nice, refactoring still requires manually scouring for the older ones are rewriting them again. Moreover these errors can not be found in the build step and will easily mess up your stylesheets.

I have found a promising solution. Which has type safety when using css-in-js. But the basic structure remains same.

The way I organize the themes is to declare the variables as a js object.

```ts
const theme = {
}
```

---

# Resources


- https://tonsky.me/
Cool dark mode with round light where the mouse is.
Don't know about accessibility but emulates the torchlight under a quilt nicely.



Loading

0 comments on commit e0a5b6a

Please sign in to comment.