speedscope/utils.ts
Jamie Wong aee4534de4
Integrate jest and add first test file (#51)
This sets up the first bit of test infrastructure to write and run unit tests on every PR

Related to #14 
Fixes #7
2018-05-29 14:34:09 -07:00

68 lines
1.8 KiB
TypeScript

export function lastOf<T>(ts: T[]): T | null {
return ts[ts.length - 1] || null
}
export function sortBy<T>(ts: T[], key: (t: T) => number | string): void {
function comparator(a: T, b: T) {
return key(a) < key(b) ? -1 : 1
}
ts.sort(comparator)
}
export function getOrInsert<K, V>(map: Map<K, V>, k: K, fallback: (k: K) => V): V {
if (!map.has(k)) map.set(k, fallback(k))
return map.get(k)!
}
export function getOrElse<K, V>(map: Map<K, V>, k: K, fallback: (k: K) => V): V {
if (!map.has(k)) return fallback(k)
return map.get(k)!
}
export function getOrThrow<K, V>(map: Map<K, V>, k: K): V {
if (!map.has(k)) {
throw new Error(`Expected key ${k}`)
}
return map.get(k)!
}
export function* itMap<T, U>(it: Iterable<T>, f: (t: T) => U): Iterable<U> {
for (let t of it) {
yield f(t)
}
}
export function itForEach<T>(it: Iterable<T>, f: (t: T) => void): void {
for (let t of it) {
f(t)
}
}
export function itReduce<T, U>(it: Iterable<T>, f: (a: U, b: T) => U, init: U): U {
let accum: U = init
for (let t of it) {
accum = f(accum, t)
}
return accum
}
export function zeroPad(s: string, width: number) {
return new Array(Math.max(width - s.length, 0) + 1).join('0') + s
}
// NOTE: This blindly assumes the same result across contexts.
const measureTextCache = new Map<string, number>()
let lastDevicePixelRatio = -1
export function cachedMeasureTextWidth(ctx: CanvasRenderingContext2D, text: string): number {
if (window.devicePixelRatio !== lastDevicePixelRatio) {
// This cache is no longer valid!
measureTextCache.clear()
lastDevicePixelRatio = window.devicePixelRatio
}
if (!measureTextCache.has(text)) {
measureTextCache.set(text, ctx.measureText(text).width)
}
return measureTextCache.get(text)!
}