Relative Units

Definitions

Absolute units
Definite - they do not change in context:
  • Absoulte units:
    • px: pixel - difficult to measure now with high-res screens. 96px is about 1 physical inch on the screen.
    • mm: millimeter
    • cm: centimeter
    • Q: quarter-millimeter
    • in: inch
    • pt: point
    • pc: pica
  • 12pt is equal to 16px
Computed value
Absolute value that the browser computes for values declared using relative units. When an element has a value defined using a length (px, rem, em, etc), its computed value is inherited by its child elements.
Length
Formal name for a CSS element that has a unit, which is formally called a distance measurement.
Relative units
Different values depending on the context they are called in
  • They are evaluated by the browser to an absolute value
  • rems and ems are the most common
Responsive design
When styles respond differently based on the size of the browser window.

Suggested units

Youtube link

UnitCSS property
remfont-size
document flow (consistent spacing)
padding
margin
emmedia queries
document flow (with more space)
pxshadow-box

border
border-radius
%page/container widths
  • height: use min-height() when you need to set a height so that the content does not overflow at the bottom if the viewport size changes.

Relative units

  • rem for properties that you do not want to scale.
  • em for properties that you want to scale.

Styling with relative units is helpful because you do not know the absolute size of the screen that your document will display on - you don’t know exactly how to size your elements

  • Ex: Set a font size with relative units so it scales proportionately with the size of the window
  • Ex: Set size of everything relative to the base font size and then resize everything by changing the base font size only

ems

Ems are the most common relative length unit, comes from typography.

Uses the local font size to determine its computed value:

  • the computed value of a font size is derived from its inherited value
  • if font size is 1.5em, then the size of 1em for that element is 24px. (16px * 1.2 = 24px)
    • If font-size is 16px and the padding is 2em, then the padding is 32px
  • tricky for nested elements that use ems for font-size and other properties, like padding
    • If em size is smaller than 1, child elements computed value is smaller (0.8em * 0.8em = 0.64em)
    • If em size is larger than 1, child elements computed value is larger (1.5em * 1.5em = 2.25em)

DO NOT use ems for font-size. If you define the font-size with an em, then the size of the em is derived from the inherited font size. If you use ems to define other properties like padding, then the browser first calculates the font size, then calculates the padding with that font size. So, font-size and padding might have the same declared value but different computed values.

.padded {
  font-size: 16px;
  padding: 1em;     /* computes to 16px */
/*padding: 1.5em;   computes to 24px */
}

rems

Use rem for font sizes.

Rem is short for “root em”, so a rem is relative to the font-size value in the root. The root is the html tag, which is the root of the HTML document. You can access the html tag with the :root pseudo-class selector so you can have the specificity of a class element rather than a tag.

Because rems are relative to the root element, it has the same computed value throughout the stylesheet.

This article says to use rem for fonts and px for everything else. This is because padding and margin scale along with the font. You might want to use rem instead of px at times.

Viewport-relative units

Examples and implementations in this CSS tricks article, Fun With Viewport Units.

Use cases:

The viewport is the area in the browser window where the web page is displayed - NOT including the address bar, shortcuts, toolbars, etc. The following are the basic units:

  • vh: 1% of the viewport height
  • vw: 1% of the viewport width
  • vmin: 1% of the smaller dimension, height or width
  • vmax: 1% of the larger dimension, height or width

So, 25vh is 25% of the viewport’s height.

Viewport units are good for large hero images. However, this is an issue with mobile devices because some mobile devices have a feature that hides the address or nav bar to save space. This causes viewport-relative items to change size on the screen. This is called layout thrashing.

Also, remember that viewports do not take scrollbars into account.

vmin

vmin lets you set the height or width depending on the size of the viewport. This example sets the height to 90vh if the screen height is smaller than the width, and sets it to 90vw if the width is smaller than the height:

.div {
    width: 90vmin;
    height: 90vmin;
    background-color: #369;
}

Large and Small units

Viewport units were used for heroes, but some mobile devices dynamically hide menus when you are not scrolling and then display them when you are scrolling. This can cause the content on the screen to jump if you use viewport units. This is called layout thrashing.

This is largely resolved in the mobile browsers, but it might still be an issue.

To address this, CSS came up with large and small viewports:

  • large is when all UX components are hidden. Prepend an l to the unit (lvh)
  • small is when UX components are visible. Prepend an s to the unit (svh)
    • When in doubt, use small viewport units

This is not used very often.

When a viewport size changes because the device hides a UX component, you can use the large (for screens without the address bar):

  • lvw
  • lvh
  • lvmin
  • lvmax

Or small (for screens with the address bar):

  • svw
  • svh
  • svmin
  • svmax

line-height

Some properties accept unitless values, but you can also use 0 without a unit, because 0 is equal to 0 of any length.

Accepts both units and unitless values, but prefer unitless because they are inherited differently:

  • Unitless values are recalculated for all child elements
    • Child elements inherit computed values if the value is assigned with lengths (units)
    • This causes issues with line-height

To calculate the line-height, multiply the font-size by the line-height value:

  • Ex: 16px (1rem) * 1.1 = 17.6px
/* total line height is 38.4px */
body {
    line-height: 1.2;
}

.about-us {
    font-size: 2em;
}

Custom properties (variables)

Create dynamic and context-based styling:

  • declare a variable and assign a value to reference throughout your stylesheet
  • more versatile than preprocessor variables
  • create a variable with double hyphens (--var-name) and assign with var()
    • var() accepts a second optional fallback value
  • Set global vars in the :root selector so they’re available for the entire page
  • These vars have scope - you can redifine variables within rulesets
:root {
    --main-font: Helvetica, Arial, sans-serif;
    --brand-color: #369;
}

p {
    font-family: var(--main-font, sans-serif);
    color: var(--brand-color, blue);
}

.dark {
    margin-top: 2em;
    padding: 1em;
    background-color: #999;
    --main-bg: #333;            /* redefine global vars */
    --main-color: #fff;
}

Themes with custom properties

You can add dark and light themes with custom properties. First, create the theme colors with class styles on the :root element:

:root.dark {
  --border-btn: 1px solid rgb(220, 220, 220);
  --color-base-bg: rgb(18, 18, 18);
  --color-base-text: rgb(240, 240, 240);
  --color-btn-bg: rgb(36, 36, 36);
}

:root.light {
  --border-btn: 1px solid rgb(36, 36, 36);
  --color-base-bg: rgb(240, 240, 240);
  --color-base-text: rgb(18, 18, 18);
  --color-btn-bg: rgb(220, 220, 220);
}

/* apply custom props to elements below */

Then, use JS to grab the root element (documentElement), and toggle the theme:

let setTheme = () => {
  const root = document.documentElement;
  const newTheme = root.className === "dark" ? "light" : "dark";
  root.className = newTheme;

  document.querySelector(".theme-name").textContent = newTheme;
};

document.querySelector(".theme-toggle").addEventListener("click", setTheme);