DEV Community

Silvestar Bistrović
Silvestar Bistrović

Posted on • Edited on • Originally published at silvestar.codes

Building an animated sticky header with custom offset

Making a website with Hugo cover

This post was originally published on silvestar.codes.

Sticky elements are used all across the web. With the appearance of the position: sticky property, we could create sticky elements without JavaScript plugins. But creating sticky elements with animations couldn't be achieved without a little bit of JavaScript.

The specifications

The task is to build a sticky header that appears when the user scrolls past the static header or on a custom offset position. When scrolling down, the static header should stay at its place, while the sticky header should roll down. When scrolling up, the sticky header should disappear, and the static header should roll down. Note that the content of the static header and the sticky header differs.

The solution

To make it easier to understand the specifications, see this pen that contains the full solution:

The breakdown

For a better understanding of how this code works, we should examine the HTML structure.

HTML

Here's the simplified preview of the HTML structure:

<main>
  <header>
    <div class="header header--alpha">
      ...
    </div>
    <div class="header header--beta">
      ...
    </div>
  </header>
  <div class="fake-header"></div>
  <article>
    ... 
  </article>
</main>
Enter fullscreen mode Exit fullscreen mode

Inside the main element, we have header, .fake-header and article elements. To make position: sticky work, the sticky element should be inside the scrollable element — the main element in our case.

Header holds two elements:

  • .header--alpha, the static header, and
  • .header--beta, the sticky header.

CSS

First, let's style our wrapper element. It should have position relative by default, and sticky once the offset is reached.

header {
  top: 0;
  position: relative;
  overflow: hidden;
}

header.sticky {
  position: sticky;
}
Enter fullscreen mode Exit fullscreen mode

The property top: 0 would make sure our header is stuck to the top, and the property overflow: hidden hides everything outside of the header element.

Next, we should define the behavior of the header parts.

.header--alpha {
  transition: .225s ease-out;
}

.sticky .header--alpha {
  opacity: .5;
  transform: translateY(-100%);
  transition: none;
}
Enter fullscreen mode Exit fullscreen mode

.header--alpha, our static header is visible by default. When the user scrolls past the offset, and the wrapper header element becomes sticky, it will be translated outside of the wrapper element instantaneously without any transition effect. Note that element will be transitioned when it goes to the original position when sticky effect won't be active.

.header--beta {
  position: absolute;
  top: 0;
  left: 0;
  right: 0;
  opacity: .5;
  transform: translateY(-100%);
}

.sticky .header--beta {
  opacity: 1;
  transform: translateY(0);
  transition: .35s ease-out;
}
Enter fullscreen mode Exit fullscreen mode

.header--beta, our sticky header is positioned absolutely and translated outside of the wrapper header element. When the sticky effect is activated, it will be translated and transitioned in the original position.

JavaScript

You might have noticed the .fake-header element.

.fake-header {
  height: 1px;
  position: relative;
}
Enter fullscreen mode Exit fullscreen mode

This element serves as an offset for the scroll. When it reaches the top of the viewport, the header becomes sticky. And when the header becomes sticky, the fake element will be pushed up by the height of the header plus one extra pixel. When the user scrolls up and reaches the fake element in its new position, the sticky effect will be turned off.

const $realSticky = document.querySelector("header");
const $fakeSticky = document.querySelector(".fake-header");

const stickyHeader = () => function() {
  const sr1 = $fakeSticky.getBoundingClientRect();
  const sr2 = $realSticky.getBoundingClientRect();

  if (sr1.top > 0) {
    $realSticky.classList.remove("sticky");
    $fakeSticky.style.top = 0;
  } else {
    $realSticky.classList.add("sticky");
    $fakeSticky.style.top = `-${sr2.height + 1}px`;
  }
};

window.addEventListener("scroll", _.debounce(stickyHeader(), 15));
Enter fullscreen mode Exit fullscreen mode

In this example, lodash debounce function is used to execute the task on scroll event.

I am aware the I could write better JS code and avoid adding the .fake-header element, but I wanted to create a demo without too many calculations.

Extra

Since the static header determines the height of the wrapper element, we have a problem with selecting or clicking on the content immediately after the sticky header, once it appears.

We could use pointer-events to fix this issue:

header {
  pointer-events: none;
}

.header {
  pointer-events: all;
}
Enter fullscreen mode Exit fullscreen mode

First we would set pointer-events to none to the header element to disable click/hover events, and then we would set the pointer-events to all to reenable them again.

Conclusion

You could see the full demo on my Codepen.

This demo uses position: sticky, and pointer-events properties that have limited support.

I am really excited to see how fast CSS is moving. Brand new properties, like position: sticky and pointer-events allows us to create better experience for the end user more easily and natively. And support is getting better and better every day. 🏆

Top comments (4)

Collapse
 
equinusocio profile image
Mattia Astorino

Doesn’t work on mobile

Collapse
 
starbist profile image
Silvestar Bistrović

Hmmm, which browser and phone? It works on Chrome my P20 Lite. 🤔

Collapse
 
equinusocio profile image
Mattia Astorino • Edited

Safari on iOS. I think it may be a codepen issue

Thread Thread
 
starbist profile image
Silvestar Bistrović

Thank you for your feedback. 👍