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> <pre class="brush:php;toolbar:false">const dialog = document.getElementById('dialog'); const dialogButton = document.getElementById('dialogButton'); dialogButton.addEventListener('click', () => dialog.showModal());
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());
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.
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!
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> <pre class="brush:php;toolbar:false">#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; }
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.
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.
<main> <dialog> <pre class="brush:php;toolbar:false">const dialog = document.getElementById('dialog'); const dialogButton = document.getElementById('dialogButton'); dialogButton.addEventListener('click', () => dialog.showModal());
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.
dialogButton.addEventListener('click', () => dialog.showModal()); dialogButton.addEventListener('click', () => dialog.show());
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.
<main> <dialog> <pre class="brush:php;toolbar:false">#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; }
The behavior looks like this:
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[open]::backdrop { background: rgba(255, 0, 0, 60%); }
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.
dialog.close();
Two significant things have been done:
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).
I added an attribute called closing when the user clicks. It is removed using the time parameter mentioned above.
/** @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);
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...
The drawer opens above the screen at full height. So let's make a new drawer element from scratch.
<main> <dialog> <pre class="brush:php;toolbar:false">const dialog = document.getElementById('dialog'); const dialogButton = document.getElementById('dialogButton'); dialogButton.addEventListener('click', () => dialog.showModal());
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!
dialogButton.addEventListener('click', () => dialog.showModal()); dialogButton.addEventListener('click', () => dialog.show());
Now it's simple: just add the animations using the same function you created in the step-by-step dialog and you're done!
<main> <dialog> <pre class="brush:php;toolbar:false">#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; }
#dialog[open]::backdrop { background: rgba(255, 0, 0, 60%); }
<dialog> <p><img src="https://img.php.cn/upload/article/000/000/000/173086397449699.jpg" alt="EXPLORING HTML ELEMENTS: <DIALOG>"></p> <h2> Thank you for reading </h2> <p>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.</p>
The above is the detailed content of EXPLORING HTML ELEMENTS: