// Copyright 2015 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. export function anyToString(x: any): string { return "" + x; } function computeScrollTop(container, element) { const height = container.offsetHeight; const margin = Math.floor(height / 4); const pos = element.offsetTop; const currentScrollTop = container.scrollTop; if (pos < currentScrollTop + margin) { return Math.max(0, pos - margin); } else if (pos > (currentScrollTop + 3 * margin)) { return Math.max(0, pos - 3 * margin); } return pos; } export class ViewElements { container: HTMLElement; scrollTop: number; constructor(container: HTMLElement) { this.container = container; this.scrollTop = undefined; } consider(element, doConsider) { if (!doConsider) return; const newScrollTop = computeScrollTop(this.container, element); if (isNaN(newScrollTop)) { console.log("NOO"); } if (this.scrollTop === undefined) { this.scrollTop = newScrollTop; } else { this.scrollTop = Math.min(this.scrollTop, newScrollTop); } } apply(doApply) { if (!doApply || this.scrollTop === undefined) return; this.container.scrollTop = this.scrollTop; } } export function sortUnique<T>(arr: Array<T>, f: (a: T, b: T) => number, equal: (a: T, b: T) => boolean) { if (arr.length == 0) return arr; arr = arr.sort(f); const ret = [arr[0]]; for (let i = 1; i < arr.length; i++) { if (!equal(arr[i - 1], arr[i])) { ret.push(arr[i]); } } return ret; } // Partial application without binding the receiver export function partial(f: any, ...arguments1: Array<any>) { return function (this: any, ...arguments2: Array<any>) { f.apply(this, [...arguments1, ...arguments2]); }; } export function isIterable(obj: any): obj is Iterable<any> { return obj != null && obj != undefined && typeof obj != 'string' && typeof obj[Symbol.iterator] === 'function'; } export function alignUp(raw: number, multiple: number): number { return Math.floor((raw + multiple - 1) / multiple) * multiple; } export function measureText(text: string) { const textMeasure = document.getElementById('text-measure'); if (textMeasure instanceof SVGTSpanElement) { textMeasure.textContent = text; return { width: textMeasure.getBBox().width, height: textMeasure.getBBox().height, }; } return { width: 0, height: 0 }; } // Interpolate between the given start and end values by a fraction of val/max. export function interpolate(val: number, max: number, start: number, end: number) { return start + (end - start) * (val / max); }