Skip to main content

Styling

CSS Post-processors and UI Libraries to make components.

Money Tracker

This application uses Nx and Angular with Tailwind CSS and Sass to style components, although Angular works with any tool that outputs CSS.

Tailwind

Tailwind CSS dependency is included for the whole project:

package.json
{
"dependencies": {
"tailwindcss": "^3.4.12"
}
}

The main configuration file is in the main web application source directory in its default form, see Nx docs:

apps/money-tracker-ui/tailwind.config.js
const { createGlobPatternsForDependencies } = require('@nrwl/angular/tailwind');
const { join } = require('path');

module.exports = {
content: [
join(__dirname, 'src/**/!(*.stories|*.spec).{ts,html}'),
...createGlobPatternsForDependencies(__dirname),
],
theme: {
extend: {},
},
plugins: [],
};

Sass

Sass aka Syntactically Awesome Style Sheets dependency is included for the whole project as well:

package.json
{
"dependencies": {
"sass": "^1.79.3",
"sass-loader": "^13.3.3"
}
}

Tailwind CSS is imported in styles.sass:

apps/money-tracker-ui/src/styles.sass
@tailwind base
@tailwind components
@tailwind utilities

So both preprocessors are available in the project and in its components.

Material UI Theme

The Material Indigo theme is included in project.json with other CSS files, for example, with styles.css and styles.sass:

apps/money-tracker-ui/project.json
{
"styles": [
"./node_modules/@angular/material/prebuilt-themes/indigo-pink.css",
"apps/money-tracker-ui/src/styles.css",
"apps/money-tracker-ui/src/styles.sass"
]
}

View Encapsulation

By default, Angular uses emulated encapsulation so that a component's styles only apply to elements defined in that component's template.

More information: Angular Styling Components

Layout

Since it is an Angular-based application, it uses Angular Material for layouts, themes and ready-to-use components.

For the layout in particular see mat-grid-system.

Pomodoro

The application is using React and Webpack; therefore, the latter takes care of the Leaner Style Sheets aka Less configuration. Less allows variables, mixins, nested classes, operations, functions and many more if compared to conventional CSS.

Less

We have CSS and Less files configured for loading and processing in Webpack, but it would be possible to configure Sass or any other pre-processor in the same manner:

cfg/webpack.server.config.js
{ 
module: {
rules: [
{
test: /\.less$/,
use: [
{
loader: "css-loader",
options: {
modules: {
mode: "local",
localIdentName: "[name]__[local]--[hash:base64:5]",
},
onlyLocals: true
}
},
"less-loader"
]
}]
}
}

The configuration uses css-loader library which is enlisted in dev dependencies of package.json file:

package.json
{
"devDependencies": {
"css-loader": "^3.6.0"
}
}

CSS Modules

Components styles encapsulation is done with the help of CSS modules in the Webpack excerpt above, see Webpack CSS-Loader for more options:

cfg/webpack.server.config.js
{
options: {
modules: {
mode: "local"
localIdentName: "[name]__[local]--[hash:base64:5]"
}
onlyLocals: true
}
}

It is necessary to declare modules to TypeScript as well since it doesn't know what to do with *.css (or *.less) files out of the box:

src/types/global.d.ts
declare module '*.less' {
const styles: {[key: string]: string};
export = styles;
}

declare module "*.css" {
const styles: { [key: string]: string };
export = styles;
}
Can be improved

CSS modules can be typified, see here

CSS Variables and Themes

Pomodoro extensively uses CSS Variables in the global CSS file, using them to create switchable themes, which override the variables with values from the theme palettes:

src/main.global.css
:root {
--OFF: ;
--ON: initial;

--black: #333333;
--whitelightness: 100%;
--white: hsl(0, 0%, var(--whitelightness));
--greyF3: #f3f3f3;
--grey33: #333;
--greyEC: #ececec;
--greyCC: #cccccc;
--greyF4: hsl(0, 0%, calc(var(--whitelightness) - 4%));
--greyDE: #DEDEDE;
--greyEE: #EEEEEE;
}

For instance, a theme could be put down like below, note how var() works when it chooses a correct colour for the active theme:

@media (prefers-color-scheme: light) {
:root {
--background: var(--light, var(--white)) var(--dark, var(--grey33)) var(--system, var(--white));
--backgroundSecondary: var(--light, var(--greyF4)) var(--dark, var(--grey66)) var(--system, var(--greyF4));
--foreground: var(--light, var(--grey33)) var(--dark, var(--white)) var(--system, var(--grey33));
--accent: var(--light, var(--red)) var(--dark, var(--brightBlue)) var(--system, var(--red));
}
}

It then can be switched to dark or back to light using Redux store variables:

export type RootState = {
theme: ETheme;
}

The actual switch in the DOM tree happens in one of the wrapper components - <Layout />:

src/shared/components/Layout/Layout.tsx
useLayoutEffect(() => {
if (theme === ETheme.LIGHT) {
document.documentElement.classList.remove(themeDark);
document.documentElement.classList.add(themeLight);
document.documentElement.classList.remove(themeSystem);
} else if (theme === ETheme.DARK) {
document.documentElement.classList.remove(themeLight);
document.documentElement.classList.add(themeDark);
document.documentElement.classList.remove(themeSystem);
} else {
document.documentElement.classList.remove(themeLight);
document.documentElement.classList.remove(themeDark);
document.documentElement.classList.add(themeSystem);
}
}, [theme]);

Layout

Pomodoro is using CSS Flexbox, which also has nice documentation here.

The typical CSS class then would start with display property and the setting of the flow:

  .title {
display: flex;
flex-flow: row nowrap;
align-items: center;
}

Cosmic

Cosmic is using React and Vite with pure CSS and CSS modules to style the components.

Tailwind

Tailwind is installed as a PostCSS plugin as it is the most seamless way to integrate it with build tools, see more. Vite understands PostCSS config automatically if present:

package.json
{
"devDependencies": {
"autoprefixer": "^10.4.20",
"postcss": "^8.4.49",
"tailwindcss": "^3.4.16"
}
}

Added to PostCSS configuration:

postcss.config.js
export default {
plugins: {
tailwindcss: {},
autoprefixer: {},
},
}

Template paths are configured for Tailwind CSS:

tailwind.config.js
/** @type {import('tailwindcss').Config} */
export default {
content: [
"./index.html",
"./src/**/*.{js,ts,jsx,tsx}",
],
theme: {
extend: {},
},
plugins: [],
}

And finally, directives are included into the main CSS file for the project:

src/index.css
@tailwind base;
@tailwind components;
@tailwind utilities;

CSS Modules

Again automatically, Vite understands any CSS file ending with .module.css as a CSS modules file.

Layout

Although CSS Flexbox is also configured for this application, it makes a much heavier use of Tailwind CSS classes for layout. In fact, Tailwind also offers CSS Flexbox grid classes which also can be used in isolation for every component.