UNPKG

@rimbu/graph

Version:

Immutable Graph data structures for TypeScript

117 lines (98 loc) 3.05 kB
import type { ArrayNonEmpty } from '@rimbu/common'; import { Reducer, type StreamSource } from '@rimbu/stream'; import { isEmptyStreamSourceInstance } from '@rimbu/stream/custom'; import { GraphBuilder, GraphEmpty, GraphNonEmpty } from '@rimbu/graph/custom'; import type { GraphBase, GraphElement, WithGraphValues, } from '../../common/index.mjs'; export interface GraphTypesContextImpl extends GraphBase.Types { readonly context: GraphContext<this['_N'], string, boolean>; } export class GraphContext< UN, TT extends string, Dir extends boolean, Tp extends GraphTypesContextImpl = GraphTypesContextImpl, > implements GraphBase.Context<UN, Tp> { readonly _fixedType!: any; readonly _empty: WithGraphValues<Tp, UN, any>['normal']; constructor( readonly isDirected: Dir, readonly typeTag: TT, readonly linkMapContext: WithGraphValues<Tp, UN, any>['linkMapContext'], readonly linkConnectionsContext: WithGraphValues< Tp, UN, any >['linkConnectionsContext'] ) { this._empty = Object.freeze(new GraphEmpty(isDirected, this)) as any; } isNonEmptyInstance( source: any ): source is WithGraphValues<Tp, UN, any>['nonEmpty'] { return source instanceof GraphNonEmpty; } readonly empty = <N extends UN>(): any => { return this._empty; }; readonly from: any = <N extends UN>( ...sources: ArrayNonEmpty<StreamSource<GraphElement<N>>> ): any => { let builder = this.builder(); let i = -1; const length = sources.length; while (++i < length) { const source = sources[i]; if (isEmptyStreamSourceInstance(source)) continue; if ( builder.isEmpty && this.isNonEmptyInstance(source) && source.context === this ) { if (i === length - 1) return source; builder = source.toBuilder(); continue; } builder.addGraphElements(source); } return builder.build(); }; // prettier-ignore readonly of = <N,>(...values: ArrayNonEmpty<GraphElement<N>>): any => { return this.from(values).assumeNonEmpty(); }; readonly builder = (): any => { return new GraphBuilder(this.isDirected, this); }; reducer = <N extends UN>(source?: StreamSource<GraphElement<N>>): any => { return Reducer.create( () => undefined === source ? this.builder() : this.from(source).toBuilder(), (builder, entry) => { builder.addGraphElement(entry); return builder; }, (builder) => builder.build() ); }; createBuilder<N extends UN>( source?: WithGraphValues<Tp, N, any>['nonEmpty'] ): WithGraphValues<Tp, N, any>['builder'] { return new GraphBuilder<N, Tp>(this.isDirected, this, source) as any; } createNonEmpty<N extends UN>( linkMap: WithGraphValues<Tp, N, any>['linkMapNonEmpty'], connectionSize: number ): WithGraphValues<Tp, N, any>['nonEmpty'] { return new GraphNonEmpty<N, Tp>( this.isDirected, this, linkMap, connectionSize ) as any; } }