2022-07-07 05:19:05 +08:00
|
|
|
|
import { DateTime } from "luxon";
|
|
|
|
|
|
|
|
|
|
export default class CommonHelper {
|
|
|
|
|
/**
|
|
|
|
|
* Checks whether value is plain object.
|
|
|
|
|
*
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static isObject(value) {
|
|
|
|
|
return value !== null && typeof value === "object" && value.constructor === Object;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Checks whether a value is empty. The following values are considered as empty:
|
|
|
|
|
* - null
|
|
|
|
|
* - undefined
|
|
|
|
|
* - empty string
|
|
|
|
|
* - empty array
|
|
|
|
|
* - empty object
|
|
|
|
|
* - zero uuid, time and dates
|
|
|
|
|
*
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static isEmpty(value) {
|
|
|
|
|
return (
|
|
|
|
|
(value === "") ||
|
|
|
|
|
(value === null) ||
|
|
|
|
|
(value === "00000000-0000-0000-0000-000000000000") || // zero uuid
|
2022-10-30 16:28:14 +08:00
|
|
|
|
(value === "0001-01-01 00:00:00.000Z") || // zero datetime
|
2022-07-07 05:19:05 +08:00
|
|
|
|
(value === "0001-01-01") || // zero date
|
|
|
|
|
(typeof value === "undefined") ||
|
|
|
|
|
(Array.isArray(value) && value.length === 0) ||
|
|
|
|
|
(CommonHelper.isObject(value) && Object.keys(value).length === 0)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Checks whether the provided dom element is a form field (input, textarea, select).
|
|
|
|
|
*
|
|
|
|
|
* @param {Node} element
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static isInput(element) {
|
|
|
|
|
let tagName = element && element.tagName ? element.tagName.toLowerCase() : "";
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
tagName === "input" ||
|
|
|
|
|
tagName === "select" ||
|
|
|
|
|
tagName === "textarea" ||
|
|
|
|
|
element.isContentEditable
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Checks if an element is a common focusable one.
|
|
|
|
|
*
|
|
|
|
|
* @param {Node} element
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static isFocusable(element) {
|
|
|
|
|
let tagName = element && element.tagName ? element.tagName.toLowerCase() : "";
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
CommonHelper.isInput(element) ||
|
|
|
|
|
tagName === "button" ||
|
|
|
|
|
tagName === "a" ||
|
|
|
|
|
tagName === "details" ||
|
|
|
|
|
element.tabIndex >= 0
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Check if `obj` has at least one none empty property.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} obj
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static hasNonEmptyProps(obj) {
|
|
|
|
|
for (let i in obj) {
|
|
|
|
|
if (!CommonHelper.isEmpty(obj[i])) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Normalizes and returns arr as a valid array instance (if not already).
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} arr
|
2022-07-18 05:16:09 +08:00
|
|
|
|
* @param {Boolean} [allowEmpty]
|
2022-07-07 05:19:05 +08:00
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
2022-07-18 05:16:09 +08:00
|
|
|
|
static toArray(arr, allowEmpty = false) {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
if (Array.isArray(arr)) {
|
|
|
|
|
return arr;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-18 05:16:09 +08:00
|
|
|
|
return (allowEmpty || !CommonHelper.isEmpty(arr)) && typeof arr !== "undefined" ? [arr] : [];
|
2022-07-07 05:19:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Loosely checks if value exists in an array.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} arr
|
|
|
|
|
* @param {String} value
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static inArray(arr, value) {
|
2022-08-10 18:22:27 +08:00
|
|
|
|
arr = Array.isArray(arr) ? arr : [];
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
for (let i = arr.length - 1; i >= 0; i--) {
|
|
|
|
|
if (arr[i] == value) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Removes single element from array by loosely comparying values.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} arr
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
*/
|
|
|
|
|
static removeByValue(arr, value) {
|
2022-08-10 18:22:27 +08:00
|
|
|
|
arr = Array.isArray(arr) ? arr : [];
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
for (let i = arr.length - 1; i >= 0; i--) {
|
|
|
|
|
if (arr[i] == value) {
|
|
|
|
|
arr.splice(i, 1);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Adds `value` in `arr` only if it's not added already.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} arr
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
*/
|
|
|
|
|
static pushUnique(arr, value) {
|
|
|
|
|
if (!CommonHelper.inArray(arr, value)) {
|
|
|
|
|
arr.push(value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns single element from objects array by matching its key value.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} objectsArr
|
|
|
|
|
* @param {Mixed} key
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static findByKey(objectsArr, key, value) {
|
2022-08-10 18:22:27 +08:00
|
|
|
|
objectsArr = Array.isArray(objectsArr) ? objectsArr : [];
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
for (let i in objectsArr) {
|
|
|
|
|
if (objectsArr[i][key] == value) {
|
|
|
|
|
return objectsArr[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Group objects array by a specific key.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} objectsArr
|
|
|
|
|
* @param {String} key
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static groupByKey(objectsArr, key) {
|
2022-08-10 18:22:27 +08:00
|
|
|
|
objectsArr = Array.isArray(objectsArr) ? objectsArr : [];
|
|
|
|
|
|
|
|
|
|
const result = {};
|
2022-07-07 05:19:05 +08:00
|
|
|
|
|
|
|
|
|
for (let i in objectsArr) {
|
|
|
|
|
result[objectsArr[i][key]] = result[objectsArr[i][key]] || [];
|
|
|
|
|
|
|
|
|
|
result[objectsArr[i][key]].push(objectsArr[i]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Removes single element from objects array by matching an item"s property value.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} objectsArr
|
|
|
|
|
* @param {String} key
|
|
|
|
|
* @param {Mixed} value
|
|
|
|
|
*/
|
|
|
|
|
static removeByKey(objectsArr, key, value) {
|
|
|
|
|
for (let i in objectsArr) {
|
|
|
|
|
if (objectsArr[i][key] == value) {
|
|
|
|
|
objectsArr.splice(i, 1);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Adds or replace an object array element by comparing its key value.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} objectsArr
|
|
|
|
|
* @param {Object} item
|
|
|
|
|
* @param {Mixed} [key]
|
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
|
|
|
|
static pushOrReplaceByKey(objectsArr, item, key = "id") {
|
|
|
|
|
for (let i = objectsArr.length - 1; i >= 0; i--) {
|
|
|
|
|
if (objectsArr[i][key] == item[key]) {
|
|
|
|
|
objectsArr[i] = item; // replace
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
objectsArr.push(item);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Filters and returns a new objects array with duplicated elements removed.
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} objectsArr
|
|
|
|
|
* @param {String} key
|
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
|
|
|
|
static filterDuplicatesByKey(objectsArr, key = "id") {
|
2022-08-10 18:22:27 +08:00
|
|
|
|
objectsArr = Array.isArray(objectsArr) ? objectsArr : [];
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
const uniqueMap = {};
|
2022-08-10 18:22:27 +08:00
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
for (const item of objectsArr) {
|
|
|
|
|
uniqueMap[item[key]] = item;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Object.values(uniqueMap)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Filters and returns a new object with removed redacted props.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} obj
|
|
|
|
|
* @param {String} [mask] Default to '******'
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static filterRedactedProps(obj, mask = "******") {
|
|
|
|
|
const result = JSON.parse(JSON.stringify(obj || {}));
|
|
|
|
|
|
|
|
|
|
for (let prop in result) {
|
|
|
|
|
if (typeof result[prop] === 'object' && result[prop] !== null) {
|
|
|
|
|
result[prop] = CommonHelper.filterRedactedProps(result[prop], mask)
|
|
|
|
|
} else if (result[prop] === mask) {
|
|
|
|
|
delete result[prop];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Safely access nested object/array key with dot-notation.
|
|
|
|
|
*
|
|
|
|
|
* @example
|
|
|
|
|
* ```javascript
|
|
|
|
|
* var myObj = {a: {b: {c: 3}}}
|
|
|
|
|
* this.getNestedVal(myObj, "a.b.c"); // returns 3
|
|
|
|
|
* this.getNestedVal(myObj, "a.b.c.d"); // returns null
|
|
|
|
|
* this.getNestedVal(myObj, "a.b.c.d", -1); // returns -1
|
|
|
|
|
* ```
|
|
|
|
|
*
|
|
|
|
|
* @param {Object|Array} data
|
|
|
|
|
* @param {string} path
|
|
|
|
|
* @param {Mixed} [defaultVal]
|
|
|
|
|
* @param {String} [delimiter]
|
|
|
|
|
* @return {Mixed}
|
|
|
|
|
*/
|
|
|
|
|
static getNestedVal(data, path, defaultVal = null, delimiter = ".") {
|
|
|
|
|
let result = data || {};
|
2022-08-15 00:30:45 +08:00
|
|
|
|
let parts = (path || '').split(delimiter);
|
2022-07-07 05:19:05 +08:00
|
|
|
|
|
|
|
|
|
for (const part of parts) {
|
|
|
|
|
if (
|
|
|
|
|
(!CommonHelper.isObject(result) && !Array.isArray(result)) ||
|
|
|
|
|
typeof result[part] === "undefined"
|
|
|
|
|
) {
|
|
|
|
|
return defaultVal;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = result[part];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Sets a new value to an object (or array) by its key path.
|
|
|
|
|
*
|
|
|
|
|
* @example
|
|
|
|
|
* ```javascript
|
|
|
|
|
* this.setByPath({}, "a.b.c", 1); // results in {a: b: {c: 1}}
|
|
|
|
|
* this.setByPath({a: {b: {c: 3}}}, "a.b", 4); // results in {a: {b: 4}}
|
|
|
|
|
* ```
|
|
|
|
|
*
|
|
|
|
|
* @param {Array|Object} data
|
|
|
|
|
* @param {string} path
|
|
|
|
|
* @param {String} delimiter
|
|
|
|
|
*/
|
|
|
|
|
static setByPath(data, path, newValue, delimiter = ".") {
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (data === null || typeof data !== 'object') {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
console.warn("setByPath: data not an object or array.");
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let result = data;
|
|
|
|
|
let parts = path.split(delimiter);
|
|
|
|
|
let lastPart = parts.pop();
|
|
|
|
|
|
|
|
|
|
for (const part of parts) {
|
|
|
|
|
if (
|
|
|
|
|
(!CommonHelper.isObject(result) && !Array.isArray(result)) ||
|
|
|
|
|
(!CommonHelper.isObject(result[part]) && !Array.isArray(result[part]))
|
|
|
|
|
) {
|
|
|
|
|
result[part] = {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = result[part];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result[lastPart] = newValue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Recursively delete element from an object (or array) by its key path.
|
|
|
|
|
* Empty array or object elements from the parents chain will be also removed.
|
|
|
|
|
*
|
|
|
|
|
* @example
|
|
|
|
|
* ```javascript
|
|
|
|
|
* this.deleteByPath({a: {b: {c: 3}}}, "a.b.c"); // results in {}
|
|
|
|
|
* this.deleteByPath({a: {b: {c: 3, d: 4}}}, "a.b.c"); // results in {a: {b: {d: 4}}}
|
|
|
|
|
* ```
|
|
|
|
|
*
|
|
|
|
|
* @param {Array|Object} data
|
|
|
|
|
* @param {string} path
|
|
|
|
|
* @param {String} delimiter
|
|
|
|
|
*/
|
|
|
|
|
static deleteByPath(data, path, delimiter = ".") {
|
|
|
|
|
let result = data || {};
|
2022-08-15 00:30:45 +08:00
|
|
|
|
let parts = (path || '').split(delimiter);
|
2022-07-07 05:19:05 +08:00
|
|
|
|
let lastPart = parts.pop();
|
|
|
|
|
|
|
|
|
|
for (const part of parts) {
|
|
|
|
|
if (
|
|
|
|
|
(!CommonHelper.isObject(result) && !Array.isArray(result)) ||
|
|
|
|
|
(!CommonHelper.isObject(result[part]) && !Array.isArray(result[part]))
|
|
|
|
|
) {
|
|
|
|
|
result[part] = {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = result[part];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Array.isArray(result)) {
|
|
|
|
|
result.splice(lastPart, 1);
|
|
|
|
|
} else if (CommonHelper.isObject(result)) {
|
|
|
|
|
delete (result[lastPart]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// cleanup the parents chain
|
|
|
|
|
if (
|
|
|
|
|
parts.length > 0 &&
|
|
|
|
|
(
|
|
|
|
|
(Array.isArray(result) && !result.length) ||
|
|
|
|
|
(CommonHelper.isObject(result) && !Object.keys(result).length)
|
|
|
|
|
) &&
|
|
|
|
|
(
|
|
|
|
|
(Array.isArray(data) && data.length > 0) ||
|
|
|
|
|
(CommonHelper.isObject(data) && Object.keys(data).length > 0)
|
|
|
|
|
)
|
|
|
|
|
) {
|
|
|
|
|
CommonHelper.deleteByPath(data, parts.join(delimiter), delimiter);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Generates random string (suitable for elements id and keys).
|
|
|
|
|
*
|
|
|
|
|
* @param {Number} [length] Results string length (default 10)
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static randomString(length) {
|
|
|
|
|
length = length || 10;
|
|
|
|
|
|
|
|
|
|
let result = "";
|
|
|
|
|
let alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
|
|
|
|
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
|
result += alphabet.charAt(Math.floor(Math.random() * alphabet.length));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Converts and normalizes string into a sentence.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} str
|
|
|
|
|
* @param {Boolean} [stopCheck]
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static sentenize(str, stopCheck = true) {
|
|
|
|
|
if (typeof str !== "string") {
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
str = str.trim().split("_").join(" ");
|
|
|
|
|
if (str === "") {
|
|
|
|
|
return str;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
str = str[0].toUpperCase() + str.substring(1);
|
|
|
|
|
|
|
|
|
|
if (stopCheck) {
|
|
|
|
|
let lastChar = str[str.length - 1];
|
|
|
|
|
if (lastChar !== "." && lastChar !== "?" && lastChar !== "!") {
|
|
|
|
|
str += ".";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return str
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Normalizes and converts the provided string to a slug.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} str
|
|
|
|
|
* @param {String} [delimiter]
|
|
|
|
|
* @param {Array} [preserved]
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static slugify(str, delimiter = '_', preserved = ['.', '=', '-']) {
|
|
|
|
|
if (str === '') {
|
|
|
|
|
return '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// special characters
|
|
|
|
|
const specialCharsMap = {
|
|
|
|
|
'a': /а|à|á|å|â/gi,
|
|
|
|
|
'b': /б/gi,
|
|
|
|
|
'c': /ц|ç/gi,
|
|
|
|
|
'd': /д/gi,
|
|
|
|
|
'e': /е|è|é|ê|ẽ|ë/gi,
|
|
|
|
|
'f': /ф/gi,
|
|
|
|
|
'g': /г/gi,
|
|
|
|
|
'h': /х/gi,
|
|
|
|
|
'i': /й|и|ì|í|î/gi,
|
|
|
|
|
'j': /ж/gi,
|
|
|
|
|
'k': /к/gi,
|
|
|
|
|
'l': /л/gi,
|
|
|
|
|
'm': /м/gi,
|
|
|
|
|
'n': /н|ñ/gi,
|
|
|
|
|
'o': /о|ò|ó|ô|ø/gi,
|
|
|
|
|
'p': /п/gi,
|
|
|
|
|
'q': /я/gi,
|
|
|
|
|
'r': /р/gi,
|
|
|
|
|
's': /с/gi,
|
|
|
|
|
't': /т/gi,
|
|
|
|
|
'u': /ю|ù|ú|ů|û/gi,
|
|
|
|
|
'v': /в/gi,
|
|
|
|
|
'w': /в/gi,
|
|
|
|
|
'x': /ь/gi,
|
|
|
|
|
'y': /ъ/gi,
|
|
|
|
|
'z': /з/gi,
|
|
|
|
|
'ae': /ä|æ/gi,
|
|
|
|
|
'oe': /ö/gi,
|
|
|
|
|
'ue': /ü/gi,
|
|
|
|
|
'Ae': /Ä/gi,
|
|
|
|
|
'Ue': /Ü/gi,
|
|
|
|
|
'Oe': /Ö/gi,
|
|
|
|
|
'ss': /ß/gi,
|
|
|
|
|
'and': /&/gi
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// replace special characters
|
|
|
|
|
for (let k in specialCharsMap) {
|
|
|
|
|
str = str.replace(specialCharsMap[k], k);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const slug = str
|
|
|
|
|
.replace(new RegExp('[' + preserved.join('') + ']', 'g'), ' ') // replace preserved characters with spaces
|
|
|
|
|
.replace(/[^\w\ ]/gi, '') // replaces all non-alphanumeric with empty string
|
|
|
|
|
.replace(/\s+/g, delimiter); // collapse whitespaces and replace with `delimiter`
|
|
|
|
|
|
|
|
|
|
return slug.charAt(0).toLowerCase() + slug.slice(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns `str` with escaped regexp characters, making it safe to
|
|
|
|
|
* embed in regexp as a whole literal expression.
|
|
|
|
|
*
|
|
|
|
|
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions#escaping
|
|
|
|
|
* @param {String} str
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static escapeRegExp(str) {
|
|
|
|
|
return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Splits `str` and returns its non empty parts as an array.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} str
|
|
|
|
|
* @param {String} [separator]
|
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
|
|
|
|
static splitNonEmpty(str, separator = ",") {
|
|
|
|
|
const items = (str || "").split(separator);
|
|
|
|
|
const result = [];
|
|
|
|
|
|
|
|
|
|
for (let item of items) {
|
|
|
|
|
item = item.trim();
|
|
|
|
|
if (!CommonHelper.isEmpty(item)) {
|
|
|
|
|
result.push(item);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a concatenated `items` string.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} items
|
|
|
|
|
* @param {String} [separator]
|
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
|
|
|
|
static joinNonEmpty(items, separator = ", ") {
|
|
|
|
|
const result = [];
|
|
|
|
|
|
|
|
|
|
for (let item of items) {
|
|
|
|
|
item = typeof item === "string" ? item.trim() : "";
|
|
|
|
|
if (!CommonHelper.isEmpty(item)) {
|
|
|
|
|
result.push(item);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result.join(separator);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a DateTime instance from a date object/string.
|
|
|
|
|
*
|
|
|
|
|
* @param {String|Date} date
|
|
|
|
|
* @return {DateTime}
|
|
|
|
|
*/
|
|
|
|
|
static getDateTime(date) {
|
|
|
|
|
if (typeof date === 'string') {
|
2022-10-30 16:28:14 +08:00
|
|
|
|
const formats = {
|
|
|
|
|
19: "yyyy-MM-dd HH:mm:ss",
|
|
|
|
|
23: "yyyy-MM-dd HH:mm:ss.SSS",
|
|
|
|
|
20: "yyyy-MM-dd HH:mm:ssZ",
|
|
|
|
|
24: "yyyy-MM-dd HH:mm:ss.SSSZ",
|
|
|
|
|
}
|
|
|
|
|
const format = formats[date.length] || formats[19];
|
2022-07-07 05:19:05 +08:00
|
|
|
|
return DateTime.fromFormat(date, format, { zone: 'UTC' });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return DateTime.fromJSDate(date);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns formatted datetime string in the UTC timezone.
|
|
|
|
|
*
|
|
|
|
|
* @param {String|Date} date
|
|
|
|
|
* @param {String} [format] The result format (see https://moment.github.io/luxon/#/parsing?id=table-of-tokens)
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static formatToUTCDate(date, format = 'yyyy-MM-dd HH:mm:ss') {
|
|
|
|
|
return CommonHelper.getDateTime(date).toUTC().toFormat(format);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns formatted datetime string in the local timezone.
|
|
|
|
|
*
|
|
|
|
|
* @param {String|Date} date
|
|
|
|
|
* @param {String} [format] The result format (see https://moment.github.io/luxon/#/parsing?id=table-of-tokens)
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static formatToLocalDate(date, format = 'yyyy-MM-dd HH:mm:ss') {
|
|
|
|
|
return CommonHelper.getDateTime(date).toLocal().toFormat(format);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Copies text to the user clipboard.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} text
|
|
|
|
|
* @return {Promise}
|
|
|
|
|
*/
|
|
|
|
|
static async copyToClipboard(text) {
|
|
|
|
|
text = "" + text // ensure that text is string
|
|
|
|
|
|
|
|
|
|
if (!text.length || !window?.navigator?.clipboard) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return window.navigator.clipboard.writeText(text).catch((err) => {
|
|
|
|
|
console.warn("Failed to copy.", err);
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-05 11:00:38 +08:00
|
|
|
|
/**
|
|
|
|
|
* Downloads a json file created from the provide object.
|
|
|
|
|
*
|
|
|
|
|
* @param {mixed} obj The JS object to download.
|
|
|
|
|
* @param {String} name The result file name.
|
|
|
|
|
*/
|
|
|
|
|
static downloadJson(obj, name) {
|
|
|
|
|
const encodedObj = "data:text/json;charset=utf-8," + encodeURIComponent(JSON.stringify(obj, null, 2));
|
|
|
|
|
|
|
|
|
|
const tempLink = document.createElement('a');
|
|
|
|
|
tempLink.setAttribute("href", encodedObj);
|
|
|
|
|
tempLink.setAttribute("download", name + ".json");
|
|
|
|
|
tempLink.click();
|
|
|
|
|
tempLink.remove();
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
/**
|
|
|
|
|
* Parses and returns the decoded jwt payload data.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} jwt
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static getJWTPayload(jwt) {
|
|
|
|
|
const raw = (jwt || '').split(".")[1] || '';
|
|
|
|
|
if (raw === "") {
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const encodedPayload = decodeURIComponent(atob(raw));
|
|
|
|
|
return JSON.parse(encodedPayload) || {};
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.warn("Failed to parse JWT payload data.", err);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Loosely check if a file is an image based on its filename extension.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} filename
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static hasImageExtension(filename) {
|
2022-07-21 17:56:17 +08:00
|
|
|
|
return /\.jpg|\.jpeg|\.png|\.svg|\.gif|\.webp|\.avif$/.test(filename)
|
2022-07-07 05:19:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Creates a thumbnail from `File` with the specified `width` and `height` params.
|
|
|
|
|
* Returns a `Promise` with the generated base64 url.
|
|
|
|
|
*
|
|
|
|
|
* @param {File} file
|
|
|
|
|
* @param {Number} [width]
|
|
|
|
|
* @param {Number} [height]
|
|
|
|
|
* @return {Promise}
|
|
|
|
|
*/
|
|
|
|
|
static generateThumb(file, width = 100, height = 100) {
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
let reader = new FileReader();
|
|
|
|
|
|
|
|
|
|
reader.onload = function(e) {
|
|
|
|
|
let img = new Image();
|
|
|
|
|
|
|
|
|
|
img.onload = function() {
|
|
|
|
|
let canvas = document.createElement("canvas");
|
|
|
|
|
let ctx = canvas.getContext("2d");
|
|
|
|
|
let imgWidth = img.width;
|
|
|
|
|
let imgHeight = img.height;
|
|
|
|
|
|
|
|
|
|
canvas.width = width;
|
|
|
|
|
canvas.height = height;
|
|
|
|
|
|
|
|
|
|
ctx.drawImage(
|
|
|
|
|
img,
|
|
|
|
|
imgWidth > imgHeight ? (imgWidth - imgHeight) / 2 : 0,
|
|
|
|
|
0, // top aligned
|
|
|
|
|
// imgHeight > imgWidth ? (imgHeight - imgWidth) / 2 : 0,
|
|
|
|
|
imgWidth > imgHeight ? imgHeight : imgWidth,
|
|
|
|
|
imgWidth > imgHeight ? imgHeight : imgWidth,
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
width,
|
|
|
|
|
height
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return resolve(canvas.toDataURL(file.type));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
img.src = e.target.result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reader.readAsDataURL(file);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Normalizes and append a value to the provided form data.
|
|
|
|
|
*
|
|
|
|
|
* @param {FormData} formData
|
|
|
|
|
* @param {string} key
|
|
|
|
|
* @param {mixed} value
|
|
|
|
|
*/
|
|
|
|
|
static addValueToFormData(formData, key, value) {
|
|
|
|
|
if (typeof value === "undefined") {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (CommonHelper.isEmpty(value)) {
|
|
|
|
|
formData.append(key, "");
|
|
|
|
|
} else if (Array.isArray(value)) {
|
|
|
|
|
for (const item of value) {
|
|
|
|
|
CommonHelper.addValueToFormData(formData, key, item);
|
|
|
|
|
}
|
|
|
|
|
} else if (value instanceof File) {
|
|
|
|
|
formData.append(key, value);
|
|
|
|
|
} else if (value instanceof Date) {
|
|
|
|
|
formData.append(key, value.toISOString());
|
|
|
|
|
} else if (CommonHelper.isObject(value)) {
|
|
|
|
|
formData.append(key, JSON.stringify(value));
|
|
|
|
|
} else {
|
|
|
|
|
formData.append(key, "" + value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns the default Flatpickr initialization options.
|
|
|
|
|
*
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static defaultFlatpickrOptions() {
|
|
|
|
|
return {
|
|
|
|
|
dateFormat: "Y-m-d H:i:S",
|
|
|
|
|
disableMobile: true,
|
|
|
|
|
allowInput: true,
|
|
|
|
|
enableTime: true,
|
|
|
|
|
time_24hr: true,
|
|
|
|
|
locale: {
|
|
|
|
|
firstDayOfWeek: 1,
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a dummy collection record object.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} collection
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static dummyCollectionRecord(collection) {
|
|
|
|
|
const fields = collection?.schema || [];
|
|
|
|
|
|
|
|
|
|
const dummy = {
|
|
|
|
|
"id": "RECORD_ID",
|
2022-10-30 16:28:14 +08:00
|
|
|
|
"collectionId": collection?.id,
|
|
|
|
|
"collectionName": collection?.name,
|
2022-11-06 21:48:27 +08:00
|
|
|
|
"created": "2022-01-01 01:00:00.123Z",
|
|
|
|
|
"updated": "2022-01-01 23:59:59.456Z",
|
2022-07-07 05:19:05 +08:00
|
|
|
|
};
|
|
|
|
|
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (collection?.isAuth) {
|
|
|
|
|
dummy["username"] = "username123";
|
|
|
|
|
dummy["verified"] = false;
|
|
|
|
|
dummy["emailVisibility"] = true;
|
|
|
|
|
dummy["email"] = "test@example.com";
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
for (const field of fields) {
|
|
|
|
|
let val = null;
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (field.type === "number") {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
val = 123;
|
|
|
|
|
} else if (field.type === "date") {
|
2022-10-30 16:28:14 +08:00
|
|
|
|
val = "2022-01-01 10:00:00.123Z";
|
2022-07-07 05:19:05 +08:00
|
|
|
|
} else if (field.type === "bool") {
|
|
|
|
|
val = true;
|
|
|
|
|
} else if (field.type === "email") {
|
|
|
|
|
val = "test@example.com";
|
|
|
|
|
} else if (field.type === "url") {
|
|
|
|
|
val = "https://example.com";
|
|
|
|
|
} else if (field.type === "json") {
|
2022-10-30 16:28:14 +08:00
|
|
|
|
val = 'JSON';
|
2022-07-07 05:19:05 +08:00
|
|
|
|
} else if (field.type === "file") {
|
|
|
|
|
val = 'filename.jpg';
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (field.options?.maxSelect !== 1) {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
val = [val];
|
|
|
|
|
}
|
|
|
|
|
} else if (field.type === "select") {
|
|
|
|
|
val = field.options?.values?.[0];
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (field.options?.maxSelect !== 1) {
|
|
|
|
|
val = [val];
|
|
|
|
|
}
|
|
|
|
|
} else if (field.type === "relation") {
|
|
|
|
|
val = 'RELATION_RECORD_ID';
|
|
|
|
|
if (field.options?.maxSelect !== 1) {
|
|
|
|
|
val = [val];
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
val = "test";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dummy[field.name] = val;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dummy;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a dummy collection schema data object.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} collection
|
|
|
|
|
* @return {Object}
|
|
|
|
|
*/
|
|
|
|
|
static dummyCollectionSchemaData(collection) {
|
|
|
|
|
const fields = collection?.schema || [];
|
|
|
|
|
|
|
|
|
|
const dummy = {};
|
|
|
|
|
|
|
|
|
|
for (const field of fields) {
|
|
|
|
|
let val = null;
|
|
|
|
|
|
|
|
|
|
if (field.type === "number") {
|
|
|
|
|
val = 123;
|
|
|
|
|
} else if (field.type === "date") {
|
|
|
|
|
val = "2022-01-01 10:00:00.123Z";
|
|
|
|
|
} else if (field.type === "bool") {
|
|
|
|
|
val = true;
|
|
|
|
|
} else if (field.type === "email") {
|
|
|
|
|
val = "test@example.com";
|
|
|
|
|
} else if (field.type === "url") {
|
|
|
|
|
val = "https://example.com";
|
|
|
|
|
} else if (field.type === "json") {
|
|
|
|
|
val = 'JSON';
|
|
|
|
|
} else if (field.type === "file") {
|
|
|
|
|
continue; // currently file upload is supported only via FormData
|
|
|
|
|
} else if (field.type === "select") {
|
|
|
|
|
val = field.options?.values?.[0];
|
|
|
|
|
if (field.options?.maxSelect !== 1) {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
val = [val];
|
|
|
|
|
}
|
2022-10-30 16:28:14 +08:00
|
|
|
|
} else if (field.type === "relation") {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
val = 'RELATION_RECORD_ID';
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (field.options?.maxSelect !== 1) {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
val = [val];
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
val = "test";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dummy[field.name] = val;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dummy;
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-30 16:28:14 +08:00
|
|
|
|
/**
|
|
|
|
|
* Returns a collection type icon.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} type
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static getCollectionTypeIcon(type) {
|
|
|
|
|
switch (type?.toLowerCase()) {
|
|
|
|
|
case "auth":
|
|
|
|
|
return "ri-group-line";
|
|
|
|
|
case "single":
|
|
|
|
|
return "ri-file-list-2-line";
|
|
|
|
|
default:
|
|
|
|
|
return "ri-folder-2-line";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-07 05:19:05 +08:00
|
|
|
|
/**
|
|
|
|
|
* Returns a field type icon.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} type
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static getFieldTypeIcon(type) {
|
|
|
|
|
switch (type?.toLowerCase()) {
|
|
|
|
|
case "primary":
|
|
|
|
|
return "ri-key-line";
|
|
|
|
|
case "text":
|
|
|
|
|
return "ri-text";
|
|
|
|
|
case "number":
|
|
|
|
|
return "ri-hashtag";
|
|
|
|
|
case "date":
|
|
|
|
|
return "ri-calendar-line";
|
|
|
|
|
case "bool":
|
|
|
|
|
return "ri-toggle-line";
|
|
|
|
|
case "email":
|
|
|
|
|
return "ri-mail-line";
|
|
|
|
|
case "url":
|
|
|
|
|
return "ri-link";
|
|
|
|
|
case "select":
|
|
|
|
|
return "ri-list-check";
|
|
|
|
|
case "json":
|
|
|
|
|
return "ri-braces-line";
|
|
|
|
|
case "file":
|
|
|
|
|
return "ri-image-line";
|
|
|
|
|
case "relation":
|
|
|
|
|
return "ri-mind-map";
|
|
|
|
|
case "user":
|
|
|
|
|
return "ri-user-line";
|
|
|
|
|
default:
|
|
|
|
|
return "ri-star-s-line";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns the field value base type as text.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} field
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static getFieldValueType(field) {
|
2022-10-30 16:28:14 +08:00
|
|
|
|
switch (field?.type) {
|
2022-07-07 05:19:05 +08:00
|
|
|
|
case 'bool':
|
|
|
|
|
return 'Boolean';
|
|
|
|
|
case 'number':
|
|
|
|
|
return 'Number';
|
|
|
|
|
case 'file':
|
|
|
|
|
return 'File';
|
|
|
|
|
case 'select':
|
|
|
|
|
case 'relation':
|
2022-10-30 16:28:14 +08:00
|
|
|
|
if (field?.options?.maxSelect === 1) {
|
|
|
|
|
return 'String';
|
2022-07-07 05:19:05 +08:00
|
|
|
|
}
|
2022-10-30 16:28:14 +08:00
|
|
|
|
return 'Array<String>';
|
2022-07-07 05:19:05 +08:00
|
|
|
|
default:
|
|
|
|
|
return 'String';
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-08-10 21:16:59 +08:00
|
|
|
|
|
2022-10-30 16:28:14 +08:00
|
|
|
|
/**
|
|
|
|
|
* Returns the zero-default string value of the provided field.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} field
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static zeroDefaultStr(field) {
|
|
|
|
|
if (field?.type === "number") {
|
|
|
|
|
return "0";
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-03 17:36:59 +08:00
|
|
|
|
if (field?.type === "bool") {
|
|
|
|
|
return "false";
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-30 16:28:14 +08:00
|
|
|
|
// array value
|
|
|
|
|
if (["select", "relation", "file"].includes(field?.type) && field?.options?.maxSelect != 1) {
|
|
|
|
|
return "[]";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return '""';
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-10 21:16:59 +08:00
|
|
|
|
/**
|
|
|
|
|
* Returns an API url address extract from the current running instance.
|
|
|
|
|
*
|
|
|
|
|
* @param {String} fallback Fallback url that will be used if the extractions fail.
|
|
|
|
|
* @return {String}
|
|
|
|
|
*/
|
|
|
|
|
static getApiExampleUrl(fallback) {
|
|
|
|
|
let url = window.location.href.substring(0, window.location.href.indexOf("/_")) || fallback || '/';
|
|
|
|
|
|
|
|
|
|
// for broader compatibility replace localhost with 127.0.0.1
|
|
|
|
|
// (see https://github.com/pocketbase/js-sdk/issues/21)
|
|
|
|
|
return url.replace('//localhost', '//127.0.0.1');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Checks if the provided 2 collections has any change (ignoring root schema fields order).
|
|
|
|
|
*
|
|
|
|
|
* @param {Collection} oldCollection
|
|
|
|
|
* @param {Collection} newCollection
|
|
|
|
|
* @param {Boolean} withDeleteMissing Skip missing schema fields from the newCollection.
|
|
|
|
|
* @return {Boolean}
|
|
|
|
|
*/
|
|
|
|
|
static hasCollectionChanges(oldCollection, newCollection, withDeleteMissing = false) {
|
|
|
|
|
oldCollection = oldCollection || {};
|
|
|
|
|
newCollection = newCollection || {};
|
|
|
|
|
|
|
|
|
|
if (oldCollection.id != newCollection.id) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (let prop in oldCollection) {
|
|
|
|
|
if (prop !== 'schema' && JSON.stringify(oldCollection[prop]) !== JSON.stringify(newCollection[prop])) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const oldSchema = Array.isArray(oldCollection.schema) ? oldCollection.schema : [];
|
|
|
|
|
const newSchema = Array.isArray(newCollection.schema) ? newCollection.schema : [];
|
|
|
|
|
const removedFields = oldSchema.filter((oldField) => {
|
|
|
|
|
return oldField?.id && !CommonHelper.findByKey(newSchema, "id", oldField.id);
|
|
|
|
|
});
|
|
|
|
|
const addedFields = newSchema.filter((newField) => {
|
|
|
|
|
return newField?.id && !CommonHelper.findByKey(oldSchema, "id", newField.id);
|
|
|
|
|
});
|
|
|
|
|
const changedFields = newSchema.filter((newField) => {
|
|
|
|
|
const oldField = CommonHelper.isObject(newField) && CommonHelper.findByKey(oldSchema, "id", newField.id);
|
|
|
|
|
if (!oldField) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (let prop in oldField) {
|
|
|
|
|
if (JSON.stringify(newField[prop]) != JSON.stringify(oldField[prop])) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return !!(
|
|
|
|
|
addedFields.length ||
|
|
|
|
|
changedFields.length ||
|
|
|
|
|
(withDeleteMissing && removedFields.length)
|
|
|
|
|
);
|
|
|
|
|
}
|
2022-10-30 16:28:14 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Groups and sorts collections array by type (auth, single, base).
|
|
|
|
|
*
|
|
|
|
|
* @param {Array} collections
|
|
|
|
|
* @return {Array}
|
|
|
|
|
*/
|
|
|
|
|
static sortCollections(collections = []) {
|
|
|
|
|
const authCollections = [];
|
|
|
|
|
const singleCollections = [];
|
|
|
|
|
const baseCollections = [];
|
|
|
|
|
|
|
|
|
|
for (const colelction of collections) {
|
|
|
|
|
if (colelction.type == 'auth') {
|
|
|
|
|
authCollections.push(colelction);
|
|
|
|
|
} else if (colelction.type == 'single') {
|
|
|
|
|
singleCollections.push(colelction);
|
|
|
|
|
} else {
|
|
|
|
|
baseCollections.push(colelction);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [].concat(authCollections, singleCollections, baseCollections);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* "Yield" to the main thread to break long runing task into smaller ones.
|
|
|
|
|
*
|
|
|
|
|
* (see https://web.dev/optimize-long-tasks/)
|
|
|
|
|
*/
|
|
|
|
|
static yieldToMain() {
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
setTimeout(resolve, 0);
|
|
|
|
|
});
|
|
|
|
|
}
|
2022-07-07 05:19:05 +08:00
|
|
|
|
}
|