Sticky Headers And Full-Height Elements: A Tricky Combination

Sticky Headers And Full-Height Elements: A Tricky Combination

Sticky Headers And Full-Height Elements: A Tricky Combination

Philip Braunen

2024-09-05T09:00:00+00:00
2025-03-06T17:04:34+00:00

I was recently asked by a student to help with a seemingly simple problem. She’d been working on a website for a coffee shop that sports a sticky header, and she wanted the hero section right underneath that header to span the rest of the available vertical space in the viewport.

Here’s a visual demo of the desired effect for clarity.

Looks like it should be easy enough, right? I was sure (read: overconfident) that the problem would only take a couple of minutes to solve, only to find it was a much deeper well than I’d assumed.

Before we dive in, let’s take a quick look at the initial markup and CSS to see what we’re working with:

<body>
  <header class="header">Header Content</header>
  <section class="hero">Hero Content</section>
  <main class="main">Main Content</main>
</body>
.header {
  position: sticky;
  top: 0; /* Offset, otherwise it won't stick! */
}

/* etc. */

With those declarations, the .header will stick to the top of the page. And yet the .hero element below it remains intrinsically sized. This is what we want to change.

The Low-Hanging Fruit

The first impulse you might have, as I did, is to enclose the header and hero in some sort of parent container and give that container 100vh to make it span the viewport. After that, we could use Flexbox to distribute the children and make the hero grow to fill the remaining space.

<body>
  <div class="container">
    <header class="header">Header Content</header>
    <section class="hero">Hero Content</section>
  </div>
  <main class="main">Main Content</main>
</body>
.container {
  height: 100vh;
  display: flex;
  flex-direction: column;
}

.hero {
  flex-grow: 1;
}

/* etc. */

This looks correct at first glance, but watch what happens when scrolling past the hero.

See the Pen [Attempt #1: Container + Flexbox [forked]](https://codepen.io/smashingmag/pen/yLdQgQo) by Philip.

See the Pen Attempt #1: Container + Flexbox [forked] by Philip.

The sticky header gets trapped in its parent container! But.. why?

If you’re anything like me, this behavior is unintuitive, at least initially. You may have heard that sticky is a combination of relative and fixed positioning, meaning it participates in the normal flow of the document but only until it hits the edges of its scrolling container, at which point it becomes fixed. While viewing sticky as a combination of other values can be a useful mnemonic, it fails to capture one important difference between sticky and fixed elements:

A position: fixed element doesn’t care about the parent it’s nested in or any of its ancestors. It will break out of the normal flow of the document and place itself directly offset from the viewport, as though glued in place a certain distance from the edge of the screen.

Conversely, a position: sticky element will be pushed along with the edges of the viewport (or next closest scrolling container), but it will never escape the boundaries of its direct parent. Well, at least if you don’t count visually transform-ing it. So a better way to think about it might be, to steal from Chris Coyier, that position: sticky is, in a sense, a locally scoped position: fixed.” This is an intentional design decision, one that allows for section-specific sticky headers like the ones made famous by alphabetical lists in mobile interfaces.

See the Pen [Sticky Section Headers [forked]](https://codepen.io/smashingmag/pen/OJeaWrM) by Philip.

See the Pen Sticky Section Headers [forked] by Philip.

Okay, so this approach is a no-go for our predicament. We need to find a solution that doesn’t involve a container around the header.

Fixed, But Not Solved

Maybe we can make our lives a bit simpler. Instead of a container, what if we gave the .header element a fixed height of, say, 150px? Then, all we have to do is define the .hero element’s height as height: calc(100vh - 150px).

See the Pen [Attempt #2: Fixed Height + Calc() [forked]](https://codepen.io/smashingmag/pen/yLdQgGz) by Philip.

See the Pen Attempt #2: Fixed Height + Calc() [forked] by Philip.

This approach kinda works, but the downsides are more insidious than our last attempt because they may not be immediately apparent. You probably noticed that the header is too tall, and we’d wanna do some math to decide on a better height.

Thinking ahead a bit,

  • What if the .header’s children need to wrap or rearrange themselves at different screen sizes or grow to maintain legibility on mobile?
  • What if JavaScript is manipulating the contents?

All of these things could subtly change the .header’s ideal size, and chasing the right height values for each scenario has the potential to spiral into a maintenance nightmare of unmanageable breakpoints and magic numbers — especially if we consider this needs to be done not only for the .header but also the .hero element that depends on it.

I would argue that this workaround also just feels wrong. Fixed heights break one of the main affordances of CSS layout — the way elements automatically grow and shrink to adapt to their contents — and not relying on this usually makes our lives harder, not simpler.

So, we’re left with…

A Novel Approach

Now that we’ve figured out the constraints we’re working with, another way to phrase the problem is that we want the .header and .hero to collectively span 100vh without sizing the elements explicitly or wrapping them in a container. Ideally, we’d find something that already is 100vh and align them to that. This is where it dawned on me that display: grid may provide just what we need!

Let’s try this: We declare display: grid on the body element and add another element before the .header that we’ll call .above-the-fold-spacer. This new element gets a height of 100vh and spans the grid’s entire width. Next, we’ll tell our spacer that it should take up two grid rows and we’ll anchor it to the top of the page.

This element must be entirely empty because we don’t ever want it to be visible or to register to screen readers. We’re merely using it as a crutch to tell the grid how to behave.

<body>
  <!-- This spacer provides the height we want -->
  <div class="above-the-fold-spacer"></div>

  <!-- These two elements will place themselves on top of the spacer -->
  <header class="header">Header Content</header>
  <section class="hero">Hero Content</section>

  <!-- The rest of the page stays unaffected -->
  <main class="main">Main Content</main>
</body>
body {
  display: grid;
}

.above-the-fold-spacer {
  height: 100vh;
  /* Span from the first to the last grid column line */
  /* (Negative numbers count from the end of the grid) */
  grid-column: 1 / -1;
  /* Start at the first grid row line, and take up 2 rows */
  grid-row: 1 / span 2; 
}

/* etc. */

This is the magic ingredient.

By adding the spacer, we’ve created two grid rows that together take up exactly 100vh. Now, all that’s left to do, in essence, is to tell the .header and .hero elements to align themselves to those existing rows. We do have to tell them to start at the same grid column line as the .above-the-fold-spacer element so that they won’t try to sit next to it. But with that done… ta-da!

See the Pen [The Solution: Grid Alignment [forked]](https://codepen.io/smashingmag/pen/YzoRNdo) by Philip.

See the Pen The Solution: Grid Alignment [forked] by Philip.

The reason this works is that a grid container can have multiple children occupying the same cell overlaid on top of each other. In a situation like that, the tallest child element defines the grid row’s overall height — or, in this case, the combined height of the two rows (100vh).

To control how exactly the two visible elements divvy up the available space between themselves, we can use the grid-template-rows property. I made it so that the first row uses min-content rather than 1fr. This is necessary so that the .header doesn’t take up the same amount of space as the .hero but instead only takes what it needs and lets the hero have the rest.

Here’s our full solution:


body {
  display: grid;
  grid-template-rows: min-content 1fr;
}

.above-the-fold-spacer {
  height: 100vh;
  grid-column: 1 / -1;
  grid-row: 1 / span 2;
}

.header {
  position: sticky;
  top: 0;
  grid-column-start: 1;
  grid-row-start: 1;
}

.hero {
  grid-column-start: 1;
  grid-row-start: 2;
}

And voila: A sticky header of arbitrary size above a hero that grows to fill the remaining visible space!

Caveats and Final Thoughts

It’s worth noting that the HTML order of the elements matters here. If we define .above-the-fold-spacer after our .hero section, it will overlay and block access to the elements underneath. We can work around this by declaring either order: -1, z-index: -1, or visibility: hidden.

Keep in mind that this is a simple example. If you were to add a sidebar to the left of your page, for example, you’d need to adjust at which column the elements start. Still, in the majority of cases, using a CSS Grid approach is likely to be less troublesome than the Sisyphean task of manually managing and coordinating the height values of multiple elements.

Another upside of this approach is that it’s adaptable. If you decide you want a group of three elements to take up the screen’s height rather than two, then you’d make the invisible spacer span three rows and assign the visible elements to the appropriate one. Even if the hero element’s content causes its height to exceed 100vh, the grid adapts without breaking anything. It’s even well-supported in all modern browsers.

The more I think about this technique, the more I’m persuaded that it’s actually quite clean. Then again, you know how lawyers can talk themselves into their own arguments? If you can think of an even simpler solution I’ve overlooked, feel free to reach out and let me know!

Smashing Editorial
(gg, yk)
5 Best WordPress Backup Plugins for Data Security (2024)

Ever wondered what would happen if your WordPress site suddenly disappeared?

It’s a nightmare scenario, but one that can be easily avoided with the right backup solution in place. For anyone managing a WordPress site, whether it’s a personal blog or a full-scale business, having a reliable backup is crucial.

This post highlights some of the best WordPress backup plugins available, focusing on their key features and how they can protect your site from unexpected data loss, security threats, and other digital disasters.

Let’s take a closer look at how these tools can help keep your WordPress site secure and your data intact.

BlogVault

BlogVault
Pros
  • Complete solution with backups, staging, and security
  • Reliable restore process with a high success rate
  • Integrated staging for safe testing
  • Real-time backups for WooCommerce sites
  • Automated malware scanning
Cons
  • Limited options for backup storage locations
  • High cost might not suit budget-conscious users

BlogVault is a robust WordPress backup plugin that’s ideal for business-critical websites. It offers features like incremental backups, secure cloud storage, and one-click restore, making sure your site can bounce back quickly from any issues. With a strong track record of successful backups, BlogVault provides the reliability you need.

What sets BlogVault apart is its strong data security, with encrypted backups stored across multiple data centers. The plugin also makes it easy to create staging sites, so you can test changes before making them live. Its migration tool is another standout feature, allowing you to move your site to a new host with minimal effort.

Who should use this?

BlogVault is perfect for businesses that need reliable backups, especially if downtime is not an option. It’s also great for developers who need staging environments or anyone who wants comprehensive backup and security for their WordPress site.

Visit BlogVault

Solid Backups

Solid Backups
Pros
  • Easy-to-use interface with clear options
  • Efficient incremental backups
Cons
  • No free version or trial available

Previously known as BackupBuddy, Solid Backups has been around since 2010, helping WordPress users keep their sites safe. This plugin allows you to back up your entire site, including your database and files, with just a few clicks. You can choose to back up everything or only specific files and folders, and send these backups to secure locations like Amazon S3, Dropbox, or even your email.

Solid Backups also makes it easy to restore your site, whether you need to recover a single file or roll back your database. You can set up automatic backups on a schedule that works for you, so you don’t have to worry about losing any data.

Who should use this?

Solid Backups is ideal for anyone who wants a dependable and user-friendly backup solution. It’s particularly useful if you manage multiple sites or prefer to set up backups once and let the plugin handle the rest.

Visit Solid Backups

UpdraftPlus

UpdraftPlus
Pros
  • Free version includes scheduled and incremental backups
  • Works with many cloud storage services
  • Light on server resources
  • Easy setup for beginners
  • Premium version offers extra features like site migration
Cons
  • User interface might seem complex at first
  • Advanced features require the premium version

UpdraftPlus is a widely trusted WordPress backup plugin, known for its reliability and ease of use. Whether you’re looking to create a complete backup of your website or just want to save specific parts like plugins, themes, or databases, UpdraftPlus makes it simple. You can choose to store your backups locally on your computer or in the cloud, with support for services like Dropbox, Google Drive, and Amazon S3.

This plugin offers both scheduled and on-demand backups, giving you the flexibility to secure your site whenever needed. While the free version covers essential features, upgrading to the premium version unlocks advanced options like incremental backups and seamless site migration. The straightforward setup ensures that even beginners can quickly get started with protecting their WordPress site.

Who should use this?

UpdraftPlus is a good choice for anyone looking for a reliable backup plugin that’s easy to use. Whether you’re new to WordPress or have been using it for years, this plugin can help keep your site’s data safe, especially if you want to store backups in more than one place.

Visit UpdraftPlus

Jetpack VaultPress Backups

Jetpack VaultPress Backups
Pros
  • Real-time backups for every site change
  • Quick recovery, even during hosting issues
  • User-friendly interface
Cons
  • Can’t manually choose specific files or directories to back up
  • No option to select your storage provider
  • Could be expensive for large sites

Following UpdraftPlus, Jetpack VaultPress Backups offers a robust solution for WordPress and WooCommerce sites, prioritizing data safety and easy recovery. Unlike many backup options, it provides real-time backups stored securely in the Jetpack Cloud, ensuring that every change on your site is instantly protected.

If your site encounters an issue, you can restore it swiftly with just a few clicks, even if your hosting service is experiencing downtime. With automated backups and a 30-day archive, Jetpack VaultPress Backups delivers peace of mind, making it a dependable choice for those who require continuous site protection.

Who should use this?

This plugin is particularly well-suited for users who prioritize security and need a hands-off, automated backup solution, especially those managing online stores or frequently updated sites.

Visit Jetpack VaultPress Backups

BackWPup

BackWPup
Pros
  • Flexible backup options
  • Proven reliability
  • Easy scheduling for automated backups
Cons
  • Advanced features require the premium version

As you explore backup solutions, BackWPup stands out as a straightforward yet powerful plugin for WordPress users. It simplifies the backup process, allowing you to easily secure your entire site, with storage options ranging from Dropbox to S3 and FTP. Additionally, BackWPup includes restore capabilities in its free version, making it a comprehensive choice for safeguarding your site.

Designed with ease of use in mind, BackWPup caters to both beginners and experienced users alike. You can schedule automatic backups and choose your preferred storage locations, ensuring that your site’s data remains safe and accessible whenever you need it.

Who should use this?

BackWPup is great for anyone looking for a straightforward backup and restore plugin, whether you’re running a small blog or managing several sites.

Visit BackWPup

The post 5 Best WordPress Backup Plugins for Data Security (2024) appeared first on Hongkiat.