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.
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.
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
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.
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:
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
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.
Chunking
Chunking means using scheduling APIs to split work and distribute it over time to have less frame drops.
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.
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 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.
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).
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.
Strategies:
Name | Priority | Render Method | Scheduling | Render Deadline |
---|---|---|---|---|
"immediate" | 1 | 🠗 detectChanges | postMessage | 0ms |
"userBlocking" | 2 | 🠗 detectChanges | postMessage | 250ms |
"normal" | 3 | 🠗 detectChanges | postMessage | 5000ms |
"low" | 4 | 🠗 detectChanges | postMessage | 10000ms |
"idle" | 5 | 🠗 detectChanges | postMessage | ❌ |
Immediate
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 Method | Scheduling | Render Deadline |
---|---|---|
🠗 detectChanges | postMessage | 0ms |
Usecase:
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
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 Method | Scheduling | Render Deadline |
---|---|---|
🠗 detectChanges | postMessage | 250ms |
Usecase:
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
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 Method | Scheduling | Render Deadline |
---|---|---|
🠗 detectChanges | postMessage | 5000ms |
Usecase:
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
Work that is typically not visible to the user or initiated by the user.
Render Method | Scheduling | Render Deadline |
---|---|---|
🠗 detectChanges | postMessage | 10000ms |
Usecase:
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
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 Method | Scheduling | Render Deadline |
---|---|---|
🠗 detectChanges | postMessage | ❌ |
Usecase:
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.