data:image/s3,"s3://crabby-images/918c4/918c42c87ec71b87bf81df5938d1457bebf228f1" alt=""
Around the end of the first "CSS to The Rescue" lecture (given by Sanne 't Hooft and Nils Blinder), we were informed about the final assignment, containing 4 choices
- Control Panel
- Firework Show
- Paper-airplane
- Rubix Cube
I picked fireworks for my project, but I wanted to spice it up. Just before the assessment, Sanne t'Hooft showed me some of his work that totally blew me away. It hardly had any JS, and I was like, "Wow!". My favorite one was Freddy's Fabulous Fish Tanks It's like a mix of a control panel and fireworks, even though the fish obviously don't explode. But when you mix up all the species, they create this incredible display.
data:image/s3,"s3://crabby-images/36403/3640317bbdaeaf3685a08eea09572613364eb49a" alt="SchermÂafbeelding 2024-03-13 om 23 19 17"
I went for a similar idea, but with birds this time. Picture a bunch of birds flying at different speeds. They put on a colorful show just like fireworks.They create a colorful spectacle just like fireworks. Each bird flies at a different speed, forming a diverse mix that can mesmerize the user. But here's the twist: the user can shoot the birds from the sky. Some are easier targets than others because of their speed. When shot, the birds simply fall off.
data:image/s3,"s3://crabby-images/8b3b1/8b3b1459fcec6478f53f143e94369b91d17c64da" alt=""
Here are some sketches to give you an idea of what I want to achieve. Each house has a different type of bird, each faster than the last. When you hover over these houses, you can see the birds peeking outClicking on several birdhouses will launch them to the center of the screen. One can add as much birds as they want to, but also shoot them down. Turning them red and drop them down.
I could consider using three different types of aninmations:
- One taking off, flying to the circle
- It will then start flying in circles
- And when it's shot, the death animation plays
data:image/s3,"s3://crabby-images/e96c0/e96c0f5116a0a24953db6a76be7f88fe5e5255a7" alt=""
It was time to get to coding! The list-items part creates a list of birds with radio buttons for selection. Each list-items element represents a list item and contains an input radio button along with the name of the bird displayed as a paragraph. Each radio button has a unique name attribute to group them together, allowing the user to select only one bird at a time.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Testing Has</title>
<link rel="stylesheet" href="./style.css">
</head>
<body>
<header></header>
<main>
<ul>
<li>
<input type="radio" name="addBird1">
<p>bird1</p>
</li>
<li>
<input type="radio" name="addBird2">
<p>bird2</p>
</li>
<li>
<input type="radio" name="addBird3">
<p>bird3</p>
</li>
<li>
<input type="radio" name="addBird4">
<p>bird4</p>
</li>
<li>
<input type="radio" name="addBird5">
<p>bird5</p>
</li>
<li>
<input type="radio" name="addBird6">
<p>bird6</p>
</li>
<li>
<input type="radio" name="addBird7">
<p>bird7</p>
</li>
<li>
<input type="radio" name="addBird8">
<p>bird8</p>
</li>
</ul>
</main>
<footer></footer>
</body>
</html>
data:image/s3,"s3://crabby-images/aca38/aca38025a0c5a59068c18b706b13688be6d77eaf" alt="SchermÂafbeelding 2024-03-14 om 06 00 42"
Regarding the CSS, I recognized that I would likely need to employ repetitive selectors to achieve the functionality I desired. However, this approach wasn't practical. Additionally, it would complicate any future changes, as identifying what adjustments I wanted to make would become akin to solving a large puzzle.
section:has( input[name^="addBird"]:checked) + article p {
display: block;
}
section:has( input[name="addBird2"]:checked) + article p:nth-of-type(2) {
display: block;
}
section input:nth-of-type(2) {
display: none;
}
section input:nth-of-type(3) {
display: none;
}
section input:nth-of-type(4) {
display: none;
}
section input:nth-of-type(5) {
display: none;
}
/* And so on, up to 19 */
section:has(input:first-of-type:checked) input:nth-of-type(2){
display: block;
}
section:has(input:nth-of-type(2):checked) input:nth-of-type(3){
display: block;
}
section:has(input:nth-of-type(3):checked) input:nth-of-type(4){
display: block;
}
section:has(input:nth-of-type(4):checked) input:nth-of-type(5){
display: block;
}
/* And once again, up to 19 */
With Sanne's help, I created multiple variables for the list items, intending to create a method where my birds would dynamically appear with an added 'flying' animation. Each bird's speed and position are controlled by these custom variables. The list is styled as a grid, with radio buttons hidden by default. When a radio button is checked, the corresponding bird's name is displayed and animated to simulate flying. The animation moves the bird vertically while increasing its font size based on its position in the list.
li:nth-child(1) { --i:1; --x:10; --y:46; }
li:nth-child(2) { --i:2; --x:50; --y:68; }
li:nth-child(3) { --i:3; --x:23; --y:23; }
li:nth-child(4) { --i:4; --x:67; --y:97; }
li:nth-child(6) { --i:6; --x:91; --y:24; }
li:nth-child(5) { --i:5; --x:12; --y:56; }
li:nth-child(7) { --i:7; --x:33; --y:89; }
li:nth-child(8) { --i:8; --x:47; --y:35; }
li:nth-child(10) { --i:10; --x:13; --y:23; }
ul {
display: grid;
justify-items: start;
margin:0;
padding:0;
list-style:none;
}
ul li {
grid-column-start:1;
grid-row-start:1;
}
ul input {
display:none;
}
input[name="addBird1"]:not(:checked) {
display:block;
}
li:has(input:checked) + li input:not(:checked) {
display:block;
}
p {
display: none;
}
input:checked + p {
display:block;
animation: flying 1s ease;
animation-iteration-count: infinite;
font-size:calc(1em * var(--i));
position: fixed;
left: calc(1vw * var(--x));
top: calc(1dvh * var(--y));
}
@keyframes flying {
0% {
translate: 0 10rem;
}
100% {
translate: 0 1rem;
}
}
data:image/s3,"s3://crabby-images/f822c/f822c6440e7b3c534bcd6469f09cacc070c29887" alt=""
It was time to get creative with styling! Although I'm still refining the HTML, I decided to start adding some design elements. After all, that's the essence of the course we're following, right? I searched online for some birdhouse SVGs, and I even found one with a matching bird.
I also edited the list-items, with the actual bird.
<li>
<input type="radio" name="addBird5">
<p>bird5</p>
</li>
<li>
<input type="radio" name="addBird8">
<img src="./imgs/birdie.png" alt="">
</li>
I also edited the list-items variables with additional filters! Each time a bird would appear, it would be displayed in a different color. When a user selects a bird, it gracefully flies onto the screen with a smooth animation, while its size adjusts based on its position in the list. The radio buttons don't look like typical buttons; they're transformed into adorable birdhouses, adding a delightful touch to the interface. By hiding the default appearance of the radio buttons and cleverly inserting the birdhouse backgrounds, we ensure a cohesive and visually appealing design. Plus, organizing everything in a neat grid layout adds structure and makes navigation a breeze. Together, these CSS styles create a playful and engaging user experience that captures the essence of our webpage beautifully.
li:nth-child(1) { --i:1; --x:0; --y:0; filter: invert(14%) sepia(86%) saturate(6754%) hue-rotate(359deg) brightness(98%) contrast(115%);}
li:nth-child(2) { --i:2; --x:50; --y:68; filter: invert(89%) sepia(26%) saturate(2802%) hue-rotate(114deg) brightness(103%) contrast(101%);}
li:nth-child(3) { --i:3; --x:23; --y:23; filter: invert(10%) sepia(90%) saturate(7456%) hue-rotate(273deg) brightness(101%) contrast(121%);}
li:nth-child(4) { --i:4; --x:67; --y:97; filter: invert(76%) sepia(53%) saturate(598%) hue-rotate(344deg) brightness(100%) contrast(92%);}
li:nth-child(6) { --i:6; --x:91; --y:24; filter: invert(93%) sepia(24%) saturate(629%) hue-rotate(357deg) brightness(93%) contrast(102%);}
li:nth-child(5) { --i:5; --x:12; --y:56; filter: invert(90%) sepia(20%) saturate(900%) hue-rotate(57deg) brightness(101%) contrast(97%);}
li:nth-child(7) { --i:7; --x:33; --y:89; filter: invert(56%) sepia(75%) saturate(6663%) hue-rotate(324deg) brightness(98%) contrast(95%);}
li:nth-child(8) { --i:8; --x:47; --y:35; filter: invert(38%) sepia(100%) saturate(912%) hue-rotate(345deg) brightness(101%) contrast(102%);}
li:nth-child(10) { --i:10; --x:13; --y:23; filter: invert(48%) sepia(100%) saturate(5349%) hue-rotate(285deg) brightness(95%) contrast(131%);}
[type=radio] {
appearance: none;
}
[type=radio]::before {
content: "";
width: 4rem;
height: 4rem;
background-image: url(./imgs/birdhouse.svg);
background-repeat: no-repeat;
position: absolute;
top: 2px;
left: 2px;
z-index: -1;
}
input:checked+img{
display: block;
animation: flying 1s ease;
animation-fill-mode: forwards;
font-size: calc(1em * var(--i));
width: 3rem;
position: fixed;
left: calc(1vw * var(--x));
top: calc(1dvh * var(--y));
-webkit-transform: scaleX(-1);
transform: scaleX(-1);
}
data:image/s3,"s3://crabby-images/6e2ab/6e2ab8c5da46c5efe5bf789d91d30ac2ecaa74b4" alt=""
data:image/s3,"s3://crabby-images/0ce72/0ce722781818c5ff2f3ae016862b82332add2627" alt=""
Sanne helped me sketch out the birds, their homes, and the areas they're supposed to fly in on paper. We made sure to consider everything, like margins, padding, and how long the animations should last. Before, I was using an offset-path, but it didn't work out as I wanted. Sunflower (The Yellowbird) ended up flying in the wrong direction, and changing the viewport messed up everything's position.
data:image/s3,"s3://crabby-images/2ca93/2ca93788cb384b406d1d94f30ff4045642f203af" alt=""
ul:first-of-type {
--houseSize: 5em;
--birdSize:calc( var(--houseSize) - var(--housePadding) * 2);
--houseTop: calc( var(--navPadding) + var(--houseSize) * .5 );
--housePadding: 1em;
--houseLeft: calc(
var(--navPadding)
+
(var(--i) - .5) * var(--houseSize)
+
(var(--i) - 1) * var(--navGap)
);
position: relative;
/* background-color: red; */
background-image:
url("./birdhouse-hover.svg"),
url("./birdhouse.svg");
background-repeat: no-repeat;
background-position:center;
background-size: 0 0, contain;
margin: 0;
padding: 0;
list-style: none;
width: var(--houseSize);
aspect-ratio: 1;
display: grid;
}
This code snippet styles the first ul element in a document, primarily for rendering a grid of birdhouses. It sets custom CSS variables to define sizes and positions relative to the birdhouses and birds, such as --houseSize for the size of the birdhouse and --birdSize for the bird's size. Additionally, it calculates the top and left positions of each birdhouse based on the number of birdhouses and the gap between them. The background image of each birdhouse is specified with a hover effect using SVG images. Margins, padding, and list styles are reset, while the width of the birdhouse and its aspect ratio are defined. Finally, the display is set to a grid layout for optimal alignment of birdhouses.
I decided to jazz up the style by adding a background. That's when I thought about clouds. I thought, why not animate clouds and a sun? And when you switch on the day-night mode, instead of just a quick change, imagine everything shifting slowly, almost like a theatrical performance. It's all about timing, making sure each element transitions smoothly.
data:image/s3,"s3://crabby-images/68e31/68e31efb72318498879881c814ad42bf51253b28" alt=""
data:image/s3,"s3://crabby-images/9b25a/9b25a6961fd0aedf9bb905715e3e287bf0b88bdd" alt=""
After the success of the first bird, I decided to add a second one to make things more entertaining. Teal is a cheerful bird that enjoys flying in loops and is slightly faster than Sunflower (the Yellow Bird).
data:image/s3,"s3://crabby-images/81e28/81e282856cdf15e3dbaf57df03d411e12d967139" alt=""
Paradise is a lightning-fast flyer! This vibrant bird thrives on zipping past slower companions. While occasionally displaying a touch of cockiness, Paradise remains ever ready to assist fellow birds in times of need.
data:image/s3,"s3://crabby-images/f6e4e/f6e4e428aa8e36190d5e163d73a37a7be7ba1469" alt=""
CANCELLED
data:image/s3,"s3://crabby-images/fb35a/fb35a0890fe5d24d75746805c487d4cbec6ee076" alt=""
body {
background-image: url(./imgs/background.svg);
background-size: cover;
}
data:image/s3,"s3://crabby-images/7cd0e/7cd0e5e547b12aa257e47a3c185a5250eff53e83" alt="SchermÂafbeelding 2024-03-19 om 11 07 52"
After that, I threw in some more scenery like clouds, the sun, and the first birdhouse. So basically, when the user clicks on the house, a bird pops up and flies to the center of the screen, adjusting for the width of their browser.
data:image/s3,"s3://crabby-images/66b68/66b68863bca5cf1295ebd608837605dba8151ef6" alt="SchermÂafbeelding 2024-03-19 om 11 19 59"
data:image/s3,"s3://crabby-images/679d5/679d592c0182f686f14084b2097cf19e8bfe32e4" alt=""
Style queries, also known as container queries, are a feature in CSS that let you check the size and style of a parent element. They've recently become supported across all modern browsers. With these queries, you can now not only check the size but also the style values of a parent element. This means you have more control over how styles behave in CSS, allowing for better organization between the logic and design of an application. While style queries currently only work with custom CSS property values, they're still handy for combining styles and keeping data separate from design. They're particularly useful for creating reusable components with different styles and making targeted style changes based on specific conditions or states.
You can read more about it at Developer Chrome's article, which is also the source of this little summary
For a more visual example, visit this
First, I'll create a checkbox and place it in the same parent (NAV) as the birds, within its own respective separate div.
<div>
<input type="checkbox" name="dayAndNight">
</div>
I note down the selector, including its only property, which is the night variable.
html:has( [ name="dayAndNight"]:checked ) {
--dayAndNight:night;
}
Now to define everything with the style query
@container style(--dayAndNight:night) {
body {
background-color: rgb(0, 0, 66);
transition: 1s 2s;
}
/*********/
/* STARS */
/*********/
/*|*****************************************|*/
/*| CREDITS FOR THE STARS: SANNE 'T HOOFT |*/
/*|*****************************************|*/
body::after{
opacity:1;
scale:1;
box-shadow:
-41vw -25vh 0 0 #ffff,
-30vw -48vh 0 0 #ffff,
-22vw 44vh 0 0 #ffff,
-19vw 37vh 0 0 #ffff,
53vw -13vh 0 0 #ffff,
48vw -36vh 0 0 #ffff,
34vw 12vh 0 0 #ffff,
27vw -25vh 0 0 #ffff,
15vw -41vh 0 0 #ffff,
-46vw 44vh 0 0 #ffff,
-36vw -20vh 0 0 #ffff,
-25vw 23vh 0 0 #ffff,
-17vw -39vh 0 0 #ffff,
44vw -32vh 0 0 #ffff,
38vw -18vh 0 0 #ffff,
23vw 11vh 0 0 #ffff,
19vw -37vh 0 0 #ffff,
-42vw 40vh 0 0 #ffff,
-30vw -26vh 0 0 #ffff,
-21vw 19vh 0 0 #ffff,
-11vw 25vh 0 0 #ffff,
40vw 38vh 0 0 #ffff,
32vw 44vh 0 0 #ffff,
29vw -27vh 0 0 #ffff,
13vw -33vh 0 0 #ffff,
-48vw -46vh 0 0 #ffff,
-34vw -12vh 0 0 #ffff,
-27vw 45vh 0 0 #ffff,
-15vw -11vh 0 0 #ffff,
46vw -34vh 0 0 #ffff,
36vw 23vh 0 0 #ffff,
25vw 49vh 0 0 #ffff,
17vw -12vh 0 0 #ffff,
-44vw 28vh 0 0 #ffff,
-28vw -21vh 0 0 #ffff,
-23vw 37vh 0 0 #ffff,
-19vw -30vh 0 0 #ffff,
42vw -16vh 0 0 #ffff,
30vw 49vh 0 0 #ffff,
21vw -25vh 0 0 #ffff,
10vw 18vh 0 0 #ffff;
filter:
drop-shadow(19vw 23vh 0 #fff9)
drop-shadow(-23vw -37vh 0 #fff9)
drop-shadow(13vw 53vh 0 #fff9)
drop-shadow(-59vw -11vh 0 #fff9)
drop-shadow(67vw 17vh 0 #fff9);
transition-delay: 2s;
}
/* titel */
h1 {
font-size: 5rem;
transform-origin: center;
opacity: 0;
pointer-events: none;
transition: 0.5s
}
/* hoekwolken 1 tot 3 */
main > img:nth-of-type(-n + 3) {
bottom: -40rem;
transition: 1s calc(var(--index) * 0.1s);
}
/* bewegende wolken 4 tot 8 */
main > img:nth-of-type(n + 4):nth-of-type(-n + 8) {
top: -30rem;
transition: 1s calc(var(--index) * 0.1s);
}
/* zon/maan */
main > img:nth-of-type(9) {
--width-sun:75vmin;
filter: contrast(10);
filter: grayscale(1);
width: var(--width-sun);
top: calc( (100dvh - var(--width-sun)) / 2);
left: calc( (100vw - var(--width-sun)) / 2);
transition: 1s 3s all;
}
/*************/
/* NAVIGATIE */
/*************/
/* day/night switch */
input[name="dayAndNight"] {
background-color: rgb(30, 30, 68);
background-image: url("./imgs/moon.png");
}
/* ZZZ */
@keyframes fadeIn {
0%, 70%, 100% {
opacity: 0;
scale:0;
}
65% {
opacity: 1;
scale:1;
}
}
ul:nth-of-type(1):hover ~ p {
position: absolute;
display: block;
margin: 0;
font-weight: 900;
font-family: cloudy;
color: white;
pointer-events: none;
user-select: none;
animation: fadeIn 2s linear infinite both;
}
ul:nth-of-type(1):hover ~ p:nth-of-type(1) {
left:45%;
bottom:1em;
animation-delay: 0s;
}
ul:nth-of-type(1):hover ~ p:nth-of-type(2) {
left:60%;
bottom:1.875rem;
font-size: 1.25rem;
animation-delay: .25s;
}
ul:nth-of-type(1):hover ~ p:nth-of-type(3) {
left:75%;
bottom:3rem;
font-size: 1.5rem;
animation-delay: .5s;
}
/***********/
/* HUISJES */
/***********/
main nav ul {
filter: brightness(20%);
transition: 1s 2.5s all;
}
main nav ul li label:first-of-type {
pointer-events: none;
}
}
data:image/s3,"s3://crabby-images/48dfc/48dfc610cc0d96234683876043d64eee4432a7d4" alt=""
data:image/s3,"s3://crabby-images/b50de/b50dea7a8d62b6cc7f433c6761159f2ecff92329" alt=""
Alright, this is what I had in mind for the clouds and the sun and moon: CSS nesting! It's when you can put CSS rules inside other rules, making your styles more organized. It's like putting smaller rules inside bigger ones. This is handy for complex HTML structures because you don't have to repeat the parent selector over and over again. Instead, you just nest the child selector inside the parent's curly braces. This makes your CSS code easier to read and manage. You usually do this using preprocessors like Sass or Less, which turn your nested CSS into regular CSS that browsers can use.
Once again, a major shoutout to Sanne who helped me sorting this out. It was quite the hassle but through his help, I've been able to achieve my desired results
main > img {
position: absolute;
z-index:20;
pointer-events: none;
user-select: none;
/* hoekwolken 1 tot 3 */
&:nth-of-type(-n + 3) {
bottom: 0;
opacity: 0.5;
transition: 0.5s;
}
&:nth-of-type(1) {
width: 70%;
}
&:nth-of-type(2) {
width: 100%;
right: 0;
}
&:nth-of-type(3) {
width: 100%;
left: 0;
}
/* bewegende wolken 4 tot 8 */
&:nth-of-type(n + 4):nth-of-type(-n + 8) {
width: 30%;
transition: 1s;
}
&:nth-of-type(4) {
top: 10%;
left: 50%;
}
&:nth-of-type(5) {
top: 50%;
left: 50%;
}
&:nth-of-type(6) {
top: 30%;
left: 15%;
}
&:nth-of-type(7) {
top: 20%;
right: 20%;
}
&:nth-of-type(8) {
top: 20%;
right: 90%;
}
@media (width > 48em) {
@keyframes movingClouds {
from {
translate: 0 0;
}
to {
translate: -5rem 0;
}
}
&:nth-of-type(n + 4):nth-of-type(-n + 8) {
animation: movingClouds var(--dur) infinite alternate-reverse ease-in-out;
}
&:nth-of-type(4) {
width: 300px;
--dur:25s;
}
&:nth-of-type(5) {
width: 300px;
--dur:30s;
}
&:nth-of-type(6) {
width: 300px;
--dur:70s;
}
&:nth-of-type(7) {
width: 200px;
--dur:50s;
}
&:nth-of-type(8) {
width: 200px;
--dur:30s;
}
}
/* zon/maan 9 */
&:nth-of-type(9){
top: 10%;
left: 80%;
z-index:10;
width: 10%;
transition: 1s;
}
}
Meet Edward The Eagle. The predator that terrifies every living bird in the sky. No one dares to share the sky with him, and whenever other birds see him, they immediately faint.
data:image/s3,"s3://crabby-images/d98e5/d98e52644d44c332cb738564350ab28e19081e91" alt=""
Each bird house in the CSS
/*********************/
/* First Bird House */
/*********************/
section:nth-of-type(1) ul {
--i:1;
--radius: 20vmin;
--flying:10s;
background-image:
url("./imgs/birdhouse-hover.svg"),
url("./imgs/birdhouse.svg");
}
/**********************/
/* Second Bird House */
/**********************/
section:nth-of-type(2) ul {
--i:2;
--radius: 27vmin;
--flying:7s;
background-image:
url("./imgs/birdhouse-house_2.svg"),
url("./imgs/birdhouse-2.svg");
}
/**********************/
/* Third Bird House */
/**********************/
section:nth-of-type(3) ul {
--i:3;
--radius: 35vmin;
--flying:5s;
background-image:
url("./imgs/birdhouse-house_3.svg"),
url("./imgs/birdhouse-3.svg");
}
h4 {
display: none;
}
section:nth-of-type(4) ul {
--i:4;
--radius: 35vmin;
--flying:5s;
background-image:
url("./imgs/birdhouse-house_4.svg"),
url("./imgs/birdhouse-4.svg");
}
ul:has( label:nth-of-type(1):hover ),
ul:has( label:nth-of-type(1):focus-within ) {
background-size: contain;
}
li {
z-index: 50;
grid-area: 1/1;
padding: var(--housePadding);
pointer-events: none;
display:grid;
}
/********************************************************************************************/
/* Don't panic about the selector below! */
/* When you the eagle's house is checked, the rest of the birds inside the other ul's die. */
/* However, when the eagle gets checked, the birds can be once again added */
/********************************************************************************************/
html:has( section:nth-of-type(4) :checked ):not( :has( label:nth-of-type(2) :checked)) section:nth-of-type(-n + 3) label:nth-of-type(1):has(:checked) + label:nth-of-type(2) {
animation-play-state: paused, paused, running;
}