Skip to content

mesteche/react_stream

Repository files navigation

react_stream

sample demo app of react and async iterators

usage

Install the dependencies npm install

Run the app npm start

Open your browser at: http://localhost:3000/

Now try to edit src/index.js and comment the line 34:

await new Promise(requestAnimationFrame)

Now try commenting line 39:

debounce(idleIterator),

/!\ This is only holds true because we are manually calling react's render method. If you use react's internal setState (call it 600 times in a loop) or redux's dispatch (same thing), react will not render 600 times, or even 20 time but only once. The lesson is : don't render manually, react is already perfomant.

What happened ?

On its own, react can manage to render 60fps on firefox, but it still renders each and every frames. It means that if you have 600 updates of the state, react will take about 10 sec to get up to date with the last version.

The line 34 ensures that react won't try to render more than once per animation frame, so the rest of the app have more CPU time available.

The line 39 is the real magic. It will only pass the last version of the state when requested. If more updates are coming from the state, it will keep the last and drop the rest. So that means that some updates won't be rendered. Most of them actually, it may render only 15 or 20 frames over 600, but it does so in a few hundred milliseconds, so it still is around 60fps.

But it doesn't matter since these updates are building on top of each other, the last update is always the complete state up to date, and the last update is always rendered. And since we are rendering one time per frame, we are already rendering the maximum number of fps that the browser can display.

Another thing to notice: it will only try to get updates from the state if the app as nothing more important to do (is idle). So it doesn't take precious CPU just to drop some state updates.

If you wonder how it works, look at the code in src/shared/stream/time.js