DEV Community

Kapil Thukral
Kapil Thukral

Posted on

Demystifying the Browser Render Cycle: useLayoutEffect, requestAnimationFrame, and Layout Thrashing

As React developers, we often live in a comfortable world of state, props, and virtual DOMs. React handles the heavy lifting of updating the UI, and we rarely have to think about how pixels actually hit the glass.

However, when you need to build high-performance animations, drag-and-drop interfaces, or perfectly positioned tooltips, the abstraction leaks. You suddenly find yourself staring at weird UI flashes, dropped frames, or laggy layouts.

To fix these issues, you have to master three things: the Browser Render Cycle, React’s useLayoutEffect, and the native requestAnimationFrame API.

In this deep dive, we will map out exactly how these pieces fit together, explore the mechanics of Layout Thrashing, and learn how to write buttery-smooth UI code.


1. The Anatomy of a Single Browser Frame

To understand why our code can slow down a website, we first need to look at the browser's daily routine. Modern screens refresh 60 to 120 times per second. That means for a 60Hz monitor, the browser has exactly 16.6 milliseconds to process code and redraw the screen for a single frame.

When a piece of JavaScript alters something on a webpage, it kicks off a strict, linear pipeline called the Browser Critical Rendering Path.

  1. JavaScript: JavaScript executes and mutates the DOM tree structure or changes CSS classes.
  2. Style (Recalculate Styles): The browser matches CSS selectors against the new DOM structure to figure out which rules apply to which elements.
  3. Layout (Reflow): The browser calculates the geometry of the page. It figures out exactly how many pixels wide an element is, where it sits relative to its parent, and how it impacts surrounding elements.
  4. Paint: The browser fills in the pixels. It draws backgrounds, text colors, borders, shadows, and images. This happens across multiple separate "layers" in memory.
  5. Composite: The browser takes all those separate layers, layers them in the correct $Z$-index order, and sends them to the GPU to be drawn onto your monitor.

2. Where React and the Event Loop Fit In

When you write React code, your component updates run inside the JavaScript phase of the Event Loop. However, the Event Loop handles different types of code execution using distinct queues.

Let's look at the absolute chronological timeline of a single frame execution when a React state change occurs:

Phase A: The Execution Window

  • Macrotask Execution: The loop picks up a standard task (like a user clicking a button).
  • The Microtask Flush: As soon as that task finishes, the browser pauses everything to empty the Microtask Queue. This is where React’s Render and Commit phases live. React updates its internal state, calculates the virtual DOM, and writes the new nodes to the real DOM.

Phase B: The useLayoutEffect Interrupt

  • Because useLayoutEffect is synchronous, it fires immediately after React writes to the real DOM, but inside this same JavaScript execution window. The browser's layout and paint steps have not happened yet.

Phase C: The Render Pipeline

Once all JavaScript and microtasks are completely finished, the browser thread takes control to prepare the visual frame:

  • requestAnimationFrame (rAF): The browser executes any queued rAF callbacks. This is the last chance for JavaScript to run right before the browser calculates geometry.
  • Style & Layout: The browser calculates the positions and sizes of the elements.
  • Paint & Composite: Pixels hit the screen.

Phase D: The Post-Paint Window

  • useEffect Fires: Now that the user is looking at the updated screen, React asynchronously shoots off your standard useEffect hooks in the background so they don't block visual performance.

3. The Showdown: useLayoutEffect vs requestAnimationFrame

Both tools are used to synchronize code with the display, but they serve completely different masters.

Feature useLayoutEffect requestAnimationFrame (rAF)
Ecosystem React-specific hook. Native Browser Web API.
Execution Timing Post-DOM mutation, Pre-Paint (Microtask window). Right before the browser's native Style/Layout steps.
Blocking Behavior Synchronous. Halts the browser from printing pixels until it finishes. Asynchronous. Runs inside the browser's natural frame pacing.
Primary Use Case Fixing visual flickers by measuring/adjusting layout before rendering. High-performance, continuous UI animations and transitions.

The "Next Frame" Catch with rAF

If you schedule a requestAnimationFrame inside a standard event listener, it runs in that same frame's pipeline.

However, if you schedule a rAF inside useLayoutEffect, you are calling it at a very late stage in the JavaScript window. The browser has already locked in its pipeline plan for the current frame. As a result, the rAF callback is pushed to the next frame.

If you use rAF to change styles inside useLayoutEffect, the user will see the unadjusted layout for exactly one frame (a visual flicker) before the rAF updates the styles on the next frame.


4. Understanding Layout Thrashing

To optimize animations, you must avoid the silent performance killer: Layout Thrashing.

By default, the browser is incredibly lazy—and that's a good thing. When you write code like this:

div1.style.width = '200px';
div2.style.height = '300px';

Enter fullscreen mode Exit fullscreen mode

The browser doesn't calculate the layout twice. It simply marks its current layout data structure as "dirty". It waits until all your JavaScript code is done, and then performs a single, optimized Layout calculation right before painting.

What is Forced Synchronous Layout?

Layout Thrashing happens when you force the browser to perform this layout calculation over and over again inside a single block of JavaScript code. This occurs when you Write a style change, and then immediately Read a layout property.

Consider this code running inside a hook:

// ❌ LAYOUT THRASHING
const w1 = element1.offsetWidth;       // 1. Browser layout is clean, returns value instantly.
element1.style.width = (w1 + 10) + 'px'; // 2. DOM is mutated. Layout is now marked "DIRTY".

const w2 = element2.offsetWidth;       // 3. BOOM! Browser needs to return an accurate number,
                                        //    but layout is dirty. It is FORCED to pause your JS 
                                        //    and run an emergency, synchronous Layout calculation.
element2.style.width = (w2 + 10) + 'px'; // 4. Layout is marked dirty AGAIN.

Enter fullscreen mode Exit fullscreen mode

If you repeat this pattern inside a loop or across multiple components, you force the browser to compute the geometry of the entire webpage multiple times in a few milliseconds. This ruins your 16.6ms frame budget, causing severe animation lag (jank).


5. Why Reading Values in useLayoutEffect Causes Thrashing

When React executes your components, it processes them sequentially. If multiple child components contain useLayoutEffect hooks that individually read geometry and write style changes, they will trigger layout thrashing automatically.

// Component A (Child 1)
useLayoutEffect(() => {
  const width = boxRef.current.offsetWidth; // Read (Layout clean)
  boxRef.current.style.left = `${width}px`;  // Write (Layout becomes DIRTY)
}, []);

// Component B (Child 2)
useLayoutEffect(() => {
  const height = menuRef.current.offsetHeight; // Read (CRITICAL: Layout is dirty! Forces Reflow)
  menuRef.current.style.top = `${height}px`;    // Write (Layout becomes dirty again)
}, []);

Enter fullscreen mode Exit fullscreen mode

Because useLayoutEffect runs synchronously right after React commits updates, the second component attempts to read from a DOM tree that was just marked "dirty" by the first component. The browser has no choice but to halt execution and perform an emergency reflow.


6. How to Avoid Layout Thrashing

The golden rule for maintaining smooth frame rates is simple: Batch your reads first, and batch your writes last.

The Correct useLayoutEffect Pattern (For Instant Adjustments)

If you need to adjust positioning before an element appears on screen to prevent a flicker, keep all reads together at the top of your effect, and apply your writes at the very end.

//  PERFECTLY BATCHED
useLayoutEffect(() => {
  // 1. Do all READS first while layout is clean
  const width1 = div1.offsetWidth;
  const width2 = div2.offsetWidth;

  // 2. Do all WRITES at the end
  div1.style.transform = `translateX(${width1}px)`;
  div2.style.transform = `translateX(${width2}px)`;
}, []);

Enter fullscreen mode Exit fullscreen mode

The Hybrid Pattern (For Kicking off Smooth Animations)

If you are measuring an element to kick off a fluid, moving animation where a 1-frame delay is acceptable, you can read inside useLayoutEffect and defer the write to requestAnimationFrame.

This unblocks React's render phase completely and hands the styling work directly to the browser's native animation loop:

// ⚡ OPTIMIZED FOR ANIMATIONS
useLayoutEffect(() => {
  // Read when the layout is clean
  const currentHeight = elementRef.current.offsetHeight;

  // Defer the write to the browser's native animation timeline
  const frameId = requestAnimationFrame(() => {
    elementRef.current.style.height = `${currentHeight + 100}px`;
    elementRef.current.style.transition = 'height 300ms ease';
  });

  return () => cancelAnimationFrame(frameId);
}, []);

Enter fullscreen mode Exit fullscreen mode

By respecting the browser’s render pipeline and keeping your reads and writes separated, you ensure your React applications remain highly responsive and visually seamless.


Usecase: Positioning a tooltip around button

To position a tooltip perfectly relative to a target element (like a button), you must use useLayoutEffect.

If you use standard useEffect, the tooltip will render briefly at a default position (e.g., top-left corner or stacked awkwardly in the DOM) before jumping to the correct coordinates, causing a jarring visual flash.

useLayoutEffect lets you measure the button, calculate the math, and place the tooltip before the browser paints the screen, making the appearance completely seamless.

Here is a complete, production-ready example of how to implement this without causing layout thrashing:

import React, { useState, useLayoutEffect, useRef } from 'react';

function TooltipButton() {
  const [showTooltip, setShowTooltip] = useState(false);

  // Refs to target the real DOM elements
  const buttonRef = useRef(null);
  const tooltipRef = useRef(null);

  useLayoutEffect(() => {
    // If the tooltip isn't open, there is nothing to measure
    if (!showTooltip || !buttonRef.current || !tooltipRef.current) return;

    const button = buttonRef.current;
    const tooltip = tooltipRef.current;

    // --- STEP 1: BATCHED READS (Layout is clean) ---
    // Get the bounding boxes of both elements from browser memory
    const buttonRect = button.getBoundingClientRect();
    const tooltipRect = tooltip.getBoundingClientRect();

    // --- STEP 2: CALCULATE MATH (Pure JS calculation) ---
    // Position the tooltip directly above the center of the button
    const topPosition = buttonRect.top - tooltipRect.height - 8; // 8px spacing
    const leftPosition = buttonRect.left + (buttonRect.width / 2) - (tooltipRect.width / 2);

    // --- STEP 3: BATCHED WRITES (Layout becomes dirty) ---
    // Apply styles directly to the DOM nodes right before the browser paints
    tooltip.style.top = `${topPosition + window.scrollY}px`;
    tooltip.style.left = `${leftPosition + window.scrollX}px`;

  }, [showTooltip]); // Re-run calculations whenever the tooltip is toggled open

  return (
    <div style={{ padding: '100px', textAlign: 'center' }}>
      <button 
        ref={buttonRef}
        onMouseEnter={() => setShowTooltip(true)}
        onMouseLeave={() => setShowTooltip(false)}
        className="trigger-button"
      >
        Hover over me
      </button>

      {showTooltip && (
        <div 
          ref={tooltipRef} 
          className="tooltip-box"
          style={{
            position: 'absolute',
            backgroundColor: '#333',
            color: '#fff',
            padding: '8px 12px',
            borderRadius: '4px',
            fontSize: '14px',
            pointerEvents: 'none',
            whiteSpace: 'nowrap',
            // Start invisible or unpositioned so it doesn't flash 
            // if layout changes drastically
            top: 0,
            left: 0, 
          }}
        >
          Dynamic Tooltip Text!
        </div>
      )}
    </div>
  );
}

export default TooltipButton;

Enter fullscreen mode Exit fullscreen mode

Why this follows the Rules perfectly:

  1. It Prevents the Flicker: When showTooltip becomes true, React inserts the tooltip markup into the DOM. Before the browser draws it, useLayoutEffect halts the paint train.
  2. It Follows the Read-then-Write Pattern: Notice how button.getBoundingClientRect() and tooltip.getBoundingClientRect() are invoked right next to each other at the very beginning. This forces exactly one synchronous layout calculation.
  3. It Finishes with the Writes: The styles are injected at the very end. The browser receives the style modifications, closes the useLayoutEffect execution window, and proceeds to draw the tooltip exactly where it belongs on its very first visual frame.

Usecase: Update progressbar as the user scrolls through a blog

Creating a smooth, high-performance Custom Scroll Animation or a Progress Indicator that moves as the user scrolls down a page.

Handling scroll events in JavaScript is notoriously heavy. If a user scrolls quickly, the browser can fire the onScroll event up to 100 times per second. If you try to update the DOM (like modifying the width of a progress bar) on every single scroll event, you will overwhelm the browser thread, causing Layout Thrashing and severe visual stuttering ("jank").

By using requestAnimationFrame, you can decouple the rapid scroll events from the actual DOM updates, ensuring your animation code runs exactly once per browser frame.


Here is how you can use requestAnimationFrame in React to build a high-performance reading progress bar that fills up as the user scrolls down an article.

import React, { useState, useEffect, useRef } from 'react';

function ScrollProgressBar() {
  const [scrollProgress, setScrollProgress] = useState(0);

  // A ref to keep track of whether a frame has already been requested.
  // This acts as a lock mechanism to prevent spamming the browser.
  const rafTick = useRef(false);

  useEffect(() => {
    const handleScroll = () => {
      // If a frame calculation is already scheduled, ignore incoming scroll events
      if (rafTick.current) return;

      // Lock: We are now scheduling a frame update
      rafTick.current = true;

      // Schedule the visual update right before the next browser paint
      requestAnimationFrame(() => {
        const totalHeight = document.documentElement.scrollHeight - window.innerHeight;

        if (totalHeight > 0) {
          const progress = (window.scrollY / totalHeight) * 100;
          setScrollProgress(progress);
        }

        // Unlock: The frame has painted, we can accept a new scroll update now
        rafTick.current = false;
      });
    };

    // Listen to the window scroll event
    window.addEventListener('scroll', handleScroll, { passive: true });

    // Clean up the event listener on unmount
    return () => window.removeEventListener('scroll', handleScroll);
  }, []);

  return (
    <div>
      {/* The Progress Bar Container */}
      <div style={{
        position: 'fixed',
        top: 0,
        left: 0,
        width: '100%',
        height: '5px',
        backgroundColor: '#e0e0e0',
        zIndex: 1000
      }}>
        {/* The Animated Filling Bar */}
        <div style={{
          height: '100%',
          width: `${scrollProgress}%`,
          backgroundColor: '#007bff',
          // Notice we don't need a CSS transition here! 
          // rAF updates fast enough to feel naturally smooth.
        }} />
      </div>

      {/* Placeholder content to make the page scrollable */}
      <div style={{ padding: '20px', lineHeight: '2' }}>
        <h1>High-Performance Scroll Tracking</h1>
        <p style={{ height: '2000px' }}>Scroll down to see the progress bar animate...</p>
      </div>
    </div>
  );
}

export default ScrollProgressBar;

Enter fullscreen mode Exit fullscreen mode

Why requestAnimationFrame is the Perfect Fit Here

  1. It Implements "Throttling" Automatically: By using the rafTick.current boolean lock, we ignore the dozens of extra scroll events fired by the browser between frames. We only compute the scroll math when the browser is actively ready to draw a new frame.
  2. Perfect Frame Alignment: If the user is running a 144Hz gaming monitor, requestAnimationFrame will naturally execute 144 times a second. If they are on a 60Hz phone, it executes 60 times a second. The animation effortlessly self-corrects to match the hardware's native refresh rate.
  3. Power Saving: If the user switches tabs or minimizes the browser, requestAnimationFrame automatically pauses itself in the background. Standard setTimeout loops or raw scroll events keep chugging along in memory, draining the device's battery unnecessarily.

Top comments (0)