UNPKG

@blueprintjs/table

Version:

Scalable interactive table component

372 lines (339 loc) 13.5 kB
/* * Copyright 2016 Palantir Technologies, Inc. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { IconSize } from "@blueprintjs/core"; // used to exclude icons from column header measure export const CLASSNAME_EXCLUDED_FROM_TEXT_MEASUREMENT = "bp-table-text-no-measure"; // supposed width of the icons placeholder const EXCLUDED_ICON_PLACEHOLDER_WIDTH = IconSize.STANDARD; /** * Since Firefox doesn't provide a computed "font" property, we manually * construct it using the ordered properties that can be specifed in CSS. */ const CSS_FONT_PROPERTIES = ["font-style", "font-variant", "font-weight", "font-size", "font-family"]; // the functions using these interfaces now live in core. it's not clear how to // import interfaces from core and re-export them here, so just redefine them. export interface KeyAllowlist<T> { include: Array<keyof T>; } export interface KeyDenylist<T> { exclude: Array<keyof T>; } // table is nearly deprecated, let's not block on code coverage /* istanbul ignore next */ export const Utils = { /** * Invokes the callback `n` times, collecting the results in an array, which * is the return value. Similar to _.times */ times<T>(n: number, callback: (i: number) => T): T[] { if (n < 0) { throw new Error("[Blueprint] times() cannot be called with negative numbers."); } const result: T[] = Array(n); for (let index = 0; index < n; index++) { result[index] = callback(index); } return result; }, /** * Takes an array of numbers, returns an array of numbers of the same length in which each * value is the sum of current and previous values in the input array. * * Example input: [10, 20, 50] * output: [10, 30, 80] */ accumulate(numbers: number[]): number[] { const result = []; let sum = 0; for (const num of numbers) { sum += num; result.push(sum); } return result; }, /** * Returns traditional spreadsheet-style column names * e.g. (A, B, ..., Z, AA, AB, ..., ZZ, AAA, AAB, ...). * * Note that this isn't technically mathematically equivalent to base 26 since * there is no zero element. */ toBase26Alpha: (num: number) => { let str = ""; while (true) { const letter = num % 26; str = String.fromCharCode(65 + letter) + str; num = num - letter; if (num <= 0) { return str; } num = num / 26 - 1; } }, /** * Returns traditional spreadsheet-style cell names * e.g. (A1, B2, ..., Z44, AA1) with rows 1-indexed. */ toBase26CellName: (rowIndex: number, columnIndex: number) => { return `${Utils.toBase26Alpha(columnIndex)}${rowIndex + 1}`; }, /** * Performs the binary search algorithm to find the index of the `value` * parameter in a sorted list of numbers. If `value` is not in the list, the * index where `value` can be inserted to maintain the sort is returned. * * Unlike a typical binary search implementation, we use a `lookup` * callback to access the sorted list of numbers instead of an array. This * avoids additional storage overhead. * * We use this to, for example, find the index of a row/col given its client * coordinate. * * Adapted from lodash https://github.com/lodash/lodash/blob/4.11.2/lodash.js#L3579 * * @param value - the query value * @param high - the length of the sorted list of numbers * @param lookup - returns the number from the list at the supplied index */ binarySearch(value: number, high: number, lookup: (index: number) => number): number { let low = 0; while (low < high) { const mid = Math.floor((low + high) / 2.0); const computed = lookup(mid); if (computed < value) { low = mid + 1; } else { high = mid; } } return high; }, /** * Takes in one full array of values and one sparse array of the same * length and type. Returns a copy of the `defaults` array, where each * value is replaced with the corresponding non-null value at the same * index in `sparseOverrides`. * * @param defaults - the full array of default values * @param sparseOverrides - the sparse array of override values */ assignSparseValues<T>(defaults: T[], sparseOverrides: Array<T | undefined | null>) { if (sparseOverrides == null || defaults.length !== sparseOverrides.length) { return defaults; } defaults = defaults.slice(); for (let i = 0; i < defaults.length; i++) { const override = sparseOverrides[i]; if (override != null) { defaults[i] = override; } } return defaults; }, /** * Measures the bounds of supplied element's textContent. * We use the computed font from the supplied element and a non-DOM canvas * context to measure the text. */ measureElementTextContent(element: Element): TextMetrics { const context = document.createElement("canvas").getContext("2d")!; const style = getComputedStyle(element, null); context.font = CSS_FONT_PROPERTIES.map(prop => style.getPropertyValue(prop)).join(" "); return measureTextContentWithExclusions(context, element); }, /** * Given a number, returns a value that is clamped within a * minimum/maximum bounded range. The minimum and maximum are optional. If * either is missing, that extrema limit is not applied. * * Assumes max >= min. */ clamp(value: number, min?: number, max?: number): number { if (min != null && value < min) { value = min; } if (max != null && value > max) { value = max; } return value; }, /** * When reordering a contiguous block of rows or columns to a new index, we show a preview guide * at the absolute index in the original ordering but emit the new index in the reordered list. * This function converts an absolute "guide" index to a relative "reordered" index. * * Example: Say we want to move the first three columns two spots to the right. While we drag, a * vertical guide is shown to preview where we'll be dropping the columns. (In the following * ASCII art, `*` denotes a selected column, `·` denotes a cell border, and `|` denotes a * vertical guide). * * Before mousedown: * · 0 · 1 · 2 · 3 · 4 · 5 · * * * * * * During mousemove two spots to the right: * · 0 · 1 · 2 · 3 · 4 | 5 · * * * * * * After mouseup: * · 3 · 4 · 0 · 1 · 2 · 5 · * * * * * * Note that moving the three columns beyond index 4 effectively moves them two spots rightward. * * In this case, the inputs to this function would be: * - oldIndex: 0 (the left-most index of the selected column range in the original ordering) * - newIndex: 5 (the index on whose left boundary the guide appears in the original ordering) * - length: 3 (the number of columns to move) * * The return value will then be 2, the left-most index of the columns in the new ordering. */ guideIndexToReorderedIndex(oldIndex: number, newIndex: number, length: number): number { if (newIndex < oldIndex) { return newIndex; } else if (oldIndex <= newIndex && newIndex < oldIndex + length) { return oldIndex; } else { return Math.max(0, newIndex - length); } }, /** * When reordering a contiguous block of rows or columns to a new index, we show a preview guide * at the absolute index in the original ordering but emit the new index in the reordered list. * This function converts a relative "reordered"" index to an absolute "guide" index. * * For the scenario in the example above, the inputs to this function would be: * - oldIndex: 0 (the left-most index of the selected column range in the original ordering) * - newIndex: 2 (the left-most index of the selected column range in the new ordering) * - length: 3 (the number of columns to move) * * The return value will then be 5, the index on whose left boundary the guide should appear in * the original ordering. */ reorderedIndexToGuideIndex(oldIndex: number, newIndex: number, length: number): number { return newIndex <= oldIndex ? newIndex : newIndex + length; }, /** * Returns a copy of the provided array with the `length` contiguous elements starting at the * `from` index reordered to start at the `to` index. * * For example, given the array [A,B,C,D,E,F], reordering the 3 contiguous elements starting at * index 1 (B, C, and D) to start at index 2 would yield [A,E,B,C,D,F]. */ reorderArray<T>(array: T[], from: number, to: number, length = 1): T[] | undefined { if (length === 0 || length === array.length || from === to) { // return an unchanged copy return array.slice(); } if (length < 0 || length > array.length || from + length > array.length) { return undefined; } const before = array.slice(0, from); const within = array.slice(from, from + length); const after = array.slice(from + length); const result = []; let i = 0; let b = 0; let w = 0; let a = 0; while (i < to) { if (b < before.length) { result.push(before[b]); b += 1; } else { result.push(after[a]); a += 1; } i += 1; } while (w < length) { result.push(within[w]); w += 1; i += 1; } while (i < array.length) { if (b < before.length) { result.push(before[b]); b += 1; } else { result.push(after[a]); a += 1; } i += 1; } return result; }, /** * Returns true if the mouse event was triggered by the left mouse button. */ isLeftClick(event: MouseEvent): boolean { return event.button === 0; }, getApproxCellHeight( cellText: string, columnWidth: number, approxCharWidth: number, approxLineHeight: number, horizontalPadding: number, numBufferLines: number, ): number { const numCharsInCell = cellText == null ? 0 : cellText.length; const actualCellWidth = columnWidth; const availableCellWidth = actualCellWidth - horizontalPadding; const approxCharsPerLine = availableCellWidth / approxCharWidth; const approxNumLinesDesired = Math.ceil(numCharsInCell / approxCharsPerLine) + numBufferLines; const approxCellHeight = approxNumLinesDesired * approxLineHeight; return approxCellHeight; }, /** * Shallow comparison of potentially sparse arrays. * * @returns true if the array values are equal */ compareSparseArrays( a: Array<number | null | undefined> | undefined, b: Array<number | null | undefined> | undefined, ): boolean { return ( a !== undefined && b !== undefined && a.length === b.length && a.every((aValue, index) => aValue === b[index]) ); }, }; // table is nearly deprecated, let's not block on code coverage /* istanbul ignore next */ /** * Wrapper around Canvas measureText which applies some extra logic to optionally * exclude an element's text from the computation. */ function measureTextContentWithExclusions(context: CanvasRenderingContext2D, element: Element): TextMetrics { const elementsToExclude = element.querySelectorAll(`.${CLASSNAME_EXCLUDED_FROM_TEXT_MEASUREMENT}`); let excludedElementsWidth = 0; if (elementsToExclude && elementsToExclude.length) { elementsToExclude.forEach(e => { const excludedMetrics = context.measureText(e.textContent ?? ""); excludedElementsWidth += excludedMetrics.width - EXCLUDED_ICON_PLACEHOLDER_WIDTH; }); } const metrics = context.measureText(element.textContent ?? ""); const metricsWithExclusions = { ...metrics, width: metrics.width - excludedElementsWidth, }; return metricsWithExclusions; }