loading
Cascade - CSS Fundamentals

Cascade - CSS Fundamentals

bottom up, the core of CSS

                        <p>Let’s begin with the cascade, the <strong><em>C</em></strong> in CSS. How it works and how to work with it practically.</p>

<h2 id="the-cascade">The Cascade</h2>

<blockquote>
  <p>The <strong>cascade</strong> is an algorithm that defines how to combine property values originating from different sources. It lies at the core of CSS, as emphasized by the name: <strong>Cascading</strong> Style Sheets.</p>
</blockquote>

<p>Fundamentally, CSS is about declaring style rules. There are often several ways to accomplish the same thing in CSS. Depending on which solution you use, you may get wildly different results when the structure of the HTML changes, or when the styles are applied to different pages. A key part of CSS development comes down to writing rules in such a way that they’re predicatable. While predicting how rules behave requires an understanding of the cascade.</p>

<p>When two or more rules target the same element on the page, the rules may provide conflicting declarations. For example in the following markups:</p>

<div class="language-html highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">&lt;header</span> <span class="na">class=</span><span class="s">"page-header"</span><span class="nt">&gt;</span>
  <span class="nt">&lt;h1</span> <span class="na">id=</span><span class="s">"page-title"</span> <span class="na">class=</span><span class="s">"title"</span><span class="nt">&gt;</span>Wombat Coffee Roasters<span class="nt">&lt;/h1&gt;</span>
<span class="nt">&lt;/header&gt;</span>
</code></pre></div></div>

<p>with the following CSS rules:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">h1</span> <span class="p">{</span>
  <span class="nl">font-family</span><span class="p">:</span> <span class="nb">serif</span><span class="p">;</span>
<span class="p">}</span>
<span class="nf">#page-title</span> <span class="p">{</span>
  <span class="nl">font-family</span><span class="p">:</span> <span class="nb">sans-serif</span><span class="p">;</span>
<span class="p">}</span>
<span class="nc">.title</span> <span class="p">{</span>
  <span class="nl">font-family</span><span class="p">:</span> <span class="nb">monospace</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The title can’t have three fonts at one time. So, which one will it be?</p>

<p>To determine the answer, you should know that the browser follows a set of rules, so the result is predictable. In this case, the rules dictate that the second declaration, which has an ID selector, wins; the title will have a <code class="language-plaintext highlighter-rouge">sans-serif</code> font.</p>

<p>The <em>cascade</em> is the name for this set of rules. It determines how conflicts are resolved, and it’s a fundamental part of how the language works. When declarations conflict, the cascade considers three things to resolve the difference:</p>

<ul>
  <li><em>Stylesheet origin</em> - where the styles come from. Your styles are applied in conjunction with the browser’s default styles.</li>
  <li><em>Selector specificity</em> - which selectors take precedence over which.</li>
  <li><em>Source order</em> - order in which styles are declared in the stylesheet.</li>
</ul>

<p><img src="https://i.loli.net/2019/06/21/5d0c9ef87a11b62779.png" alt="High-level flowchart of the cascade showing declaration precedence - CSS in Depth (2018)" /></p>

<p>These rules allow browsers to behave predicatably when resolving any ambiguity in the CSS.</p>

<blockquote>
  <p>Note that <a href="https://developer.mozilla.org/en-US/docs/Web/CSS/@import"><code class="language-plaintext highlighter-rouge">@import</code></a> and <a href="https://developer.mozilla.org/en-US/docs/Web/CSS/@charset"><code class="language-plaintext highlighter-rouge">@charset</code></a> obey specific algorithms and aren’t affected by the cascade algorithm.</p>
</blockquote>

<h2 id="stylesheet-origin">Stylesheet Origin</h2>

<p>There’re different types, or origins, of stylesheets. Yours are called <em>author</em> styles; there are also user agent styles, which are the browser’s default styles. User agent styles have lower priority.</p>

<blockquote>
  <p>Some browsers let users define a <em>user stylesheet</em>. This is considered a third origin, with a priority between user agent and author styles.</p>
</blockquote>

<p>The user agent styles set things you typically want, so they don’t do anything entirely unexpected. When you don’t like what they do to a certain property, just set your own values in your stylesheet.</p>

<h3 id="important-declarations">!important Declarations</h3>

<p>A declaratiion can be marked important by adding <code class="language-plaintext highlighter-rouge">!important</code> to the end of the declaration, before the semicolon:</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">red</span> <span class="o">!</span><span class="nt">important</span><span class="o">;</span>
</code></pre></div></div>

<p>These declarations marked with <code class="language-plaintext highlighter-rouge">!important</code> are treated as a higher-priority origin, so the overall order of preference, in decreasing order, is this:</p>

<ol>
  <li>Author important;</li>
  <li>Author;</li>
  <li>User agent.</li>
</ol>

<p>The cascade independently resolves conflicts for every property of every elements on the page.</p>

<h3 id="specificity">Specificity</h3>

<p>If conflicting declarations can’t be resolved based on their origin, the browser next tries to resolve them by looking at their <em>specificity</em>. The browsers evaluates specificity in two parts: styles applied inline in the HTML and styles applied using a selector.</p>

<h4 id="inline-styles">Inline Styles</h4>

<p>If you use an HTML <code class="language-plaintext highlighter-rouge">style</code> attribute to apply styles, the declarations are applied only to the element. These are, in effect, “scoped” declarations, which override any declarations applied from your stylesheet or a <code class="language-plaintext highlighter-rouge">&lt;style&gt;</code> tag.</p>

<p>To override inline declarations in your stylesheet, you’ll need to add an <code class="language-plaintext highlighter-rouge">!important</code> to the declaration, shifting it into a higher-priority origin. If the inline styles are marked important, then nothing can override them.</p>

<h4 id="selector-specificity">Selector Specificity</h4>

<p>The second part of specificity is determined by the selectors. Different types of selectors have different specificities. An ID selector has a higher specificity than a class selector. Similarly, a class selector has a higher specificity than a tag selector (also called a <em>type selector</em>).</p>

<p>The exact rules of specificity are:</p>

<ol>
  <li>If a selector has more IDs, it wins (that is, it is more specific).</li>
  <li>If that results in a tie, the selector with the most classes wins.</li>
  <li>If that results in a tie, the selector with the most tag names wins.</li>
</ol>

<p>For example:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="c">/* (1) four tags */</span>
<span class="nt">html</span> <span class="nt">body</span> <span class="nt">header</span> <span class="nt">h1</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">blue</span><span class="p">;</span>
<span class="p">}</span>

<span class="c">/* (2) three tags and one class */</span>
<span class="nt">body</span> <span class="nt">header</span><span class="nc">.page-header</span> <span class="nt">h1</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">orange</span><span class="p">;</span>
<span class="p">}</span>

<span class="c">/* (3) two classes */</span>
<span class="nc">.page-header</span> <span class="nc">.title</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">green</span><span class="p">;</span>
<span class="p">}</span>

<span class="c">/* (4) one ID */</span>
<span class="nf">#page-title</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">red</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The most specific selector here is (4), with one ID. The next specific is (3), with two class names. Selector (3) has a higher specificity than selector (2), despite its length: two classes are more specific than one class. The least specific selector is (1), even it has four element types (tag names).</p>

<blockquote>
  <p>Pseudo-class selectors (<em>e.g.</em>, <code class="language-plaintext highlighter-rouge">:hover</code>) and attribute selectors (<em>e.g.</em>, <code class="language-plaintext highlighter-rouge">[type="input"]</code>) each have the same specific as a class selector. The universal selector (<code class="language-plaintext highlighter-rouge">*</code>) and combinators (<code class="language-plaintext highlighter-rouge">&gt;</code>, <code class="language-plaintext highlighter-rouge">+</code>, and <code class="language-plaintext highlighter-rouge">~</code>) have no effect on specificity.</p>
</blockquote>

<p>If you add a declaration to your CSS and it seems have no effect, often it’s because a more specific rule is overriding it.</p>

<h4 id="a-notation-for-specificity">A Notation for Specificity</h4>

<p>A common way to indicate specificity is in a number form, often with commas between each number. For example, “1,2,2” indicates a specificity of one ID, two classes, and two tags. IDs having the highest priority are listed first, followed by classes, then tags.</p>

<p>The selector <code class="language-plaintext highlighter-rouge">#page-header #page-title</code> has two IDs, no classes, and no tags. We can say this selector has a specificity of “2,0,0”. The <code class="language-plaintext highlighter-rouge">ul li</code>, with two tags but no IDs or classes, has a specificity of “0,0,2”.</p>

<p>It now becomes a matter of comparing the numbers to determine which selector is more specific. A specificity of “1,0,0” takes precedence over a specificity of “0,2,2” and ever over “0,10,0”.</p>

<p>Occasionally, people use a four-number notation with a 0 or 1 in the most significant digit to represent whether a declaration is applied via inline styles. In this case, an inline style has a specificity of “1,0,0,0”. This would override styles applied via selectors, which could be indicated as having specificities of “0,1,2,0” or something similar.</p>

<h4 id="specificity-considerations">Specificity Considerations</h4>

<p>To raise the specificity, the quickest fix is to add an <code class="language-plaintext highlighter-rouge">!important</code> to the declaration you want to favor. This works because the <code class="language-plaintext highlighter-rouge">!important</code> annotation raises the declaration to a higher priority origin. Sure, it’s easy, but it’s also a naive fix and not recommended.</p>

<p>Instead of <em>raising</em> the specificity of the selector, we can also <em>lower</em> the specificity of the selector to get the same result. It is generally best to keep specificity low when you can, so when you need to override something, your options are open.</p>

<h3 id="source-order-of-stylesheet">Source Order of Stylesheet</h3>

<p>The third and final step to resolving the cascade is source order. If the origin and the specificity are the same, then the declaration that appears later in the stylesheet, or appears in a stylesheet included later on the page, takes precedence. This means you can manipulate the source order to style your featured link. If you make the two conflicting selectors equal in specificity, then whichever appears last wins.</p>

<h4 id="link-styles-and-source-order">Link Styles and Source Order</h4>

<p>For styling links (<code class="language-plaintext highlighter-rouge">&lt;a&gt;</code> tags), it should go in a certain order. That’s because source order affects the cascade. This listing shows styles for links on a page in the “corret” order, the so-called “<strong>L</strong>o<strong>V</strong>e/<strong>HA</strong>te” order:</p>

<div class="language-css highlighter-rouge"><div class="highlight"><pre class="highlight"><code><span class="nt">a</span><span class="nd">:link</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">blue</span><span class="p">;</span>
  <span class="nl">text-decoration</span><span class="p">:</span> <span class="nb">none</span><span class="p">;</span>
<span class="p">}</span>
<span class="nt">a</span><span class="nd">:visited</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">purple</span><span class="p">;</span>
<span class="p">}</span>
<span class="nt">a</span><span class="nd">:hover</span> <span class="p">{</span>
  <span class="nl">text-decoration</span><span class="p">:</span> <span class="nb">underline</span><span class="p">;</span>
<span class="p">}</span>
<span class="nt">a</span><span class="nd">:active</span> <span class="p">{</span>
  <span class="nl">color</span><span class="p">:</span> <span class="no">red</span><span class="p">;</span>
<span class="p">}</span>
</code></pre></div></div>

<p>The cascade is the reason this order matters: given the same specificity, later styles override earlier styles. If two or more of these states are true of one element at the same time, the last one override the others. If the user hovers over a visited link, the hover styles take precedence. If the user activates the link (<em>i.e.</em>, clicks it) while hovering over it, the active styles take precedence.</p>

<h4 id="cascaded-values">Cascaded Values</h4>

<p>The browser follows these three steps — origin, specificity, and source order to resolve every property for every element on the page. A declaration that “wins” the cascade is called a <em>cascaded value</em>. There’s at most one cascaded value per property per element.</p>

<h2 id="inheritance">Inheritance</h2>

<p>If an element has no cascaded value for a given property, it may inherit one from an ancestor element. It’s common to apply a <code class="language-plaintext highlighter-rouge">font-family</code> to the <code class="language-plaintext highlighter-rouge">&lt;body&gt;</code> element. All the ancestor elements within will then inherit this font.</p>

<p>Note that <strong>NOT</strong> all properties are inherited, however. By default, only certain ones are. In general, these are the properties you’ll <em>want</em> to be inherited. They are primarily properties pertaining to text: <code class="language-plaintext highlighter-rouge">color</code>, <code class="language-plaintext highlighter-rouge">font</code>, <code class="language-plaintext highlighter-rouge">font-family</code>, <code class="language-plaintext highlighter-rouge">font-size</code>, <code class="language-plaintext highlighter-rouge">font-weight</code>, <code class="language-plaintext highlighter-rouge">font-variant</code>, <code class="language-plaintext highlighter-rouge">font-style</code>, <code class="language-plaintext highlighter-rouge">line-height</code>, <code class="language-plaintext highlighter-rouge">letter-spacing</code>, <code class="language-plaintext highlighter-rouge">text-align</code>, <code class="language-plaintext highlighter-rouge">text-indent</code>, <code class="language-plaintext highlighter-rouge">text-transform</code>, <code class="language-plaintext highlighter-rouge">white-space</code>, and <code class="language-plaintext highlighter-rouge">word-spacing</code>.</p>

<p>A few others inherit as well, such as the list properties: <code class="language-plaintext highlighter-rouge">list-style</code>, <code class="language-plaintext highlighter-rouge">list-style-type</code>, <code class="language-plaintext highlighter-rouge">list-style-position</code>, and <code class="language-plaintext highlighter-rouge">list-style-image</code>. The table border properties, <code class="language-plaintext highlighter-rouge">border-collapse</code> and <code class="language-plaintext highlighter-rouge">border-spacing</code>, are also inherited.</p>

<p>Above is not quite a comprehensive list, but very nearly.</p>

<h2 id="special-values">Special Values</h2>

<p>There are two special values that you can apply to any property to help manipulate the cascade: <code class="language-plaintext highlighter-rouge">inherit</code> and <code class="language-plaintext highlighter-rouge">initial</code>.</p>

<p>Using the <code class="language-plaintext highlighter-rouge">inherit</code> keyword, you can override another value with this, and it will cause the element to inherit that value from its parent.</p>

<p>Sometimes you’ll find you have styles applied to an element that you want to undo. You can do this by specifying the keyword <code class="language-plaintext highlighter-rouge">initial</code>. Every CSS property has an initial, or default, value. If you assign the value <code class="language-plaintext highlighter-rouge">initial</code> to that property, then it effectively resets to its default value. The benefit of this is you don’t have to think about it as much. If you want to remove a border from an element, set <code class="language-plaintext highlighter-rouge">border: initial</code>. If you want to restore an element to its default width, set <code class="language-plaintext highlighter-rouge">width: initial</code>.</p>

<blockquote>
  <p>Declaring <code class="language-plaintext highlighter-rouge">display: initial</code> is equivalent to <code class="language-plaintext highlighter-rouge">display: inline</code>. It won’t evaluate to <code class="language-plaintext highlighter-rouge">display: block</code> regardless of what type of element you apply it to. That’s because <code class="language-plaintext highlighter-rouge">initial</code> resets to the initial value for the property, not the element.</p>
</blockquote>


                        

林宏

Frank Lin

Hey, there! This is Frank Lin (@flinhong), one of the 1.4 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 templating language that Jekyll uses to process pages on your site. With Liquid you can output an modify variables, have logic statements inside your pages and loop over content.

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?