606 lines
16 KiB
TypeScript
606 lines
16 KiB
TypeScript
import type { NameValue, TypeAndName, TypeAndDisplayName} from '@/core/base.ts';
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type
|
|
export function isFunction(val: unknown): val is Function {
|
|
return typeof(val) === 'function';
|
|
}
|
|
|
|
export function isDefined<T>(val: T | null | undefined): val is T {
|
|
return val !== null && typeof(val) !== 'undefined';
|
|
}
|
|
|
|
export function isObject(val: unknown): val is object {
|
|
return val !== null && typeof(val) === 'object' && !isArray(val);
|
|
}
|
|
|
|
export function isArray<T>(val: unknown): val is T[] {
|
|
if (isFunction(Array.isArray)) {
|
|
return Array.isArray(val);
|
|
}
|
|
|
|
return Object.prototype.toString.call(val) === '[object Array]';
|
|
}
|
|
|
|
export function isString(val: unknown): val is string {
|
|
return typeof(val) === 'string';
|
|
}
|
|
|
|
export function isNumber(val: unknown): val is number {
|
|
return typeof(val) === 'number';
|
|
}
|
|
|
|
export function isInteger(val: unknown): val is number {
|
|
return Number.isInteger(val);
|
|
}
|
|
|
|
export function isBoolean(val: unknown): val is boolean {
|
|
return typeof(val) === 'boolean';
|
|
}
|
|
|
|
export function isYearMonth(val: unknown): val is string {
|
|
if (!isString(val)) {
|
|
return false;
|
|
}
|
|
|
|
const items = val.split('-');
|
|
|
|
if (items.length !== 2) {
|
|
return false;
|
|
}
|
|
|
|
return !!parseInt(items[0]) && !!parseInt(items[1]);
|
|
}
|
|
|
|
export function isEquals(obj1: unknown, obj2: unknown): boolean {
|
|
if (obj1 === obj2) {
|
|
return true;
|
|
}
|
|
|
|
if (isArray(obj1) && isArray(obj2)) {
|
|
const arr1 = obj1;
|
|
const arr2 = obj2;
|
|
|
|
if (arr1.length !== arr2.length) {
|
|
return false;
|
|
}
|
|
|
|
for (let i = 0; i < arr1.length; i++) {
|
|
if (!isEquals(arr1[i], arr2[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
} else if (isObject(obj1) && isObject(obj2)) {
|
|
const keys1 = Object.keys(obj1);
|
|
const keys2 = Object.keys(obj2);
|
|
|
|
if (keys1.length !== keys2.length) {
|
|
return false;
|
|
}
|
|
|
|
const keyExistsMap2 = new Map<string, boolean>();
|
|
|
|
for (let i = 0; i < keys2.length; i++) {
|
|
const key = keys2[i];
|
|
|
|
keyExistsMap2.set(key, true);
|
|
}
|
|
|
|
for (let i = 0; i < keys1.length; i++) {
|
|
const key = keys1[i];
|
|
|
|
if (!keyExistsMap2.get(key)) {
|
|
return false;
|
|
}
|
|
|
|
if (!isEquals((obj1 as Record<string, unknown>)[key], (obj2 as Record<string, unknown>)[key])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
} else {
|
|
return obj1 === obj2;
|
|
}
|
|
}
|
|
|
|
export function isYearMonthEquals(val1: unknown, val2: unknown): boolean {
|
|
if (typeof(val1) !== 'string' || typeof(val2) !== 'string') {
|
|
return false;
|
|
}
|
|
|
|
const items1 = val1.split('-');
|
|
const items2 = val2.split('-');
|
|
|
|
if (items1.length !== 2 || items2.length !== 2) {
|
|
return false;
|
|
}
|
|
|
|
return (!!parseInt(items1[0]) && !!parseInt(items1[1])) && (parseInt(items1[0]) === parseInt(items2[0])) && (parseInt(items1[1]) === parseInt(items2[1]));
|
|
}
|
|
|
|
export function isArray1SubsetOfArray2<T>(array1: T[], array2: T[]): boolean {
|
|
if (array1.length > array2.length) {
|
|
return false;
|
|
}
|
|
|
|
const array2ValuesMap: Map<T, boolean> = new Map<T, boolean>();
|
|
|
|
for (let i = 0; i < array2.length; i++) {
|
|
array2ValuesMap.set(array2[i], true);
|
|
}
|
|
|
|
for (let i = 0; i < array1.length; i++) {
|
|
if (!array2ValuesMap.get(array1[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
export function isObjectEmpty(obj: object): boolean {
|
|
if (!obj) {
|
|
return true;
|
|
}
|
|
|
|
for (const field in obj) {
|
|
if (!Object.prototype.hasOwnProperty.call(obj, field)) {
|
|
continue;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
export function ofObject<T>(object: T): T {
|
|
return object;
|
|
}
|
|
|
|
export function getNumberValue(value: unknown, defaultValue: number): number {
|
|
if (isString(value)) {
|
|
return parseInt(value, 10);
|
|
} else if (isNumber(value)) {
|
|
return value;
|
|
} else {
|
|
return defaultValue;
|
|
}
|
|
}
|
|
|
|
export function sortNumbersArray(array: number[]): number[] {
|
|
return array.sort(function (num1, num2) {
|
|
return num1 - num2;
|
|
});
|
|
}
|
|
|
|
export function getObjectOwnFieldCount(object: object): number {
|
|
let count = 0;
|
|
|
|
if (!object || !isObject(object)) {
|
|
return count;
|
|
}
|
|
|
|
for (const field in object) {
|
|
if (!Object.prototype.hasOwnProperty.call(object, field)) {
|
|
continue;
|
|
}
|
|
|
|
count++;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
export function replaceAll(value: string, originalValue: string, targetValue: string): string {
|
|
// Escape special characters in originalValue to safely use it in a regex pattern.
|
|
// This ensures that characters like . (dot), * (asterisk), +, ?, etc. are treated literally,
|
|
// rather than as special regex symbols.
|
|
const escapedOriginalValue = originalValue.replace(/([.*+?^=!:${}()|\-/\\])/g, '\\$1');
|
|
|
|
return value.replace(new RegExp(escapedOriginalValue, 'g'), targetValue);
|
|
}
|
|
|
|
export function removeAll(value: string, originalValue: string): string {
|
|
return replaceAll(value, originalValue, '');
|
|
}
|
|
|
|
export function limitText(value: string, maxLength: number): string {
|
|
let length = 0;
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
const c = value.charCodeAt(i);
|
|
|
|
if ((c >= 0x0001 && c <= 0x007e) || (0xff60 <= c && c <= 0xff9f)) {
|
|
length++;
|
|
} else {
|
|
length += 2;
|
|
}
|
|
}
|
|
|
|
if (length <= maxLength || maxLength <= 3) {
|
|
return value;
|
|
}
|
|
|
|
return value.substring(0, maxLength - 3) + '...';
|
|
}
|
|
|
|
export function base64encode(arrayBuffer: ArrayBuffer): string {
|
|
return btoa(String.fromCharCode.apply(null, Array.from(new Uint8Array(arrayBuffer))));
|
|
}
|
|
|
|
export function base64decode(str: string): string {
|
|
if (!str) {
|
|
return '';
|
|
}
|
|
|
|
return atob(str);
|
|
}
|
|
|
|
export function arrayBufferToString(arrayBuffer: ArrayBuffer): string {
|
|
return String.fromCharCode.apply(null, Array.from(new Uint8Array(arrayBuffer)));
|
|
}
|
|
|
|
export function stringToArrayBuffer(str: string): ArrayBuffer {
|
|
return Uint8Array.from(str, c => c.charCodeAt(0)).buffer;
|
|
}
|
|
|
|
export function getFirstVisibleItem<T>(items: Record<string, T>[] | Record<string, Record<string, T>>, hiddenField: string): Record<string, T> | null {
|
|
if (isArray(items) && items.length > 0) {
|
|
const arr = items as Record<string, T>[];
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
if (hiddenField && arr[i][hiddenField]) {
|
|
continue;
|
|
}
|
|
|
|
return arr[i];
|
|
}
|
|
} else if (isObject(items)) {
|
|
const obj = items as Record<string, Record<string, T>>;
|
|
|
|
for (const field in obj) {
|
|
if (!Object.prototype.hasOwnProperty.call(obj, field)) {
|
|
continue;
|
|
}
|
|
|
|
if (hiddenField && obj[field][hiddenField]) {
|
|
continue;
|
|
}
|
|
|
|
return obj[field];
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function getItemByKeyValue<T>(src: Record<string, T>[] | Record<string, Record<string, T>>, value: T, keyField: string): Record<string, T> | null {
|
|
if (isArray(src)) {
|
|
const arr = src as Record<string, T>[];
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
const item = arr[i];
|
|
|
|
if (item[keyField] === value) {
|
|
return item;
|
|
}
|
|
}
|
|
} else if (isObject(src)) {
|
|
const obj = src as Record<string, Record<string, T>>;
|
|
|
|
for (const field in obj) {
|
|
if (!Object.prototype.hasOwnProperty.call(src, field)) {
|
|
continue;
|
|
}
|
|
|
|
const item = obj[field];
|
|
|
|
if (item[keyField] === value) {
|
|
return item;
|
|
}
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function findNameByValue(items: NameValue[], value: string): string | null {
|
|
for (const item of items) {
|
|
if (item.value === value) {
|
|
return item.name;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function findNameByType(items: TypeAndName[], type: number): string | null {
|
|
for (const item of items) {
|
|
if (item.type === type) {
|
|
return item.name;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function findDisplayNameByType(items: TypeAndDisplayName[], type: number): string | null {
|
|
for (const item of items) {
|
|
if (item.type === type) {
|
|
return item.displayName;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function getNameByKeyValue<V, N>(src: Record<string, N | V>[] | Record<string, Record<string, N | V>>, value: V, keyField: string | null, nameField: string, defaultName?: N): N | undefined {
|
|
if (isArray(src)) {
|
|
const arr = src as Record<string, N | V>[];
|
|
|
|
if (keyField) {
|
|
for (let i = 0; i < arr.length; i++) {
|
|
const option = arr[i];
|
|
|
|
if (option[keyField] === value) {
|
|
return option[nameField] as N;
|
|
}
|
|
}
|
|
} else if (isNumber(value)) {
|
|
const index = value;
|
|
|
|
if (arr[index]) {
|
|
const option = arr[index];
|
|
|
|
return option[nameField] as N;
|
|
}
|
|
}
|
|
} else if (isObject(src)) {
|
|
const obj = src as Record<string, Record<string, N | V>>;
|
|
|
|
if (keyField) {
|
|
for (const key in obj) {
|
|
if (!Object.prototype.hasOwnProperty.call(src, key)) {
|
|
continue;
|
|
}
|
|
|
|
const option = obj[key];
|
|
|
|
if (option[keyField] === value) {
|
|
return option[nameField] as N;
|
|
}
|
|
}
|
|
} else if (isString(value)) {
|
|
const key = value;
|
|
|
|
if (obj[key]) {
|
|
const option = obj[key];
|
|
|
|
return option[nameField] as N;
|
|
}
|
|
}
|
|
}
|
|
|
|
return defaultName;
|
|
}
|
|
|
|
export function arrayContainsFieldValue<T>(array: Record<string, T>[], fieldName: string, value: T): boolean {
|
|
if (!value || !array || !array.length) {
|
|
return false;
|
|
}
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
if (array[i][fieldName] === value) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export function objectFieldToArrayItem(object: object): string[] {
|
|
const ret: string[] = [];
|
|
|
|
for (const field in object) {
|
|
if (!Object.prototype.hasOwnProperty.call(object, field)) {
|
|
continue;
|
|
}
|
|
|
|
ret.push(field);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
export function arrayItemToObjectField<T>(array: string[], value: T): Record<string, T> {
|
|
const ret: Record<string, T> = {};
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
ret[array[i]] = value;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
export function splitItemsToMap(str: string | undefined | null, separator: string): Record<string, boolean> {
|
|
const ret: Record<string, boolean> = {};
|
|
|
|
if (!str) {
|
|
return ret;
|
|
}
|
|
|
|
const items = str.split(separator);
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
if (items[i]) {
|
|
ret[items[i]] = true;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
export function countSplitItems(str: string | undefined | null, separator: string): number {
|
|
if (!str) {
|
|
return 0;
|
|
}
|
|
|
|
const items = str.split(separator);
|
|
let count = 0;
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
if (items[i]) {
|
|
count++;
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
export function categorizedArrayToPlainArray<T>(object: Record<string, T[]>): T[] {
|
|
const ret: T[] = [];
|
|
|
|
for (const field in object) {
|
|
if (!Object.prototype.hasOwnProperty.call(object, field)) {
|
|
continue;
|
|
}
|
|
|
|
const array = object[field];
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
ret.push(array[i]);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
export function selectAll(filterItemIds: Record<string, boolean>, allItemsMap: { [key: string]: { id: string } }): void {
|
|
for (const itemId in filterItemIds) {
|
|
if (!Object.prototype.hasOwnProperty.call(filterItemIds, itemId)) {
|
|
continue;
|
|
}
|
|
|
|
const item = allItemsMap[itemId];
|
|
|
|
if (item) {
|
|
filterItemIds[item.id] = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
export function selectNone(filterItemIds: Record<string, boolean>, allItemsMap: { [key: string]: { id: string } }): void {
|
|
for (const itemId in filterItemIds) {
|
|
if (!Object.prototype.hasOwnProperty.call(filterItemIds, itemId)) {
|
|
continue;
|
|
}
|
|
|
|
const item = allItemsMap[itemId];
|
|
|
|
if (item) {
|
|
filterItemIds[item.id] = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
export function selectInvert(filterItemIds: Record<string, boolean>, allItemsMap: { [key: string]: { id: string } }): void {
|
|
for (const itemId in filterItemIds) {
|
|
if (!Object.prototype.hasOwnProperty.call(filterItemIds, itemId)) {
|
|
continue;
|
|
}
|
|
|
|
const item = allItemsMap[itemId];
|
|
|
|
if (item) {
|
|
filterItemIds[item.id] = !filterItemIds[item.id];
|
|
}
|
|
}
|
|
}
|
|
|
|
export function isPrimaryItemHasSecondaryValue(primaryItem: Record<string, Record<string, unknown>[]>, primarySubItemsField: string, secondaryValueField: string | undefined, secondaryHiddenField: string | undefined, secondaryValue: unknown): boolean {
|
|
for (let i = 0; i < primaryItem[primarySubItemsField].length; i++) {
|
|
const secondaryItem = primaryItem[primarySubItemsField][i];
|
|
|
|
if (secondaryHiddenField && secondaryItem[secondaryHiddenField]) {
|
|
continue;
|
|
}
|
|
|
|
if (secondaryValueField && secondaryItem[secondaryValueField] === secondaryValue) {
|
|
return true;
|
|
} else if (!secondaryValueField && secondaryItem === secondaryValue) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export function getPrimaryValueBySecondaryValue<T>(items: Record<string, Record<string, T>[]>[] | Record<string, Record<string, Record<string, T>[]>>, primarySubItemsField: string | undefined, primaryValueField: string | undefined, primaryHiddenField: string | undefined, secondaryValueField: string | undefined, secondaryHiddenField: string | undefined, secondaryValue: T): Record<string, T>[] | Record<string, Record<string, T>[]> | null {
|
|
if (primarySubItemsField) {
|
|
if (isArray(items)) {
|
|
const arr = items as Record<string, Record<string, T>[]>[];
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
const primaryItem = arr[i];
|
|
|
|
if (primaryHiddenField && primaryItem[primaryHiddenField]) {
|
|
continue;
|
|
}
|
|
|
|
if (isPrimaryItemHasSecondaryValue(primaryItem, primarySubItemsField, secondaryValueField, secondaryHiddenField, secondaryValue)) {
|
|
if (primaryValueField) {
|
|
return primaryItem[primaryValueField];
|
|
} else {
|
|
return primaryItem;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
const obj = items as Record<string, Record<string, Record<string, T>[]>>;
|
|
|
|
for (const field in obj) {
|
|
if (!Object.prototype.hasOwnProperty.call(obj, field)) {
|
|
continue;
|
|
}
|
|
|
|
const primaryItem = obj[field];
|
|
|
|
if (primaryHiddenField && primaryItem[primaryHiddenField]) {
|
|
continue;
|
|
}
|
|
|
|
if (isPrimaryItemHasSecondaryValue(primaryItem, primarySubItemsField, secondaryValueField, secondaryHiddenField, secondaryValue)) {
|
|
if (primaryValueField) {
|
|
return primaryItem[primaryValueField];
|
|
} else {
|
|
return primaryItem;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function arrangeArrayWithNewStartIndex<T>(array: T[], startIndex: number): T[] {
|
|
if (startIndex <= 0 || startIndex >= array.length) {
|
|
return array;
|
|
}
|
|
|
|
const newArray: T[] = [];
|
|
|
|
for (let i = startIndex; i < array.length; i++) {
|
|
newArray.push(array[i]);
|
|
}
|
|
|
|
for (let i = 0; i < startIndex; i++) {
|
|
newArray.push(array[i]);
|
|
}
|
|
|
|
return newArray;
|
|
}
|