Flux.ai is a complex, single-page web app that helps electronic engineers design printed circuit boards. As with any professional design tool, it needs to be fast, precise and predictable.
Early versions of Flux.ai were noticeably slow. We set a goal of making all our user interactions take less than 100ms, supported by science. We started a plethora of projects to make that happen, one of which I wrote about in another blog post. To make sure these efforts paid off, we created a measurement tool, log-time-to-next-idle, that is the subject of this blog post.
We've been using it to track the performance of around 50 key user interactions over 18 months, such as:
We're very happy with how it is working. And now we've shared it with the world as an open-source package.
A user interaction is defined here as any input the user provides—mouse click, key press, and so on—coupled with the intended effects of the input—menu opened, element added, and so on. The performance of a user interaction is simply the time between the start of the input to the end of the effects. For example, clicking on a select box will open a menu. The interaction is done when the menu has finished loading.
A subset of the interaction time is defined as “frozen time”. This is the interval following the user input when there are zero screen updates––no animation frames.
Although the concept of a user interaction is easy to define intuitively, the end state can be hard to define formally. In modern reactive UIs, any part of the UI can freely change in response to an update of a store of application state (Redux, Zustand, and so forth). The initial handler of some user input doesn't know all the downstream effects of its execution and so it can't mark the end of the interaction.

In the diagram above, it is hard to know which among the N components will determine the "end" of the interaction. You may not even know what are all the N components that react to some user inptu. To deal with this problem, we took a shortcut. Assuming that...
...we can leverage requestIdleCallback to mark when an interaction is done.
Similarly, we can leverage requestAnimationFrame to indicate when an interaction has yielded control back to the main thread, unfreezing the UI.
In short, log-time-to-next-idle measures user interactions by queuing a requestAnimationFrame and a requestIdleCallback at the start of an interaction, then recording the time when the callbacks fire.
Just put a call to logTimeToNextIdle at the start of any event handler. Here is an example of logging the time it takes to switch fictional tabs in a React app.
log-time-to-next-idle deals with overlapping interactions by cancelling earlier queued callbacks. In other words, the last interaction "wins" and any previous in-progress interaction is ignored. For ease of interpretation, you should try to measure debounced interactions that happen one at a time.
log-time-to-next-idle will store measured intervals in the browser using window.performance.measure (if available). The intervals will then show up in the profiler (if available).
Here is an example that logs to the console when in dev build mode, and logs to somewhere in the cloud when in prod build.
We hope this measurment function log-time-to-next-idle will be useful to you and your app, as well as the general approach of defining interaction end points in terms of CPU usage. Check out the open-source package and please post any feedback there.

A practical guide to calculating PCB trace resistance, covering the core formula, how geometry affects resistance, worked examples, and design tips to minimize voltage drop and heat.

A practical guide to diagnosing and fixing PCB failures, covering common symptoms, a step-by-step debugging workflow, essential tools (multimeter, oscilloscope, logic analyzer, thermal camera), a pre-power-up checklist, and the design mistakes that most often lead to broken boards.

A practical guide to PCB impedance control, covering why it matters for signal integrity, the four physical variables that shape trace impedance, and how to enforce impedance targets from stackup planning through routing and fabrication.

A practical guide to reducing EMI in PCB design through grounding, return path control, shielding, and layout best practices. Covers EMC compliance with CISPR 32 and FCC Part 15.

A step-by-step guide to designing accurate PCB footprints — covering pads, silkscreen, courtyards, IPC-7351 density levels, origin setup, and common mistakes to avoid.

A practical guide to PCB grounding techniques — ground planes, return paths, star grounding, and analog/digital partitioning — with best practices for reducing noise and improving signal stability.

A practical guide to designing multilayer PCB stackups for signal integrity, EMI control, and stable power delivery. Covers layer types, controlled impedance, common mistakes, and how modern tools simplify the process.

A look at how AI is reshaping PCB design by automating routing, placement, and signal integrity checks so engineers can focus on architecture and higher-level decisions.