CSS selectors
// updated 2025-05-09 12:01
A CSS selector refers to a syntax that represents a specific group of related objects on a webpage. We use this convention in order to give each item in the group the same styling:
- e.g. Every second-level heading (<h2>) can enjoy the same font sizings or line heights (or whatever) with a concise, one-line CSS rule!
In this page, we will go over the basic structure of a CSS selector and then look at the following selector types:
- element
- class (.)
- id (#)
- attribute ([])- specific value
- partial match (^, *, $)
 
- child (>)
- descendant ( )
- sibling (+, ~)
Structure
The abstract template for two CSS selectors and their properties:
1(selector1) {
2  (property1) : (value1); 
3}
4
5(selector2) {
6  (property2) : (value2);
7}
8
9...Example
For example, to make a link red with no underline:
1a {
2  color: red;
3  text-decoration: none; 
4}Selector types
Element selectors
To update the style for a top-level element, such as an <h1> or <a> we simply use the HTML element's name as the selector, followed by curly braces, e.g.:
1h1 {
2  ...
3}1a {
2  ...
3}Class selectors
To update the style for any element that has a specific class, e.g. in HTML:
1<!-- index.html -->
2<html>
3  
4  <head>
5    <link rel="stylesheet" href="style.css" />
6  </head>
7  
8  <body>
9    <h1 class="redFont">This will be red!</h1>
10    <p class="redFont">This will also be red!</p>
11  </body>
12  
13</html>Then, in CSS, we would use the dot notation to select all items of a class:
1/* style.css */
2
3.redFont { 
4  color: red;
5}With that class selector, both the h1 and p will have red font. This also means that we don't need to restrict the red font to just h1 or make all h1 elements have a red font. We will see more flexibility that comes with CSS!
ID selectors
To update the style for a unique element on the page, e.g.:
1<!-- index.html -->
2<html>
3  
4  <head>
5    <link rel="stylesheet" href="style.css" />
6  </head>
7  
8  <body>
9    <section id="my-map">
10      ...
11    </section>
12  </body>
13  
14</html>We would use the hashtag notation as such:
1/* style.css */
2
3#my-map {
4  height: 600px;
5  width: 800px;
6}This works well when we want to restrict a certain styling to one and only one element of an HTML page!
Attribute selectors
To update the style for any element that has a specific attribute, e.g.
1<!-- index.html -->
2<html>
3  
4  <head>
5    <link rel="stylesheet" href="style.css" />
6  </head>
7  
8  <body>
9    <section data-status="danger">
10      ...
11    </section>
12    <section data-status="warning">
13      ...
14    </section>
15  </body>
16  
17</html>We would use the square bracket notation:
1/* style.css */
2
3[data-status] {
4  border: 2px;
5  border-style: solid;
6  border-color: black;  
7}
8
9[data-status="danger"] {
10  background-color: red;
11}
12
13[data-status="warning"] {
14  background-color: orange;
15}Note that we could specify attribute values in our selectors to further customize each value of an HTML tag attribute.
Partial match selectors
Looking at the attribute selectors above, we can see this has many fine-grained implications:
1/* style.css */
2
3/* starts with "war" */
4[data-status^="war"] { ... } 
5
6/* contains "nin" */
7[data-status*="nin"] { ... }
8
9/* ends with "ing" */
10[data-status$="ing"] { ... }In the example above, any element with a data-status value of:
- warningwould take the styles in all three declaration blocks
- churningwould take the styles in only the last two blocks
- wartimewould take the style in only the first block
An mnemonic to remember the symbols:
- a ^resembles a "hat" (i.e. "heads with" or "begins with")
- a *resembles a "star" (i.e. "starring" or "featuring")
- a $for "the bottom line" (i.e. "ending with")
Child selectors
Let us define the parent element and child element as such:
1<div> <!-- parent -->
2  <p> <!-- child -->
3    ...
4  </p>
5</div>Then, to target only the child element, we would simply use the "greater than" notation:
1div > p {
2  ...
3}This would target only the p tag within the div tag and nothing else within the div.
Descendant selectors
To target "grand-children" and "great-grand-children" elements (and so on), we would just keep chaining the "greater than" notation:
1/* span as the grandchild of div */
2div > p > span {
3  ...
4}
5
6/* strong as the great-grandchild of div */
7div > p > span > strong {
8  ...
9}In order to select all descendants of an ancestor element, regardless of "generation", simply place them side by side:
1main p {
2  ...
3}Thus, that would target all p tags in the main HTML tag, regardless of whether p is a child, grand-child, great-grand-child, or however far nested p is from main!
Sibling selectors
Suppose that we have an element on the same level as another element, e.g.:
1<header>
2  <h2>...</h2>
3  <p>...</p>
4  <aside>...</p>
5</header>To reach an adjacent element on the same level as another element, we can use the "plus" notation:
1h2 + p {
2  ...
3}That CSS would then target any p tag that lies right after an h2 tag. Here, we get into some really refined and specific CSS selection! When working with CSS "in the real world", this kind of selection seldom happens but it exists for us when we need it! 
To reach a non-adjacent element on the same level, we would use the "tilde" notation:
1h2 ~ aside {
2  ...
3}That should about cover most of the selectors a front-end developer would encounter on a daily basis! Of course, much to anyone's amazement, even more CSS selectors exist. At that point, we would simply search (or ask AI) for them when the need arises!