chorus/src/bones/collection/Collection.ts

1379 lines
40 KiB
TypeScript

import {AsyncPipe, Pipeline} from '../Pipe'
import type {Unsubscribe, Subscription} from '../Reactive'
export type CollectionItem<T> = T
export type MaybeCollectionItem<T> = CollectionItem<T> | undefined
export type KeyFunction<T, T2> = (item: CollectionItem<T>, index: number) => CollectionItem<T2>
export type KeyReducerFunction<T, T2> = (current: any, item: CollectionItem<T>, index: number) => T2
export type CollectionFunction<T, T2> = (items: Collection<T>) => T2
export type KeyOperator<T, T2> = keyof T | KeyFunction<T, T2>
export type AssociatedCollectionItem<T2, T> = { key: T2, item: CollectionItem<T> }
export type CollectionComparable<T> = CollectionItem<T>[] | Collection<T>
export type DeterminesEquality<T> = (item: CollectionItem<T>, other: any) => boolean
export type CollectionIndex = number
export type MaybeCollectionIndex = CollectionIndex | undefined
export type ComparisonFunction<T> = (item: CollectionItem<T>, otherItem: CollectionItem<T>) => number
export type Collectable<T> = CollectionItem<T>[] | Collection<T>
import {type WhereOperator, applyWhere, whereMatch } from './where'
import type {Awaitable, Awaited, Either, Maybe, MethodsOf, MethodType} from '../types'
import {isLeft, right, unright} from '../types'
import {AsyncCollection} from './AsyncCollection'
import {ArrayIterable} from './ArrayIterable'
export const collect = <T>(items: CollectionItem<T>[] = []): Collection<T> => Collection.collect(items)
const toString = (item: unknown): string => String(item)
/**
* A helper class for working with arrays of items in a more robust fashion.
* Provides helpers for accessing sub-keys, filtering, piping, and aggregate functions.
*/
export class Collection<T> {
private storedItems: CollectionItem<T>[] = []
private pushSubscribers: Subscription<T>[] = []
/**
* Create a new collection from an array of items.
* @param items
*/
public static collect<T2>(items: CollectionItem<T2>[]): Collection<T2> {
return new Collection(items)
}
/**
* Create a new collection from an item or array of items.
* Filters out undefined items.
* @param itemOrItems
*/
public static normalize<T2>(itemOrItems: Collection<T2> | (CollectionItem<T2>)[] | CollectionItem<T2>): Collection<T2> {
if ( itemOrItems instanceof Collection ) {
return itemOrItems
}
if ( !Array.isArray(itemOrItems) ) {
itemOrItems = [itemOrItems]
}
return new Collection<T2>(itemOrItems)
}
/**
* Create a collection of "undefined" elements of a given size.
* @param size
*/
public static size(size: number): Collection<undefined> {
const arr = Array(size).fill(undefined)
return new Collection<undefined>(arr)
}
/**
* Fill a new collection of the given size with the given item.
* @param size
* @param item
*/
public static fill<T2>(size: number, item: T2): Collection<T2> {
const arr = Array(size).fill(item)
return new Collection<T2>(arr)
}
constructor(
/**
* The items to base the collection on.
*/
items?: CollectionItem<T>[],
) {
if ( items ) {
this.storedItems = items
}
}
private allOperator<T2>(key: KeyOperator<T, T2>): CollectionItem<T2>[] {
let items: CollectionItem<T2>[] = []
if ( typeof key === 'function' ) {
items = this.storedItems.map(key)
} else {
items = this.storedItems.map((item: CollectionItem<T>) => (<any>item)[key])
}
return items
}
private allAsNumbers<T2>(key: KeyOperator<T, T2>): number[] {
return this.allOperator(key).map(value => Number(value))
}
private allAssociated<T2>(key: KeyOperator<T, T2>): AssociatedCollectionItem<T2, T>[] {
const associatedItems: AssociatedCollectionItem<T2, T>[] = []
const items = [...this.storedItems]
if ( typeof key === 'function' ) {
items.map((item, index) => {
const keyItem = key(item, index)
associatedItems.push({
key: keyItem,
item,
})
})
} else {
items.map(item => {
associatedItems.push({
key: (<any>item)[key],
item,
})
})
}
return associatedItems
}
/**
* Cast the collection to an array.
*/
all(): CollectionItem<T>[] {
return [...this.storedItems]
}
/**
* Get the average value of the items or one of their keys.
* @param key
*/
average<T2>(key?: KeyOperator<T, T2>): number {
let items
if ( key ) {
items = this.allAsNumbers(key)
} else {
items = this.storedItems.map(x => Number(x))
}
if ( items.length === 0 ) {
return 0
}
const sum = items.reduce((prev, curr) => prev + curr)
return sum / items.length
}
/**
* Get the median value of the items or one of their keys.
* @param key
*/
median<T2>(key?: KeyOperator<T, T2>): number {
let items
if ( key ) {
items = this.allAsNumbers(key).sort((a, b) => a - b)
} else {
items = this.storedItems.map(x => Number(x)).sort((a, b) => a - b)
}
const middle = Math.floor((items.length - 1) / 2)
if ( items.length % 2 ) {
return items[middle]
} else {
return (items[middle] + items[middle + 1]) / 2
}
}
/**
* Get the mode of the items or one of their keys.
* @param key
*/
mode<T2>(key?: KeyOperator<T, T2>): number {
let items
if ( key ) {
items = this.allAsNumbers(key).sort((a, b) => a - b)
} else {
items = this.storedItems.map(x => Number(x)).sort((a, b) => a - b)
}
const counts: any = {}
for ( const item of items ) {
counts[item] = (counts[item] ?? -1) + 1
}
return Number(Object.keys(counts).reduce((a, b) => counts[a] > counts[b] ? a : b)[0])
}
/**
* Collapse a (potentially nested) collection of items down to a single dimension.
*/
collapse(): Collection<any> {
const newItems: CollectionItem<T>[] = []
const items = [...this.storedItems]
const getLayer = (current: CollectionItem<T>|CollectionItem<T>[]) => {
if ( typeof (<any>current)[Symbol.iterator] === 'function' ) {
for (const item of (current as any)) {
if (Array.isArray(item)) {
getLayer(item)
} else {
newItems.push(item)
}
}
}
}
getLayer(items)
return new Collection(newItems)
}
/**
* Returns true if the given key matches the given condition for any item in the collection.
*
* @example
* ```typescript
* const userExists = users.contains('username', '=', 'jdoe')
* ```
*
* @param key
* @param operator
* @param operand
*/
contains<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): boolean {
const associate = this.allAssociated(key)
const matches = applyWhere(associate, operator, operand)
return matches.length > 0
}
// TODO crossJoin
/**
* Create a copy of this collection.
* Does NOT deep copy the underlying items.
*/
clone(): Collection<T> {
return new Collection<T>(this.storedItems)
}
/**
* Return a collection of items that ARE in this collection, but NOT in the `items` collection.
* @param items
*/
diff<T2>(items: CollectionComparable<T|T2>): Collection<T> {
const exclude = items instanceof Collection ? items.all() : items
const matches = []
for ( const item of [...this.storedItems] ) {
if ( !exclude.includes(item) ) {
matches.push(item)
}
}
return new Collection(matches)
}
/**
* Like diff, but mutates the current collection.
* @param items
*/
diffInPlace<T2>(items: CollectionComparable<T|T2>): this {
const exclude = items instanceof Collection ? items.all() : items
this.storedItems = this.storedItems.filter(item => !exclude.includes(item))
return this
}
/**
* Return a collection of items that ARE in this collection, but NOT In the `items` collection
* using a helper function to determine whether two items are equal.
*
* @example
* ```typescript
* potentialUsers.diffUsing(storedUsers, (u1, u2) => u1.username.toLowerCase() === u2.username.toLowerCase())
* ```
*
* @param items
* @param compare
*/
diffUsing<T2>(items: CollectionComparable<T|T2>, compare: DeterminesEquality<T>): Collection<T> {
const exclude = items instanceof Collection ? items.all() : items
const matches = []
for ( const item of [...this.storedItems] ) {
if ( !exclude.some(exc => compare(item, exc)) ) {
matches.push(item)
}
}
return new Collection(matches)
}
/**
* Returns true if the given function returns truthy for any item in the collection.
* Stops executing if a single truth case is found.
* @param func
*/
some(func: (item: T) => Maybe<boolean>): boolean {
return this.storedItems.some(func)
}
/**
* Execute the function for every item in the collection.
* @param func
*/
each<T2>(func: KeyFunction<T, T2>): void {
this.storedItems.map(func)
}
/**
* Create a new collection by mapping the items in this collection using the given function.
* @param func
*/
map<T2>(func: KeyFunction<T, T2>): Collection<T2> {
const newItems: CollectionItem<T2>[] = []
this.each(((item, index) => {
newItems.push(func(item, index))
}))
return new Collection<T2>(newItems)
}
/**
* Create a new collection by mapping the items in this collection using the given function
* where the function returns an Either. The collection is all Right instances. If a Left
* is encountered, that value is returned.
* @param func
*/
mapRight<TLeft, TRight>(func: KeyFunction<T, Either<TLeft, TRight>>): Either<TLeft, Collection<TRight>> {
const newItems: CollectionItem<TRight>[] = []
for ( let i = 0; i < this.length; i += 1 ) {
const result = func(this.storedItems[i], i)
if ( isLeft(result) ) {
return result
}
newItems.push(unright(result))
}
return right(new Collection<TRight>(newItems))
}
/**
* Create a new collection by mapping the items in this collection using the given function
* where the function returns an Either. The collection is all Right instances. If a Left
* is encountered, that value is returned.
* @param func
*/
async asyncMapRight<TLeft, TRight>(func: KeyFunction<T, Awaitable<Either<TLeft, TRight>>>): Promise<Either<TLeft, Collection<TRight>>> {
const newItems: CollectionItem<TRight>[] = []
for ( let i = 0; i < this.length; i += 1 ) {
const result = await func(this.storedItems[i], i)
if ( isLeft(result) ) {
return result
}
newItems.push(unright(result))
}
return right(new Collection<TRight>(newItems))
}
/**
* Get the collection as an AsyncCollection.
*/
toAsync(): AsyncCollection<T> {
const iter = new ArrayIterable([...this.storedItems])
return new AsyncCollection<T>(iter)
}
/**
* Map a method on the underlying type, passing it any required parameters.
* This is delightfully type-safe.
* @param method
* @param params
*/
mapCall<T2 extends MethodsOf<T>>(method: T2, ...params: Parameters<MethodType<T, T2>>): Collection<ReturnType<MethodType<T, T2>>> {
// This is dumb, but I'm not sure how else to resolve it. The types check out, but TypeScript loses track of the fact that
// typeof x[method] === MethodType<T, T2>, so it assumes we're indexing an object incorrectly.
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
return this.map((x: T) => x[method](...params))
}
/**
* Shortcut for .mapCall(...).awaitAll().
* @param method
* @param params
*/
async awaitMapCall<T2 extends MethodsOf<T>>(method: T2, ...params: Parameters<MethodType<T, T2>>): Promise<Collection<Awaited<ReturnType<MethodType<T, T2>>>>> {
return this.mapCall(method, ...params).awaitAll()
}
/**
* Await all values in the collection.
*/
async awaitAll(): Promise<Collection<Awaited<T>>> {
return this.promiseMap(async x => x as Awaited<T>)
}
/**
* Map each element in the collection to a string.
*/
strings(): Collection<string> {
return this.map<string>(x => String(x))
}
/**
* Create a new collection by mapping the items in this collection using the given function,
* excluding any for which the function returns undefined.
* @param func
*/
partialMap<T2>(func: KeyFunction<T, T2 | undefined>): Collection<NonNullable<T2>> {
const newItems: CollectionItem<NonNullable<T2>>[] = []
this.each(((item, index) => {
const result = func(item, index)
if ( typeof result !== 'undefined' ) {
newItems.push(result as NonNullable<T2>)
}
}))
return new Collection<NonNullable<T2>>(newItems)
}
/**
* Convert this collection to an object keyed by the given field.
*
* @example
* ```typescript
* const users = collect([{uid: 1, name: 'John'}, {uid: 2, name: 'Jane'}])
* users.keyBy('name') // => {John: {uid: 1, name: 'John'}, Jane: {uid: 2, name: 'Jane'}}
* ```
*
* @param key
*/
keyBy(key: KeyOperator<T, string>): {[key: string]: T} {
const obj: {[key: string]: T} = {}
this.allAssociated(key).forEach(assoc => {
obj[assoc.key] = assoc.item
})
return obj
}
/**
* Convert this collection to an object keyed by the given field, whose values are
* the output of the `value` operator.
*
* @example
* ```typescript
* const users = collect([{uid: 1, name: 'John'}, {uid: 2, name: 'Jane'}])
* users.keyMap('name', 'uid') // => {John: 1, Jane: 2}
* ```
*
* @param key
* @param value
*/
keyMap<T2>(key: KeyOperator<T, string>, value: KeyOperator<T, T2>): {[key: string]: T2} {
const obj: {[key: string]: T2} = {}
let i = -1
this.allAssociated(key).forEach(assoc => {
i += 1
if ( typeof value === 'function' ) {
obj[assoc.key] = value(assoc.item, i)
} else {
obj[assoc.key] = (assoc.item[value] as any) as T2
}
})
return obj
}
/**
* Returns true if the given function returns true for every item in the collection.
* @param func
*/
every<T2>(func: KeyFunction<T, T2>): boolean {
return this.storedItems.every(func)
}
everyWhere<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): boolean {
const items = this.allAssociated(key)
return items.every(item => whereMatch(item, operator, operand))
}
/**
* Return a new collection filtered by the given function.
* @param func
*/
filter<T2>(func?: KeyFunction<T, T2>): Collection<T> {
return new Collection(this.storedItems.filter(func ?? Boolean))
}
/**
* Like filter, but inverted. That is, removes items that DO match the criterion.
* @param func
*/
filterOut<T2>(func?: KeyFunction<T, T2>): Collection<T> {
return this.filter((...args) => !(func ?? Boolean)(...args))
}
whereDefined(): Collection<NonNullable<T>> {
return this.filter() as unknown as Collection<NonNullable<T>>
}
/**
* Returns the index of the record for which the given function returns true, if such an index exists.
* @param func
*/
find<T2>(func: KeyFunction<T, T2>): number | undefined {
let foundIndex: number | undefined = undefined
this.storedItems.some((item, index) => {
if ( func(item, index) ) {
foundIndex = index
return true
}
})
return foundIndex
}
/**
* When `bool` is truthy, execute the callback, passing in the collection.
* Can be used for functional-style chained calls.
* @param bool
* @param then
*/
when<T2>(bool: boolean, then: CollectionFunction<T, T2>): Collection<T> {
if ( bool ) {
then(this)
}
return this
}
/**
* When `bool` is falsy, execute the callback, passing in the collection.
* Can be used for functional-style chained calls.
* @param bool
* @param then
*/
unless<T2>(bool: boolean, then: CollectionFunction<T, T2>): Collection<T> {
if ( !bool ) {
then(this)
}
return this
}
/**
* Filter the collection by the given where-condition.
*
* @example
* ```typescript
* const users = collect([
* {uid: 1, name: 'John'},
* {uid: 2, name: 'Jane'},
* {uid: 3, name: 'James'},
* ])
*
* users.where('uid', '<', 3) // => Collection[{uid: 1, name: 'John'}, {uid: 2, name: 'Jane'}]
* ```
*
* @param key
* @param operator
* @param operand
*/
where<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): Collection<T> {
const items = this.allAssociated(key)
return new Collection(applyWhere(items, operator, operand))
}
/**
* Filter the collection by the inverse of the given where-condition.
* @param key
* @param operator
* @param operand
*/
whereNot<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): Collection<T> {
return this.diff(this.where(key, operator, operand))
}
/**
* Filter the collection for all records where the given key is in a set of items.
* @param key
* @param items
*/
whereIn<T2>(key: KeyOperator<T, T2>, items: CollectionComparable<T2>): Collection<T> {
const allowed = items instanceof Collection ? items.all() : items
const matches = []
for ( const { key: search, item } of this.allAssociated(key) ) {
if ( allowed.includes(search) ) {
matches.push(item)
}
}
return new Collection(matches)
}
/**
* Filter the collection for all records where the given key is NOT in a set of items.
* @param key
* @param items
*/
whereNotIn<T2>(key: KeyOperator<T, T2>, items: CollectionComparable<T2>): Collection<T> {
return this.diff(this.whereIn(key, items))
}
/**
* Return the first item in the collection, if it exists.
*/
first(): MaybeCollectionItem<T> {
if ( this.length > 0 ) {
return this.storedItems[0]
}
}
/**
* Return the first item in the collection that matches the given where-condition.
* @param key
* @param operator
* @param operand
*/
firstWhere<T2>(key: KeyOperator<T, T2>, operator: WhereOperator = '=', operand: any = true): MaybeCollectionItem<T> {
const items = this.where(key, operator, operand).all()
if ( items.length > 0 ) {
return items[0]
}
}
/**
* Return the first item in the collection that does NOT match the given where-condition.
* @param key
* @param operator
* @param operand
*/
firstWhereNot<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): MaybeCollectionItem<T> {
const items = this.whereNot(key, operator, operand).all()
if ( items.length > 0 ) {
return items[0]
}
}
/**
* Get the number of items in the collection.
*/
get length(): number {
return this.storedItems.length
}
/**
* Get the number of items in the collection.
*/
count(): number {
return this.storedItems.length
}
// TODO flatten - depth
/**
* Get the item at the given index in the collection. If none exists,
* return the fallback.
* @param index
* @param fallback
*/
get(index: number, fallback?: T): MaybeCollectionItem<T> {
if ( this.length > index ) {
return this.storedItems[index]
} else {
return fallback
}
}
/**
* Return the item at the given index in the collection, if it exists.
* @param index
*/
at(index: number): MaybeCollectionItem<T> {
return this.get(index)
}
/**
* Return an object mapping key values to arrays of records with that key.
*
* @example
* ```typescript
* const users = collect([
* {uid: 1, name: 'John', type: 'admin'},
* {uid: 2, name: 'Jane', type: 'user'},
* {uid: 3, name: 'James', type: 'user'},
* ])
*
* users.groupBy('type') // => {admin: [{uid: 1, ...}], user: [{uid: 2, ...}, {uid: 3, ...}]}
* ```
*
* @param key
*/
groupBy<T2>(key: KeyOperator<T, T2>): any {
const items = this.allAssociated(key)
const groups: any = {}
for ( const item of items ) {
const itemKey = String(item.key)
if ( !groups[itemKey] ) {
groups[itemKey] = []
}
groups[itemKey].push(item.item)
}
return groups
}
/**
* Return an object mapping the given key to the record with that key.
* See `keyBy()`.
* @param key
*/
associate<T2>(key: KeyOperator<T, T2>): any {
const items = this.allAssociated(key)
const values: any = {}
for ( const item of items ) {
values[String(item.key)] = item.item
}
return values
}
/**
* Join the items in the collection to a string delimited by the given delimiter.
* @param delimiter
*/
join(delimiter: string): string {
return this.storedItems.join(delimiter)
}
/**
* Join the items in the collection to a string delimited by the given delimiter.
* @param delimiter
*/
implode(delimiter: string): string {
return this.join(delimiter)
}
/**
* Return a collection of the items that exist in both this collection and this collection,
* (optionally) using the given key to compare.
* @param items
* @param key
*/
intersect(items: CollectionComparable<T>, key?: KeyOperator<T, T>): Collection<T> {
const compare = items instanceof Collection ? items.all() : items
const intersect = []
let allItems
if ( key ) {
allItems = this.allAssociated(key)
} else {
allItems = this.storedItems.map(item => {
return {
key: item,
item,
}
})
}
for ( const item of allItems ) {
if ( compare.includes(item.key) ) {
intersect.push(item.item)
}
}
return new Collection(intersect)
}
/**
* True if the collection has no items.
*/
isEmpty(): boolean {
return this.length < 1
}
/**
* True if the collection has at least one item.
*/
isNotEmpty(): boolean {
return this.length > 0
}
/**
* Return the last item in the collection.
*/
last(): MaybeCollectionItem<T> {
if ( this.length > 0 ) {
return this.storedItems.reverse()[0]
}
}
/**
* Return the last item in the collection that matches the given where-condition.
* @param key
* @param operator
* @param operand
*/
lastWhere<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): MaybeCollectionItem<T> {
const items = this.where(key, operator, operand).all()
if ( items.length > 0 ) {
return items.reverse()[0]
}
}
/**
* Return the last item in the collection that does NOT match the given where-condition.
* @param key
* @param operator
* @param operand
*/
lastWhereNot<T2>(key: KeyOperator<T, T2>, operator: WhereOperator, operand?: unknown): MaybeCollectionItem<T> {
const items = this.whereNot(key, operator, operand).all()
if ( items.length > 0 ) {
return items.reverse()[0]
}
}
/**
* Map the collection to a collection of the values of the key.
*
* @example
* ```typescript
* const users = collect([
* {uid: 1, name: 'John', type: 'admin'},
* {uid: 2, name: 'Jane', type: 'user'},
* {uid: 3, name: 'James', type: 'user'},
* ])
*
* users.pluck('name') // => Collection['John', 'Jane', 'James']
* ```
*
* @param key
*/
pluck<T2 extends keyof T>(key: T2): Collection<T[T2]> {
return new Collection<T[T2]>(this.allOperator(key))
}
/**
* Return the max value of the given key.
* @param key
*/
max<T2 = T>(key: KeyOperator<T, T2>): number {
const values = this.allAsNumbers(key)
return Math.max(...values)
}
/**
* Return the item with the max value of the given key.
* @param key
*/
whereMax<T2>(key: KeyOperator<T, T2>): Collection<T> {
return this.where(key, '=', this.max(key))
}
/**
* Return the min value of the given key.
* @param key
*/
min<T2>(key: KeyOperator<T, T2>): number {
const values = this.allAsNumbers(key)
return Math.min(...values)
}
/**
* Return the item with the min value of the given key.
* @param key
*/
whereMin<T2>(key: KeyOperator<T, T2>): Collection<T> {
return this.where(key, '=', this.min(key))
}
/**
* Get a new collection containing both the items in this collection, and the `items` collection.
* @param items
*/
merge<T2>(items: CollectionComparable<T2>): Collection<T|T2> {
const merge = items instanceof Collection ? items.all() : items
return new Collection([...this.storedItems, ...merge])
}
/**
* Return every nth item in the collection.
*
* @example
* ```
* const items = collect(['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i'])
*
* items.nth(3) // => Collection['a', 'd', 'g']
* ```
*
* @param n
*/
nth(n: number): Collection<T> {
const matches: CollectionItem<T>[] = []
let current = 1
this.storedItems.forEach(item => {
if ( current === 1 ) {
matches.push(item)
}
current += 1
if ( current > n ) {
current = 1
}
})
return new Collection(matches)
}
/**
* Return the items that should exist on the given page, assuming there are `perPage` many items on a single page.
* @param page
* @param perPage
*/
forPage(page: number, perPage: number): Collection<T> {
const start = page * perPage - perPage
const end = page * perPage
return new Collection(this.storedItems.slice(start, end))
}
/**
* Return a new Pipe of this collection.
*/
pipeTo<TOut>(pipeline: Pipeline<this, TOut>): TOut {
return pipeline.apply(this)
}
/** Build and apply a pipeline. */
pipe<TOut>(builder: (pipeline: Pipeline<this, this>) => Pipeline<this, TOut>): TOut {
return builder(Pipeline.id()).apply(this)
}
/**
* Return a new AsyncPipe of this collection.
*/
asyncPipe(): AsyncPipe<Collection<T>> {
return AsyncPipe.wrap(this)
}
/**
* Remove the last item from this collection.
*/
pop(): MaybeCollectionItem<T> {
if ( this.length > 0 ) {
return this.storedItems.pop()
}
}
/**
* Add the given item to the beginning of this collection.
* @param item
*/
prepend(item: CollectionItem<T>): Collection<T> {
this.storedItems = [item, ...this.storedItems]
this.callPushSubscribers(item)
return this
}
/**
* Add the given item to the end of this collection.
* @param item
*/
push(item: CollectionItem<T>): Collection<T> {
this.storedItems.push(item)
this.callPushSubscribers(item)
return this
}
/**
* Subscribe to listen for items being added to the collection.
* @param sub
*/
push$(sub: Subscription<T>): Unsubscribe {
this.pushSubscribers.push(sub)
return {
unsubscribe: () => this.pushSubscribers = this.pushSubscribers.filter(x => x !== sub),
}
}
/** Helper to notify subscribers that an item has been pushed to the collection. */
private callPushSubscribers(item: T): void {
this.pushSubscribers
.forEach(sub => {
if ( typeof sub === 'object' ) {
sub?.next?.(item)
} else {
sub(item)
}
})
}
/**
* Push the given items to the end of this collection.
* Unlike `merge()`, this mutates the current collection's items.
* @param items
*/
concat(items: CollectionComparable<T>): Collection<T> {
const concats = items instanceof Collection ? items.all() : items
for ( const item of concats ) {
this.storedItems.push(item)
this.callPushSubscribers(item)
}
return this
}
/**
* Insert the given item into this collection at the specified index.
* @param index
* @param item
*/
put(index: number, item: CollectionItem<T>): Collection<T> {
const newItems = []
let inserted = false
this.storedItems.forEach((existing, existingIndex) => {
if ( existingIndex === index ) {
newItems.push(item)
inserted = true
}
newItems.push(existing)
})
if ( !inserted ) {
newItems.push(item)
}
return new Collection(newItems)
}
/**
* Return `n` many randomly-selected items from this collection.
* @param n
*/
random(n: number): Collection<T> {
const randomItems: CollectionItem<T>[] = []
const all = this.storedItems
if ( n > this.length ) {
n = this.length
}
while ( randomItems.length < n ) {
const item = all[Math.floor(Math.random() * all.length)]
if ( !randomItems.includes(item) ) {
randomItems.push(item)
}
}
return new Collection(randomItems)
}
/**
* Reduce this collection to a single value using the given reducer function.
*
* @example
* ```typescript
* const items = collect([1, 3, 5, 7])
*
* items.reduce((sum, item) => sum + item, 0) // => 16
* ```
*
* @param reducer
* @param initialValue
*/
reduce<T2>(reducer: KeyReducerFunction<T, T2>, initialValue: T2): T2
reduce<T2>(reducer: KeyReducerFunction<T, T2>, initialValue?: T2): T2 | undefined {
let currentValue = initialValue
this.storedItems.forEach((item, index) => {
currentValue = reducer(currentValue, item, index)
})
return currentValue
}
/**
* Return a new collection of items that fail the given truth-test function.
* @param truthTestFunction
*/
reject<T2>(truthTestFunction: KeyFunction<T, T2>): Collection<T> {
const rejected = this.storedItems.filter((item, index) => {
return !truthTestFunction(item, index)
})
return new Collection(rejected)
}
/**
* Return a new collection whose items are in the reverse order of the current one.
*/
reverse(): Collection<T> {
return new Collection([...this.storedItems.reverse()])
}
/**
* Try to find the given item in the collection. If it exists, return the index.
* @param item
*/
search(item: CollectionItem<T>): MaybeCollectionIndex {
let foundIndex
this.storedItems.some((possibleItem, index) => {
if ( possibleItem === item ) {
foundIndex = index
return true
}
})
return foundIndex
}
/**
* Remove and return the first item in the collection, if it exists.
*/
shift(): MaybeCollectionItem<T> {
if ( this.length > 0 ) {
return this.storedItems.shift()
}
}
/**
* Shuffle the items in this collection to a random order.
*/
shuffle(): Collection<T> {
const items = [...this.storedItems]
for ( let i = items.length - 1; i > 0; i-- ) {
const j = Math.floor(Math.random() * (i + 1))
;[items[i], items[j]] = [items[j], items[i]]
}
return new Collection(items)
}
/**
* Return a sub-set of this collection between the given index ranges.
* @param start
* @param end
*/
slice(start: number, end: number): Collection<T> {
return new Collection(this.storedItems.slice(start, end))
}
// TODO split
// TODO chunk
/**
* Sort the collection (optionally) using the given comparison function.
* @param comparisonFunction
*/
sort(comparisonFunction?: ComparisonFunction<T>): Collection<T> {
const items = this.storedItems
if ( comparisonFunction ) {
items.sort(comparisonFunction)
} else {
items.sort()
}
return new Collection(items)
}
/**
* Sort the collection (optionally) using the given key operator.
* @param key
*/
sortBy<T2>(key?: KeyOperator<T, T2>): Collection<T> {
let items: any[]
if ( key ) {
items = this.allAssociated(key)
} else {
items = this.storedItems.map(item => {
return { key: item,
item }
})
}
items.sort((a: any, b: any) => {
if ( a.key > b.key ) {
return 1
} else if ( a.key < b.key ) {
return -1
} else {
return 0
}
})
return new Collection(items.map((item: AssociatedCollectionItem<T2, T>) => item.item))
}
/**
* Identical to `sort()`, but in reverse order.
* @param comparisonFunction
*/
sortDesc(comparisonFunction?: ComparisonFunction<T>): Collection<T> {
return this.sort(comparisonFunction).reverse()
}
/**
* Identical to `sortBy()`, but in reverse order.
* @param key
*/
sortByDesc<T2>(key?: KeyOperator<T, T2>): Collection<T> {
return this.sortBy(key).reverse()
}
/**
* Remove `deleteCount` many items from the collection, starting at the `start` index.
* @param start
* @param deleteCount
*/
splice(start: CollectionIndex, deleteCount?: number): Collection<T> {
return new Collection([...this.storedItems].splice(start, deleteCount))
}
/**
* Return the sum of the items in the collection, optionally by key.
*
* @example
* ```typescript
* const items = collect([{k1: 1}, {k1: 3}, {k1: 5}, {k1: 7}])
*
* items.sum('k1') // => 16
* ```
*
* @param key
*/
sum<T2>(key?: KeyOperator<T, T2>): number {
let items
if ( key ) {
items = this.allAsNumbers(key)
} else {
items = this.storedItems.map(x => Number(x))
}
return items.reduce((prev, curr) => prev + curr)
}
/**
* Return a collection of the first `limit` many items in this collection.
* If `limit` is negative, returns the last `limit` many items.
* @param limit
*/
take(limit: number): Collection<T> {
if ( limit === 0 ) {
return new Collection()
} else if ( limit > 0 ) {
return new Collection(this.storedItems.slice(0, limit))
} else {
return new Collection(this.storedItems.reverse().slice(0, -1 * limit)
.reverse())
}
}
/**
* Apply the given function to this collection then return the collection.
* This is intended to help with chaining.
*
* @example
* ```typescript
* collection.tap(coll => {
* coll.push(item)
* })
* .where('someKey', '>', 4)
* // ... &c.
* ```
*
* @param func
*/
tap<T2>(func: CollectionFunction<T, T2>): this {
func(this)
return this
}
/**
* Return all distinct items in this collection. If a key is specified, returns
* all unique values of that key.
*
* @example
* ```typescript
* const users = collect([
* {uid: 1, name: 'John', type: 'admin'},
* {uid: 2, name: 'Jane', type: 'user'},
* {uid: 3, name: 'James', type: 'user'},
* ])
*
* users.unique('type') // => Collection['admin', 'user']
* ```
*
* @param key
*/
unique<T2>(key?: KeyOperator<T, T2>): Collection<T|T2> {
const has: CollectionItem<T|T2>[] = []
let items
if ( key ) {
items = this.allOperator<T2>(key)
} else {
items = [...this.storedItems]
}
for ( const item of items ) {
if ( !has.includes(item) ) {
has.push(item)
}
}
return new Collection(has)
}
/**
* Returns true if the given item is in this collection.
* @param item
*/
includes(item: CollectionItem<T>): boolean {
return this.storedItems.includes(item)
}
/**
* Add on to the end of this collection as many `value` items as necessary until the collection is `length` long.
* @param length
* @param value
*/
pad(length: number, value: CollectionItem<T>): Collection<T> {
const items = [...this.storedItems]
while ( items.length < length ) {
items.push(value)
}
return new Collection(items)
}
/**
* Cast the collection to an array.
*/
toArray(recursive = true): any[] {
const returns: any = []
for ( const item of this.storedItems ) {
if ( recursive && item instanceof Collection ) {
returns.push(item.toArray())
} else {
returns.push(item)
}
}
return returns
}
/**
* Cast the collection to a JSON string, optionally specifying the replacer and indentation.
* @param replacer
* @param space
*/
toJSON(replacer = undefined, space = 4): string {
return JSON.stringify(this.toArray(), replacer, space)
}
// TODO getIterator
// TODO getCachingIterator
[Symbol.iterator](): Iterator<T> {
const items = this.storedItems
let currentIndex = 0
return {
next() {
const item = items[currentIndex]
currentIndex += 1
return {
done: currentIndex > items.length,
value: item,
}
},
}
}
/**
* Like `map()`, but the callback can be async.
*
* @example
* A trivial example, but demonstrative:
*
* ```typescript
* const collection = collect([1, 2, 3])
*
* collection.map(async item => item + 1) // => Collection[Promise<1>, Promise<2>, Promise<3>]
*
* collection.promiseMap(async item => item + 1) // => Promise<Collection[2, 3, 4]>
* ```
*
* @param func
*/
async promiseMap<T2>(func: KeyFunction<T, T2 | Promise<T2>>): Promise<Collection<T2>> {
return new Collection<T2>(await Promise.all(
this.map(func).toArray(),
))
}
}