Spécificité

La spécificité est le moyen par lequel les navigateurs décident quelles valeurs de propriété CSS d'un élément sont les plus pertinentes et seront appliquées. La spécificité est basée sur les règles, ciblant un élément, composées par différents types de sélecteurs CSS.

Détermination de la spécificité

La spécificité est l'importance donnée à une déclaration CSS, déterminée par le nombre de types de sélecteur dans le sélecteur correspondant. Lorsque plusieurs déclarations ont une même spécificité, la dernière déclaration trouvée dans la CSS est appliquée à l'élément. La spécificité ne s'applique que lorsque le même élément est ciblé par plusieurs déclarations. Selon les règles CSS, les éléments ciblés directement seront toujours prioritaires par rapport aux règles héritées par un élément de leurs ancêtres.

Note : la proximité des éléments dans l'arborescence du document n'a aucun effet sur la spécificité.

Types de sélecteurs

Les types de sélecteurs de la liste suivante sont présentés dans l'ordre de spécificité croissante :

  1. Sélecteurs de type (ex. h1) et pseudo-éléments (ex. ::before).
  2. Sélecteurs de classe (ex. .exemple), sélecteurs d'attributs (ex. [type="radio"]) et pseudo-classes (ex. :hover).
  3. Sélecteurs d'identifiant (ex. #exemple).

Sélecteur universel (*), combinateurs (+, >, ~, " ", ||) et pseudo-classe de négation (:not()) n'ont aucun effet sur la spécificité (cependant, les sélecteurs déclarés à l’intérieur de :not() ont un effet).

Pour davantage d'informations, veuillez visiter : "Spécificité" dans "Cascade et héritage", vous pouvez également visiter : https://specifishity.com

Les styles ajoutés à un élément dans la balise HTML (ex. : style="font-weight: bold;") sont prioritaires par rapport à tout style provenant d'une feuille de style externe et peuvent donc être considérés comme ayant la plus grande spécificité.

L'exception !important

Lorsqu'une règle important est utilisée dans une déclaration de style, cette déclaration est prioritaire par rapport à toute autre déclaration. Même si, techniquement, !important n'a rien à voir avec la spécificité, elle interagit directement avec elle. Cependant, utiliser !important, est une mauvaise pratique et devrait être évité car cela rend le débogage difficile en affectant le traitement naturel en cascade dans vos feuilles de styles. Lorsque deux déclarations en conflit avec la règle !important sont appliquées au même élément, la déclaration avec la plus haute spécificité sera prise en compte.

Quelques règles de base :

  • Toujours trouver un moyen d'utiliser la spécificité avant de se résoudre à utiliser !important
  • Utiliser !important uniquement sur des CSS spécifiques à une page pour remplacer des CSS étrangères (provenant de bibliothèques externes comme Bootstrap ou normalize.css).
  • Ne jamais utiliser !important lorsque vous codez un plugin, une extension.
  • Ne jamais utiliser !important sur des CSS appliquées à un site.

Au lieu d'utiliser !important, envisager les points suivants :

  1. Utiliser mieux la cascade CSS
  2. Utiliser davantage des règles spécifiques. En indiquant un ou plusieurs éléments avant l'élément sélectionné, la règle devient plus spécifique et sa priorité est accrue :
    <div id="test">
      <span>Texte</span>
    </div>
    
    div#test span { color: green; }
    div span { color: blue; }
    span { color: red; }
    
    L'ordre n'a aucune importance, le texte sera en vert (green) car cette règle est plus spécifique. (Également, la règle pour bleu est prioritaire par rapport à celle pour rouge, là encore l'ordre n'a pas d'importance)
  3. En rapport avec le point 2, le fait de dupliquer des sélecteurs simples afin d''augmenter la spécificité lorsque vous n'avez aucun autre moyen de le faire, n'a aucun sens.
    #monId#monId span { color: yellow; }
    .maClasse.maClasse span { color: orange; }
    

Utilisation de !important

A) Prendre le dessus sur les styles dans les balises HTML

L'effet de votre fichier CSS global qui définit les aspects visuels de votre site de manière générale peut être supplanté par les styles indiqués directement dans les balises HTML. Les deux pratiques que sont les styles dans les balises HTML et l'utilisation de !important sont considérées comme étant très mauvaises, mais parfois, vous devez pouvoir prioriser la dernière par rapport à la première.

Dans ce cas, vous pouvez définir certains styles comme étant !important dans votre fichier CSS global, et ainsi les rendre prioritaires par rapport au style définit dans les balises HTML.

<div class="toto" style="color: red;">De quelle couleur suis-je ?</div>
.toto[style*="color: red"] {
  color: firebrick !important;
}

Beaucoup de frameworks JavaScript et de bibliothèques ajoutent des styles dans les balises HTML. En utilisant !important avec un sélecteur ciblant précisément il est possible de passer outre les styles dans ces balises HTML.

B) Prendre le dessus sur la haute spécificité
#unElement p {
  color: blue;
}

p.genial {
  color: red;
}

Comment faites-vous pour rendre les paragraphes de classe genial toujours en rouge, même ceux qui se trouvent dans #unElement ? Sans !important, la première règle aura davantage de spécificité et prendra le dessus sur la seconde règle.

Prendre le dessus sur !important

A) Ajouter une autre règle CSS avec !important et, soit donner au sélecteur une spécificité plus haute (en ajoutant une balise, un identifiant ou une classe au sélecteur), soit ajouter une règle CSS avec le même sélecteur plus loin dans le fichier CSS. Cela fonctionne car lorsqu'on est en présence d'une égalité de spécificité, la dernière règle définie est prise en compte.

Quelques exemples avec une spécificité accrue :

table td    { height: 50px !important; }
.monTableau td { height: 50px !important; }
#monTableau td { height: 50px !important; }

B) Ou ajouter le même sélecteur après le sélecteur existant&nbsp;:

```css
td { height: 50px !important; }

C) Ou, de préférence, réécrire la règle d'origine afin d'éviter totalement l'utilisation de !important.

[id="unElement"] p {
  color: blue;
}

p.genial {
  color: red;
}

Inclure un identifiant comme un sélecteur d'attribut au lieu d'un sélecteur d'identifiant (`#someElement`) lui donne la même spécificité qu'une classe. Les deux sélecteurs ci-dessus ont à présent la même importance. Lors d'une égalité de spécificité, la dernière règle définie est prise en compte.

#### Pour davantage d'informations, visiter&nbsp;:

- <https://stackoverflow.com/questions/3706819/what-are-the-implications-of-using-important-in-css>
- <https://stackoverflow.com/questions/9245353/what-does-important-in-css-mean>
- <https://stackoverflow.com/questions/5701149/when-to-use-important-property-in-css>
- <https://stackoverflow.com/questions/11178673/how-to-override-important>
- <https://stackoverflow.com/questions/2042497/when-to-use-important-to-save-the-day-when-working-with-css>

### Les exceptions :is() et :not()

La pseudo-classe de correspondance :is() 
    Experimental
 et la pseudo-classe de négation :not() _ne_ sont _pas_ considérées comme pseudo-classes dans le calcul de la spécificité. Mais les sélecteurs placés dans la pseudo-classe comptent comme des sélecteurs normaux lorsqu'on détermine le nombre de [types de sélecteurs](#types_de_sélecteurs).

Ce bout de CSS&hellip;

```css
div.outer p {
  color: orange;
}

div:not(.outer) p {
  color: blueviolet;
}

…lorsqu'il est utilisé avec l'HTML suivant…

<div class="outer">
  <p>Ceci est dans la div externe.</p>
  <div class="inner">
    <p>Ce texte est dans la div interne.</p>
  </div>
</div>

…s'affiche sur l'écran ainsi :

L'exception :where()

La pseudo-classe d'ajustement de spécificité :where() Experimental a toujours une spécificité remplacée par zéro.

Ce bout de CSS…

div:where(.externe) p {
  color: orange;
}

div p {
  color: blueviolet;
}

…utilisé avec l'HTML suivant…

<div class="externe">
  <p>Ceci est dans la div externe.</p>
  <div class="interne">
    <p>Ce texte est dans la div interne.</p>
  </div>
</div>

&hellip;s'affiche sur l'écran ainsi&nbsp;:



### Spécificité liée à la forme

La spécificité est basée sur la forme d'un sélecteur. Dans le cas suivant, le sélecteur `*[id="toto"]` compte pour un sélecteur d'attribut lorsqu'on détermine la spécificité du sélecteur, même s'il cible un identifiant.

Les styles CSS suivants&hellip;

```css
*#toto {
  color: green;
}

*[id="toto"] {
  color: purple;
}

…utilisés avec ce balisage…

<p id="toto">Je suis un texte d’exemple.</p>

…ressemblera finalement à cela :

Cela est dû au fait qu’il corresponde au même élément mais que le sélecteur d’identifiant a une spécificité plus haute.

Ignorance de la proximité dans l'arborescence

La proximité d'un élément avec d'autres éléments qui sont référencés dans un sélecteur n'a aucun impact sur la spécificité. La déclaration de style suivante…

body h1 {
  color: green;
}

html h1 {
  color: purple;
}

…avec l'HTML suivant…

<html>
  <body>
    <h1>Ici un titre !</h1>
  </body>
</html>

…s'affichera ainsi :

Cela est dû au fait que les deux déclarations ont le même nombre de types de sélecteurs, mais le sélecteur html h1 est déclaré en dernier.

Éléments ciblés directement vs styles hérités

Les styles pour un élément directement ciblé prendront toujours le dessus sur les styles hérités, sans prise en compte de la spécificité de la règle héritée. Cette CSS…

#parent {
  color: green;
}

h1 {
  color: purple;
}

…avec l'HTML suivant…

<html>
  <body id="parent">
    <h1>Ici un titre !</h1>
  </body>
</html>

…s'affichera également ainsi :

Ceci est dû au fait que le sélecteur h1 cible l'élément spécifiquement, mais le sélecteur vert (green) est uniquement hérité de ses parents.

Caractéristiques

Voir aussi