Skip to main content

Basic CSS

The basics you might not know

CSS Variables

CSS variables cascade down like any other property, meaning all children have access to the same CSS variables their parent has set on it.

When referencing the value of a CSS variable using var(), you can also provide a default value if the variable doesn't exist, like var(--name, #ff0000).

:root {
--light-gray: #ccc;
}

p {
color: var(--light-grey, #f0f0f0); /* No --light-grey, so #f0f0f0 is
used as a fallback value */
}

Emmet Crash Course

Here is how to use emmet to quickly scaffold out HTML:

  • h1.className : the class attribute shortcut. Creates h1 tag with the specified class.
  • > : create child operator.
  • + : adjacent element operator
  • $: autoincrement, like 1,2,3 ...
  • h1*8 : creates 8 <h1> tags.
  • h1{someText} : creates an <h1> tag and populates it with the provided text.

And here are some examples:

  • ul>li: creates an <ul> element with a child <li>
  • ul>li*3: creates an <ul> element with three children <li>
  • ul>li{item$}*3: creates an <ul> element with three children <li>, with text "item1", "item2", "item3".
  • main>aside+nav: creates a <main> element with an <aside> and <nav> as children.

Images

You should always style any media elements like images or videos with display: block to avoid the weird quirks with inline styling. And then always set their max-width: 100% and height: auto to avoid overfilling their container.

Here are the best practices for images:

  1. Always give them block display
  2. Make sure to always nest them in a div container.
  3. Always set a height and width attribute on them to avoid layout shift.

Here are the base styles to set for all images:

img {
display: block;
height: auto;
max-width: 100%;
}

Object fit with images

img {
max-width: 100%;
height: 300px;
display: block;
object-fit: cover;
}

After setting a fixed width and height on an image, you can use the object-fit property to change the behavior of how the image fills its container.

Margin collapsing

When two block level elements have vertical margin, like a margin-top and a margin-bottom pushing against each other, the margins collapse to use the larger margin. Only vertical margins collapse.

<style>
.h1 {
margin-bottom: 10px;
}
.h2 {
margin-top: 100px;
}
</style>
<h1 class="h1">hello</h1>
<h2 class="h2">hello 2</h2>

In the example above, the resulting margin between the <h1> and <h2> is not 110px, but rather the larger of the two, which is 100px.

background position

When using the keywords, they align the corners of the image with the corners of the container.

A background-position: top left makes the top left corner of the image align with the top left corner of the container.

Z-index

Z-index is dependent on stacking layers. If you have two containers, one with a z-index: 2 and another like z-index: 1, all of the children of those containers are basically subject to that same stacking order.

Even if the container with z-index: 1 has a child with z-index: 9999, it will not rise above the container with z-index: 2 because it effectively has a max z-index = 1. This is the concept known as a stacking context.

NOTE

Obeying the stacking context All children have the effective max z-index of their container, if it has a z-index. z-index values are not global.

Any element container with a z-index set on it creates a stacking context that all of its children now obey. For any children, setting a high z-index only helps them rise above other children of the container.

This is how you can set a stacking context (just set non-static position and z-index):

.some-element {
position: relative;
z-index: 1;
}

To avoid stacking context issues, don't use z-indexes. By default, all elements in the document are in the same stacking context.

blend modes

Blend Modes  |  web.dev

mix-blend-mode applies blending to the whole element while background-blend-mode applies blending only to the background of an element, not affecting inner content.

Here are some values you can set for the blend modes:

  • multiply : lights get lighter, darks get darker. Results in darker image
  • screen : Results in lighter image
  • overlay : a mix between multiply and screen
  • darken: selects the darker of the multiple color for the luminosity color value of the resulting color.

background blend mode

The background-blend-mode property mixes together all the backgrounds of an element together in a way that you specify.

The best use case for background-blend-mode is when you have both a background-image and a background-color set on the same element, and then you can blend them together for the color to bleed into the image.

You can provide multiple background images to the background property, and you can blend them together with the background-blend-mode property.

body {
background-image: url('https://imgur.com/Fn9FQwT.jpg'), url('https://imgur.com/VfcgZZ9.jpg');
background-size: cover;
background-blend-mode: difference;
}

The image-set() function

The image-set() function allows you to provide different versions of an image for different device resolutions. This can be useful if you want to use a higher resolution image for devices with high pixel densities, such as smartphones and tablets, while using a lower resolution image for devices with lower pixel densities.

Here’s an example of how to use the image-set() function in a background-image declaration:

body {
background-image: image-set(
'/path/to/image-lowres.jpg' 1x,
'/path/to/image-highres.jpg' 2x
);
}

The image-set() function takes a series of image URLs and resolutions as arguments.

In this example, we’re providing two versions of the same image: a low-resolution version for devices with a pixel density of 1x, and a high-resolution version for devices with a pixel density of 2x. The browser will choose the appropriate version of the image based on the device’s pixel density.

Filters

The filter property applies filter functions to the selected elements. The backdrop-filter property applies filter functions only to the background of the element, so any text and inner content is unaffected.

  • The blur() filter blurs an element

    .my-element {
    filter: blur(0.25rem);
    }
  • The brightness() filter increases or decreases brightness

    • Use values below 100% to decrease brightness, and above to increase brightness
    .my-element {
    filter: brightness(80%); // set brightness to 80%
    }
  • The invert() filter inverts the colors

    .my-element {
    filter: invert(1); // invert colors
    }
  • The opacity() filter changes the opacity of the image

    .my-element {
    filter: opacity(0.3);
    }

Drop shadow

codepen embed

When we want to add shadows to transparent images and have the shadow appear on the subject of photo rather than the container, we can use the drop-shadow() CSS filter

.my-image {
filter: drop-shadow(0px 0px 10px rgba(0 0 0 / 30%))
}
  • Use drop-shadow() filter on transparent image pngs.

Transitions

You can set transitions on the normal selector to act as exit transitions and also add transitions on the psuedoselector states like hover for enter transitions:

.my-element {
background: red;

/* This transition is applied on the "exit" transition, when mouse leaves */
transition: background 2000ms ease-in;
}

.my-element:hover {
background: blue;

/* This transition is applied on the "enter" transition, when you mouse over */
transition: background 150ms ease;
}

It’s important to remember you have to accomodate for users who don’t want animations:

/*
If the user has expressed their preference for
reduced motion, then don't use transitions.
*/
@media (prefers-reduced-motion: reduce) {
.my-element {
transition: none;
}
}

Text Orientation

The writing-mode property changes the flow of text

  • writing-mode: vertical-lr : text is displayed vertically as if it were turned 90 degrees.

The text-orientation property changes whether text is horizontal or vertical in conjunction with writing-mode

  • text-orentation: mixed : text is horizontal
  • text-orientation: upright: text is vertical

This is what happens when you have these properties:

.text {
writing-mode: vertical-rl
text-orientation: mixed
}

Mixed text orientation example

.text {
writing-mode: vertical-rl
text-orientation: upright
}

upright text orientation

Flexbox Tutorial

Controlling the size of flex items

Flex-grow

By default, all flex-children will never grow past their original size. We use the flex-grow property to allow flex children to grow past their original size.

The flex-grow property controls how the element will fill out the space of the container as the container size changes, relative to other flex-children.

  • By default, all flex children have a flex-grow: 0, meaning that their initial size do not change at all.
  • A flex child with flex-grow: 1 will try to take up as much space as possible.

Flex-shrink

By default, all flex-children will never grow past their original size, but will shrink if necessary.

The flex-shrink property controls how flex-items shrink relative to other flex-children.

  • By default, all flex children have a flex-shrink: 1: meaning they will shrink if necessary to stay in the container.

Flex-basis

The flex-basis property controls the minimum width or height for a flexbox item, making sure that the item can never shrink below what you set.

Setting a flex-basis: 150px for a flex child means that flex child will never shrink below 150px.

WARNING

You need to set box sizing correctly It is advisable to set border-sizing: border-box to not get janky behavior with padding and flex-basis.

Flex

The flex property is a shorthand for flex-grow, flex-shrink, and flex-basis.

flex: flex-grow flex-shrink flex-basis;

The default value for flex is flex: 0 1 auto, meaning that flex items don't grow, they only shrink, and that their minimum width/height is whatever space their content inherently takes up.

A common split layout where each child takes up as much space as possible within their row is as follows:

.split-item {
flex: 0 1 50%; /* only take up max 50% space, but shrink if necessary */
}

Grid Tutorial

Setting up rows and columns

grid-template-columns and grid-template-rows decide the number of columns and the number of rows along with their respectives widths/heights. You can use normal css measurement values, but you can also use fr fraction values, which will distribute in a fractional matter.

fractions

grid-template-columns: 1fr 1fr 1fr;
  • Makes 3 columns, each 1/3 width

repeat

grid-template-columns: repeat(2, 200px);
  • makes 2 columns, each 200px wide

repeat with minmax

The minmax(min, max) function sets a min and max value to choose from. You can combine this with repeat to set minimum widths for columns and rows, etc.

The real power of minmax comes from using dynamic values as the minimum or maximum, like so:

  • minmax(250px, 1fr): minimum is 250px, max is 1fr (as much space as possible)
  • minmax(auto, 300px): max is 300px, min is just inherent size of content
.min-width-columns {
/* 3 columns, min-width 250px, max 1fr */
grid-template-columns: repeat(3, minmax(250px, 1fr));
}

Assigning grid items to space

Use the grid-row and grid-column properties to define how many cells wide and tall a grid item should be.

Grid lines start at 1, and you can refer to the end as -1.

.box:first-child {
grid-column: 4 / span 1;
grid-row: 3 / span 1;
}

TIP

Absolute positioning You can achieve an absolute positioning effect by stacking grid items in the same cell, stacking each grid item on top of each to produce the illusion of absolute positioning.

Naming grid lines

.container {
display: grid;
grid-template-columns: [col1-start] 1fr [col1-end col2-start] 1fr [col2-end col3-start] 1fr [col3-end];
grid-template-rows: [row1-start] 1fr [row1-end row2-start] 1fr [row2-end row3-start] 1fr [row3-end];
}

Inside brackets [] before each of the values defining the size of rows or columns, we can name our grid lines and refer to their names instead of the numbers.

  • We can define multiple names for the same grid line by separating names with a space.
.container {
display: grid;
grid-template-columns: [col1-start] 1fr [col1-end col2-start] 1fr [col2-end col3-start] 1fr [col3-end];
grid-template-rows: [row1-start] 1fr [row1-end row2-start] 1fr [row2-end row3-start] 1fr [row3-end];
}

.box:first-child {
grid-column: col3-start / col3-end;
grid-row: row2-start / row3-end;
}

Using grid template areas

.container {
grid-template-rows: repeat(4, 100px);
grid-template-columns: repeat(3, 1fr);
grid-template-areas:
"a a b"
"c c b"
"d e f"
"g g f";
}

You can visually set up your layout with grid-template-areas, and then you assign your grid-items to certain areas in the layout using the grid-area property.

  • Each string corresponds to a row
  • Each value you put in a string corresponds to a column.
  • You cannot use numbers to represent the columns in the grid-template-areas property.
body {
background-color: black;
}

.grid-item {
background-color: blue;
min-height: 100px;
border: 3px solid red;
}

.grid-container {
display: grid;
border: 10px solid yellow;
grid-auto-rows: minmax(150px, auto);
gap: 50px 50px;

/* 1. define how the grid should look like */
grid-template-areas:
"header header"
"sidebar content";
}

/* 2. assign each grid item to their space using their grid are name */
.grid-item-1 {
grid-area: header;
}

.grid-item-2 {
grid-area: sidebar;
}

.grid-item-3 {
grid-area: content;
}

Implicit Grid

When you define a certain number of columns and rows, and the grid is completely filled up, and you add more grid items to the grid, the implicit grid kicks in.

The default behavior is to add an extra row to the grid and just start adding elements in there, but you can configure that behavior with these properties:

  • grid-auto-rows: defines the height of the implicit row new grid items are added into. The default value is to only fit the content of the grid item added in.
  • grid-auto-flow: determines how new grid items get added to the grid. The default behavior is row, meaning an extra implicit row is added, but you can change it to column to make extra implicit columns be added.
  • grid-auto-columns: defines the width of the implicit column new grid items are added into. The default value is to only fit the content of the grid item added in.

Content Layout in the grid

When dealing with content layout, you have two aspects to worry about:

  1. Aligning the grid item within their grid cell/area
  2. Aligning the grid items with respect to each other and the grid layout.

grid item content alignment

The below properties are used for aligning the a grid item within their grid cell

  • align-items : vertically aligns content within a grid cell. Default is stretch
  • justify-items : horizontally aligns content within a grid cell. Default is start
  • align-self : vertically aligns content for a specific grid item
  • justify-self : horizontally aligns content for a specific grid item

grid item alignment

  • justify-content : controls how grid items are aligned on the horizontal axis. Works only if using non-fraction values for the grid-template-columns. Default is flex-start.
  • align-content : controls how grid items are aligned on the vertical axis. Default is stretch. Only works if you give a defined height to the grid container.

short-hand alignment

  • place-items: shorthand for setting the same value for both justify-items and align-items
  • place-content: shorthand for setting the same value for both justify-content and align-content

auto-fit and auto-fill

Use the auto-fit and auto-fill properties in combination with repeat() to provide automatically change the column and grid numbers based on the size of the grid items.

WARNING

the only use case These keywords only work when the grid container has a defined height and width

  • auto-fill adds new columns whenever it gets the chance to.
  • auto-fit grows grid-items to fill the entire container space
.example {
grid-template-columns: repeat(auto-fit, minmax(100px, 1fr));
grid-template-columns: repeat(auto-fill, minmax(100px, 1fr));
}

TIP

auto-fit is always a good default.

Card layout with auto-fill

For a responsive column layout, we should follow this formula:

.example {
grid-template-columns: repeat(auto-fill min-width-item);
}

To find the minimum width of a grid item in our layout, we find the max width of our container, subtract it by the gap amount times the number of gaps, and then divide it by the number of columns we want when the container reaches its max size.

Let's dive into an example:

.container {
max-width: 1170px;
margin: 2rem auto;
display: grid;
gap: 2rem;
grid-template-columns: repeat(
auto-fill,
minmax(calc((1170px - (2rem * 2)) / 3), 1fr)
);
}

something

  • 1170px is our max width, and from that we'll subtract the two 2rem gap values to just get the width of all the three cards.
  • Then we divide by 3 to get the individual width of a card.
  • So card in a 3-column layout with 2rem will have a minimum width of 368px.

New CSS Features

interpolate-size

Adding the interpolate-size: allow-keywords on the :root pseudoselector allows you to animate height and width to keywords like auto or none, which is pretty useful.

/* 1) always add this */
:root {
interpolate-size: allow-keywords;
}

nav a {
width: 80px;
transition: width 0.3s;
&:hover {
width: auto;
}
}

Entry and Exit Animations

dialog {
--duration: 0.34s;

transition:
translate var(--duration) ease-in-out,
scale var(--duration) ease-in-out,
filter var(--duration) ease-in-out,
display var(--duration) ease-in-out allow-discrete;

&[open] {

/* Post-Entry (Normal) State */
translate: 0 0;
scale: 1;
filter: blur(0);

/* Pre-Entry State */
@starting-style {
translate: 0 8vh;
scale: 1.15;
filter: blur(8px);
}
}

/* Exiting State */
&:not([open]) {
translate: 0 -8vh;
scale: 1.15;
filter: blur(8px);
}
}

content-visibility

The content-visibility CSS property helps hide below-the-fold content and lazy load it instead, which can help improve rendering speed.

.render-me-later {
content-visibility: auto;
contain-intrinsic-size: auto 1000px; /* estimate 1000px height to prevent jitters. */
}

@property

CSS introduced @property variables, which are typed CSS variables you can animate. Here is the basic syntax for creating them:

@property --property-name {
syntax: "<color>";
inherits: false;
initial-value: red;
}
  • syntax: the variable type. Here are the different types you have access to:
    • "<color>": color type
    • "<number>": number type
    • "<percentage>": percentage type
    • "<angle>": angle type
  • inherits: whether or not this variable should inherit from another variable.
  • initial-value: the initial value for the variable.

You can then use the property variable as a normal CSS variable:

@property --brand {
syntax: "<color>";
inherits: false;
initial-value: red;
}

.container {
--brand: orange;
background-color: var(--brand);
}

The main use case of property variables is using them in animations:

@property --shine-1 {
syntax: "<color>";
inherits: false;
initial-value: #ffbbc0;
}

@property --shine-2 {
syntax: "<color>";
inherits: false;
initial-value: #c0aecb;
}

@keyframes animate-color-1 {
from {
--shine-1: initial;
}
to {
--shine-1: orange;
}
}

@keyframes animate-color-2 {
from {
--shine-2: initial;
}
to {
--shine-2: hotpink;
}
}

You can also register them in JavaScript:

CSS.registerProperty({
name: "--my-color",
syntax: "<color>",
inherits: false,
initialValue: "#c0ffee",
});

light-dark()

The light-dark(light_val, dark_val) function returns the first parameter if the current color scheme is in light mode, and returns the second parameter if the current color scheme is in dark mode.

:root {
color-scheme: light dark;
}

.omni-theme {
/* If in light mode, return black. Else return white. */
color: light-dark(black, white);
}

Container queries

Container queries are a way of responsively styling containers based on their width, not the width of the window. This helps ensure more reusable code and reliable responsiveness.

To use container queries, follow these steps:

  1. Declare your container to use container queries by using the container-type CSS property and give it a container-name. Here are the different values you can provide for the container-type:
    • inline-size means you want to track only width
    • size means you want to track both width and height (rarely used).
.container {
container-type: inline-size;
container-name: containerName;
}
  1. Create a container query to style the container based on a breakpoint.
@container containerName (max-width: 800px) {
/* something... */
}

TIP

Container queries are really useful whenever you have a resizable element that you want to style differently when its size changes.

Layers

CSS layers allow you to ignore specificity of certain selectors and instead apply styling based on layer priority.

You define layers with an @layer directive and then put all your CSS rulesets inside of it to include those styles in that layer.

  • All rules not included inside a layer have the highest priority

A common pattern is to first have a one-liner that defines the order of layer priority at the top of your file, where the last layer before the semicolon has the highest priority.

In the example above, a body < p would have higher specificity than the .text selector, but because it is in the first-defined layer, it has less priority than any styles in later defined layers. so the text appears green

<style>
/* new layer has highest priority */
@layer base, new;

@layer base {
body < p {
color: black;
}
}

@layer new {
.text {
color: green;
}
}
</style>
<body>
<p class="text">bruh</p>
</body>

NOTE

Tip Having base styles be the first layer and utility class layers be the last is a great away to avoid specificity headaches.

Dark Mode

Dark mode is a simple technique that only requires a few steps. It involves setting global CSS variables that you use everywhere in your application for styling, and then creating a class that overrides those CSS variables when applied.

  1. Create all the CSS variables you use for your app in the :root selector.
:root {
--text-color: black;
}
  1. Create a .dark class that overrides those exact same CSS variables with different values
.dark {
--text-color: white;
}
  1. Use JavaScript to programmatically add and remove the .dark class from the html element. Combine with local storage to save the user's preference.

CSS Tips

CSS Functions

  • min(val1, val2): returns the minimum value. This is useful for container width styles, like so:
.container {
width: min(90%, 1280px); /* max-width is 1280, else width = 90% */
}
  • max(val1, val2) : returns the maximum value from the two passed in.

Quick Tips

:focus-visible

Use the :focus-visible pseudo-selector when styling focus for text inputs in forms. It results so that the style only applies when the user uses the keyboard to focus on the input, which is what you always want.

input:focus {
outline: none;
}

input:focus-visible {
outline: 5px solid black;
}

The :any-link selector selects all anchor tags that have valid href link values, and does not style any invalid links.

Comparison operator in media queries

@media (width <= 768px) {
/* styles here */
}

Writing mode to change text direction

You can use the writing-mode property to display text vertically.

.vertical-text {
writing-mode: vertical-lr;
}

Flip horizontally

You can flip an image or any element by using transform: translateX(-1).

Resize with resize

Resize an element using the resize property. This property only works on elements where the overflow is not set to visible.

  • resize: horizontal : resizes only horizontally
  • resize: vertical: resizes only vertically
  • resize: both: resizes both vertically and horizontally.

Scroll padding

Use the scroll-padding-top and scroll-margin-top properties to add scroll padding to prevent content at the top of a page being hidden by a fixed navbar.

text-wrap for headings

  1. Use the text-wrap: balance property to make sure that the text in headings wraps normally and does not stretch across the page. It offers a nicer look.
h1, h2, h3, h4, h5, h6 {
text-wrap: balance;
}

Advanced Selectors

has, is, not

  • The :has pseudoselector lets you style parent elements based on some state of a child element.
  • The :not pseudoselector basically does the opposite of the selection

These advanced CSS psuedoselectors are functions that take in multiple selectors.

  • :is() : selects an element if it matches any of the selectors inside the parenthesis
  • :has() : selects an element if it has any of the specified selectors as its children. It’s useful because it styles the selected element, not its children.
/* If .container has an h2 descendant */
.container:has(h2) {
background: green;
}

/* If .content has an h2 direct child */
.content:has(> h2) {
background: green;
}

/* styles all images without an alt attribute set */
img:not([alt]) {
border: 10px solid red;
}

In the above example, the .container elements will be styled with a green background only if they have an <h2> element as one of their children.

You can even nest these selectors inside each other:

/* If the container does not have an h2 inside it... */
.container:not(:has(h2)) {
}

You can use the ::is psuedoselector to provide a list of elements and selectors to match. This reduces the amount of code you have to write.

.post :is(h2, li, img) {
// styles here
}

nth child

/* affects children 1-3 */
li:nth-child(-n + 3) {
text-decoration: underline;
}

/* affects children 2-5 */
li:nth-child(n + 2):nth-child(-n + 5) {
color: #2563eb;
}

/* affects the last 2 children */
li:nth-last-child(-n + 2) {
text-decoration-line: line-through;
}