Home Blog CSS shadow play – Scroll shadows with «animation-timeline»

CSS shadow play – Scroll shadows with “animation-timeline”

Jeff Chi Avatar

“Whoever wants a beautiful shadow must not focus on the shadow, but on the body. If the shadow is right, it is because the form is right.”

from “Master Lü’s Spring and Autumn Annals

Scroll shadows are a neat trick in UX design to indicate that there is more content lying beyond the edge of a container. While native scroll bars serve the same purpose, they can sometimes look rather subtle, depending on the device and browser.

You might have found your way here because you were trying to implement CSS scroll shadows, but ran into the problem of the content covering the effect. I have come up with a new solution using the CSS property animation-timeline – and I am glad to share it with you in this article.

⚠️

At the time of publication of this article, animation-timeline is only available in Chromium-based browsers such as Google Chrome, Microsoft Edge or Opera. See: https://caniuse.com/mdn-css_properties_animation-timeline

Fortunately, scroll shadows are mainly an aesthetic improvement and not an essential function. We can look forward to a future in which all browsers can display this cool trick.

Shadows that reveal, not conceal

A common CSS trick allows us to add scroll shadows to containers. They fade out when you scroll to the end of the container, in both directions. This helps to indicate that the end of the content has also been reached.

Video example

Full CSS code
@media ( prefers-reduced-motion: no-preference ) {
  .scroll-container {
    --white: #fff;
    --black: #000;
    --scroll-shadow-size: 1rem;

    background:
      /* Gradient 1:
      - White
      - On top of the stack
      - Sticks to the top of the container */
      linear-gradient(
        180deg,
        var( --white ) var( --scroll-shadow-size ),
        transparent
      ) center top,

      /* Gradient 2:
      - White
      - Second place on the stack
      - Sticks to the bottom of the container */
      linear-gradient(
        0deg,
        var( --white ) var( --scroll-shadow-size ),
        transparent
      ) center bottom,

      /* Gradient 3:
      - Black
      - Lower on the stack
      - Sticks to the top and scrolls with the viewport */
      linear-gradient(
        180deg,
        color-mix( in srgb, var( --black ), transparent 50% ),
        transparent
      ) center top,

      /* Gradient 4:
      - Black
      - Lower on the stack
      - Sticks to the bottom and scrolls with the viewport */
      linear-gradient(
        0deg,
        color-mix( in srgb, var( --black ), transparent 50% ),
        transparent
      ) center bottom,

      /* Background 5:
      Regular white background for full container */
      var(--white);
    background-size:
      100% calc( 2 * var( --scroll-shadow-size ) ),
      100% calc( 2 * var( --scroll-shadow-size ) ),
      100% var( --scroll-shadow-size ),
      100% var( --scroll-shadow-size );
    background-attachment: local, local, scroll, scroll;
  }
}Code language: CSS (css)

With all the techniques I am going to show you, I will always check for prefers-reduced-motion. We do not want to show the animated shadows if someone has turned off animations and effects in their browser.

🎨

In the CSS snippets, I have generally broken down the code to the lines that are necessary to demonstrate the principles. The embedded examples contain additional styles – like paddings, margins and other effects – just to make them a little prettier.

Behind the curtain

I did not come up with this trick myself – you will find it quite a lot on the internet. The idea is clever: we set the rarely used CSS property background-attachment for four separate background gradients. Two will follow the scroll movement, while the two others remain fixed at the ends of the container.

Video example

--white: #0f0;
--black: #f00;Code language: CSS (css)

In this example, I highlighted the moving gradients in red and the fixed ones in green. Because the fixed gradients are declared later, they lie “behind” the moving ones and get covered by them once you reach the container’s ends.

The trouble comes with backgrounds

This works fine – until the scroll container contains elements with backgrounds or also images. These will overlay the gradients and break the effect.

Video example

A new solution thanks to “animation-timeline”

Video example

Full CSS code
@keyframes scroll-indicator {
  0% { 
    opacity: 0; 
  }
  15% { 
    opacity: 1; 
  }
  100% { 
    /* `animation-fill-mode` does not work here, so we have to define the end state. */
    opacity: 1;
  } 
}

@media ( prefers-reduced-motion: no-preference ) {
  @supports ( ( animation-timeline: scroll() ) ) {
    .scroll-container {
      --scroll-indicator-size: 1rem;

      position: relative;
      scroll-timeline-name: --scroll-timeline;
      scroll-timeline-axis: block;

      &::before,
      &::after {
        content: "";
        position: sticky;
        display: block;
        width: 100%;
        height: var( --scroll-shadow-size );
        opacity: 0;
        pointer-events: none;
        animation-name: scroll-indicator;
        animation-timeline: --scroll-timeline;
      }

      &::before {
        top: 0;
        background:
          linear-gradient(
            180deg,
            color-mix( in srgb, var( --black ), transparent 50% ),
            transparent
          );
      }

      &::after {
        animation-direction: reverse;
        bottom: 0;
        background:
          linear-gradient(
            0deg,
            color-mix( in srgb, var( --black ), transparent 50% ),
            transparent
          );
      }
    }
  }
}Code language: CSS (css)

Time for our actual trick: With the property animation-timeline we can define timelines for CSS animations to follow. This is often combined with scroll-timeline to link the animations to the users’ scrolling behaviour.

🖱️

Want to learn more about “scroll-driven animations”? Check out the demo page that we prepared with some examples and explanations.

scroll-timeline-name: --scroll-timeline;
scroll-timeline-axis: block;Code language: HTTP (http)

In these lines, we define a custom timeline called --scroll-timeline for the scroll container. The property scroll-timeline-axis: block specifies that this new timeline should be aligned with the “block” axis (the vertical y-axis for most languages).

animation-name: scroll-indicator;
animation-timeline: --scroll-timeline;Code language: CSS (css)

The scroll shadows are set on the two ::before and ::after pseudo-elements, which are attached to the top and bottom of the scroll container using position: sticky. We assign an animation to them called scroll-indicator, controlled by the animation timeline we just defined as --scroll-timeline.

@keyframes scroll-indicator {
  0% { 
    opacity: 0; 
  }
  15% { 
    opacity: 1; 
  }
  100% { 
    /* `animation-fill-mode` does not work here, so we have to define the end state. */
    opacity: 1;
  } 
}Code language: CSS (css)

The animation does nothing more than to show and hide the pseudo-elements. At 0% scroll progress they will have 0% opacity – being invisible – and at 100% scroll progress they will also have 100% opacity so you can see them again. That’s all there is to it.

Because we do not want both scroll shadows to disappear at the same time, the bottom shadow on the ::after pseudo-element is given the property animation-direction: reverse. And since we also do not want the shadows to obstruct interactions with the content, we assign them pointer-events: none.

💡

With regular, time-based animations, animation-fill-mode: both would ensure that the animation retains its final state after it has finished. This does not seem to work with animation-timeline animations, so we need to define the 100% state explicitly …

Taking a horizontal look

Of course, you can apply the same effect to horizontal scroll containers. In the following example, I use the “old” method again with background-attachment.

Video example

Full CSS code
@media ( prefers-reduced-motion: no-preference ) {
  .scroll-container {
    background:
      linear-gradient(
        90deg, /* Rotated to be vertical */
        var( --white ) var( --scroll-shadow-size ),
        transparent
      ) center left, /* `left` instead of `top` */

      linear-gradient(
        -90deg, /* Rotated to be vertical */
        var( --white ) var( --scroll-shadow-size ),
        transparent
      ) center right, /* `right` instead of `bottom` */

      linear-gradient(
        90deg, /* Rotated to be vertical */
        color-mix( in srgb, var( --black ), transparent 50% ),
        transparent
      ) center left, /* `left` instead of `top` */

      linear-gradient(
        -90deg, /* Rotated to be vertical */
        color-mix( in srgb, var( --black ), transparent 50% ),
        transparent
      ) center right, /* `right` instead of `bottom` */

      var(--white);
    background-size:
      /* Switched widths and heights */
      calc( 2 * var( --scroll-shadow-size ) ) 100%,
      calc( 2 * var( --scroll-shadow-size ) ) 100%,
      var( --scroll-shadow-size ) 100%,
      var( --scroll-shadow-size ) 100%;
    background-attachment: local, local, scroll, scroll;
    background-repeat: no-repeat;
  }
}
Code language: CSS (css)

And naturally, as soon as we add content with backgrounds, we run into our familiar problem once more.

Video example

To infinity – and beyond?

No need to worry – our new method using animation-timeline will also work for horizontal scroll containers. To accomplish this, we simply have to change scroll-timeline-axis from block to inline.

Video example

Full CSS code
@media ( prefers-reduced-motion: no-preference ) {
  @supports ( ( animation-timeline: scroll() ) ) {
    .scroll-container {
      --scroll-indicator-size: 1rem;

      position: relative;
      overflow-y: clip; /* This is new! */
      scroll-timeline-name: --scroll-timeline;
      scroll-timeline-axis: inline;

      &::before,
      &::after {
        content: "";
        position: sticky;
        display: block;
        width: var( --scroll-shadow-size ); /* `width` instead of `height` */
        height: calc( infinity * 1px ); /* This is where it gets wild … */
        opacity: 0;
        pointer-events: none;
        animation-name: scroll-indicator;
        animation-timeline: --scroll-timeline;
      }

      &::before {
        left: 0; /* `left` instead of `top` */'
        margin-block-end: calc( -infinity * 1px ); /* I'll explain in just a minute … */
        background:
          linear-gradient(
            90deg, /* Rotated to be vertical */
            color-mix( in srgb, var( --black ), transparent 50% ),
            transparent
          );
      }

      &::after {
        animation-direction: reverse;
        left: calc( 100% - var( --scroll-shadow-size ) ); /* Instead of `bottom: 0` */
        margin-block-start: calc( -infinity * 1px ); /* Trust me!!! */
        background:
          linear-gradient(
            -90deg, /* Rotated to be vertical */
            color-mix( in srgb, var( --black ), transparent 50% ),
            transparent
          );
      }    
    }
  }
}
Code language: CSS (css)

However, when we scroll along the x-axis, we need to add one more trick: although the browser always knows what width: 100% means, height: 100% has no effect without a fixed value. This is why we cannot just stretch the pseudo-elements to the correct size. Time to get creative.

.scroll-container {
  overflow-y: clip;

  &::before,
  &::after {
    height: calc( infinity * 1px );
  }

  &::before {
    margin-block-end: calc( -infinity * 1px );
  }

  &::after {
    margin-block-start: calc( -infinity * 1px );
  }
}Code language: CSS (css)

That’s right: We assign infinite heights to our pseudo-elements. Then we move them into the visible area by applying negative infinite margins. This way they will always cover the entire scroll container – sounds a little crazy, but it does the trick.

Infinite possibilities

This was the first time I used animation-timeline in real life. Exactly these kinds of challenges are what I enjoy most in my work as a frontend developer: discovering a new CSS property, experimenting with it, and then suddenly solving all my problems in a clever way.

Creative puzzle-solving like this keeps your mind sharp and is a welcome change of scenery when you have been staring at databases and deployments for too long. So never stop playing around – who knows what else you might discover in the shadows …