mana-common
Version:
Common utils for mana
118 lines (99 loc) • 3.12 kB
text/typescript
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation and others. All rights reserved.
* Licensed under the MIT License. See https://github.com/Microsoft/vscode/blob/master/LICENSE.txt for license information.
*--------------------------------------------------------------------------------------------*/
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Event, Emitter } from './event';
export type CancellationToken = {
readonly isCancellationRequested: boolean;
/*
* An event emitted when cancellation is requested
* @event
*/
readonly onCancellationRequested: Event<void>;
};
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const shortcutEvent: Event<void> = Object.freeze(
Object.assign(
function (callback: any, context?: any): any {
const handle = setTimeout(callback.bind(context), 0);
return {
dispose(): void {
clearTimeout(handle);
},
};
},
{ maxListeners: 0 },
),
);
export namespace CancellationToken {
export const None: CancellationToken = Object.freeze({
isCancellationRequested: false,
onCancellationRequested: Event.None,
});
export const Cancelled: CancellationToken = Object.freeze({
isCancellationRequested: true,
onCancellationRequested: shortcutEvent,
});
}
class MutableToken implements CancellationToken {
private _isCancelled: boolean = false;
private _emitter: Emitter<void> | undefined;
public cancel(): void {
if (!this._isCancelled) {
this._isCancelled = true;
if (this._emitter) {
this._emitter.fire(undefined);
this._emitter = undefined;
}
}
}
get isCancellationRequested(): boolean {
return this._isCancelled;
}
get onCancellationRequested(): Event<void> {
if (this._isCancelled) {
return shortcutEvent;
}
if (!this._emitter) {
this._emitter = new Emitter<void>();
}
return this._emitter.event;
}
}
export class CancellationTokenSource {
private _token?: CancellationToken;
get token(): CancellationToken {
if (!this._token) {
// be lazy and create the token only when
// actually needed
this._token = new MutableToken();
}
return this._token;
}
cancel(): void {
if (!this._token) {
// save an object by returning the default
// cancelled token when cancellation happens
// before someone asks for the token
this._token = CancellationToken.Cancelled;
} else if (this._token !== CancellationToken.Cancelled) {
(<MutableToken>this._token).cancel();
}
}
dispose(): void {
this.cancel();
}
}
const cancelledMessage = 'Cancelled';
export function cancelled(): Error {
return new Error(cancelledMessage);
}
export function isCancelled(err: Error | undefined): boolean {
return !!err && err.message === cancelledMessage;
}
export function checkCancelled(token?: CancellationToken): void {
if (!!token && token.isCancellationRequested) {
throw cancelled();
}
}