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:
{
"dependencies": {
"tailwindcss": "^3.4.12"
}
}
The main configuration file is in the main web application source directory in its default form, see Nx docs:
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:
{
"dependencies": {
"sass": "^1.79.3",
"sass-loader": "^13.3.3"
}
}
Tailwind CSS is imported in 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
:
{
"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:
{
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:
{
"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:
{
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:
declare module '*.less' {
const styles: {[key: string]: string};
export = styles;
}
declare module "*.css" {
const styles: { [key: string]: string };
export = styles;
}
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:
: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 />
:
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:
{
"devDependencies": {
"autoprefixer": "^10.4.20",
"postcss": "^8.4.49",
"tailwindcss": "^3.4.16"
}
}
Added to PostCSS configuration:
export default {
plugins: {
tailwindcss: {},
autoprefixer: {},
},
}
Template paths are configured for Tailwind CSS:
/** @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:
@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.