| /** |
| * @license |
| * Copyright 2017 Google LLC |
| * SPDX-License-Identifier: Apache-2.0 |
| */ |
| |
| import {computeDisplayPath} from './path-list-util'; |
| |
| /** |
| * Returns a count plus string that is pluralized when necessary. |
| */ |
| export function pluralize(count: number, noun: string): string { |
| if (count === 0) return ''; |
| return `${count} ${noun}` + (count > 1 ? 's' : ''); |
| } |
| |
| export function charsOnly(s: string): string { |
| return s.replace(/[^a-zA-Z]+/g, ''); |
| } |
| |
| export function isCharacterLetter(ch: string): boolean { |
| return ch.length === 1 && ch.toLowerCase() !== ch.toUpperCase(); |
| } |
| |
| export function isUpperCase(ch: string): boolean { |
| return ch === ch.toUpperCase(); |
| } |
| |
| export function ordinal(n?: number): string { |
| if (n === undefined) return ''; |
| if (n % 10 === 1 && n % 100 !== 11) return `${n}st`; |
| if (n % 10 === 2 && n % 100 !== 12) return `${n}nd`; |
| if (n % 10 === 3 && n % 100 !== 13) return `${n}rd`; |
| return `${n}th`; |
| } |
| |
| /** Escape operator value to avoid affecting overall query. |
| * |
| * Escapes quotes (") and backslashes (\). Wraps in quotes so the value can |
| * contain spaces and colons. |
| */ |
| export function escapeAndWrapSearchOperatorValue(value: string): string { |
| return `"${value.replaceAll('\\', '\\\\').replaceAll('"', '\\"')}"`; |
| } |
| |
| /** |
| * This converts any inputed value into string. |
| * |
| * This is so typescript checker doesn't fail. |
| */ |
| export function convertToString(key?: unknown) { |
| return key !== undefined ? String(key) : ''; |
| } |
| |
| export function capitalizeFirstLetter(str: string) { |
| return str.charAt(0).toUpperCase() + str.slice(1); |
| } |
| |
| /** |
| * Converts the items into a sentence-friendly format. Examples: |
| * listForSentence(["Foo", "Bar", "Baz"]) |
| * => 'Foo, Bar, and Baz' |
| * listForSentence(["Foo", "Bar"]) |
| * => 'Foo and Bar' |
| * listForSentence(["Foo"]) |
| * => 'Foo' |
| */ |
| export function listForSentence(items: string[]): string { |
| if (items.length < 2) return items.join(''); |
| if (items.length === 2) return items.join(' and '); |
| |
| const firstItems = items.slice(0, items.length - 1); |
| const lastItem = items[items.length - 1]; |
| return `${firstItems.join(', ')}, and ${lastItem}`; |
| } |
| |
| /** |
| * Separates a path into: |
| * - The part that matches another path, |
| * - The part that does not match the other path, |
| * - The file name |
| * |
| * For example: |
| * diffFilePaths('same/part/new/part/foo.js', 'same/part/different/foo.js'); |
| * yields: { |
| * matchingFolders: 'same/part/', |
| * newFolders: 'new/part/', |
| * fileName: 'foo.js', |
| * } |
| */ |
| export function diffFilePaths(filePath: string, otherFilePath?: string) { |
| // Separate each string into an array of folder names + file name. |
| const displayPath = computeDisplayPath(filePath); |
| const previousFileDisplayPath = computeDisplayPath(otherFilePath); |
| const displayPathParts = displayPath.split('/'); |
| const previousFileDisplayPathParts = previousFileDisplayPath.split('/'); |
| |
| // Construct separate strings for matching folders, new folders, and file |
| // name. |
| const firstDifferencePartIndex = displayPathParts.findIndex( |
| (part, index) => previousFileDisplayPathParts[index] !== part |
| ); |
| const matchingSection = displayPathParts |
| .slice(0, firstDifferencePartIndex) |
| .join('/'); |
| const newFolderSection = displayPathParts |
| .slice(firstDifferencePartIndex, -1) |
| .join('/'); |
| const fileNameSection = displayPathParts[displayPathParts.length - 1]; |
| |
| // Note: folder sections need '/' appended back. |
| return { |
| matchingFolders: matchingSection.length > 0 ? `${matchingSection}/` : '', |
| newFolders: newFolderSection.length > 0 ? `${newFolderSection}/` : '', |
| fileName: fileNameSection, |
| }; |
| } |