focalboard/webapp/src/utils.ts

346 lines
11 KiB
TypeScript
Raw Normal View History

2020-10-20 21:50:53 +02:00
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
2020-10-20 21:52:56 +02:00
import marked from 'marked'
import {IntlShape} from 'react-intl'
2020-10-08 18:21:27 +02:00
declare global {
2020-10-20 21:50:53 +02:00
interface Window {
msCrypto: Crypto
}
2020-10-08 18:21:27 +02:00
}
const IconClass = 'octo-icon'
const OpenButtonClass = 'open-button'
const SpacerClass = 'octo-spacer'
const HorizontalGripClass = 'HorizontalGrip'
2020-10-08 18:21:27 +02:00
class Utils {
2020-10-20 22:26:06 +02:00
static createGuid(): string {
2020-10-20 21:50:53 +02:00
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')
2020-10-20 22:26:06 +02:00
template.innerHTML = html.trim()
2020-10-20 21:50:53 +02:00
return template.content.firstChild as HTMLElement
}
static getElementById(elementId: string): HTMLElement {
const element = document.getElementById(elementId)
Utils.assert(element, `getElementById "${elementId}$`)
return element!
}
2020-10-20 22:26:06 +02:00
static htmlEncode(text: string): string {
2020-10-20 21:50:53 +02:00
return String(text).replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;')
}
2020-10-20 22:26:06 +02:00
static htmlDecode(text: string): string {
2020-10-20 21:50:53 +02:00
return String(text).replace(/&amp;/g, '&').replace(/&lt;/g, '<').replace(/&gt;/g, '>').replace(/&quot;/g, '"')
}
// re-use canvas object for better performance
2021-05-14 00:17:07 +02:00
static canvas : HTMLCanvasElement | undefined
static getTextWidth(displayText: string, fontDescriptor: string): number {
if (displayText !== '') {
2021-05-14 00:17:07 +02:00
if (!this.canvas) {
this.canvas = document.createElement('canvas') as HTMLCanvasElement
}
const context = this.canvas.getContext('2d')
if (context) {
context.font = fontDescriptor
const metrics = context.measureText(displayText)
return Math.ceil(metrics.width)
}
}
return 0
}
static getFontAndPaddingFromCell = (cell: Element) : {fontDescriptor: string, padding: number} => {
const style = getComputedStyle(cell)
const padding = Utils.getHorizontalPadding(style)
return Utils.getFontAndPaddingFromChildren(cell.children, padding)
}
// recursive routine to determine the padding and font from its children
// specifically for the table view
static getFontAndPaddingFromChildren = (children: HTMLCollection, pad: number) : {fontDescriptor: string, padding: number} => {
const myResults = {
fontDescriptor: '',
padding: pad,
}
Array.from(children).forEach((element) => {
switch (element.className) {
case IconClass:
case HorizontalGripClass:
myResults.padding += element.clientWidth
break
case SpacerClass:
case OpenButtonClass:
break
default: {
const style = getComputedStyle(element)
myResults.fontDescriptor = style.font
myResults.padding += Utils.getHorizontalPadding(style)
const childResults = Utils.getFontAndPaddingFromChildren(element.children, myResults.padding)
if (childResults.fontDescriptor !== '') {
myResults.fontDescriptor = childResults.fontDescriptor
myResults.padding = childResults.padding
}
}
}
})
return myResults
}
static getHorizontalPadding = (style: CSSStyleDeclaration): number => {
return parseInt(style.paddingLeft, 10) + parseInt(style.paddingRight, 10) + parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10) + parseInt(style.borderLeft, 10) + parseInt(style.borderRight, 10)
}
2020-10-20 21:50:53 +02:00
// Markdown
static htmlFromMarkdown(text: string): string {
// HACKHACK: Somehow, marked doesn't encode angle brackets
2020-10-22 01:06:11 +02:00
const renderer = new marked.Renderer()
2021-06-15 09:34:59 +02:00
if ((window as any).openInNewBrowser) {
2021-06-18 16:46:57 +02:00
renderer.link = (href, title, contents) => `<a target="_blank" rel="noreferrer" href="${encodeURI(href || '')}" title="${title ? encodeURI(title) : ''}" onclick="event.stopPropagation(); openInNewBrowser && openInNewBrowser(event.target.href);">${contents}</a>`
2021-06-15 09:34:59 +02:00
}
2020-11-18 20:01:07 +01:00
const html = marked(text.replace(/</g, '&lt;'), {renderer, breaks: true})
2021-06-15 09:34:59 +02:00
return html.trim()
2020-10-20 21:50:53 +02:00
}
// Date and Time
static displayDate(date: Date, intl: IntlShape): string {
const text = intl.formatDate(date, {year: 'numeric', month: 'short', day: '2-digit'})
2020-10-20 21:50:53 +02:00
return text
}
static displayDateTime(date: Date, intl: IntlShape): string {
const text = intl.formatDate(date, {
year: 'numeric',
month: 'short',
day: '2-digit',
hour: 'numeric',
minute: 'numeric',
})
2020-10-20 21:50:53 +02:00
return text
}
2020-12-18 21:52:45 +01:00
static sleep(miliseconds: number): Promise<void> {
return new Promise((resolve) => setTimeout(resolve, miliseconds))
}
2020-10-20 21:50:53 +02:00
// Errors
2020-10-20 22:26:06 +02:00
static assertValue(valueObject: any): void {
2020-10-20 21:50:53 +02:00
const name = Object.keys(valueObject)[0]
const value = valueObject[name]
if (!value) {
Utils.logError(`ASSERT VALUE [${name}]`)
}
}
2020-10-20 22:26:06 +02:00
static assert(condition: any, tag = ''): void {
2020-10-20 21:50:53 +02:00
/// #!if ENV !== "production"
if (!condition) {
Utils.logError(`ASSERT [${tag ?? new Error().stack}]`)
}
/// #!endif
}
2020-10-20 22:26:06 +02:00
static assertFailure(tag = ''): void {
2020-10-20 21:50:53 +02:00
/// #!if ENV !== "production"
Utils.assert(false, tag)
/// #!endif
}
2020-10-20 22:26:06 +02:00
static log(message: string): void {
2020-10-20 21:50:53 +02:00
/// #!if ENV !== "production"
const timestamp = (Date.now() / 1000).toFixed(2)
2020-10-20 22:26:06 +02:00
// eslint-disable-next-line no-console
2020-10-20 21:50:53 +02:00
console.log(`[${timestamp}] ${message}`)
/// #!endif
}
2020-10-20 22:26:06 +02:00
static logError(message: string): void {
2020-10-20 21:50:53 +02:00
/// #!if ENV !== "production"
const timestamp = (Date.now() / 1000).toFixed(2)
2020-10-20 22:26:06 +02:00
// eslint-disable-next-line no-console
2020-10-20 21:50:53 +02:00
console.error(`[${timestamp}] ${message}`)
/// #!endif
}
// favicon
2020-10-20 22:26:06 +02:00
static setFavicon(icon?: string): void {
2020-10-22 18:46:06 +02:00
const href = icon ? `data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100"><text y=".9em" font-size="90">${icon}</text></svg>` : ''
2020-10-20 21:50:53 +02:00
const link = (document.querySelector("link[rel*='icon']") || document.createElement('link')) as HTMLLinkElement
2020-10-20 21:52:56 +02:00
link.type = 'image/x-icon'
link.rel = 'shortcut icon'
2020-10-20 21:50:53 +02:00
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())
}
}
static ensureProtocol(url: string): string {
return url.match(/^.+:\/\//) ? url : `https://${url}`
}
2020-10-20 21:50:53 +02:00
// File names
2020-10-20 22:26:06 +02:00
static sanitizeFilename(filename: string): string {
2020-10-20 21:50:53 +02:00
// TODO: Use an industrial-strength sanitizer
let sanitizedFilename = filename
const illegalCharacters = ['\\', '/', '?', ':', '<', '>', '*', '|', '"', '.']
illegalCharacters.forEach((character) => {
sanitizedFilename = sanitizedFilename.replace(character, '')
2020-10-20 21:52:56 +02:00
})
2020-10-20 21:50:53 +02:00
return sanitizedFilename
}
// File picker
static selectLocalFile(onSelect?: (file: File) => void, accept = '.jpg,.jpeg,.png'): void {
const input = document.createElement('input')
2020-10-20 21:52:56 +02:00
input.type = 'file'
2020-10-20 21:50:53 +02:00
input.accept = accept
input.onchange = async () => {
const file = input.files![0]
onSelect?.(file)
2020-10-20 21:52:56 +02:00
}
2020-10-20 21:50:53 +02:00
2020-10-20 21:52:56 +02:00
input.style.display = 'none'
2020-10-20 21:50:53 +02:00
document.body.appendChild(input)
input.click()
// TODO: Remove or reuse input
}
// Arrays
2020-12-07 21:13:20 +01:00
static arraysEqual(a: readonly any[], b: readonly any[]): boolean {
2020-10-20 21:50:53 +02:00
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
}
2020-12-18 21:52:45 +01:00
static arrayMove(arr: any[], srcIndex: number, destIndex: number): void {
arr.splice(destIndex, 0, arr.splice(srcIndex, 1)[0])
}
2021-01-13 20:01:33 +01:00
// 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) {
2021-06-22 17:26:00 +02:00
if (finalPath.indexOf('/') === 0) {
finalPath = finalPath.slice(1)
}
return window.location.origin + '/' + finalPath
}
return finalPath
}
2020-10-08 18:21:27 +02:00
}
2020-10-20 21:50:53 +02:00
export {Utils}