// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import marked from 'marked' import {IntlShape} from 'react-intl' declare global { interface Window { msCrypto: Crypto } } class Utils { static createGuid(): string { const crypto = window.crypto || window.msCrypto function randomDigit() { if (crypto && crypto.getRandomValues) { const rands = new Uint8Array(1) crypto.getRandomValues(rands) return (rands[0] % 16).toString(16) } return (Math.floor((Math.random() * 16))).toString(16) } return 'xxxxxxxx-xxxx-4xxx-8xxx-xxxxxxxxxxxx'.replace(/x/g, randomDigit) } static htmlToElement(html: string): HTMLElement { const template = document.createElement('template') template.innerHTML = html.trim() return template.content.firstChild as HTMLElement } static getElementById(elementId: string): HTMLElement { const element = document.getElementById(elementId) Utils.assert(element, `getElementById "${elementId}$`) return element! } static htmlEncode(text: string): string { return String(text).replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"') } static htmlDecode(text: string): string { return String(text).replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/"/g, '"') } // Markdown static htmlFromMarkdown(text: string): string { // HACKHACK: Somehow, marked doesn't encode angle brackets const renderer = new marked.Renderer() renderer.link = (href, title, contents) => `${contents}` const html = marked(text.replace(/ { return new Promise((resolve) => setTimeout(resolve, miliseconds)) } // Errors static assertValue(valueObject: any): void { const name = Object.keys(valueObject)[0] const value = valueObject[name] if (!value) { Utils.logError(`ASSERT VALUE [${name}]`) } } static assert(condition: any, tag = ''): void { /// #!if ENV !== "production" if (!condition) { Utils.logError(`ASSERT [${tag ?? new Error().stack}]`) } /// #!endif } static assertFailure(tag = ''): void { /// #!if ENV !== "production" Utils.assert(false, tag) /// #!endif } static log(message: string): void { /// #!if ENV !== "production" const timestamp = (Date.now() / 1000).toFixed(2) // eslint-disable-next-line no-console console.log(`[${timestamp}] ${message}`) /// #!endif } static logError(message: string): void { /// #!if ENV !== "production" const timestamp = (Date.now() / 1000).toFixed(2) // eslint-disable-next-line no-console console.error(`[${timestamp}] ${message}`) /// #!endif } // favicon static setFavicon(icon?: string): void { const href = icon ? `data:image/svg+xml,${icon}` : '' const link = (document.querySelector("link[rel*='icon']") || document.createElement('link')) as HTMLLinkElement link.type = 'image/x-icon' link.rel = 'shortcut icon' link.href = href document.getElementsByTagName('head')[0].appendChild(link) } // URL static replaceUrlQueryParam(paramName: string, value?: string): void { const queryString = new URLSearchParams(window.location.search) const currentValue = queryString.get(paramName) || '' if (currentValue !== value) { const newUrl = new URL(window.location.toString()) if (value) { newUrl.searchParams.set(paramName, value) } else { newUrl.searchParams.delete(paramName) } window.history.pushState({}, document.title, newUrl.toString()) } } // File names static sanitizeFilename(filename: string): string { // TODO: Use an industrial-strength sanitizer let sanitizedFilename = filename const illegalCharacters = ['\\', '/', '?', ':', '<', '>', '*', '|', '"', '.'] illegalCharacters.forEach((character) => { sanitizedFilename = sanitizedFilename.replace(character, '') }) return sanitizedFilename } // File picker static selectLocalFile(onSelect?: (file: File) => void, accept = '.jpg,.jpeg,.png'): void { const input = document.createElement('input') input.type = 'file' input.accept = accept input.onchange = async () => { const file = input.files![0] onSelect?.(file) } input.style.display = 'none' document.body.appendChild(input) input.click() // TODO: Remove or reuse input } // Arrays static arraysEqual(a: readonly any[], b: readonly any[]): boolean { if (a === b) { return true } if (a === null || b === null) { return false } if (a === undefined || b === undefined) { return false } if (a.length !== b.length) { return false } for (let i = 0; i < a.length; ++i) { if (a[i] !== b[i]) { return false } } return true } static arrayMove(arr: any[], srcIndex: number, destIndex: number): void { arr.splice(destIndex, 0, arr.splice(srcIndex, 1)[0]) } // Clipboard static copyTextToClipboard(text: string): boolean { const textField = document.createElement('textarea') textField.innerText = text textField.style.position = 'fixed' textField.style.opacity = '0' document.body.appendChild(textField) textField.select() let result = false try { result = document.execCommand('copy') } catch (err) { Utils.logError(`copyTextToClipboard ERROR: ${err}`) result = false } textField.remove() return result } static isMobile(): boolean { const toMatch = [ /Android/i, /webOS/i, /iPhone/i, /iPad/i, /iPod/i, /BlackBerry/i, /Windows Phone/i, ] return toMatch.some((toMatchItem) => { return navigator.userAgent.match(toMatchItem) }) } static getBaseURL(absolute?: boolean): string { let baseURL = (window as any).baseURL || '' baseURL = baseURL.replace(/\/+$/, '') if (baseURL.indexOf('/') === 0) { baseURL = baseURL.slice(1) } if (absolute) { return window.location.origin + '/' + baseURL } return baseURL } static buildURL(path: string, absolute?: boolean): string { const baseURL = this.getBaseURL() let finalPath = baseURL + path if (path.indexOf('/') !== 0) { finalPath = baseURL + '/' + path } if (absolute) { return window.location.origin + '/' + finalPath } return finalPath } } export {Utils}