EXPLORING HTML ELEMENTS: <DIALOG>

miguelccampos

Miguel Campos

Posted on November 5, 2024

EXPLORING HTML ELEMENTS: <DIALOG>

DEMO

Introduction

OPEN DIALOG AND DRAWER WITH ANIMATIONS

The dialog element has been widely supported by all browsers for some time now, and I'm here to show you how this element can greatly improve the productivity and quality of your code, regardless of the framework you're using.

I'm using plain JavaScript to demonstrate that you don't need many lines of code to create beautiful and simple components.
Using this element is quite simple:

<main>
  <dialog id="dialog">
     <h1>DIALOG</h1>
  </dialog>
  <button id="dialog button">OPEN DIALOG</button>
</main>
Enter fullscreen mode Exit fullscreen mode
const dialog = document.getElementById('dialog');
const dialogButton = document.getElementById('dialogButton');

dialogButton.addEventListener('click', () => dialog.showModal());
Enter fullscreen mode Exit fullscreen mode

As you can see above, you just need to take the reference of the dialog element and call one of these two functions:

dialogButton.addEventListener('click', () => dialog.showModal());
dialogButton.addEventListener('click', () => dialog.show());
Enter fullscreen mode Exit fullscreen mode

But you might be wondering:

What is the difference between these two functions?

Well, to understand that, we have to understand a little about layers and what the top-layer is.

Layers

Let me give you an example of something you may have come across before when using a drawer component. You call the component normally in your html and it ends up not opening on the whole screen, only in the context you placed it, so you end up having to place your drawer in index.html for it to open in the whole viewport or fill your css with position and z-index to create an overlapping context.

By using layers you can make this easier, but you also end up polluting your css document a bit. By using showModal() it opens the dialog element in that top-layer already with a separate context without any configuration required. If you use show() it doesn't call the dialog in that top-layer and you can have the normal behavior of the viewport.

The top layer is a “layer” that expands over the entire viewport and overlays all the layers created by you or your web browser. It helps a lot and avoids the infamous excessive prop drilling.

Okay, okay, but how do I style this element? It must be very difficult!

Not at all! It's as simple as centering a div... Maybe a bit more!

Dialog / Modals

Dialog image

For more in-depth stylization I recommend you to see this video from Kevin Powell.

To style the dialog is simple just put id/class in the element and use CSS.

<main>
  <dialog id="dialog">
     <h1>DIALOG</h1>
  </dialog>
  <button id="dialogButton">OPEN DIALOG</button>
</main>
Enter fullscreen mode Exit fullscreen mode
#dialog {
  border: none;
  width: 20rem;
  height: 20rem;
  border-radius: 1.69%; /* NICE! */
}

#dialog[open] {
  /* OMG! I know how to center a div */
  display: flex;
  justify-content: center;
  align-items: center;
}
Enter fullscreen mode Exit fullscreen mode

As you can see above, I used #dialog[open] to modify the positioning of the elements. When you create a dialog it comes by default with the display disabled.

HTML DIALOG ELEMENT

This attribute is only added when you call one of the two functions.
So when styling any dialog element, always try to use the name of your class/id and the attribute to avoid display problems.
And as you can see above, the dialog already comes with a ::backdrop effect and it's also very simple to modify it.

BACKDROP CHANGED TO RED

#dialog[open]::backdrop {
  background: rgba(255, 0, 0, 60%);
}
Enter fullscreen mode Exit fullscreen mode

Show! Super easy! But how do I close this modal? When I click on the backdrop nothing happens!

Let's go! By default the element doesn't come with this possibility. If you want to close the modal you have to call the function to close it using an internal button or something like that.

dialog.close();
Enter fullscreen mode Exit fullscreen mode

But if you want to replicate this effect of closing by clicking outside, you can use a simple function to identify the edges of the dialog element to know if you've clicked outside or not.

/** @type {HTMLDialogElement} */
const dialog = document.getElementById('dialog');

/**
 * Adds a close behavior to clicking outside dialog element
 * @param {HTMLDialogElement} dialogRef - The refer to element Dialog.
 * @example
 * const dialog = document.querySelector('#dialog');
 * closeDialogClickingOutside(dialog)
 */
 const closeByClickingOutside = (dialogRef) => {
  dialogRef.addEventListener('click', (ev) => {
    if (
      ev.offsetX < 0 ||
      ev.offsetX > ev.target.offsetWidth ||
      ev.offsetY < 0 ||
      ev.offsetY > ev.target.offsetHeight
    ) {
      dialogRef.close();
    }
  });
};

closeByClickingOutside(dialog);
Enter fullscreen mode Exit fullscreen mode

The behavior looks like this:

MODAL OPEN AND CLOSE IN THE BACKGROUND RED

I recommend you to see the Popover API if this is what you want to do constantly.

Man, that's ugly! My library is already animated!

Don't worry about it! Let's add a simple fade-in animation.

#dialog {
  border: none;
  width: 20rem;
  height: 20rem;
  border-radius: 1.69%; /* nice */
}

#dialog[open]::backdrop {
  background: rgba(255, 0, 0, 60%);
}

#dialog[open],
#dialog[open]::backdrop {
  animation: fadeIn 1s forwards; /* Não se esqueça de colocar fowards*/
  /* omg i now i know how to center a div */
  display: flex;
  justify-content: center;
  align-items: center;
}

@keyframes fadeIn {
  from {
    opacity: 0;
  }
  to {
    opacity: 1;
  }
}
Enter fullscreen mode Exit fullscreen mode

ANIMATED MODAL OPEN

Adding the fade-out animation is a little more difficult because you don't have a way of identifying when the user is going to close the modal, so we'll have to make a change to our script file.

/** @type {HTMLDialogElement} */
const dialog = document.getElementById('dialog');
/** @type {HTMLButtonElement} */
const dialogButton = document.getElementById('dialogButton');

/**
 * Adds a close behavior to clicking outside dialog element
 * @param {HTMLDialogElement} dialogRef - The refer to element Dialog.
 * @example
 * const dialog = document.querySelector('#dialog');
 * closeDialogClickingOutside(dialog)
 */
const closeByClickingOutside = (dialogRef, closingAnimationDuration = 0) => {
  dialogRef.addEventListener('click', (ev) => {
    if (
      ev.offsetX < 0 ||
      ev.offsetX > ev.target.offsetWidth ||
      ev.offsetY < 0 ||
      ev.offsetY > ev.target.offsetHeight
    ) {
      dialogRef.setAttribute('closing', '');
      setTimeout(() => {
        dialogRef.close();
        dialogRef.removeAttribute('closing');
      }, closingAnimationDuration);
    }
  });
};

closeByClickingOutside(dialog, 500);
Enter fullscreen mode Exit fullscreen mode

Two significant things have been done:

  1. A new parameter has been added to determine the duration of the animation to close the modal (after some searching I found this video from Kevin Powell).

  2. I added an attribute called closing when the user clicks. It is removed using the time parameter mentioned above.

#dialog {
  border: none;
  width: 20rem;
  height: 20rem;
  border-radius: 1.69%; /* nice */
}

#dialog[open]::backdrop {
  background: rgba(255, 0, 0, 60%);
}

#dialog[closing]::backdrop {
  background: rgba(0, 0, 255, 90%);
}

#dialog[open],
#dialog[open]::backdrop {
  animation: fadeIn 1s forwards;
  /* omg i now i know how to center a div */
  display: flex;
  justify-content: center;
  align-items: center;
}

#dialog[closing],
#dialog[closing]::backdrop {
  animation: fadeOut 0.5s forwards;
}

@keyframes fadeIn {
  from {
    opacity: 0;
  }
  to {
    opacity: 1;
  }
}

@keyframes fadeOut {
  from {
    opacity: 1;
  }
  to {
    opacity: 0;
  }
}
Enter fullscreen mode Exit fullscreen mode

OPEN AND CLOSE MODAL ANIMATION

And there we have our fully animated dialog without any libraries!

That nice! But how did you make a drawer using this element?

Well, to achieve this we have to take the default styles of the dialog element and overwrite them...

Drawer

The drawer opens above the screen at full height. So let's make a new drawer element from scratch.

<dialog id="drawer">
  <h1>DRAWER</h1>
</dialog>
<button id="drawerButton">OPEN DRAWER</button>
Enter fullscreen mode Exit fullscreen mode

It's important to know that the dialog is centered using the margin and not the flexbox.

DIALOG ELEMENT

So to move the element you just need to change the margin.

#drawer {
  margin: 0;
  margin-left: 0;
  margin-right: auto;
}
Enter fullscreen mode Exit fullscreen mode

It's important to note that to set the drawer to the full height of the screen you have to overwrite the dialog's default max-height!

#drawer {
  margin: 0;
  margin-left: 0;
  margin-right: auto;
  height: 100vh;
  max-height: 100vh; /** IMPORTANTE !!!*/
  border: 0;
}
Enter fullscreen mode Exit fullscreen mode

DIALOG INTERNAL CSS

Now it's simple: just add the animations using the same function you created in the step-by-step dialog and you're done!

/** @type {HTMLDialogElement} */
const drawer = document.getElementById('drawer');
/** @type {HTMLButtonElement} */
const drawerButton = document.getElementById('drawerButton');

drawerButton.addEventListener('click', () => drawer.showModal());

/**
 * Add eventListener to close dialog clicking on outerbox
 * @param {HTMLDialogElement} dialogRef - The DialogReference.
 * @param {number|undefined} closingAnimationDuration - Time for closingAnimation duration in miliseconds
 * @example
 * const dialog = document.querySelector('#dialog');
 * closeDialogClickingOutside(dialog, 500)
 */
const closeByClickingOutside = (dialogRef, closingAnimationDuration = 0) => {
  dialogRef.addEventListener('click', (ev) => {
    if (
      ev.offsetX < 0 ||
      ev.offsetX > ev.target.offsetWidth ||
      ev.offsetY < 0 ||
      ev.offsetY > ev.target.offsetHeight
    ) {
      dialogRef.setAttribute('closing', '');
      setTimeout(() => {
        dialogRef.close();
        dialogRef.removeAttribute('closing');
      }, closingAnimationDuration);
    }
  });
};

closeByClickingOutside(drawer, 500);
Enter fullscreen mode Exit fullscreen mode
#drawer {
  margin: 0;
  margin-left: 0;
  margin-right: auto;
  height: 100vh;
  max-height: 100vh;
}

#drawer[open] {
  animation: openDrawer 0.5s forwards;
}

#drawer[closing] {
  animation: closingDrawer linear 0.5s forwards;
}

@keyframes openDrawer {
  from {
    transform: translateX(-100%);
  }
  to {
    transform: translateX(0);
  }
}

@keyframes closingDrawer {
  from {
    transform: translateX(0);
  }
  to {
    transform: translateX(-100%);
  }
}
Enter fullscreen mode Exit fullscreen mode
<dialog id="drawer">
    <h1>DRAWER</h1>
</dialog>
<button id="drawerButton">OPEN DRAWER</button>
Enter fullscreen mode Exit fullscreen mode

DRAWER WITH ANIMATION

Thank you for reading

This is my first post here so thank you for reading and if I've helped in any way I recommend you follow me on linkedin my first language is Portuguese so most of my posts are in Portuguese but I am trying to post something in English too. I intend to post similar things from now on, if you have any recommendations on how to improve this modal and drawer strategy you can send them to me there.

💖 💪 🙅 🚩
miguelccampos
Miguel Campos

Posted on November 5, 2024

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related