-
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcustom.ts
87 lines (81 loc) · 2.4 KB
/
custom.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
import { imap } from "./itertools.ts";
import { flatten } from "./more-itertools.ts";
import type { Maybe, NullableKeys, Predicate } from "./types.ts";
function isDefined<T>(x: T): boolean {
return x !== undefined;
}
/**
* Returns an iterable, filtering out any `undefined` values from the input
* iterable. This function is useful to convert a list of `Maybe<T>`'s to
* a list of `T`'s, discarding all the undefined values:
*
* >>> compact([1, 2, undefined, 3])
* [1, 2, 3]
*/
export function* icompact<T>(iterable: Iterable<Maybe<T>>): Iterable<T> {
for (const item of iterable) {
if (typeof item !== "undefined") {
yield item;
}
}
}
/**
* See icompact().
*/
export function compact<T>(iterable: Iterable<Maybe<T>>): Array<T> {
return Array.from(icompact(iterable));
}
/**
* Removes all undefined values from the given object. Returns a new object.
*
* >>> compactObject({ a: 1, b: undefined, c: 0 })
* { a: 1, c: 0 }
*/
export function compactObject<O extends Record<string, unknown>>(
obj: O,
): Omit<O, NullableKeys<O>> {
// deno-lint-ignore no-explicit-any
const result: any = {};
for (const [key, value] of Object.entries(obj)) {
if (typeof value !== "undefined") {
result[key] = value;
}
}
return result;
}
/**
* Returns the first item in the iterable for which the predicate holds, if
* any. If no such item exists, `undefined` is returned. The default
* predicate is any defined value.
*/
export function first<T>(
iterable: Iterable<T>,
keyFn?: Predicate<T>,
): Maybe<T> {
const fn = keyFn || isDefined;
for (const value of iterable) {
if (fn(value)) {
return value;
}
}
return undefined;
}
/**
* Returns 0 or more values for every value in the given iterable.
* Technically, it's just calling map(), followed by flatten(), but it's a very
* useful operation if you want to map over a structure, but not have a 1:1
* input-output mapping. Instead, if you want to potentially return 0 or more
* values per input element, use flatmap():
*
* For example, to return all numbers `n` in the input iterable `n` times:
*
* >>> const repeatN = n => repeat(n, n);
* >>> [...flatmap([0, 1, 2, 3, 4], repeatN)]
* [1, 2, 2, 3, 3, 3, 4, 4, 4, 4] // note: no 0
*/
export function flatmap<T, S>(
iterable: Iterable<T>,
mapper: (v: T) => Iterable<S>,
): Iterable<S> {
return flatten(imap(iterable, mapper));
}