Skip to main content

Concurrent strategies

Based on the RAIL model, e.g. if your app provides animated user feedback within more than 16ms (less than 60 frames per second), it feels laggy to the user and leads to bad UX. From the UX perspective that means users should not experience blocking periods more than 16 ms.

Concepts

There are 5 core concepts of the concurrent strategies:

  • Frame budget / Frame drop
  • Scheduling
  • Priority
  • Chunking
  • Concurrent Scheduling

Frame budget / Frame drop

The Browser has only one UI thread (main thread), meaning things happen one after another.

Users constantly interact with our site and this means if our main thread is busy, they can't interact with the page. The events like scroll or click will get delayed until the main thread is unblocked from work again and can process those interactions.

Render Strategies - Frame Drop Overview

Such situations cause problems like:

  • blocking UIs
  • animation junk
  • scroll junk or stuttering
  • delayed navigation
  • bad frame rates in general

All those problems boil down to the way the user perceives the interactions with the website. Due to the human eye and how our screens got standardized, the frame rate is defined as good if it is 60 frames per second which is a screen update every 16.6 milliseconds.

In the browser, we can see tasks in the main thread that are too long for a good framerate marked with a red triangle.

Render Strategies-Frame Drop Detail View

In the image, we see ChromeDevtools marks frames that take longer than 50ms as long task. All those tasks exceeded the input response budget.

The reason why it is 50ms and not 16.6ms is based on other things that can happen in relation to the user input. The related theory is known as the RAIL model.

Scheduling

Render Strategies - scheduling abstract diagram

When it comes to scripting work we can do 2 things to avoid that:

  • reduce scripting work and let the user interact earlier
  • chunk up work and use scheduling API's to distribute the work overtime and let the user interact in between.

It is often the case that the work just can't get reduced so we have to schedule.

Render Strategies-Scheduling Detail View

Some of the possible APIs are:

  • queueMicrotask
  • setTimeout
  • postMessage
  • requestAnimationFrame
  • requestIdleCallback

Angular did that internally in different places. One of them is in the elements package:

https://github.com/angular/angular/blob/master/packages/elements/src/component-factory-strategy.ts#L255-L267

Also, the utils file is an interesting place to look at: https://github.com/angular/angular/blob/master/packages/elements/src/utils.ts#L13-L46

A simple way to schedule work is using setTimeout.

function work(): void {
console.log('work done!');
}

const asyncId = setTimeout(work);

By calling setTimeout we can schedule the execution of the work function in the next task.

As a return value we receive the so called "asyncId" a number that serves as reference to the scheduled task.

This is important for cancellation and cleanup logic.

clearTimeout(asyncId);

If we pass the asyncId as parameter to the clearTimeout function we can cancel the scheduling and work will never get executed.

Priority

Render Strategies - priority abstract diagram png

Input handlers (tap, click etc.) often need to schedule a combination of different kinds of work:

  • kicking off some immediate work as microtasks, e.g. fetching from a local cache
  • scheduling data fetches over the network
  • rendering in the current frame, e.g. to respond to user typing, toggle the like button, start an animation when clicking on a comment list etc.
  • rendering over the course of next new frames, as fetches complete and data becomes available to prepare and render results.

To get the best user experience we should prioritize this tasks.

There are couple of scheduling APIs mentioned under scheduling. They all help to prioritize the work and define the moment of execution differently.

Render Strategies - scheduling techniques

Chunking

Chunking means using scheduling APIs to split work and distribute it over time to have less frame drops.

Render Strategies-chunking-example

All scheduling APIs can help to prioritize the work and define the moment of execution differently.

When using the requestAnimationFrame API we should know that it is not a queued system. All scheduled tasks will end up in the same task of the main thread.

Render Strategies-chunking-animation-frame

The image shows that all AnimationFrame events end up in the same task.

This scenario gets to a problem depending on:

  • the number of Angular elements
  • the amount of work done in the elements

Concurrent scheduling

concurrent scheduling - abstract diagram

Concurrent scheduling is a marketing term and simply means that there is a mechanism in place that knows how much time is spent in the current task. This number is called frame budget and measured in milliseconds. As a result of this technique we're getting prioritized user-centric scheduling behaviour's.

This enables:

  • scheduling
  • cancellation
  • fine grained prioritization
  • works distribution based on the frame budget
  • render deadlines

One of the first things to understand is the term "frame budget". It means we have a maximum time (which is globally defined) a task can take before yielding to the main thread. e.g. 1000ms/60frames=16.6666ms animations or 50ms long task.

Scheduling with notion of frame budget enables us to split work into individual browser tasks as soon as we exceed the budget. We then yield to the main thread and are interactive again until the next batch of tasks will get processed.

rx-angular-cdk-render-strategies__frame-budget

The special thing about the set of concurrent strategies is they have a render deadline. It means if the scheduled tasks in the global queue of work is not exhausted after a certain time window, we stop the chunking process. Instead all remaining work will get executed as fast as possible. This means in one synchronous block (that potentially can causes a frame drop).

Render Strategies - concurrent anatomy png Every strategy has a different render deadline. Strategies are designed from the perspective of how important the work is for the user. see: RAIL model

What concurrent scheduling does under the hood is chunking up work in cycles of scheduling, prioritization and execution based on different settings.

Render Strategies - task flow

Strategies:

NamePriorityRender MethodSchedulingRender Deadline
"immediate"1🠗 detectChangespostMessage0ms
"userBlocking"2🠗 detectChangespostMessage250ms
"normal"3🠗 detectChangespostMessage5000ms
"low"4🠗 detectChangespostMessage10000ms
"idle"5🠗 detectChangespostMessage

Render Strategies - example usage

Immediate

render-strategies-concurrent-immediate-tree

Urgent work that must happen immediately is initiated and visible by the user. This occurs right after the current task and has the highest priority.

Render MethodSchedulingRender Deadline
🠗 detectChangespostMessage0ms

render-strategies-concurrent-immediate-diagram

Usecase:

Render Strategies - immediate example

A good example here would be a tool-tip.

Tooltips should be displayed immediately on mouse over. Any delay will be very noticeable.

@Component({
selector: 'item-image',
template: ` <img [src]="src()" (mouseenter)="showTooltip()" (mouseleave)="hideTooltip()" /> `,
})
export class ItemsListComponent {
private strategyProvider = inject(RxStrategyProvider);

readonly src = input.required<string>();

showTooltip() {
this.strategyProvider
.schedule(
() => {
// create tooltip
},
{ strategy: 'immediate' },
)
.subscribe();
}

hideTooltip() {
this.strategyProvider
.schedule(
() => {
// destroy tooltip
},
{ strategy: 'immediate' },
)
.subscribe();
}
}

⚠ Notice: Be aware to avoid scheduling large or non-urgent work with immediate priority as it blocks rendering

User blocking

render-strategies-concurrent-userBlocking-tree

Critical work that must be done in the current frame, is initiated and visible by the user. DOM manipulations that should be rendered quickly. Tasks with this priority can delay current frame rendering, so this is the place for lightweight work (otherwise use "normal" priority).

Render MethodSchedulingRender Deadline
🠗 detectChangespostMessage250ms

render-strategies-concurrent-userBlocking-diagram

Usecase:

Render Strategies - userBlocking example

A good example here would be a dropdown menu.

Dropdowns should be displayed right away on user interaction.

@Component({
selector: 'item-dropdown',
template: `
<div id="collapse" (mouseenter)="showDropdown()" (mouseleave)="hideDropdown()">
{{ text() }}
</div>
`,
})
export class DropdownComponent {
private strategyProvider = inject(RxStrategyProvider);

readonly text = input.required<string>();

showDropdown() {
this.strategyProvider
.schedule(
() => {
// create dropdown
},
{ strategy: 'userBlocking' },
)
.subscribe();
}

hideDropdown() {
this.strategyProvider
.schedule(
() => {
// destroy dropdown
},
{ strategy: 'userBlocking' },
)
.subscribe();
}
}

⚠ Notice: Be aware to avoid scheduling large or non-urgent work with userBlocking priority as it blocks rendering after 250ms

Normal

render-strategies-concurrent-normal-tree

Heavy work visible to the user. For example, since it has a higher timeout, it is more suitable for the rendering of data lists.

Render MethodSchedulingRender Deadline
🠗 detectChangespostMessage5000ms

render-strategies-concurrent-normal-diagram

Usecase:

Render Strategies - normal example

For normal strategy a perfect example will be rendering of the items list.

It is often the case that rendering of big lists blocks user interactions. In combination with rxFor directive such operations become truly unblocking.

@Component({
selector: 'items-list',
template: `
<div id="items-list">
<div *rxFor="let item of state.items$; strategy: 'normal'>
<item-image [src]="item.image"></item-image>
<item-dropdown [text]="item.text"></item-dropdown>
</div>
</div>
`,
})
export class ItemsListComponent {
protected state = inject(StateService);
}

Low

render-strategies-concurrent-low-tree

Work that is typically not visible to the user or initiated by the user.

Render MethodSchedulingRender Deadline
🠗 detectChangespostMessage10000ms

render-strategies-concurrent-low-diagram

Usecase:

Render Strategies - low example

Good use case for this strategy will be lazy loading of the components. For example popup.

@Component({
selector: 'items-list',
template: `
<div id="items-list">
<div *rxFor="let item of state.items$; strategy: 'normal'>
<item-image [src]="item.image"></item-image>
<item-dropdown [text]="item.text"></item-dropdown>
</div>
</div>

<button id="addItem" (click)="openCreateItemPopup()">Create new item</button>
`,
})
export class ItemsListComponent {
protected state = inject(StateService);
private strategyProvider = inject(RxStrategyProvider);

openCreateItemPopup() {
this.strategyProvider
.schedule(
() => {
// logic to lazy load popup component
},
{ strategy: 'low' },
)
.subscribe();
}
}

⚠ Notice: This priority fits well for things that should happen but has lower priority. For any non-urgent background process idle is the best fit.

Idle

render-strategies-concurrent-idle-tree

Urgent work that should happen in the background and is not initiated but visible by the user. This occurs right after current task and has the lowest priority.

Render MethodSchedulingRender Deadline
🠗 detectChangespostMessage

render-strategies-concurrent-idle-diagram

Usecase:

Render Strategies - idle example

This strategy is especially useful for logic meant to run in the background. Good example of such interaction is background sync.

@Component({
selector: 'items-list',
template: `
<div id="items-list">
<div *rxFor="let item of state.items$; strategy: 'normal'>
{{item.name}}
</div>
</div>

<button id="addItem" (click)="openCreateItemPopup()">Create new item</button>

<div id="background-indicator">Background sync</div>
`,
})
export class ItemsListComponent {
private strategyProvider = inject(RxStrategyProvider);
private webSocket = inject(WebSocketService);
protected state = inject(StateService);

constructor() {
this.state.items$.pipe(this.strategyProvider.scheduleWith((items) => this.webSocket.syncItems(items), { strategy: 'idle' })).subscribe();
}

openCreateItemPopup() {
this.strategyProvider
.schedule(
() => {
// logic to lazy load popup component
},
{ strategy: 'low' },
)
.subscribe();
}
}

⚠ Notice: This priority fits well for low priority background processes that are not affecting user interactions.