Mimcss Guide: Activation Scheduling
We already mentioned in the previous units that activating style definitions as well as changing style properties are operations that write to the DOM and that without the proper care these operations can have adverse impact on performance such as layout thrashing. Mimcss addresses this problem by introducing scheduler types.
Scheduler Types
When the activate
function or the IStyleRule.setProp
method is called, a DOM writing operation should be performed, which will cause style and optionally layout recalculations. Normally, these recalculations will occur asynchronously after the primary JavaScript thread finishes its event loop iteration. If, however, the JavaScript code tries to read certain information from the DOM after the changes have been made the style and layout recalculations can be performed synchronously. If this happens frequently enough - especially in a loop - this can significantly degrade the page performance.
The solution to the above problem is to not execute the DOM writing operations immediately but to schedule them for after all other code in the event loop iteration has finished executing. There are several techniques for doing this and different component libraries can provide different solutions. Mimcss uses schedulers to implement scheduling techniques. All calls to the functions such as activate
, deactivate
and IStyleRule.setProp
are accumulated along with their parameters and the role of the scheduler is to execute them at the appropriate time.
A scheduler is identified by a numeric ID called scheduler type. Mimcss implements several schedulers out-of-the-box and allows callers to implement their own. The SchedulerType
enumeration provides values for the built-in scheduler types:
export const enum SchedulerType
{
Sync = 1,
AnimationFrame,
Manual,
}
The functions activate
, deactivate
and IStyleRule.setProp
have an optional parameter schedulerType
, which allows specifying the scheduler type to be used for this particular call. If this parameter is left undefined, Mimcss uses the value of the default scheduler type. Initially, the default scheduler type is set to SchedulerType.Sync
, which means that the DOM writing operations will be executed synchronously. Callers can use the setDefaultSchedulerType
function to set a different default scheduler type, for example:
setDefaultActivatrType( SchedulerType.AnimationFrame);
After the above call, all calls to the activate
, deactivate
and IStyleRule.setProp
functions that don’t use the schedulerType
parameter, will be scheduled to be executed on the next animation frame.
Note that the call to the setDefaultSchedulerType
function should occur early in the application lifetime before any activation and style property setting occurs. On the other hand, nothing stops application developers to switch to a different scheduler in the middle of the application, although it is hard to see a real case for it.
Synchronous Scheduler
The Synchronous scheduler doesn’t do any real scheduling; calls to the activate
, deactivate
and IStyleRule.setProp
functions directly make changes to the DOM. This scheduler may be useful for applications that already take care of calling the above functions in a manner that avoids unnecessary style and layout recalculations.
The Synchronous scheduler is the default scheduler unless a different scheduler is set using the setDefaultSchedulerType
function.
Animation Frame Scheduler
The Animation Frame scheduler schedules the DOM writing operations for the next animation frame by using the requestAnimationFrame
API function. No matter how many times the functions activate
, deactivate
and IStyleRule.setProp
are called since the last animation frame, all the corresponding DOM changes will be made in the next animation frame.
Manual Scheduler
While the Animation Frame scheduler can serve well under certain circumstances, it does not guarantee that layout thrashing is avoided. If the application (or some library used by the application) also uses animation frames, it is unpredictable in which order the animation frame callbacks are executed, which can lead to intermingling of DOM reads and writes.
Mimcss implements the Manual scheduler type, which can be used by applications to achieve more predictable results. With the Manual scheduler set as the default, the calls to the activate
, deactivate
and IStyleRule.setProp
functions are accumulated until the application calls the forceDOMUpdate
function. This allows applications to synchronize style changes with other DOM writing activities.
Custom Schedulers
Mimcss allows writing custom schedulers, which can be a reasonable approach for component authoring libraries or for applications that have a particular way of synchronizing DOM writing operations.
A custom scheduler is a JavaScript class that implements the IScheduler
interface. The custom scheduler object should be registered with Mimcss using the registerScheduler
function. This function returns the number identifying the new scheduler type and this number should be passed to the setDefaultSchedulerType
function in order to set the new scheduler as the default one.
Once the custom scheduler is registered and set as the default, all calls to the activate
, deactivate
, IStyleRule.setProp
, forceDOMUpdate
and cancelDOMUpdate
functions will be accumulated by the Mimcss scheduling infrastructure and the scheduler object’s methods will be called:
- the
init
method is invoked immediately upon registering the scheduler via theregisterScheduler
method. This method passes thedoDOMUpdate
callback that should be called by the scheduler when it decides that it is time to write to the DOM. - The
scheduleDOMUpdate
method is invoked when the first call to one ofactivate
,deactivate
orIStyleRule.setProp
functions is made. The custom scheduler is free to use any technique to schedule a callback or an event to be executed at some point in the future. When the callback is called or the event is fired, the custom scheduler should call thedoDOMUpdate
callback (passed in theinit
method), which writes all the accumulated changes to the DOM. - The
cancelDOMUpdate
method is invoked when the last scheduled item is removed from the internal structures maintained by Mimcss. This can happen, for example, if the calls to theactivate
anddeactivate
functions were made for the same style definition class before the callback or event scheduled in thescheduleDOMUpdate
method was executed. The custom scheduler is free to use any technique to cancel the scheduled callback or event.