import { formatValue } from 'publicodes' export function capitalise0(name: undefined): undefined export function capitalise0(name: string): string export function capitalise0(name?: string) { return name && name[0].toUpperCase() + name.slice(1) } export const debounce = (waitFor: number, fn: (...args: T[]) => void) => { let timeoutId: ReturnType return (...args: T[]) => { clearTimeout(timeoutId) timeoutId = setTimeout(() => fn(...args), waitFor) } } export const fetcher = (url: RequestInfo) => fetch(url).then((r) => r.json()) export function inIframe(): boolean { try { return window.self !== window.top } catch (e) { return true } } export function softCatch( fn: (arg: ArgType) => ReturnType ): (arg: ArgType) => ReturnType | null { return function (...args) { try { return fn(...args) } catch (e) { // eslint-disable-next-line no-console console.warn(e) return null } } } export function getSessionStorage() { // In some browsers like Brave, even just reading the variable sessionStorage // is throwing an error in the iframe, so we can't do things if sessionStorage !== undefined // and we need to wrap it in a try { } catch { } logic try { return window.sessionStorage } catch (e) { return undefined } } export const currencyFormat = (language: string) => ({ isCurrencyPrefixed: !!formatValue(12, { language, displayedUnit: '€' }).match( /^€/ ), thousandSeparator: formatValue(1000, { language }).charAt(1), decimalSeparator: formatValue(0.1, { language }).charAt(1), }) export function hash(str: string): number { let hash = 0 let chr for (let i = 0; i < str.length; i++) { chr = str.charCodeAt(i) hash = (hash << 5) - hash + chr hash |= 0 // Convert to 32bit integer } return hash } export function omit(obj: T, key: K): Omit { const returnObject = { ...obj } delete returnObject[key] return returnObject } export function isIterable(obj: unknown): obj is Iterable { return Symbol.iterator in Object(obj) } /** * Check if a key exists in the object and return its value or undefined, * usefull for type check * @param obj any object * @param key key to get value from object * @returns value or undefined */ export const getValueFrom = < T extends Record, K extends string | number | symbol >( obj: T, key: K ): Extract[K] | undefined => key in obj ? obj[key] : undefined /** * Return git branch name * @returns string */ export const getBranch = () => { let branch: string | undefined = import.meta.env.VITE_GITHUB_REF?.split( '/' )?.slice(-1)?.[0] if (branch === 'merge') { branch = import.meta.env.VITE_GITHUB_HEAD_REF } return branch ?? '' } /** * We use this function to hide some features in production while keeping them * in feature-branches. In case we do A/B testing with several branches served * in production, we should add the public faced branch names in the test below. * This is different from the import.meta.env.MODE in that a feature branch may * be build in production mode (with the NODE_ENV) but we may still want to show * or hide some features. * @returns boolean */ export const isProduction = () => import.meta.env.PROD && ['master', 'next'].includes(getBranch()) /** * Is a feature branche * @returns boolean */ export const isStaging = () => import.meta.env.PROD && !isProduction() export const isDevelopment = () => import.meta.env.DEV