Digital Powertools

Make your tools easier, faster and better

Tab tab tab - Building CSS-only tabs

A developer who just wanted a simple tab interface. Generated with ChatGPT.
A developer who just wanted a simple tab interface. Generated with ChatGPT.

Tabs are an essential UI component that allows users to navigate between different sections of content without scrolling or changing pages. While JavaScript is commonly used for tab functionality, we can create fully functional tabs using only CSS. This approach is lightweight, accessible, and works even when JavaScript is disabled.

Details and Summaries

The foundation of our CSS-only tabs will use the native <details> and <summary> HTML elements. These elements provide built-in expand/collapse functionality without any JavaScript. As a recent addition to the specification for the details element, the name attribute allows to group the details elements together. Within the group, only one single detail element can be expanded.

Let's start with a basic example:


<div class="tabs">
    <details name="tab-group" open>
        <summary>Tab 1</summary>
        <div>
            <p>This is the content for Tab 1.</p>
        </div>
    </details>

    <details name="tab-group">
        <summary>Tab 2</summary>
        <div>
            <p>This is the content for Tab 2.</p>
        </div>
    </details>

    <details name="tab-group">
        <summary>Tab 3</summary>
        <div>
            <p>This is the content for Tab 3.</p>
        </div>
    </details>
</div>
        

While this is a very minimal example, there are a couple of things to note:

We'll be using this HTML structure for all remaining examples in the rest of the article. If you're reading this in a modern browser, this plain example without any css looks like this:

Example 1
Tab 1

This is the content for Tab 1.

Tab 2

This is the content for Tab 2.

Tab 3

This is the content for Tab 3.

Going horizontal

The details are all hooked up together, but they don't look like tabs at all yet. Nothing we can't fix with some css, but let's go through it step by step.


.tabs {
    display: flex;
    flex-wrap: wrap;

    & > details {
        display: contents;

        div {
            order: 1;
        }
        &::details-content {
            order: 1;
        }
    }
    summary {
        order: 0;
    }
}
        

The current iteration will look like this, with the summaries horizontally sorted on the left followed by the active details content div:

Example 2
Tab 1

This is the content for Tab 1.

Tab 2

This is the content for Tab 2.

Tab 3

This is the content for Tab 3.

Getting pretty with it

While the current iteration looks good, it's still not looking very much like tabs. Let's add some styling to make it look a bit nicer.


.tabs {
    display: flex;
    flex-wrap: wrap;

    & > details {
        display: contents;

        div {
            order: 1;
            width: 100%;

            padding: .5rem;
            border: 1px solid var(--primary-color);
            background-color: color-mix(in srgb, var(--neutral-color) 100%, var(--secondary-color) 10%);
        }
        &[open]::details-content {
            order: 1;
            width: 100%;
        }
    }
    summary {
        order: 0;

        padding: .5rem;
        border: 1px solid var(--primary-color);
        border-bottom: none;

        [open] > & {
            z-index: 1;
            margin-bottom: -1px;
            background-color: color-mix(in srgb, var(--neutral-color) 100%, var(--secondary-color) 10%);
        }
    }
}
        
Example 3
Tab 1

This is the content for Tab 1.

Tab 2

This is the content for Tab 2.

Tab 3

This is the content for Tab 3.

Are we there yet?

That already looks like a tab interface! But there still some rough edges that you might have noticed. Let's get them fixed and push this over the finish line.


.tabs {
    display: flex;
    flex-wrap: wrap;

    & > details {
        display: contents;

        div {
            order: 1;
            width: 100%;

            padding: .5rem;
            border: 1px solid var(--primary-color);
            background-color: color-mix(in srgb, var(--neutral-color) 100%, var(--secondary-color) 10%);
        }
        &[open]::details-content {
            order: 1;
            width: 100%;
        }
    }
    summary {
        order: 0;
        display: block;
        cursor: pointer;

        padding: .5rem;
        border: 1px solid var(--primary-color);
        border-bottom: none;

        &::-webkit-details-marker,
        &::marker {
            display: none;
        }

        [open] > & {
            pointer-events: none;
            z-index: 1;
            margin-bottom: -1px;
            background-color: color-mix(in srgb, var(--neutral-color) 100%, var(--secondary-color) 10%);
        }
    }
}
        
Example 4
Tab 1

This is the content for Tab 1.

Tab 2

This is the content for Tab 2.

Tab 3

This is the content for Tab 3.

Well, there you have it. Modern, CSS-only tabs. The final result above, while still looking quite plain, is fully functional. Now it's your turn to go nuts with it. Center the summaries. Make the summaries rounded. Put items in the summaries. The sky is the limit, go make it pretty and dazzling. .

And when you have, shoot me an email or just message me on 𝕏 or LinkedIn. I would love to see what you do with it.

Want to read more? Check out my previous post Go build it. on what my optimal Dockerfile for Go applications looks like.