/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { Event, Emitter } from 'vs/base/common/event'; import { IDisposable } from 'vs/base/common/lifecycle'; export interface ISplice { readonly start: number; readonly deleteCount: number; readonly toInsert: T[]; } export interface ISpliceable { splice(start: number, deleteCount: number, toInsert: T[]): void; } export interface ISequence { readonly elements: T[]; readonly onDidSplice: Event>; } export class Sequence implements ISequence, ISpliceable { readonly elements: T[] = []; private readonly _onDidSplice = new Emitter>(); readonly onDidSplice: Event> = this._onDidSplice.event; splice(start: number, deleteCount: number, toInsert: T[] = []): void { this.elements.splice(start, deleteCount, ...toInsert); this._onDidSplice.fire({ start, deleteCount, toInsert }); } } export class SimpleSequence implements ISequence { private _elements: T[]; get elements(): T[] { return this._elements; } readonly onDidSplice: Event>; private disposable: IDisposable; constructor(elements: T[], onDidAdd: Event, onDidRemove: Event) { this._elements = [...elements]; this.onDidSplice = Event.any( Event.map(onDidAdd, e => ({ start: this.elements.length, deleteCount: 0, toInsert: [e] })), Event.map(Event.filter(Event.map(onDidRemove, e => this.elements.indexOf(e)), i => i > -1), i => ({ start: i, deleteCount: 1, toInsert: [] })) ); this.disposable = this.onDidSplice(({ start, deleteCount, toInsert }) => this._elements.splice(start, deleteCount, ...toInsert)); } dispose(): void { this.disposable.dispose(); } }