DEV Community

Olavi Haapala
Olavi Haapala

Posted on • Originally published at olavihaapala.fi on

Modern and Accessible Marquee Clone with TailwindCSS

What Even Is/Was marquee?

Marquee is a deprecated HTML element that renders scrolling content. For more information, check the MDN page for marquee for more details.

It is not recommended to use the <marquee> element even though it might still work and additionally in modern days, you rarely need anything like this. However, sometimes you do and you might need a solution to this using modern web technologies.

Example <marquee> if your browser still supports it:


Here's an example for you, that should render this text as an element with scrolling text if your browser still supports these.

If you see nothing above here, it means dev.to is understandably escaping the content and rendering HTML tags from user input is not supported. You can check this post on my personal blog in order to check how the <marquee> would work on your browser.

Problems with Marquee

  1. Not very user friendly – many people do not like scrolling text.

  2. Deprecated and not matching the modern web design.

  3. Stressful for the user who might miss some of the content and has to wait for the scrolling to start from the beginning.

  4. Not respecting user’s preference for reduced motion. Many people will feel physically ill or sick when a website has (too many) moving elements on it.

DO NOT USE JS FOR THIS

When implementing a <marquee>-like element using modern technologies, please bare in mind that you (probably) won’t need JS for the job outside of maybe using something like React to render the element.

My example implementation is using React in a Next.js project, but I chose not to use JS to achieve the wanted result. It is often better to try to achieve as much as possible without JS whenever feasible. JS is the root of all performance issues and especially when used for something that can be achieved using HTML and CSS only. I will probably write another blog post about this topic at a later point in time.

Marquee Using CSS Only

To achieve a rolling text effect, the only thing we need is some HTML and CSS. The effect can be achieved with CSS animations.

Basic Solution With HTML + CSS

In order to animate the text and get the feeling that the text “wraps around” and goes on infinitely, we need to add the elements twice to the DOM and animate them starting from different positions. First element will start from 0 and go all the way to -100% and the other one will start from 100% and go to 0. If we animate both elements with the same duration, the first element will be reset back to the initial position when the second element reaches the starting point, thus the user won’t notice when the animation starts over again.

Here is a basic example snippet to achieve a <marquee> like effect using HTML and CSS:

HTML

<article>
  <div class="wrapper">
    <ul class="marquee">
      <li>Item 0</li>
      <li>Item 1</li>
      <li>Item 2</li>
      <li>Item 3</li>
      <li>Item 4</li>
      <li>Item 5</li>
    </ul>
    <ul class="marquee2">
      <li>Item 0</li>
      <li>Item 1</li>
      <li>Item 2</li>
      <li>Item 3</li>
      <li>Item 4</li>
      <li>Item 5</li>
    </ul>
  </div>
</article>

Enter fullscreen mode Exit fullscreen mode

CSS

article {
  border: 2px solid red;
  overflow: hidden;
  white-space: nowrap;
  width: 350px;
  display: flex;
}

.wrapper {
  position: relative;
}

.marquee {
  animation: marquee 5s linear infinite;
}

.marquee2 {
  animation: marquee2 5s linear infinite;
  position: absolute;
  top: 0;
}

ul {
  display: flex;
  list-style: none;
  padding-left: 0;
}

li {
  margin: 8px;
}

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

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

Enter fullscreen mode Exit fullscreen mode

And here’s a link to a codepen with the above content and a functioning version of it:

Codepen.io basic marquee example

Using TailwindCSS

Tailwind config here:

extend: {
  animation: {
    marquee: 'marquee 30s linear infinite',
    marquee2: 'marquee2 30s linear infinite'
  },
  keyframes: {
    marquee: {
      '0%': { transform: 'translateX(0%)' },
      '100%': { transform: 'translateX(-100%)' }
    },
    marquee2: {
      '0%': { transform: 'translateX(100%)' },
      '100%': { transform: 'translateX(0%)' }
    }
  }
}

Enter fullscreen mode Exit fullscreen mode

And now we can use these animations using the classes that Tailwind generates for us:

<article class="flex whitespace-no-wrap overflow-x-hidden">
  <div class="relative">
    <ul class="flex animate:marquee">
      <li class="m-8">Item 0</li>
      <li class="m-8">Item 1</li>
      <li class="m-8">Item 2</li>
      <li class="m-8">Item 3</li>
      <li class="m-8">Item 4</li>
      <li class="m-8">Item 5</li>
    </ul>
    <ul class="flex absolute top-0 animate:marquee2">
      <li class="m-8">Item 0</li>
      <li class="m-8">Item 1</li>
      <li class="m-8">Item 2</li>
      <li class="m-8">Item 3</li>
      <li class="m-8">Item 4</li>
      <li class="m-8">Item 5</li>
    </ul>
  </div>
</article>

Enter fullscreen mode Exit fullscreen mode

And here’s a link to a codepen with the above content and a functioning version of it.

Accessibility Aspects

Having animations and motion on a webpage can make some people feel physically ill or sick. That’s why it is important that we as developers provide the users a way to opt-in to reduced motion. Many operating systems provide on option on the system level to set a prefers reduced motion setting, and you as a web developer can use that for reducing the motion using the prefers-reduced-motion media query.

Tailwind has these built-in as utility classes called motion-safe and motion-reduce which we can use to make the <marquee> clone accessible and user friendly. In order to use these, make sure you add this to your Tailwind config:

variants: {
  animation: ['motion-safe', 'motion-reduce'];
}

Enter fullscreen mode Exit fullscreen mode

Then we can enable the animation only if the user has not preferred reduced motion by using the class motion-safe:animate-marquee and motion-safe:overflow-x-hidden.

<!-- As a fallback we show scrollbar but hide overflow and animate it in case:
    - The user has not preferred reduced motion
    - The user's browser suppports prefers-reduced-motion media query 
 -->
<article
  class="flex whitespace-no-wrap overflow-x-scroll motion-safe:overflow-x-hidden"
>
  <div class="relative">
    <ul class="flex motion-safe:animate:marquee">
      <li class="m-8">Item 0</li>
      <li class="m-8">Item 1</li>
      <li class="m-8">Item 2</li>
      <li class="m-8">Item 3</li>
      <li class="m-8">Item 4</li>
      <li class="m-8">Item 5</li>
    </ul>
    <ul class="flex absolute top-0 motion-safe:animate:marquee2">
      <li class="m-8">Item 0</li>
      <li class="m-8">Item 1</li>
      <li class="m-8">Item 2</li>
      <li class="m-8">Item 3</li>
      <li class="m-8">Item 4</li>
      <li class="m-8">Item 5</li>
    </ul>
  </div>
</article>

Enter fullscreen mode Exit fullscreen mode

As another usability and accessibility improvement, we might want to pause the animation on hover by setting the animation-state to paused. In this case we use a class name to indicate clearly that when hovering over it, we pause the animation on any of its children. Why do we need to do this? Because if we would only pause the animation when hovering over the element that has either marquee or marquee2 animations, the other list, which at that point of time would be non visible, would continue animating and soon appear on the element. This would not be a nice experience as the texts would overlap.

.pause-animations-on-children-on-hover:hover * {
  animation-play-state: paused;
}

Enter fullscreen mode Exit fullscreen mode

Learn More

Here’s nice blog post on this topic by Josh Comeau:Accessible Animations in React.

Optimization

In our case we wanted to start the animation only after receiving data from an API. In this case we could use the will-change attribute to hint the browser that this element will be animated.

.will-change-transform {
  will-change: transform;
}

Enter fullscreen mode Exit fullscreen mode

See the MDN page for more information and please note the huge note there.

Important: will-change is intended to be used as a last resort, in order to try to deal with existing performance problems. It should not be used to anticipate performance problems.

Based on this note, I think I should actually not have used this, but instead measure the performance and only improve it if there are some performance problems.

Fun Fact

Even though <marquee> is deprecated, it’s really not gone, the functionality is currently available in most web browsers but it’s not recommended to use it as it can be removed any time in the future without a warning. See the MDN page for marquee for more details.

When I started drafting this blog post and used <marquee> in the title and excerpt of this post, the page contents started scrolling to left and I had to fix this by escaping the title and excerpt before rendering them as HTML.

My webpage contents started scrolling, due to  raw `<marquee>` endraw  in content

So, remember to escape your content, especially if it’s input directly from your users.

I first thought this blog post would be about implementing this in React and TailwindCSS, but as I mentioned, my solution uses no JS, so the framework used for rendering the HTML does not matter.

Top comments (0)