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
: theclass
attribute shortcut. Createsh1
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:
- Always give them block display
- Make sure to always nest them in a div container.
- Always set a
height
andwidth
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
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 imagescreen
: Results in lighter imageoverlay
: a mix between multiply and screendarken
: 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%
} - Use values below
-
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
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 horizontaltext-orientation: upright
: text is vertical
This is what happens when you have these properties:
.text {
writing-mode: vertical-rl
text-orientation: mixed
}
.text {
writing-mode: vertical-rl
text-orientation: upright
}
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 isrow
, meaning an extra implicit row is added, but you can change it tocolumn
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:
- Aligning the grid item within their grid cell/area
- 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 isstretch
justify-items
: horizontally aligns content within a grid cell. Default isstart
align-self
: vertically aligns content for a specific grid itemjustify-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 isflex-start
.align-content
: controls how grid items are aligned on the vertical axis. Default isstretch
. Only works if you give a defined height to the grid container.
short-hand alignment
place-items
: shorthand for setting the same value for bothjustify-items
andalign-items
place-content
: shorthand for setting the same value for bothjustify-content
andalign-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)
);
}
1170px
is our max width, and from that we'll subtract the two2rem
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 of368px
.
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:
- Declare your container to use container queries by using the
container-type
CSS property and give it acontainer-name
. Here are the different values you can provide for thecontainer-type
:inline-size
means you want to track only widthsize
means you want to track both width and height (rarely used).
.container {
container-type: inline-size;
container-name: containerName;
}
- 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.
- Create all the CSS variables you use for your app in the
:root
selector.
:root {
--text-color: black;
}
- Create a
.dark
class that overrides those exact same CSS variables with different values
.dark {
--text-color: white;
}
- Use JavaScript to programmatically add and remove the
.dark
class from thehtml
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;
}
:any-link
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 horizontallyresize: vertical
: resizes only verticallyresize: 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
- 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;
}