2016-09-17 01:39:19 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 OpenMarket Ltd
|
2020-06-08 18:14:10 +00:00
|
|
|
Copyright 2019, 2020 The Matrix.org Foundation C.I.C.
|
2016-09-17 01:39:19 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2023-02-13 11:39:16 +00:00
|
|
|
import { ReactElement, ReactNode } from "react";
|
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
import { _t } from "../languageHandler";
|
|
|
|
import { jsxJoin } from "./ReactUtils";
|
2019-05-20 14:16:12 +00:00
|
|
|
|
2016-09-17 01:39:19 +00:00
|
|
|
/**
|
|
|
|
* formats numbers to fit into ~3 characters, suitable for badge counts
|
|
|
|
* e.g: 999, 9.9K, 99K, 0.9M, 9.9M, 99M, 0.9B, 9.9B
|
|
|
|
*/
|
2020-06-08 18:14:10 +00:00
|
|
|
export function formatCount(count: number): string {
|
2020-06-23 15:41:36 +00:00
|
|
|
if (count < 1000) return count.toString();
|
|
|
|
if (count < 10000) return (count / 1000).toFixed(1) + "K";
|
|
|
|
if (count < 100000) return (count / 1000).toFixed(0) + "K";
|
|
|
|
if (count < 10000000) return (count / 1000000).toFixed(1) + "M";
|
|
|
|
if (count < 100000000) return (count / 1000000).toFixed(0) + "M";
|
|
|
|
return (count / 1000000000).toFixed(1) + "B"; // 10B is enough for anyone, right? :S
|
2017-01-20 14:22:27 +00:00
|
|
|
}
|
2017-05-22 11:01:09 +00:00
|
|
|
|
2020-01-24 15:13:55 +00:00
|
|
|
/**
|
|
|
|
* Format a count showing the whole number but making it a bit more readable.
|
|
|
|
* e.g: 1000 => 1,000
|
|
|
|
*/
|
2020-06-08 18:14:10 +00:00
|
|
|
export function formatCountLong(count: number): string {
|
2020-01-24 15:13:55 +00:00
|
|
|
const formatter = new Intl.NumberFormat();
|
2020-06-18 13:32:43 +00:00
|
|
|
return formatter.format(count);
|
2020-01-24 15:13:55 +00:00
|
|
|
}
|
|
|
|
|
2019-11-26 12:39:45 +00:00
|
|
|
/**
|
|
|
|
* format a size in bytes into a human readable form
|
|
|
|
* e.g: 1024 -> 1.00 KB
|
|
|
|
*/
|
2020-06-08 18:14:10 +00:00
|
|
|
export function formatBytes(bytes: number, decimals = 2): string {
|
2022-12-12 11:24:14 +00:00
|
|
|
if (bytes === 0) return "0 Bytes";
|
2019-11-26 12:39:45 +00:00
|
|
|
|
|
|
|
const k = 1024;
|
|
|
|
const dm = decimals < 0 ? 0 : decimals;
|
2022-12-12 11:24:14 +00:00
|
|
|
const sizes = ["Bytes", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"];
|
2019-11-26 12:39:45 +00:00
|
|
|
|
|
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + " " + sizes[i];
|
2019-11-26 12:39:45 +00:00
|
|
|
}
|
|
|
|
|
2017-05-22 11:01:09 +00:00
|
|
|
/**
|
|
|
|
* format a key into groups of 4 characters, for easier visual inspection
|
|
|
|
*
|
|
|
|
* @param {string} key key to format
|
|
|
|
*
|
|
|
|
* @return {string}
|
|
|
|
*/
|
2020-06-08 18:14:10 +00:00
|
|
|
export function formatCryptoKey(key: string): string {
|
2023-02-13 17:01:43 +00:00
|
|
|
return key.match(/.{1,4}/g)!.join(" ");
|
2017-05-22 11:01:09 +00:00
|
|
|
}
|
2018-10-23 08:49:44 +00:00
|
|
|
/**
|
|
|
|
* calculates a numeric hash for a given string
|
|
|
|
*
|
|
|
|
* @param {string} str string to hash
|
|
|
|
*
|
|
|
|
* @return {number}
|
|
|
|
*/
|
2020-06-08 18:14:10 +00:00
|
|
|
export function hashCode(str: string): number {
|
2018-10-23 08:49:44 +00:00
|
|
|
let hash = 0;
|
|
|
|
let i;
|
|
|
|
let chr;
|
|
|
|
if (str.length === 0) {
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
for (i = 0; i < str.length; i++) {
|
|
|
|
chr = str.charCodeAt(i);
|
2022-12-12 11:24:14 +00:00
|
|
|
hash = (hash << 5) - hash + chr;
|
2018-10-23 08:49:44 +00:00
|
|
|
hash |= 0;
|
|
|
|
}
|
|
|
|
return Math.abs(hash);
|
|
|
|
}
|
2019-04-17 08:21:30 +00:00
|
|
|
|
2020-06-08 18:14:10 +00:00
|
|
|
export function getUserNameColorClass(userId: string): string {
|
2019-04-17 08:21:30 +00:00
|
|
|
const colorNumber = (hashCode(userId) % 8) + 1;
|
|
|
|
return `mx_Username_color${colorNumber}`;
|
|
|
|
}
|
2019-05-20 14:16:12 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructs a written English string representing `items`, with an optional
|
|
|
|
* limit on the number of items included in the result. If specified and if the
|
|
|
|
* length of `items` is greater than the limit, the string "and n others" will
|
|
|
|
* be appended onto the result. If `items` is empty, returns the empty string.
|
|
|
|
* If there is only one item, return it.
|
|
|
|
* @param {string[]} items the items to construct a string from.
|
|
|
|
* @param {number?} itemLimit the number by which to limit the list.
|
|
|
|
* @returns {string} a string constructed by joining `items` with a comma
|
|
|
|
* between each item, but with the last item appended as " and [lastItem]".
|
|
|
|
*/
|
2021-09-17 10:36:22 +00:00
|
|
|
export function formatCommaSeparatedList(items: string[], itemLimit?: number): string;
|
2023-02-13 11:39:16 +00:00
|
|
|
export function formatCommaSeparatedList(items: ReactElement[], itemLimit?: number): ReactElement;
|
|
|
|
export function formatCommaSeparatedList(items: ReactNode[], itemLimit?: number): ReactNode;
|
|
|
|
export function formatCommaSeparatedList(items: ReactNode[], itemLimit?: number): ReactNode {
|
2022-12-12 11:24:14 +00:00
|
|
|
const remaining = itemLimit === undefined ? 0 : Math.max(items.length - itemLimit, 0);
|
2019-05-20 14:16:12 +00:00
|
|
|
if (items.length === 0) {
|
|
|
|
return "";
|
|
|
|
} else if (items.length === 1) {
|
|
|
|
return items[0];
|
|
|
|
} else {
|
2021-09-17 10:36:22 +00:00
|
|
|
let lastItem;
|
|
|
|
if (remaining > 0) {
|
|
|
|
items = items.slice(0, itemLimit);
|
|
|
|
} else {
|
|
|
|
lastItem = items.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
let joinedItems;
|
2022-12-12 11:24:14 +00:00
|
|
|
if (items.every((e) => typeof e === "string")) {
|
2021-09-17 10:36:22 +00:00
|
|
|
joinedItems = items.join(", ");
|
|
|
|
} else {
|
|
|
|
joinedItems = jsxJoin(items, ", ");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (remaining > 0) {
|
2022-01-10 12:57:20 +00:00
|
|
|
return _t("%(items)s and %(count)s others", { items: joinedItems, count: remaining });
|
2021-09-17 10:36:22 +00:00
|
|
|
} else {
|
|
|
|
return _t("%(items)s and %(lastItem)s", { items: joinedItems, lastItem });
|
|
|
|
}
|
2019-05-20 14:16:12 +00:00
|
|
|
}
|
|
|
|
}
|