Skip to content

Latest commit

 

History

History
312 lines (215 loc) · 12.9 KB

13-a-jednak-sie-kreci.md

File metadata and controls

312 lines (215 loc) · 12.9 KB
title layout
13. And yet it's spinning! 🪐
post

Copernicus stopped the Sun, and moved the Earth. We will move all 8 planets!

Download our package: LINK

Inside, you will find the index.html, main.js, and main.css files.

When you open the index.html file in a browser, you will see 9 planet images.

Now let's take a look at this file in a text editor.

Our planets are list elements:

<ul class="carousel">
    <li class="single-slide">…</li>
</ul>

As you can guess, our task is to create a carousel. We will start with its proper layout. We will use CSS styles for that, which we will include in the main.css file. To link the styles to our HTML file, we need to add another element between the <head></head> tags:

<link rel="stylesheet" type="text/css" href="sciezka/do/pliku.css">

If you do it correctly, you should see a starry sky.

Now let's add our styles. You can find help on this website: https://www.w3schools.com/css/. Don't hesitate to use Google search as well. To apply a style to specific elements, we need to refer to their class names. To do this in the CSS file, we need to write:

.className {
    property: value;
}

for example:

.carousel {
    background-color: green;
}

The background of the entire carousel should turn green.

Let's start by arranging the planets side by side. Each slide should have a width of 800px. Set the width of the entire carousel to approximately 7300px.

Also, let's make sure each planet is centered within slide (text-align: center;). But we still see more than one planet. So let's specify the width of the stage (.carousel-stage) and hide what doesn't fit inside (overflow: hidden;). Let's also center our stage (.carousel_stage, margin-left: auto; margin-right: auto;).

Now for the navigation. Do you see two arrows below the carousel? First, let's position them at the appropriate height, which is halfway up the carousel. Initially, give the carousel (.carousel) a position: relative. We will position the arrows relative to it. To make the arrows not appear under the carousel but "float in the air," give the navigation (.carousel-nav) a position: absolute. Let's place the arrows at the halfway point of the carousel's height. So, add top: 50% to the styles (the navigation will then be positioned at 50% of the parent's height, which has position: relative). But something doesn't quite match. The arrows are slightly too low. Exactly half their height too low. So let's apply a small transformation: transform: translateY(-50%).

The next step is to place one arrow on the right and the other on the left side of the carousel. Use float: left; and float: right;, respectively.

It's time to set our planets in motion!

First, let's link the main.js file to our page. We do it similarly to the CSS file, but we use the script tag and include the path in the src attribute.

<script src="sciezka/do/pliku.js"><script>

Go to the website, and then open the browser console. If everything is working, a message should appear.

Let's think about how our carousel should work. Imagine that it is a film strip, and at certain moments (after clicking the arrow or after a specific time has elapsed), the entire strip should move by the width of one frame (which is one slide).

So, let's go to the main.js file. Let's remove the current code. We'll start by defining our variables:

carousel for the carousel

stage for the carousel scene

prev for the "previous" button

next for the "next" button

Don't forget the keywords for defining variables (i.e., use let or const here).

Now, let's retrieve the HTML elements for our specified variables. We'll use the familiar querySelector() method, which will display the first element on the page with a specified attribute, such as a class.

var carousel = document.querySelector('.carousel');

Download the elements for the remaining defined variables (i.e., for our scene and two buttons) using the same approach.

We still have one variable to define: slide for individual carousel elements. Here, we need to retrieve all slides, so we'll use the querySelectorAll() method.

Let's do the same with all elements on the page :)

The next step is to determine the width by which our "strip" should move. As we mentioned, it's the width of one slide. Let's try to "extract" this value. We'll use the clientWidth property, which returns the width of the specified element. Let's try:

var slideWidth = slide.clientWidth;
console.log(slideWidth);

Check what appears in the console. It displays a message that the value is undefined. So let's see what the variable slide contains. It shows a list of elements. JS cannot determine the width of the list of elements because our variable slide contains an array with all elements having the class slide that it found in the document. However, our code can handle a single element, for example, the first one. The first element in the list has an index of zero, so:

var slideWidth = slide[0].clientWidth;
console.log(slideWidth);

The next step is to determine which slide is currently displayed. Initially, it will be the first slide, but as we know, in JS, the first element is element 0.

var currentIndex = 0;

What happens when we reach the last element? We should go back to the first slide. So let's find the last element. First, we'll determine the total number of elements using the length property.

var slidesNumber = slide.length - 1;

Where did the -1 come from? The Slide.length is the number of slides. which is 9. However, in JavaScript, element counting starts from 0, not 1. So the last slide won't have a value of 9 but 8.

OK, now it's time to write functions that will animate our carousel and move the entire carousel by the appropriate width. We'll use styles for this. Let's first try using CSS to move our carousel to the left by one slide, or 800px. The position, left, and right properties will help us with that.

Once you've accomplished that, go back to the JS file. We'll manipulate CSS values using JavaScript functions.

Let's create a function called goToSlide(). Its result is to be the changed value of the left property of our carousel. It is supposed to be enough to show the corresponding slide. A small hint - we will use the slideWidth variable for this and the position of the slide we want to see.

Let's start from the beginning. To change the left value of the carousel, we'll use the style.left method. This allows us to change the position of an element relative to its left edge.

function goToSlide() {
    carousel.style.left = ...;
}

Let's think about what value length should have to show the second slide. What about the third and fourth slides? Do you notice any general rule?

Yes! We multiply slideWidth by the position of the specific slide!

So let's try:

Assume that the variable index represents the position of our slide. Let's define it as 3\ (position of the 4th slide).

function goToSlide() {
    carousel.style.left = 3 * (-slideWidth);
}

Call this function in the console.

It works!

However, there's a slight problem - we have multiple slides, each with a different index. Writing a separate function for each slide would be inefficient. So let's use function parameters! With that, we can use the same function for different values.

function goToSlide(index) {
    carousel.style.left = index * (-slideWidth);
}

Let's call this function in the console by typing, for example, goToSlide(3); goToSlide(1); goToSlide(4);`.

It works! The only thing is that now the currentIndex should also change. It should be equal to the number we entered as an argument. So let's add this change to our function:

function goToSlide(index) {
    carousel.style.left = index * (-slideWidth);
    currentIndex = index;
}

Let's move on to navigation :)

Clicking on the carousel-next button should take us to a slide with an index that is 1 higher. Clicking on the carousel-prev button should take us to a slide with an index that is 1 lower than the current index.

So let's create two functions. Firstly:

 function slideToNext() {
 }

It is supposed to move the slides forward by 1 each time it is called. That is, we are using the goToSlide() function here. But what will be our argument? As we mentioned earlier, each call to our function is to move us to a slide with an index 1 greater than the index of the current slide. We store the index of the current slide in the currentIndex variable. So our argument is currentIndex + 1.

function slideToNext() {
    goToSlide(currentIndex + 1);
}

Let's do an analogy with slideToPrev.

The next step is to call both functions when clicking on buttons. Clicks are events that take place on the page. They can be triggered by the user (like a click), or some element on the page. Submitting a form, loading an image, are also events. Examples of events on the page are:

Event: Description:
blur object is no longer active
change The object has changed its content (e.g., a form field).
click click on the object
dblclick double click on the object
focus selecting an object on the site
keydown pressing a key on the keyboard
input while holding the key
keyUp releasing a key on the keyboard
load When the object has been loaded (it can even be a whole page).
mouseover when the cursor is on a particular object
mouseout when the cursor has left an object
contextmenu when right clicked and context menu appeared
wheel when you spin the mouse wheel
resize when we resize the browser window
select when the content of the object has been selected
submit when the form has been sent
unload user leaves the page
animationstart css animation will begin
animationend css animation will end

We will use the addEventListener method to keep track of whether an event has taken place.

element.addEventListener('event_as_string', what_should_happen, optionally_true_lub_false);

We will create a separate function for all events on the page. We will call it bindEvents:

function bindEvents() {
}

Let's start with the back button. It is under the prev variable. On this variable, let's call the addEventListener method:

function bindEvents() {
    prev.addEventListener();
}

And now the arguments. We want to track the event of the click, that is, the click. It is to call the slideToPrev function. Let's put it in the right place:

function bindEvents() {
    prev.addEventListener('click', slideToPrev);
}

Let's add an analogous event to the bindEvents function for the next button.

Let's call the bindEvents function to see if the buttons work :)

Great! However, look what will happen if we keep clicking next or back. - the planets disappear. Our carousel keeps moving 800px. We need to limit it. After the last planet, let it go back to the first one, and when we want to go back from the first one, let it show us the last planet.

Let's take another look at our function:

function goToSlide(index) {
    carousel.style.left = index * (-slideWidth);
    currentIndex = index;
}

Everything that happens depends on the index. So let's make it so that if the index is greater than the index of the last planet, it becomes 0, and if it is less than the index of the first planet, it becomes equal to the index of the first planet.

We will use the conditional statement (if... else) to do this. That is, if the index is less than 0 we change it to the value of slidesNumber.

function goToSlide(index) {
    if (index < 0) {
        index = slidesNumber;
    }

    carousel.style.left = index * (-slideWidth);
    currentIndex = index;
}

And if it is greater than slidesNumber - we change it to 0.

function goToSlide(index) {
    if (index < 0) {
        index = slidesNumber;
    } else if (index > slidesNumber) {
        index = 0;
    }

    carousel.style.left = index * (-slideWidth);
    currentIndex = index;
}

Let's check it out now.

Add some more life to the carousel - let it spin on its own. We'll use the already familiar setInterval method for this.

We should create an autorotate function.

function autorotate() {
}

Let the slideToNext function be executed every 4s (4000 ms):

function autorotate() {
    setInterval(slideToNext, 4000);
}

And let's call the autorotate function.

And now everything is spinning! :)