Skip to content

Scan for React performance issues and eliminate slow renders in your app

License

Notifications You must be signed in to change notification settings

aidenybai/react-scan

Repository files navigation

React Scan

React Scan detects performance issues in your React app.

Previously, tools like <Profiler />, Why Did You Render?, and React Devtools required lots of manual code change, lacked simple visual cues, or didn't have a simple, programmatic API.

Instead, React Scan automatically detects and highlights renders that cause performance issues. This shows you exactly which components you need to fix.

It's also just JavaScript, so you drop it in anywhere – script tag, npm, you name it!

React Scan in action


Engineering teams use React Scan to optimize their React apps:

                  

Looking for a more advanced version? Check out Million Lint!

Install

The fastest way to get started is via CLI:

npx react-scan@latest https://react.dev

If you want to test your app locally, add this script to your app:

<!-- import this BEFORE any scripts -->
<script src="https://unpkg.com/react-scan/dist/auto.global.js"></script>

Examples:

Next.js (pages)

Add the script tag to your pages/_document.tsx:

import { Html, Head, Main, NextScript } from 'next/document';

export default function Document() {
  return (
    <Html lang="en">
      <Head>
        <script src="https://unpkg.com/react-scan/dist/auto.global.js"></script>

        {/* rest of your scripts go under */}
      </Head>
      <body>
        <Main />
        <NextScript />
      </body>
    </Html>
  );
}
Next.js (app)

Add the script tag to your app/layout.tsx:

export default function RootLayout({
  children,
}: {
  children: React.ReactNode
}) {
  return (
    <html lang="en">
      <head>
        <script src="https://unpkg.com/react-scan/dist/auto.global.js" async />
        {/* rest of your scripts go under */}
      </head>
      <body>{children}</body>
    </html>
  )
}
Vite / Create React App

Add the script tag to your index.html:

<!doctype html>
<html lang="en">
  <head>
    <script src="https://unpkg.com/react-scan/dist/auto.global.js"></script>

    <!-- rest of your scripts go under -->
  </head>
  <body>
    <!-- ... -->
  </body>
</html>

Or, install it via npm:

npm install react-scan

Then, in your app, import this BEFORE react:

import { scan } from 'react-scan'; // import this BEFORE react
import React from 'react';

scan({
  enabled: true,
  log: true, // logs render info to console (default: false)
});

API Reference

scan(options)

Automatically scan your app for renders.

scan({
  /**
   * Enable/disable scanning
   */
  enabled: true,
  // Recommended way:
  // enabled: process.env.NODE_ENV === 'development',

  /**
   * Include children of a component applied with withScan
   */
  includeChildren: true,

  /**
   * Enable/disable geiger sound
   */
  playSound: true,

  /**
   * Log renders to the console
   */
  log: false,

  /**
   * Show toolbar bar
   */
  showToolbar: true,

  /**
   * Render count threshold, only show
   * when a component renders more than this
   */
  renderCountThreshold: 0,

  /**
   * Report data to getReport()
   */
  report: false,

  onCommitStart: () => {},
  onRender: (fiber, render) => {},
  onCommitFinish: () => {},
  onPaintStart: (outline) => {},
  onPaintFinish: (outline) => {},
});
withScan(Component, options)

Scan a specific component for renders.

function Component(props) {
  // ...
}

withScan(Component, {
  /**
   * Enable/disable scanning
   */
  enabled: true,
  // Recommended way:
  // enabled: process.env.NODE_ENV === 'development',

  /**
   * Include children of a component applied with withScan
   */
  includeChildren: true,

  /**
   * Enable/disable geiger sound
   */
  playSound: true,

  /**
   * Log renders to the console
   */
  log: false,

  /**
   * Show toolbar bar
   */
  showToolbar: true,

  /**
   * Render count threshold, only show
   * when a component renders more than this
   */
  renderCountThreshold: 0,

  /**
   * Report data to getReport()
   */
  report: false,

  onCommitStart: () => {},
  onRender: (fiber, render) => {},
  onCommitFinish: () => {},
  onPaintStart: (outline) => {},
  onPaintFinish: (outline) => {},
});
getReport()

Get a aggregated report of all components and renders.

scan({ report: true });

const report = getReport();

for (const component in report) {
  const { count, time } = report[component];

  console.log(`${component} rendered ${count} times, took ${time}ms`);
}
setOptions(options)
function Component(props) {
  // ...
}

setOptions({
  /**
   * Enable/disable scanning
   */
  enabled: true,
  // Recommended way:
  // enabled: process.env.NODE_ENV === 'development',

  /**
   * Include children of a component applied with withScan
   */
  includeChildren: true,

  /**
   * Enable/disable geiger sound
   */
  playSound: true,

  /**
   * Log renders to the console
   */
  log: false,

  /**
   * Show toolbar bar
   */
  showToolbar: true,

  /**
   * Render count threshold, only show
   * when a component renders more than this
   */
  renderCountThreshold: 0,

  /**
   * Report data to getReport()
   */
  report: false,

  onCommitStart: () => {},
  onRender: (fiber, render) => {},
  onCommitFinish: () => {},
  onPaintStart: (outline) => {},
  onPaintFinish: (outline) => {},
});
getOptions()
const {
  enabled,
  includeChildren,
  runInProduction,
  playSound,
  log,
  showToolbar,
  longTaskThreshold,
  resetCountTimeout,
} = getOptions();

Why React Scan?

React can be tricky to optimize.

The issue is that component props are compared by reference, not value. This is intentional – this way rendering can be cheap to run.

However, this makes it easy to accidentally cause unnecessary renders, making the the app slow. Even in production apps, with hundreds of engineers, can't fully optimize their apps (see GitHub, Twitter, and Instagram).

This often comes down to props that update in reference, like callbacks or object values. For example, the onClick function and style object are re-created on every render, causing ExpensiveComponent to slow down the app:

<ExpensiveComponent onClick={() => alert('hi')} style={{ color: 'purple' }} />

React Scan helps you identify these issues by automatically detecting and highlighting renders that cause performance issues. Now, instead of guessing, you can see exactly which components you need to fix.

Want to automatically fix these issues? Check out Million Lint!

FAQ

Q: Why this instead of React Devtools?

React Devtools aims to be a general purpose tool for React. However, I deal with React performance issues every day, and React Devtools doesn't fix my problems well. There's a lot of noise (no obvious distinction between unnecessary and necessary renders), and there's no programmatic API. If it sounds like you have the same problems, then React Scan may be a better choice.

Also, some personal complaints about React Devtools' highlight feature:

  • React Devtools "batches" paints, so if a component renders too fast, it will lag behind and only show 1 every second or so
  • When you scroll/resize the boxes don't update position
  • No count of how many renders there are
  • I don't know what the bad/slow renders are without inspecting
  • The menu is hidden away so it's annoying to turn on/off, user experience should be specifically tuned for debugging performance, instead of hidden behind a profiler/component tree
  • No programmatic API
  • It's stuck in a chrome extension, I want to run it anywhere on the web
  • It looks subjectively ugly (lines look fuzzy, feels sluggish)
  • I'm more ambitious with react-scan (see our roadmap)

Q: React Native wen?

Soon :)

Q: Chrome Extension wen?

Soon :)

Resources & Contributing Back

Want to try it out? Check the our demo.

Looking to contribute back? Check the Contributing Guide out.

Want to talk to the community? Hop in our Discord and share your ideas and what you've build with React Scan.

Find a bug? Head over to our issue tracker and we'll do our best to help. We love pull requests, too!

We expect all contributors to abide by the terms of our Code of Conduct.

→ Start contributing on GitHub

Roadmap

  • Scan only for unnecessary renders ("unstable" props)
  • Scan API (withScan, scan)
  • Cleanup config options
  • Chrome extension (h/t @biw)
  • Mode to highlight long tasks
  • Add context updates
  • Expose primitives / internals for advanced use cases
  • More explicit options override API (start log at certain area, stop log, etc.)
  • Don't show label if no reconciliation occurred ("client renders" in DevTools)
  • "global" counter using sessionStorage, aggregate count stats instead of immediate replacement
  • Give a general report of the app's performance
  • checkbox filtering API, leaderboard
  • Offscreen canvas on worker thread
  • heatmap decay (stacked renders will be more intense)
  • Investigate components (UI allowlist)
  • UI for turning on/off options
  • “PageSpeed insights” for React
  • React Native support
  • Name / explain the actual problem, docs
  • Simple FPS counter
  • Drag and select areas of the screen to scan
  • Long task progress bar filter
  • Report should include all renders
  • Runtime version guarding
  • React as peer dependency (lock version to range)
  • Add a funny mascot, like the "Stop I'm Changing" dude

Acknowledgments

React Scan takes inspiration from the following projects:

License

React Scan is MIT-licensed open-source software by Aiden Bai, Million Software, Inc., and contributors: