· Updated on

How to Build a Modal Video Component with Tailwind CSS and Next.js

Create a video modal in Tailwind CSS

Welcome to the second part of the series of How to Build a Video Modal Component with Tailwind CSS! In the previous part, we learned how to create a modal video component using Tailwind CSS and Alpine.js. In this article, we will level up the game and show you how to create a reusable component for Next.js using TypeScript.

Before we dive in, it’s important to note that we won’t be covering how to get started with Next.js. We recommend referring to the official documentation for that. Instead, we’ll focus on building the component using Next.js 13 with the App Router (app) and React Server Components.

If you’re interested in seeing how we previously built some modal video components in Next.js, we suggest checking out Tidy, an elegant HTML website template, and Appy, a mobile web design template!

Let’s start by sketching out our component by creating a file called modal-video.tsx and defining the function to export.

'use client'

export default function ModalVideo() {
  return (
    <div>

      {/* 1. The button */}
      {/* 2. The backdrop layer */}
      {/* 3. The modal video */}

    </div>
  )
}

For the structure, we are using the same HTML structure we created before with Tailwind CSS, remembering to replace the class attributes with className to ensure compatibility with React.

Additionally, we use the 'use client' directive at the top of the file and before any imports because we know that this component requires client-side interactivity.

Define the modal initial state with useState

Now, let’s define the initial state of the modal, whether it should be open or closed. We can achieve this by using the useState hook from React and setting the initial state to false.

By setting the initial state of the modal, we ensure that it’s closed by default. Later, we’ll create a function that updates the state of the modal when a user interacts with it. This way, the modal will only appear when a user clicks on the thumbnail.

'use client'

export default function ModalVideo() {
  const [modalOpen, setModalOpen] = useState<boolean>(false)

  return (
    <div>

      {/* 1. The button */}
      {/* 2. The backdrop layer */}
      {/* 3. The modal video */}

    </div>
  )
}

Toggling the modal state

Now that we’ve defined the initial state of the modal, let’s move on to toggling its state. To achieve this, add the button markup to our component and include an onClick event that changes the modalOpen state variable to true.

By doing this, we’ll trigger the modal to open when the user clicks on the thumbnail.

'use client'

import Image from 'next/image'
import VideoThumb from '@/public/modal-video-thumb.jpg'

export default function ModalVideo() {
  const [modalOpen, setModalOpen] = useState<boolean>(false)

  return (
    <div>

      {/* 1. The button */}
      <button
        className="relative flex justify-center items-center focus:outline-none focus-visible:ring focus-visible:ring-indigo-300 rounded-3xl group"
        onClick={() => { setModalOpen(true) }}
        aria-label="Watch the video"
      >
        <Image className="rounded-3xl shadow-2xl transition-shadow duration-300 ease-in-out" src={VideoThumb} width={768} height={432} priority alt="Modal video thumbnail" />
        {/* Play icon */}
        <svg className="absolute pointer-events-none group-hover:scale-110 transition-transform duration-300 ease-in-out" xmlns="http://www.w3.org/2000/svg" width="72" height="72">
          <circle className="fill-white" cx="36" cy="36" r="36" fillOpacity=".8" />
          <path className="fill-indigo-500 drop-shadow-2xl" d="M44 36a.999.999 0 0 0-.427-.82l-10-7A1 1 0 0 0 32 29V43a.999.999 0 0 0 1.573.82l10-7A.995.995 0 0 0 44 36V36c0 .001 0 .001 0 0Z" />
        </svg>
      </button>

      {/* 2. The backdrop layer */}
      {/* 3. The modal video */}

    </div>
  )
}

Handling modal visibility and adding enter / leave transitions

Now that we can toggle the modal state, we need to make sure that the visibility of the backdrop layer and the modal video are linked to the modalOpen state variable. Additionally, we want to add enter and leave transitions every time the modal is opened or closed.

One option is to use the React Transition Group library to achieve this, but we should also handle accessibility, and integrate the component with functions that close the modal when the backdrop layer is clicked or the escape key is pressed.

Alternatively, we can use a pre-built UI component that handles all of this for us: Headless UI. Headless UI is a library of fully accessible UI components made by the creators of Tailwind CSS. It includes a bunch of ready-to-use components, including menu, popover, tabs, dialog, and more.

To get started with Headless UI, first install it using the simple Terminal command npm install @headlessui/react@latest. Once installed, import the Dialog and Transition components and define the structure of the backdrop layer and the modal video, as shown in the following example:

'use client'

import { useState, Fragment } from 'react'
import { Dialog, DialogPanel, Transition } from '@headlessui/react'
import Image from 'next/image'

export default function ModalVideo() {
  const [modalOpen, setModalOpen] = useState<boolean>(false)

  return (
    <div>

      {/* 1. The button */}
      <button
        className="relative flex justify-center items-center focus:outline-none focus-visible:ring focus-visible:ring-indigo-300 rounded-3xl group"
        onClick={() => { setModalOpen(true) }}
        aria-label="Watch the video"
      >
        <Image className="rounded-3xl shadow-2xl transition-shadow duration-300 ease-in-out" src={thumb} width={thumbWidth} height={thumbHeight} priority alt="Modal video thumbnail" />
        {/* Play icon */}
        <svg className="absolute pointer-events-none group-hover:scale-110 transition-transform duration-300 ease-in-out" xmlns="http://www.w3.org/2000/svg" width="72" height="72">
          <circle className="fill-white" cx="36" cy="36" r="36" fillOpacity=".8" />
          <path className="fill-indigo-500 drop-shadow-2xl" d="M44 36a.999.999 0 0 0-.427-.82l-10-7A1 1 0 0 0 32 29V43a.999.999 0 0 0 1.573.82l10-7A.995.995 0 0 0 44 36V36c0 .001 0 .001 0 0Z" />
        </svg>
      </button>

      <Transition show={modalOpen} as={Fragment}>
        <Dialog onClose={() => setModalOpen(false)}>

          {/* 2. The backdrop layer */}
          <TransitionChild
            as="div"
            className="fixed inset-0 z-[99999] bg-black bg-opacity-50 transition-opacity"
            enter="transition ease-out duration-200"
            enterFrom="opacity-0"
            enterTo="opacity-100"
            leave="transition ease-out duration-100"
            leaveFrom="opacity-100"
            leaveTo="opacity-0"
            aria-hidden="true"
          />

          {/* 3. The modal video */}
          <TransitionChild
            as="div"
            className="fixed inset-0 z-[99999] flex p-6"
            enter="transition ease-out duration-300"
            enterFrom="opacity-0 scale-75"
            enterTo="opacity-100 scale-100"
            leave="transition ease-out duration-200"
            leaveFrom="opacity-100 scale-100"
            leaveTo="opacity-0 scale-75"
          >
            <div className="max-w-5xl mx-auto h-full flex items-center">
              <DialogPanel className="w-full max-h-full rounded-3xl shadow-2xl aspect-video bg-black overflow-hidden">
                <video width="1920" height="1080" loop controls>
                  <source src="/video.mp4" type="video/mp4" />
                  Your browser does not support the video tag.
                </video>
              </DialogPanel>
            </div>
          </TransitionChild>

        </Dialog>
      </Transition>

    </div>
  )
}

The Transition component is a part of the Headless UI library that allows you to animate content based on its visibility.

When the modalOpen state variable is set to true, the Transition component will show the content wrapped inside it. This is done by passing the modalOpen variable to the show property.

The Transition component has two TransitionChild components that wrap the backdrop and modal video separately. Each TransitionChild component has its own set of properties to define how it should appear and disappear when the modal opens and closes. These properties are defined in the following properties:

  • enter: Defines the CSS transition property when the component is entering.
  • enterFrom: Defines the starting state of the transition when the component is entering.
  • enterTo: Defines the ending state of the transition when the component is entering.
  • leave: Defines the CSS transition property when the component is leaving.
  • leaveFrom: Defines the starting state of the transition when the component is leaving.
  • leaveTo: Defines the ending state of the transition when the component is leaving.

Next, we have the Dialog component which includes an onClose callback that triggers when the user clicks outside the DialogPanel or presses the escape key. We’ll use this callback to set the modalOpen state to false.

At this point, our modal video is fully functional! The final step is to ensure that the video plays automatically when the modal is opened. Let’s take a look at how we can achieve this.

Playing the video automatically when the modal opens

Unlike the Alpine.js example we showed in the previous article, in Next.js, the modal content gets unmounted when the modal is closed. For this reason, we don’t need to worry about pausing the video when the modal is closed, and we’ll only focus on starting the video when it opens.

To do this, we need to reference the video and tell it to start playing. We can’t use the usual useEffect React hook because the video is inside a component that gets unmounted when the modal closes, and the ref would return null.

Instead, we’ll use the afterEnter callback in the Transition component. This callback runs after the modal has finished opening, so we can access the video element through the ref and tell it to play.

'use client'

import { useState, useRef, Fragment } from 'react'
import { Dialog, DialogPanel, Transition } from '@headlessui/react'
import Image from 'next/image'

export default function ModalVideo() {
  const [modalOpen, setModalOpen] = useState<boolean>(false)

  return (
    <div>

      {/* 1. The button */}
      <button
        className="relative flex justify-center items-center focus:outline-none focus-visible:ring focus-visible:ring-indigo-300 rounded-3xl group"
        onClick={() => { setModalOpen(true) }}
        aria-label="Watch the video"
      >
        <Image className="rounded-3xl shadow-2xl transition-shadow duration-300 ease-in-out" src={thumb} width={thumbWidth} height={thumbHeight} priority alt="Modal video thumbnail" />
        {/* Play icon */}
        <svg className="absolute pointer-events-none group-hover:scale-110 transition-transform duration-300 ease-in-out" xmlns="http://www.w3.org/2000/svg" width="72" height="72">
          <circle className="fill-white" cx="36" cy="36" r="36" fillOpacity=".8" />
          <path className="fill-indigo-500 drop-shadow-2xl" d="M44 36a.999.999 0 0 0-.427-.82l-10-7A1 1 0 0 0 32 29V43a.999.999 0 0 0 1.573.82l10-7A.995.995 0 0 0 44 36V36c0 .001 0 .001 0 0Z" />
        </svg>
      </button>

      <Transition show={modalOpen} as={Fragment} afterEnter={() => videoRef.current?.play()}>
        <Dialog initialFocus={videoRef} onClose={() => setModalOpen(false)}>

          {/* 2. The backdrop layer */}
          <TransitionChild
            as="div"
            className="fixed inset-0 z-[99999] bg-black bg-opacity-50 transition-opacity"
            enter="transition ease-out duration-200"
            enterFrom="opacity-0"
            enterTo="opacity-100"
            leave="transition ease-out duration-100"
            leaveFrom="opacity-100"
            leaveTo="opacity-0"
            aria-hidden="true"
          />

          {/* 3. The modal video */}
          <TransitionChild
            as="div"
            className="fixed inset-0 z-[99999] flex p-6"
            enter="transition ease-out duration-300"
            enterFrom="opacity-0 scale-75"
            enterTo="opacity-100 scale-100"
            leave="transition ease-out duration-200"
            leaveFrom="opacity-100 scale-100"
            leaveTo="opacity-0 scale-75"
          >
            <div className="max-w-5xl mx-auto h-full flex items-center">
              <DialogPanel className="w-full max-h-full rounded-3xl shadow-2xl aspect-video bg-black overflow-hidden">
                <video ref={videoRef} width="1920" height="1080" loop controls>
                  <source src="/video.mp4" type="video/mp4" />
                  Your browser does not support the video tag.
                </video>
              </DialogPanel>
            </div>
          </TransitionChild>

        </Dialog>
      </Transition>

    </div>
  )
}

Lastly, to ensure that the video element is focused when the modal is opened, we can pass the videoRef to the Dialog’s initialFocus property.

With that, the modal video component is now complete and can be imported and used in another component like this:

<ModalVideo />

If you only need to use the component once in your app, then you’re good to go! But if you want to reuse it with different thumbnail images and videos, a few more adjustments may be necessary.

Making the modal video component reusable

To ensure that the ModalVideo component is reusable, we should pass the following properties as props to it:

  • The source attribute (src) of the thumbnail image
  • The dimensions of the thumbnail image
  • The alternative text (alt) for the thumbnail image
  • The source attribute (src) of the video
  • The dimensions of the video
<ModalVideo
  thumb={VideoThumb}
  thumbWidth={768}
  thumbHeight={432}
  thumbAlt="Modal video thumbnail"
  video="/video.mp4"
  videoWidth={1920}
  videoHeight={1080} />

We can then read these props by listing their names in the function component. To ensure that the props are properly structured, we’ll also define an object with a TypeScript interface:

interface ModalVideoProps {
  thumb: StaticImageData
  thumbWidth: number
  thumbHeight: number
  thumbAlt: string
  video: string
  videoWidth: number
  videoHeight: number
}

export default function ModalVideo({
  thumb,
  thumbWidth,
  thumbHeight,
  thumbAlt,
  video,
  videoWidth,
  videoHeight,
}: ModalVideoProps) {
...

Conclusions

And here’s the final result of the modal video component built with Next.js, TypeScript and Tailwind CSS:

'use client'

import { useState, useRef, Fragment } from 'react'
import type { StaticImageData } from 'next/image'
import { Dialog, DialogPanel, Transition, TransitionChild } from '@headlessui/react'
import Image from 'next/image'

interface ModalVideoProps {
  thumb: StaticImageData
  thumbWidth: number
  thumbHeight: number
  thumbAlt: string
  video: string
  videoWidth: number
  videoHeight: number
}

export default function ModalVideo({
  thumb,
  thumbWidth,
  thumbHeight,
  thumbAlt,
  video,
  videoWidth,
  videoHeight,
}: ModalVideoProps) {
  const [modalOpen, setModalOpen] = useState<boolean>(false)
  const videoRef = useRef<HTMLVideoElement>(null)

  return (
    <div>

      {/* Video thumbnail */}
      <button
        className="relative flex justify-center items-center focus:outline-none focus-visible:ring focus-visible:ring-indigo-300 rounded-3xl group"
        onClick={() => { setModalOpen(true) }}
        aria-label="Watch the video"
      >
        <Image className="rounded-3xl shadow-2xl transition-shadow duration-300 ease-in-out" src={thumb} width={thumbWidth} height={thumbHeight} priority alt={thumbAlt} />
        {/* Play icon */}
        <svg className="absolute pointer-events-none group-hover:scale-110 transition-transform duration-300 ease-in-out" xmlns="http://www.w3.org/2000/svg" width="72" height="72">
          <circle className="fill-white" cx="36" cy="36" r="36" fillOpacity=".8" />
          <path className="fill-indigo-500 drop-shadow-2xl" d="M44 36a.999.999 0 0 0-.427-.82l-10-7A1 1 0 0 0 32 29V43a.999.999 0 0 0 1.573.82l10-7A.995.995 0 0 0 44 36V36c0 .001 0 .001 0 0Z" />
        </svg>
      </button>
      {/* End: Video thumbnail */}

      <Transition show={modalOpen} as={Fragment} afterEnter={() => videoRef.current?.play()}>
        <Dialog initialFocus={videoRef} onClose={() => setModalOpen(false)}>

          {/* Modal backdrop */}
          <TransitionChild
            className="fixed inset-0 z-10 bg-black bg-opacity-50 transition-opacity"
            enter="transition ease-out duration-200"
            enterFrom="opacity-0"
            enterTo="opacity-100"
            leave="transition ease-out duration-100"
            leaveFrom="opacity-100"
            leaveTo="opacity-0"
            aria-hidden="true"
          />
          {/* End: Modal backdrop */}

          {/* Modal dialog */}
          <TransitionChild
            className="fixed inset-0 z-10 flex p-6"
            enter="transition ease-out duration-300"
            enterFrom="opacity-0 scale-75"
            enterTo="opacity-100 scale-100"
            leave="transition ease-out duration-200"
            leaveFrom="opacity-100 scale-100"
            leaveTo="opacity-0 scale-75"
          >
            <div className="max-w-5xl mx-auto h-full flex items-center">
              <DialogPanel className="w-full max-h-full rounded-3xl shadow-2xl aspect-video bg-black overflow-hidden">
                <video ref={videoRef} width={videoWidth} height={videoHeight} loop controls>
                  <source src={video} type="video/mp4" />
                  Your browser does not support the video tag.
                </video>
              </DialogPanel>
            </div>
          </TransitionChild>
          {/* End: Modal dialog */}

        </Dialog>
      </Transition>

    </div>
  )
}

The nice thing about this component is that by passing props for the thumbnail and the video, you can easily use it in different parts of your application.

If you’re interested in learning how to build this component in other stacks, here are the links to the first and third parts of the series: