Relative Units - CSS Fundamentals

Relative Units - CSS Fundamentals

                        <p>In the web environment, the user can have their browser window set to any number of sizes, and the CSS has to apply to it. Also, users can resize the page after it’s opened, and the CSS needs to adjust to new constrains. This means that the browser must calculate those when the page is rendered on-screen. Relative units are one of the tools CSS provides to work for the <em>responsive</em> design.</p>

<h2 id="pixels-points-and-picas">Pixels, Points, and Picas</h2>

<p>CSS supports several absolute length units, the most common for which, and the most basic, is the pixel (px). Less common absolute units are mm (millimetre), cm (centimetre), in. (inch), pt (point — typographic term for 1/72 of an inch), and pc (pica — typographic term for 12 points). Any of these units can be translated directly to another if you want to work out the math:</p>

<span class="katex-display"><span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML" display="block"><semantics><mrow><mn>1</mn><mtext> </mtext><mi>i</mi><mi>n</mi><mi mathvariant="normal">.</mi><mo>=</mo><mn>25.4</mn><mtext> </mtext><mi>m</mi><mi>m</mi><mo>=</mo><mn>2.54</mn><mtext> </mtext><mi>c</mi><mi>m</mi><mo>=</mo><mn>6</mn><mtext> </mtext><mi>p</mi><mi>c</mi><mo>=</mo><mn>72</mn><mtext> </mtext><mi>p</mi><mi>t</mi><mo>=</mo><mn>96</mn><mtext> </mtext><mi>p</mi><mi>x</mi></mrow><annotation encoding="application/x-tex">1\ in. = 25.4\ mm = 2.54\ cm = 6\ pc = 72\ pt = 96\ px</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:0.65952em;vertical-align:0em;"></span><span class="mord">1</span><span class="mspace"> </span><span class="mord mathnormal">i</span><span class="mord mathnormal">n</span><span class="mord">.</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord">2</span><span class="mord">5</span><span class="mord">.</span><span class="mord">4</span><span class="mspace"> </span><span class="mord mathnormal">m</span><span class="mord mathnormal">m</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord">2</span><span class="mord">.</span><span class="mord">5</span><span class="mord">4</span><span class="mspace"> </span><span class="mord mathnormal">c</span><span class="mord mathnormal">m</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.8388800000000001em;vertical-align:-0.19444em;"></span><span class="mord">6</span><span class="mspace"> </span><span class="mord mathnormal">p</span><span class="mord mathnormal">c</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.8388800000000001em;vertical-align:-0.19444em;"></span><span class="mord">7</span><span class="mord">2</span><span class="mspace"> </span><span class="mord mathnormal">p</span><span class="mord mathnormal">t</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.8388800000000001em;vertical-align:-0.19444em;"></span><span class="mord">9</span><span class="mord">6</span><span class="mspace"> </span><span class="mord mathnormal">p</span><span class="mord mathnormal">x</span></span></span></span></span>

<p>therefore, 16 px is the same as 12 pt (<span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML"><semantics><mrow><mfrac><mn>16</mn><mn>96</mn></mfrac><mo>×</mo><mn>72</mn></mrow><annotation encoding="application/x-tex">\frac{16}{96}\times{72}</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:1.190108em;vertical-align:-0.345em;"></span><span class="mord"><span class="mopen nulldelimiter"></span><span class="mfrac"><span class="vlist-t vlist-t2"><span class="vlist-r"><span class="vlist" style="height:0.845108em;"><span style="top:-2.6550000000000002em;"><span class="pstrut" style="height:3em;"></span><span class="sizing reset-size6 size3 mtight"><span class="mord mtight"><span class="mord mtight">9</span><span class="mord mtight">6</span></span></span></span><span style="top:-3.23em;"><span class="pstrut" style="height:3em;"></span><span class="frac-line" style="border-bottom-width:0.04em;"></span></span><span style="top:-3.394em;"><span class="pstrut" style="height:3em;"></span><span class="sizing reset-size6 size3 mtight"><span class="mord mtight"><span class="mord mtight">1</span><span class="mord mtight">6</span></span></span></span></span><span class="vlist-s">​</span></span><span class="vlist-r"><span class="vlist" style="height:0.345em;"><span></span></span></span></span></span><span class="mclose nulldelimiter"></span></span><span class="mspace" style="margin-right:0.2222222222222222em;"></span><span class="mbin">×</span><span class="mspace" style="margin-right:0.2222222222222222em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord"><span class="mord">7</span><span class="mord">2</span></span></span></span></span>). Designers are often more familiar with the use of points, where developers are more accustomed to pixels.</p>

<p>Pixel is a slightly misleading name — a CSS pixel does not strictly equate to a monitor’s pixel. This is notably the case on high-resolution displays. Although the CSS measurements can be scaled a bit, depending on the browser, the operating system, and the hardware, 96 px is usually in the ballpark of 1 physical inch on-screen, though this can vary on certain devices or with a user’s resolution settings.</p>

<h2 id="ems-and-rems">Ems and Rems</h2>

<p>Ems, the most common relative length unit, are a measure used in typography, referring to a specified font size. In CSS, 1 em means the font size of the current element; its exact value varies depending on the element you’re applying it to.</p>

<p>For example:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nc">.padded</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">16px</span><span class="p">;</span>
  <span class="nl">padding</span><span class="p">:</span> <span class="m">1em</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>This ruleset specifies a font size of 16 px, which becomes the element’s local definition for 1 em. Then the code uses ems to specify the padding of the element, producing a rendered padding of 16 px. If another selector targets the same element and overrides it with a different font size, it’ll change the local meaning of em, and the computed padding will change to reflect that.</p>

<blockquote>
  <p>Values declared using relative units are evaluated by the browser to an absolute value, called <em>computed value</em>.</p>
</blockquote>

<h3 id="using-ems-to-define-font-size">Using Ems to Define Font-size</h3>

<p>When it comes to the <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">size</span></code> property, ems have behave a little differently. If you declare <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">size</span><span class="p">:</span> <span class="mf">1.2</span><span class="nx">em</span></code>, what does that mean? A font size can’t equal 1.2 times itself. Instead, <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">size</span></code> ems are derived from the <em>inherited</em> font size.</p>

<p>If you know the pixel-based font size you’d like, but want to specify the declaration in ems, here’s a simple formula: divide the desired pixel size by the parent (inherited) pixel size. For example, if you want a 10 px font and your element is inheriting a 12 px font, <span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML"><semantics><mrow><mn>10</mn><mi mathvariant="normal">/</mi><mn>12</mn><mo>=</mo><mn>0.8333</mn><mtext> </mtext><mi>e</mi><mi>m</mi></mrow><annotation encoding="application/x-tex">10/12 = 0.8333\ em</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:1em;vertical-align:-0.25em;"></span><span class="mord">1</span><span class="mord">0</span><span class="mord">/</span><span class="mord">1</span><span class="mord">2</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord">0</span><span class="mord">.</span><span class="mord">8</span><span class="mord">3</span><span class="mord">3</span><span class="mord">3</span><span class="mspace"> </span><span class="mord mathnormal">e</span><span class="mord mathnormal">m</span></span></span></span>. If you want a 16 px font and the parent font is 12 px, <span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML"><semantics><mrow><mn>16</mn><mi mathvariant="normal">/</mi><mn>12</mn><mo>=</mo><mn>1.3333</mn><mtext> </mtext><mi>e</mi><mi>m</mi></mrow><annotation encoding="application/x-tex">16/12 = 1.3333\ em</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:1em;vertical-align:-0.25em;"></span><span class="mord">1</span><span class="mord">6</span><span class="mord">/</span><span class="mord">1</span><span class="mord">2</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span><span class="mrel">=</span><span class="mspace" style="margin-right:0.2777777777777778em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord">1</span><span class="mord">.</span><span class="mord">3</span><span class="mord">3</span><span class="mord">3</span><span class="mord">3</span><span class="mspace"> </span><span class="mord mathnormal">e</span><span class="mord mathnormal">m</span></span></span></span>.</p>

<blockquote>
  <p>It’s helpful to know that, for most browsers, the default font size is <strong>16 px</strong>. Technically, it’s the keyword value <code class="language-javascript highlighter-rouge"><span class="nx">medium</span></code> that calculates to 16 px.</p>
</blockquote>

<h3 id="ems-for-font-size-together-with-ems-for-other-properties">Ems for Font-size Together with Ems for Other Properties</h3>

<p>What makes ems tricky is when you use them for both font size and any other properties on the same element. When you do this, the browser must calculate the font size first, and then it uses that value to calculate the other values. Both properties can have the same declared value, but they’ll have different computed values. For example:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">body</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">16px</span><span class="p">;</span>
<span class="p">}</span>

<span class="nc">.padding</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">1.2em</span><span class="p">;</span> <span class="c">/* evaluates to 19.2px */</span>
  <span class="nl">padding</span><span class="p">:</span> <span class="m">1.2em</span><span class="p">;</span> <span class="c">/* evaluates to 23.04px */</span>
  <span class="nl">background-color</span><span class="p">:</span> <span class="m">#ccc</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>In this example, <code class="language-javascript highlighter-rouge"><span class="nx">padding</span></code> has a specified value of 1.2 em. This multiplied by 19.2 px (the computed font size of the element) produces a calculated value of 23.04 px.</p>

<h3 id="the-shrinking-font-problem">The Shrinking Font Problem</h3>

<p>Ems can produce unexpected results when you use them to specify the font sizes of multiple nested elements. To know the exact value for each element, you’ll need to know its inherited font size, which, if defined on the parent element in ems, requires you to know the parent element’s inherited size, and so on up the tree.</p>

<p>This becomes quickly apparent when you use ems for the font size of lists and then nest lists several levels deep, like this:</p>

<p><img src="/assets/img/placeholder.svg" alt="lists using ems as font-size" data-src="https://i.loli.net/2019/06/27/5d14c75dd20a814482.png"></p>

<p>with CSS defined like:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">body</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">16px</span><span class="p">;</span>
<span class="p">}</span>

<span class="nt">ul</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">.8em</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>By now, it should be clear that ems are nice for element sizing, but when it comes to font size, they can get complicated. Thankfully, there is a better option — rems.</p>

<h3 id="using-rems-for-font-size">Using Rems for Font-size</h3>

<p>When the browser parses an HTML document, it creates a representation in memory of all the elements on the page. This representation is called <em>DOM</em> (Document Object Model). It’s a tree structure, where each element if represented by a node. The <code class="language-javascript highlighter-rouge"><span class="o">&lt;</span><span class="nx">html</span><span class="o">&gt;</span></code> element is the top-level (or root) node. Beneath it are its child nodes, <code class="language-javascript highlighter-rouge"><span class="o">&lt;</span><span class="nx">head</span><span class="o">&gt;</span></code> and <code class="language-javascript highlighter-rouge"><span class="o">&lt;</span><span class="nx">body</span><span class="o">&gt;</span></code>. And beneath those are their children, then their children, and so on.</p>

<p>The root node is the ancestor of all other elements in the document. It has a special pseudo-class selector (<code class="language-javascript highlighter-rouge"><span class="p">:</span><span class="nx">root</span></code>) that you can use to target it. This is equivalent to using the type selector <code class="language-javascript highlighter-rouge"><span class="nx">html</span></code> with the specificity of a class rather than a tag.</p>

<p><em>Rem</em> is short for room em. Instead of being relative to the current element, rems are relative to the root element. No matter where you apply it in the document, 1.2 rem has the same computed value: 1.2 times the font size of the root element.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">1em</span><span class="p">;</span> <span class="c">/* set to browser's default of 16px */</span>
<span class="p">}</span>

<span class="nt">ul</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">.8em</span><span class="p">;</span> <span class="c">/* evaluated to be 12.8px */</span>
<span class="p">}</span>
</code></pre></div></div>

<blockquote>
  <p><strong>Accessibility: use relative units for font size</strong></p>

  <p>Some browsers provide two ways for the user to customize the size of text:</p>

  <ol>
    <li>
<strong>Zoom</strong>. By pressing <kbd>ctrl+</kbd> or <kbd>ctrl-</kbd>, the user can zoom the page up or down. This visually scales all fonts and images and generally makes everthing on the page larger or smaller. In most browsers, this change is temporary and is only applied to the current tab.</li>
    <li>
<strong>Set default font size</strong>. In browser’s settings, you can permanently change the default font size for text. The catch is that this setting does not resize fonts defined using pixels or other absolute units.</li>
  </ol>
</blockquote>

<p>Rems simplify a lot of complexities involved with ems. In fact, they offer a good middle ground between pixels and ems by providing the benefits of relative units, but easier to work with. Does this mean you should use rems everywhere and abandon the other options?</p>

<p>In CSS, again, the answer is often, “it depends”. Rems are but one tool in your tool box. An important part of mastering CSS is learning when to use which tool. It’s nice to use rems for font sizes, pixels for borders, and ems for most other measures, especially paddings, margins, and border radius. This way, font sizes are predictable, but you’ll still get the power of ems scaling your paddings and margins. Pixels make sense for borders, particularly when you want a nice fine line.</p>

<h2 id="stop-thinking-in-pixels">Stop Thinking in Pixels</h2>

<p>One pattern, or rather, antiparttern, that has been common for the past several years is to reset the font size at the page’s root to .625 em or 62.5%. This takes the browser’s default size, 16 px, and scales it down to 10 px. This practise simplifies the math.</p>

<p>Initially, this may be convenient, but there are two problems with this approach. First, it forces you to write a lot of duplicate styles. 10 px is too small for most text, so you’ll have to override it throughout the page. The second problem is that when you do this, you’re still thinking in pixels. You might type 1.4 rem into your CSS, but in your mind, you’re still thinking “14 pixels”. On a responsive web, you should get comfortable with “fuzzy” values. It doesn’t matter how many pixels 1.5 em evaluates to; all you need to know is that it’s a bit bigger than the inherited font size.</p>

<h3 id="setting-a-sane-default-font-size">Setting a Sane Default Font Size</h3>

<p>Let’s say you want your default font size to be 14 px. Instead of setting a 10 px default then overriding it through the page, set that value at the root. The desired value divided by the inherited value — in this case, the browser’s default —  is 14/16, which equals 0.875:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">0.875em</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>Now your desired font size is applied to the whole page. You won’t need to specify it elsewhere. You’ll only need to change it in places where the design deviates from this, such as headings.</p>

<h3 id="media-query">Media Query</h3>

<p>Let’s make this a bit further. You can use some <em>media queries</em> to change the base font size, depending on the screen size.</p>

<blockquote>
  <p><em>media query</em> — An <code class="language-javascript highlighter-rouge"><span class="p">@</span><span class="nd">media</span></code> rule used to specify styles that will be applied only to certain screen size or media types (<em>e.g.</em> print or screen). This is a key component of responsive design.</p>
</blockquote>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">0.75em</span><span class="p">;</span>
<span class="p">}</span>

<span class="k">@media</span> <span class="p">(</span><span class="n">min-width</span><span class="p">:</span> <span class="m">800px</span><span class="p">)</span> <span class="p">{</span> <span class="c">/* applies only to screens 800px and wider */</span>
  <span class="nd">:root</span> <span class="p">{</span>
    <span class="nl">font-size</span><span class="p">:</span> <span class="m">0.875em</span><span class="p">;</span>
  <span class="p">}</span>
<span class="p">}</span>

<span class="k">@media</span> <span class="p">(</span><span class="n">min-width</span><span class="p">:</span> <span class="m">1200px</span><span class="p">)</span> <span class="p">{</span> <span class="c">/* applies only to screens 1200px and wider */</span>
  <span class="nd">:root</span> <span class="p">{</span>
    <span class="nl">font-size</span><span class="p">:</span> <span class="m">1em</span><span class="p">;</span>
  <span class="p">}</span>
<span class="p">}</span>
</code></pre></div></div>

<p>If you are disciplined enough to style your entire page in relative units like this, the entire page will scale up and down based on the viewport size.</p>

<h3 id="resizing-a-single-component">Resizing a Single Component</h3>

<p>For example, you have a panel and want a larger version:</p>

<p><img src="/assets/img/placeholder.svg" alt="panel, sizes" data-src="https://res.cloudinary.com/frankindev/image/upload/f_auto,dpr_auto,w_auto/v1621612956/cld/panel_sizes.png"></p>

<p>You can only add a <code class="language-javascript highlighter-rouge"><span class="nx">large</span></code> class to the second panel: <code class="language-javascript highlighter-rouge"><span class="o">&lt;</span><span class="nx">div</span> <span class="kd">class</span><span class="o">=</span><span class="dl">"</span><span class="s2">panel large</span><span class="dl">"</span><span class="o">&gt;</span></code>.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nc">.panel</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">1rem</span><span class="p">;</span>
  <span class="nl">padding</span><span class="p">:</span> <span class="m">1em</span><span class="p">;</span>
  <span class="nl">border</span><span class="p">:</span> <span class="m">1px</span> <span class="nb">solid</span> <span class="m">#999</span><span class="p">;</span>
  <span class="nl">border-radius</span><span class="p">:</span> <span class="m">0.5em</span><span class="p">;</span>
<span class="p">}</span>

<span class="nc">.panel</span> <span class="o">&gt;</span> <span class="nt">h2</span> <span class="p">{</span>
  <span class="nl">margin-top</span><span class="p">:</span> <span class="m">0</span><span class="p">;</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">0.8em</span><span class="p">;</span>
  <span class="nl">font-weight</span><span class="p">:</span> <span class="nb">bold</span><span class="p">;</span>
  <span class="nl">text-transform</span><span class="p">:</span> <span class="nb">uppercase</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>First, add the declaration <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">size</span><span class="p">:</span> <span class="mi">1</span><span class="nx">rem</span></code> to the parent element of each panel. This means each panel will establish a predictable font size for itself, no matter where it’s placed on the page. Second, redefine the heading’s font size using Ems rather than rems to make it relative to the parent’s font size that just established.</p>

<p>For the larger version of the panel, all you have to do is override the parent element’s 1 rem with another value. Because all the component’s measurements are relative to this, overriding it will resize the entire panel:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nc">.panel.large</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">1.2rem</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<h2 id="viewport-relative-units">Viewport-relative Units</h2>

<p>There are also <em>viewport-relative units</em> for defining lengths relative to the browser’s viewport.</p>

<blockquote>
  <p><em>viewport</em> — The framed area in the browser window where the web page is visible. This excludes the browser’s address bar, toolbars, and status bar, if present.</p>
</blockquote>

<ul>
  <li>
<em>vh</em> — 1/100 of the viewport height</li>
  <li>
<em>vw</em> — 1/100 of the viewport width</li>
  <li>
<em>vmin</em> — 1/100 of the smaller dimension, height or width</li>
  <li>
<em>vmax</em> — 1/100 of the larger dimension, height or width</li>
</ul>

<p>The viewport-relative lengths are great for things like making a large hero image fill the screen. You can place an image inside a long container, but setting the image height to 100 vh, makes it exactly the height of the viewport.</p>

<h3 id="using-vw-for-font-size">Using vw for Font Size</h3>

<p>Consider what would happen if you applied <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">size</span><span class="p">:</span> <span class="mi">2</span><span class="nx">vw</span></code> to an element. On a desktop monitor at 1,200 px, this evaluates to 24 px (2% of 1,200). On a tablet with a screen width of 768 px, it evaluates to about 15 px (2% of 768). Unfortunately, it scales all the way down to 7.5 px on an iPhone 6.</p>

<h3 id="using-calc-for-font-size">Using calc() for Font Size</h3>

<p>The <code class="language-javascript highlighter-rouge"><span class="nx">calc</span><span class="p">()</span></code> function lets you do basic arithmetic with two or more values. This is particularly useful for combining values that are measured in different units. This function supports addition (+), subtraction (-), multiplication (*) and division (/). The addition and subtraction operators must be surrounded by whitespace, for example, <code class="language-javascript highlighter-rouge"><span class="nx">calc</span><span class="p">(</span><span class="mi">1</span><span class="nx">em</span> <span class="o">+</span> <span class="mi">10</span><span class="nx">px</span><span class="p">)</span></code>.</p>

<p>Use <code class="language-javascript highlighter-rouge"><span class="nx">calc</span><span class="p">()</span></code> to combine ems with vw units, as:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="n">calc</span><span class="p">(</span><span class="m">0.5em</span> <span class="err">+</span> <span class="m">1vw</span><span class="p">);</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The 0.5 em here operates as a sort of minimum font size, and the 1 vw adds a repsonsive scalar. This’ll give you a base font size that scales from 11.75 px on an iPhone 6 up to 20 px in a 1,200 px browser window. It should automatically resize the font size when you resize the browser window, however, I found it was not working on Safari (Version 12.1.1 (14607.2.6.1.1)), you have to refresh the page to adjust the font size.</p>

<h2 id="unitless-numbers-and-line-height">Unitless Numbers and Line-height</h2>

<p>Some properties allow for <em>unitless</em> values. Properties that support this include <code class="language-javascript highlighter-rouge"><span class="nx">line</span><span class="o">-</span><span class="nx">height</span></code>, <code class="language-javascript highlighter-rouge"><span class="nx">z</span><span class="o">-</span><span class="nx">index</span></code>, and <code class="language-javascript highlighter-rouge"><span class="nx">font</span><span class="o">-</span><span class="nx">weight</span></code> (700 is equivalent to bold; 400 is equivalent to normal, and so on). You can also use the unitless value <code class="language-javascript highlighter-rouge"><span class="mi">0</span></code> anywhere a length unit (such as px, em, or rem) is required because, in these cases, the unit does not matter.</p>

<blockquote>
  <p>A unitless <code class="language-javascript highlighter-rouge"><span class="mi">0</span></code> can only be used for <em>length</em> values and percentages, such as in paddings, borders, and widths. It can’t be used for angular values, such as degrees or time-based values like seconds.</p>
</blockquote>

<p>The <code class="language-javascript highlighter-rouge"><span class="nx">line</span><span class="o">-</span><span class="nx">height</span></code> property is unusual in that it accepts both units and unitless values. You should typically use unitless numbers because they’re inherited differently.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">body</span> <span class="p">{</span>
  <span class="nl">line-height</span><span class="p">:</span> <span class="m">1.2</span><span class="p">;</span>
<span class="p">}</span>

<span class="nc">.element</span> <span class="p">{</span>
  <span class="nl">font-size</span><span class="p">:</span> <span class="m">2em</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The <code class="language-javascript highlighter-rouge"><span class="p">.</span><span class="nx">element</span></code> inherits a line height of 1.2. Because the font size is 32 px (<span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML"><semantics><mrow><mn>2</mn><mtext> </mtext><mi>e</mi><mi>m</mi><mo>×</mo><mn>16</mn><mtext> </mtext><mi>p</mi><mi>x</mi></mrow><annotation encoding="application/x-tex">2\ em\times16\ px</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:0.72777em;vertical-align:-0.08333em;"></span><span class="mord">2</span><span class="mspace"> </span><span class="mord mathnormal">e</span><span class="mord mathnormal">m</span><span class="mspace" style="margin-right:0.2222222222222222em;"></span><span class="mbin">×</span><span class="mspace" style="margin-right:0.2222222222222222em;"></span></span><span class="base"><span class="strut" style="height:0.8388800000000001em;vertical-align:-0.19444em;"></span><span class="mord">1</span><span class="mord">6</span><span class="mspace"> </span><span class="mord mathnormal">p</span><span class="mord mathnormal">x</span></span></span></span>), the line height is calculated locally to 38.4 px (<span class="katex"><span class="katex-mathml"><math xmlns="http://www.w3.org/1998/Math/MathML"><semantics><mrow><mn>32</mn><mtext> </mtext><mi>p</mi><mi>x</mi><mo>×</mo><mn>1.2</mn></mrow><annotation encoding="application/x-tex">32\ px\times 1.2</annotation></semantics></math></span><span class="katex-html" aria-hidden="true"><span class="base"><span class="strut" style="height:0.8388800000000001em;vertical-align:-0.19444em;"></span><span class="mord">3</span><span class="mord">2</span><span class="mspace"> </span><span class="mord mathnormal">p</span><span class="mord mathnormal">x</span><span class="mspace" style="margin-right:0.2222222222222222em;"></span><span class="mbin">×</span><span class="mspace" style="margin-right:0.2222222222222222em;"></span></span><span class="base"><span class="strut" style="height:0.64444em;vertical-align:0em;"></span><span class="mord">1</span><span class="mord">.</span><span class="mord">2</span></span></span></span>). This will leave an appropriate amount of space between lines of text.</p>

<p>When an element has a value defined using a <em>length</em> (px, em, rem, and so forth), its computed value is inherited by child elements. When units such as ems are specified for a line height, their value is calculated, and that calculated value is passed down to any inheriting children.</p>

<blockquote>
  <p><em>length</em> — The formal name for a CSS value that denotes a distance measurement. It’s a number followed by a unit, such as 5 px. Length comes in two flavors: absolute and relative. Percentages are similar to lengths, but strictly speaking, they’re <strong>NOT</strong> considered lengths.</p>
</blockquote>

<p>When you use a unitless number, that declared value is inherited, meaning its computed value is recalculated for each inheriting child element. This will almost always be the result you want.</p>

<h2 id="custom-properties-aka-css-variables">Custom Properties (<em>aka.</em> CSS Variables)</h2>

<p>In 2015, a long-awaited CSS specification titled <em>Custom Properties for Cascading Variables</em> was published as a Candidate Recommendation. This specification introduced the concept of variables to the language, which enabled a new level of dynamic, context-based styles. You can declare a variable and assign it a value; then you can reference this value throughout your stylesheet. You can use this to reduce repetition in your stylesheet, as well as some other beneficial applications.</p>

<p>At present, support for custom properties has rolled out in all major browsers except IE, check <a href="https://caniuse.com/#feat=css-variables">https://caniuse.com/#feat=css-variables</a> for up-to-date support information.</p>

<blockquote>
  <p>If you happen to use a CSS preprocessor that supports its own variables, such as Sass or Less, you may be tempted to disregard CSS variables. However, the new CSS variables are different in nature and are far more versatile than anything a preprocessor can accomplish.</p>
</blockquote>

<p>To define a custom property, you declare it much like any other CSS property:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="py">--main-font</span><span class="p">:</span> <span class="n">Helvetica</span><span class="p">,</span> <span class="n">Arial</span><span class="p">,</span> <span class="nb">sans-serif</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The name must begin with two hyphens (<code class="language-javascript highlighter-rouge"><span class="o">--</span></code>) to distinguish it from CSS properties, followed by whatever name you’d like to use. By itself, this variable declaration doesn’t do anything until we use it.</p>

<p>A function <code class="language-javascript highlighter-rouge"><span class="kd">var</span><span class="p">()</span></code> allows the use of variables.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">p</span> <span class="p">{</span>
  <span class="nl">font-family</span><span class="p">:</span> <span class="n">var</span><span class="p">(</span><span class="n">--main-font</span><span class="p">);</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The <code class="language-javascript highlighter-rouge"><span class="kd">var</span><span class="p">()</span></code> function accepts a second parameter, which specifies a fallback value. If the variable specified in the first parameter is not defined, then the second value is used. If a <code class="language-javascript highlighter-rouge"><span class="kd">var</span><span class="p">()</span></code> function evaluates to an invalid value, the property will be set to its initial value.</p>

<h3 id="changing-custom-properties-dynamically">Changing Custom Properties Dynamically</h3>

<p>You can define the same variable inside multiple selectors, and the variable will have a different value for various parts of the page.</p>

<p>For example, you can define a variable as black, and then redefine it as white inside a particular container. Then, any styles based on that variable will dynamically resolve to black if they are outside the container and to white if inside.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nd">:root</span> <span class="p">{</span>
  <span class="py">--main-bg</span><span class="p">:</span> <span class="m">#fff</span><span class="p">;</span>
  <span class="py">--main-color</span><span class="p">:</span> <span class="m">#000</span><span class="p">;</span>
<span class="p">}</span>

<span class="nc">.panel</span> <span class="p">{</span>
  <span class="nl">background-color</span><span class="p">:</span> <span class="n">var</span><span class="p">(</span><span class="n">--main-bg</span><span class="p">);</span>
  <span class="nl">color</span><span class="p">:</span> <span class="n">var</span><span class="p">(</span><span class="n">--main-color</span><span class="p">);</span>
<span class="p">}</span>

<span class="nc">.dark</span> <span class="p">{</span>
  <span class="py">--main-bg</span><span class="p">:</span> <span class="m">#333</span><span class="p">;</span>
  <span class="py">--main-color</span><span class="p">:</span> <span class="m">#fff</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>For contexts like:</p>

<div class="language-html highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">&lt;div</span> <span class="na">class=</span><span class="s">"panel"</span><span class="nt">&gt;</span>
  ...
<span class="nt">&lt;/div&gt;</span>

<span class="nt">&lt;div</span> <span class="na">class=</span><span class="s">"dark"</span><span class="nt">&gt;</span>
  <span class="nt">&lt;div</span> <span class="na">class=</span><span class="s">"panel"</span><span class="nt">&gt;</span>
    ...
  <span class="nt">&lt;/div&gt;</span>
<span class="nt">&lt;/div&gt;</span>
</code></pre></div></div>

<p>The second <code class="language-javascript highlighter-rouge"><span class="p">.</span><span class="nx">panel</span></code> will have a dark background and white text. This is because when the panel uses these variables, they’ll resolve to the values defined on the dark container, rather than on the root.</p>

<h3 id="changing-custom-properties-with-javascript">Changing Custom Properties with JavaScript</h3>

<p>The following script shows how to access and set a property on an element.</p>

<div class="language-js highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="kd">let</span> <span class="nx">rootElement</span> <span class="o">=</span> <span class="nb">document</span><span class="p">.</span><span class="nx">documentElement</span><span class="p">;</span>
<span class="kd">let</span> <span class="nx">styles</span> <span class="o">=</span> <span class="nx">getComputedStyle</span><span class="p">(</span><span class="nx">rootElement</span><span class="p">);</span>
<span class="kd">let</span> <span class="nx">mainColor</span> <span class="o">=</span> <span class="nx">styles</span><span class="p">.</span><span class="nx">getPropertyValue</span><span class="p">(</span><span class="dl">'</span><span class="s1">--main-bg</span><span class="dl">'</span><span class="p">);</span>
<span class="nx">console</span><span class="p">.</span><span class="nx">log</span><span class="p">(</span><span class="nb">String</span><span class="p">(</span><span class="nx">mainColor</span><span class="p">).</span><span class="nx">trim</span><span class="p">());</span> <span class="c1">// trims whitespace, logs "#fff"</span>

<span class="nx">rootElement</span><span class="p">.</span><span class="nx">style</span><span class="p">.</span><span class="nx">setProperty</span><span class="p">(</span><span class="dl">'</span><span class="s1">--main-bg</span><span class="dl">'</span><span class="p">,</span> <span class="dl">'</span><span class="s1">#333</span><span class="dl">'</span><span class="p">);</span> <span class="c1">// set "--main-bg"</span>
</code></pre></div></div>

<p>With this technique, you can use only a few lines of JavaScript that make changes that’ll affect a large number of elements on the page.</p>

<p>Be aware that any declaration using <code class="language-javascript highlighter-rouge"><span class="kd">var</span><span class="p">()</span></code> will be ignored by old browsers that don’t understand it. Provide a fallback behaviour for those browsers when possible.</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">color</span><span class="o">:</span> <span class="nt">black</span><span class="o">;</span>
<span class="nt">color</span><span class="o">:</span> <span class="nt">var</span><span class="o">(</span><span class="nt">--main-color</span><span class="o">);</span>
</code></pre></div></div>

                        
Ads by Google

林宏

Frank Lin

Hey, there! This is Frank Lin (@flinhong), one of the 1.41 billion 🇨🇳. This 'inDev. Journal' site holds the exploration of my quirky thoughts and random adventures through life. Hope you enjoy reading and perusing my posts.

YOU MAY ALSO LIKE

Using Liquid in Jekyll - Live with Demos

Web Notes

2016.08.20

Using Liquid in Jekyll - Live with Demos

Liquid is a simple template language that Jekyll uses to process pages for your site. With Liquid you can output complex contents without additional plugins.

Setup an IKEv2 server with StrongSwan

Tutorials

2020.01.09

Setup an IKEv2 server with StrongSwan

IKEv2, or Internet Key Exchange v2, is a protocol that allows for direct IPSec tunnelling between two points. In IKEv2 implementations, IPSec provides encryption for the network traffic. IKEv2 is natively supported on some platforms (OS X 10.11+, iOS 9.1+, and Windows 10) with no additional applications necessary, and it handles client hiccups quite smoothly.

Practising closures in JavaScript

JavaScript Notes

2018.12.17

Practising closures in JavaScript

JavaScript is a very function-oriented language. As we know, functions are first class objects and can be easily assigned to variables, passed as arguments, returned from another function invocation, or stored into data structures. A function can access variable outside of it. But what happens when an outer variable changes? Does a function get the most recent value or the one that existed when the function was created? Also, what happens when a function invoked in another place - does it get access to the outer variables of the new place?