CSS properties - content-visibility

revision:


content-visibility property

- controls whether or not an element renders its contents at all, along with forcing a strong set of containments, allowing user agents to potentially omit large swathes of layout and rendering work until it becomes needed.

It enables the user agent to skip an element's rendering work (including layout and painting) until it is needed — which makes the initial page load much faster.

When to use: 1. long webpages with heavy off-screen content, 2. scollable lists/grids, 3. elements with frequent visibility toggles.

Syntax : content-visibility = visible | auto | hidden

Property values:

visibile : no effect. The element's contents are laid out and rendered as normal.

auto : the element skips its contents. The skipped contents must not be accessible to user-agent features, such as find-in-page, tab-order navigation, etc., nor be selectable or focusable. This is similar to giving the contents "display: none".

hidden : the element turns on layout containment, style containment, and paint containment. If the element is not relevant to the user, it also skips its contents. Unlike hidden, the skipped contents must still be available as normal to user-agent features such as find-in-page, tab order navigation, etc., and must be focusable and selectable as normal.

Syntax examples

        /* Keyword values */
        content-visibility: visible;
        content-visibility: hidden;
        content-visibility: auto;
        
        /* Global values */
        content-visibility: inherit;
        content-visibility: initial;
        content-visibility: revert;
        content-visibility: revert-layer;
        content-visibility: unset;
        

JavaScript syntax: object.style.contentVisibility="visible"


how to use content-visibility in a webpage?

1. basic usage for off-screen content.

Apply content-visibility: auto to sections that are initially off-screen (e.g., below the fold). This skips rendering until the user scrolls near the element.

example

        <style>
            .section {content-visibility: auto; /* Estimate height to prevent layout shifts */ contain-intrinsic-size: 500px; margin-bottom: 20px;}
        </style>
          
        <div class="section">Section 1 (Heavy Content)</div>
        <div class="section">Section 2 (Heavy Content)</div>
    

2. use contain-intrinsic-size

Provide an estimated height/width to reserve space for off-screen elements, avoiding layout shifts when they render.

example

        <style>
            .section { content-visibility: auto; /* Syntax: contain-intrinsic-size: <width> <height>; */ contain-intrinsic-size: auto 500px; /* Auto width, 500px height */ }
        </style>
    

3. optimize scrollable containers

For elements inside scrollable containers (e.g., a list), apply content-visibility: auto to individual items.

example

        <style>
            .scroll-container {height: 300px; overflow-y: auto;}
            .item { content-visibility: auto; contain-intrinsic-size: 100px; padding: 12px;}
        </style>
          
        <div class="scroll-container">
            <div class="item">Item 1</div>
            <div class="item">Item 2</div>
            <!-- 100+ items -->
        </div>
    

4. toggle visibility efficiently

Use content-visibility: hidden (instead of "display: none") for elements that toggle visibility, preserving rendering state.

example

        <style>
            .toggle-content {content-visibility: hidden; contain-intrinsic-size: 200px; /* Reserve space if needed */ }
        </style>

        <button onclick="toggle()">Show/Hide</button>
        <div class="toggle-content" id="content">Hidden Content</div>

        <script>
            function toggle() {
                const el = document.getElementById('content');
                el.style.contentVisibility = 
                el.style.contentVisibility === 'hidden' ? 'visible' : 'hidden';
            }
        </script>
    

5. browser support & fallbacks

Supported browsers: Chromium (v85+), Firefox (v109+), Safari (partial).
Progressive enhancement: use @supports to avoid issues in unsupported browsers:


examples

This content is initially visible and can be hidden by clicking the button.

            <div>
                <div class="hidden">
                    <button class="toggle">Show</button>
                    <p> This content is initially hidden and can be shown by clicking the button.</p>
                </div>
                <div class="visible">
                    <button class="toggle">Hide</button>
                    <p>This content is initially visible and can be hidden by clicking the button.</p>
                </div>
            </div>
            <style>
                .hidden p, .visible p { contain-intrinsic-size: 0 1.1em; border: dotted 2px;}
                .hidden > p {content-visibility: hidden;}
                .visible > p {content-visibility: visible;}
            </style>
            <script>
                const handleClick = (event) => {
                    const button = event.target;
                    const div = button.parentElement;
                    button.textContent = div.classList.contains("visible") ? "Show" : "Hide";
                    div.classList.toggle("hidden");
                    div.classList.toggle("visible");
                };
    
                document.querySelectorAll("button.toggle").forEach((button) => {
                    button.addEventListener("click", handleClick);
                });
            </script>
        
Hi, I'm hidden. Notice that all of my styling is hidden as well, and that I still take up space, even though you can't see me.
Howdy, my parent element is hidden, but I'm still visible.
Hover over me to make my parent visible.
            <div class="hidden1">
                Hi, I'm hidden. Notice that all of my styling is hidden as well, and that
                 I still take up space, even though you can't see me.
                <div class="visible1">
                  Howdy, my parent element is hidden, but I'm still visible. <br>Hover
                   over me to make my parent visible.
                </div>
            </div>
            <style>
                .hidden1 {visibility: hidden; background: pink; border: 10px dotted teal;
                     padding: 10px;
                    &:hover {visibility: visible;}
                }
                .visible1 {border: 1px solid black; visibility: visible;
                }
            </style>