@theia/core
Version:
Theia is a cloud & desktop IDE framework implemented in TypeScript.
189 lines (168 loc) • 6.07 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2017 TypeFox and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************
import { Event, Emitter } from './event';
import { isFunction, isObject } from './types';
export interface Disposable {
/**
* Dispose this object.
*/
dispose(): void;
}
export namespace Disposable {
export function is(arg: unknown): arg is Disposable {
return isObject<Disposable>(arg) && isFunction(arg.dispose);
}
export function create(func: () => void): Disposable {
return { dispose: func };
}
/** Always provides a reference to a new disposable. */
export declare const NULL: Disposable;
}
/**
* Ensures that every reference to {@link Disposable.NULL} returns a new object,
* as sharing a disposable between multiple {@link DisposableCollection} can have unexpected side effects
*/
Object.defineProperty(Disposable, 'NULL', {
configurable: false,
enumerable: true,
get(): Disposable {
return { dispose: () => { } };
}
});
/**
* Utility for tracking a collection of Disposable objects.
*
* This utility provides a number of benefits over just using an array of
* Disposables:
*
* - the collection is auto-pruned when an element it contains is disposed by
* any code that has a reference to it
* - you can register to be notified when all elements in the collection have
* been disposed [1]
* - you can conveniently dispose all elements by calling dispose()
* on the collection
*
* Unlike an array, however, this utility does not give you direct access to
* its elements.
*
* Being notified when all elements are disposed is simple:
* ```
* const dc = new DisposableCollection(myDisposables);
* dc.onDispose(() => {
* console.log('All elements in the collection have been disposed');
* });
* ```
*
* [1] The collection will notify only once. It will continue to function in so
* far as accepting new Disposables and pruning them when they are disposed, but
* such activity will never result in another notification.
*/
export class DisposableCollection implements Disposable {
protected readonly disposables: Disposable[] = [];
protected readonly onDisposeEmitter = new Emitter<void>();
constructor(...toDispose: Disposable[]) {
toDispose.forEach(d => this.push(d));
}
/**
* This event is fired only once
* on first dispose of not empty collection.
*/
get onDispose(): Event<void> {
return this.onDisposeEmitter.event;
}
protected checkDisposed(): void {
if (this.disposed && !this.disposingElements) {
this.onDisposeEmitter.fire(undefined);
this.onDisposeEmitter.dispose();
}
}
get disposed(): boolean {
return this.disposables.length === 0;
}
private disposingElements = false;
dispose(): void {
if (this.disposed || this.disposingElements) {
return;
}
this.disposingElements = true;
while (!this.disposed) {
try {
this.disposables.pop()!.dispose();
} catch (e) {
console.error(e);
}
}
this.disposingElements = false;
this.checkDisposed();
}
push(disposable: Disposable): Disposable {
const disposables = this.disposables;
disposables.push(disposable);
const originalDispose = disposable.dispose.bind(disposable);
const toRemove = Disposable.create(() => {
const index = disposables.indexOf(disposable);
if (index !== -1) {
disposables.splice(index, 1);
}
this.checkDisposed();
});
disposable.dispose = () => {
toRemove.dispose();
disposable.dispose = originalDispose;
originalDispose();
};
return toRemove;
}
pushAll(disposables: Disposable[]): Disposable[] {
return disposables.map(disposable =>
this.push(disposable)
);
}
}
export type DisposableGroup = { push(disposable: Disposable): void } | { add(disposable: Disposable): void };
export namespace DisposableGroup {
export function canPush(candidate?: DisposableGroup): candidate is { push(disposable: Disposable): void } {
return Boolean(candidate && (candidate as { push(): void }).push);
}
export function canAdd(candidate?: DisposableGroup): candidate is { add(disposable: Disposable): void } {
return Boolean(candidate && (candidate as { add(): void }).add);
}
}
export function disposableTimeout(...args: Parameters<typeof setTimeout>): Disposable {
const handle = setTimeout(...args);
return { dispose: () => clearTimeout(handle) };
}
/**
* Wrapper for a {@link Disposable} that is not available immediately.
*/
export class DisposableWrapper implements Disposable {
private disposed = false;
private disposable: Disposable | undefined = undefined;
set(disposable: Disposable): void {
if (this.disposed) {
disposable.dispose();
} else {
this.disposable = disposable;
}
}
dispose(): void {
this.disposed = true;
if (this.disposable) {
this.disposable.dispose();
this.disposable = undefined;
}
}
}