@theia/core
Version:
Theia is a cloud & desktop IDE framework implemented in TypeScript.
124 lines (109 loc) • 4.22 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2018 Ericsson and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************
import { isObject, isUndefined, isUndefinedOrNull } from './types';
export function deepClone<T>(obj: T): T {
if (!isObject(obj)) {
return obj;
}
if (obj instanceof RegExp) {
return obj;
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const result: any = Array.isArray(obj) ? [] : {};
Object.keys(obj).forEach((key: string) => {
const prop = obj[key];
if (isObject(prop)) {
result[key] = deepClone(prop);
} else {
result[key] = prop;
}
});
return result;
}
export function deepFreeze<T>(obj: T): T {
if (!isObject(obj)) {
return obj;
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const stack: any[] = [obj];
while (stack.length > 0) {
const objectToFreeze = stack.shift();
Object.freeze(objectToFreeze);
for (const key in objectToFreeze) {
if (_hasOwnProperty.call(objectToFreeze, key)) {
const prop = objectToFreeze[key];
if (isObject(prop) && !Object.isFrozen(prop)) {
stack.push(prop);
}
}
}
}
return obj;
}
const _hasOwnProperty = Object.prototype.hasOwnProperty;
export function notEmpty<T>(arg: T | undefined | null): arg is T {
// eslint-disable-next-line no-null/no-null
return arg !== undefined && arg !== null;
}
/**
* `true` if the argument is an empty object. Otherwise, `false`.
*/
export function isEmpty(arg: Object): boolean {
return Object.keys(arg).length === 0 && arg.constructor === Object;
}
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation and others. All rights reserved.
* Licensed under the MIT License. See https://github.com/Microsoft/vscode/blob/master/LICENSE.txt for license information.
*--------------------------------------------------------------------------------------------*/
// Copied from https://github.com/microsoft/vscode/blob/1.72.2/src/vs/base/common/objects.ts
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function cloneAndChange(obj: any, changer: (orig: any) => any): any {
return _cloneAndChange(obj, changer, new Set());
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
function _cloneAndChange(obj: any, changer: (orig: any) => any, seen: Set<any>): any {
if (isUndefinedOrNull(obj)) {
return obj;
}
const changed = changer(obj);
if (!isUndefined(changed)) {
return changed;
}
if (Array.isArray(obj)) {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const r1: any[] = [];
for (const e of obj) {
r1.push(_cloneAndChange(e, changer, seen));
}
return r1;
}
if (isObject(obj)) {
if (seen.has(obj)) {
throw new Error('Cannot clone recursive data-structure');
}
seen.add(obj);
const r2 = {};
for (const i2 in obj) {
if (_hasOwnProperty.call(obj, i2)) {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
(r2 as any)[i2] = _cloneAndChange(obj[i2], changer, seen);
}
}
seen.delete(obj);
return r2;
}
return obj;
}