paint-brush
How to Recreate the Iconic Mr Potato Head 🥔 with Vanilla Javascriptby@daily-dev-tips
561 reads
561 reads

How to Recreate the Iconic Mr Potato Head 🥔 with Vanilla Javascript

by Daily Dev TipsNovember 2nd, 2020
Read on Terminal Reader
Read this story w/o Javascript
tldt arrow

Too Long; Didn't Read

JavaScript recreates the iconic Mr. Potato Head in JavaScript. We will have all his parts, which we can drag on his body. The result of today's article is this amazing Codepen. With a simple HTML setup we have a fairly simple setup. We use flexbox to center our two divs. The Parts container is then relative, and we add a small border to make it look nicer. The draggable function is used to change the left and top position of our part. And Set the z-index higher so it's on top.

Companies Mentioned

Mention Thumbnail
Mention Thumbnail
featured image - How to Recreate the Iconic Mr Potato Head 🥔 with Vanilla Javascript
Daily Dev Tips HackerNoon profile picture

Who doesn't like Mr. and Mrs. Potato Head!

Today we will be recreating the iconic Mr. Potato Head in JavaScript. Meaning we will have all his parts, which we can drag on his body.

The result of today's article is this amazing Codepen.

HTML Structure

As for out HTML, we have a fairly simple setup.

<div class="container">
  <div class="parts">
    <img src="https://i.imgur.com/GONNbHf.png" class="draggable" />
    <img src="https://i.imgur.com/optSzq4.png" class="draggable" />
    <img src="https://i.imgur.com/qJDxc4o.png" class="draggable" />
    <img src="https://i.imgur.com/tIZGoeR.png" class="draggable" />
    <img src="https://i.imgur.com/bKlbeXU.png" class="draggable" />
    <img src="https://i.imgur.com/eUPbX3H.png" class="draggable" />
    <img src="https://i.imgur.com/voJPsR5.png" class="draggable" />
    <img src="https://i.imgur.com/dt2gqit.png" class="draggable" />
    <img src="https://i.imgur.com/2POeyJZ.png" class="draggable" />
  </div>
  <div class="body">
    <img src="https://i.imgur.com/kXbr8Tb.png" />
  </div>
</div>

So we use the container to wrap everything, then we have our parts div, which contains each of the body-parts with a class of draggable. And we have our body, which is Mr Potato's body.

CSS Styling

We use flexbox to center our two divs.

.container {
  display: flex;
  align-items: center;
  justify-content: space-around;
  min-height: 100vh;
  background: #efefef;
}

The Parts container is then relative, and we add a small border to make it look nicer.

.container .parts {
  position: relative;
  border: 3px dashed black;
  width: 250px;
  height: 100vh;
}

Each image will be absolute so we can place it anywhere in the page.

.container .parts img {
  position: absolute;
}

Vanilla JavaScript draggable Mr. Potato Head parts

To make an actual Mr. Potato Head, we need to make sure all the parts are draggable!

I did not use the draggable element since that requires a dropzone, and it doesn't serve this article.

Let's start by getting our elements with the class draggable.

const draggableElements = document.querySelectorAll(".draggable");

Then we need to define four basic variables we will use to store our position in.

let initX, initY, firstX, firstY;

Next on our list is to loop over each element.

draggableElements.forEach((element) => {
    // Code here
});

Then we need to attach a mousedown eventListener. This will be our starting point. We will define the current x and y position by using offsetLeft and offsetTop. Then we get the mouse position x and y.

And we attach a eventListener too mousemove since that will be us, dragging a part. Once we move our mouse we call the draggable function which we will make in a second.

draggableElements.forEach((element) => {
  element.addEventListener("mousedown", function (e) {
    e.preventDefault();
    initX = this.offsetLeft;
    initY = this.offsetTop;
    firstX = e.pageX;
    firstY = e.pageY;
    this.addEventListener("mousemove", draggable, false);
  });
});

Let's get started with our draggable function.

All this function does is change the left and top position of our part. And Set the z-index higher so it's on top.

function draggable(e) {
  e.preventDefault();
  this.style.zIndex = 9;
  this.style.left = initX + e.pageX - firstX + "px";
  this.style.top = initY + e.pageY - firstY + "px";
}

We calculate the original position + the dragged amount - the initial mouse x. And the same goes for the y position.

That's cool, but we have no way of stopping it dragging now. So let's add a mouseup listener.

draggableElements.forEach((element) => {
  window.addEventListener("mouseup", function () {
      element.style.zIndex = 0;
      element.removeEventListener("mousemove", draggable, false);
    }, false);

  // Mouse down code from above
});

In this section, we add a mouseup event to our window, and once that happens, we remove the z-index from our dragging element and remove the draggable mousemove listener.

That is it. We can now drag Mr. Potato Head's parts on his body!

Thank you for reading, and let's connect!

Thank you for reading my blog. Feel free to subscribe to my email newsletter and connect on Facebook or Twitter

Previously published at https://daily-dev-tips.com/posts/vanilla-javascript-draggable-mr-potato-head-%F0%9F%A5%94/