2024-05-03 15:09:28 +00:00
|
|
|
import {
|
|
|
|
isAudioBuffer,
|
|
|
|
isAudioNode,
|
|
|
|
isAudioParam,
|
|
|
|
} from "./AdvancedTypeCheck.js";
|
2024-05-03 14:10:40 +00:00
|
|
|
import { isDefined, isObject, isUndef } from "./TypeCheck.js";
|
2019-05-22 04:14:43 +00:00
|
|
|
|
2019-07-11 13:57:06 +00:00
|
|
|
type BaseToneOptions = import("../Tone").BaseToneOptions;
|
|
|
|
|
2019-08-19 17:04:21 +00:00
|
|
|
/**
|
|
|
|
* Some objects should not be merged
|
|
|
|
*/
|
|
|
|
function noCopy(key: string, arg: any): boolean {
|
2024-05-03 15:09:28 +00:00
|
|
|
return (
|
|
|
|
key === "value" ||
|
|
|
|
isAudioParam(arg) ||
|
|
|
|
isAudioNode(arg) ||
|
|
|
|
isAudioBuffer(arg)
|
|
|
|
);
|
2019-08-19 17:04:21 +00:00
|
|
|
}
|
|
|
|
|
2024-05-01 19:55:52 +00:00
|
|
|
export function deepMerge<T>(target: T): T;
|
|
|
|
export function deepMerge<T, U>(target: T, source1: U): T & U;
|
2024-05-03 15:09:28 +00:00
|
|
|
export function deepMerge<T, U, V>(
|
|
|
|
target: T,
|
|
|
|
source1: U,
|
|
|
|
source2: V
|
|
|
|
): T & U & V;
|
|
|
|
export function deepMerge<T, U, V, W>(
|
|
|
|
target: T,
|
|
|
|
source1: U,
|
|
|
|
source2: V,
|
|
|
|
source3: W
|
|
|
|
): T & U & V & W;
|
2019-05-22 04:14:43 +00:00
|
|
|
/**
|
|
|
|
* Recursively merge an object
|
|
|
|
* @param target the object to merge into
|
|
|
|
* @param sources the source objects to merge
|
|
|
|
*/
|
2019-08-16 17:57:17 +00:00
|
|
|
export function deepMerge(target: any, ...sources: any[]): any {
|
2019-09-16 03:32:40 +00:00
|
|
|
if (!sources.length) {
|
2024-05-03 15:09:28 +00:00
|
|
|
return target;
|
2019-09-16 03:32:40 +00:00
|
|
|
}
|
2019-05-22 04:14:43 +00:00
|
|
|
const source = sources.shift();
|
|
|
|
|
|
|
|
if (isObject(target) && isObject(source)) {
|
|
|
|
for (const key in source) {
|
2019-08-19 17:04:21 +00:00
|
|
|
if (noCopy(key, source[key])) {
|
2019-07-23 21:09:11 +00:00
|
|
|
target[key] = source[key];
|
|
|
|
} else if (isObject(source[key])) {
|
2019-09-16 03:32:40 +00:00
|
|
|
if (!target[key]) {
|
2024-05-03 15:09:28 +00:00
|
|
|
Object.assign(target, { [key]: {} });
|
2019-09-16 03:32:40 +00:00
|
|
|
}
|
2019-05-22 04:14:43 +00:00
|
|
|
deepMerge(target[key], source[key] as any);
|
|
|
|
} else {
|
|
|
|
Object.assign(target, { [key]: source[key] as any });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-08-16 17:57:17 +00:00
|
|
|
// @ts-ignore
|
2019-05-22 04:14:43 +00:00
|
|
|
return deepMerge(target, ...sources);
|
|
|
|
}
|
|
|
|
|
2019-08-08 18:22:15 +00:00
|
|
|
/**
|
|
|
|
* Returns true if the two arrays have the same value for each of the elements
|
|
|
|
*/
|
2020-09-21 13:26:22 +00:00
|
|
|
export function deepEquals<T>(arrayA: T[], arrayB: T[]): boolean {
|
2024-05-03 15:09:28 +00:00
|
|
|
return (
|
|
|
|
arrayA.length === arrayB.length &&
|
|
|
|
arrayA.every((element, index) => arrayB[index] === element)
|
|
|
|
);
|
2019-08-08 18:22:15 +00:00
|
|
|
}
|
|
|
|
|
2019-05-22 04:14:43 +00:00
|
|
|
/**
|
|
|
|
* Convert an args array into an object.
|
2024-04-29 14:48:28 +00:00
|
|
|
* @internal
|
2019-05-22 04:14:43 +00:00
|
|
|
*/
|
2019-07-22 20:15:55 +00:00
|
|
|
export function optionsFromArguments<T extends object>(
|
|
|
|
defaults: T,
|
|
|
|
argsArray: IArguments,
|
2020-07-19 19:04:03 +00:00
|
|
|
keys: Array<keyof T> = [],
|
2024-05-03 15:09:28 +00:00
|
|
|
objKey?: keyof T
|
2019-07-22 20:15:55 +00:00
|
|
|
): T {
|
2020-07-19 19:04:03 +00:00
|
|
|
const opts: Partial<T> = {};
|
2019-05-22 04:14:43 +00:00
|
|
|
const args = Array.from(argsArray);
|
2019-07-22 20:15:55 +00:00
|
|
|
// if the first argument is an object and has an object key
|
|
|
|
if (isObject(args[0]) && objKey && !Reflect.has(args[0], objKey)) {
|
|
|
|
// if it's not part of the defaults
|
2024-05-03 15:09:28 +00:00
|
|
|
const partOfDefaults = Object.keys(args[0]).some((key) =>
|
|
|
|
Reflect.has(defaults, key)
|
|
|
|
);
|
2019-07-22 20:15:55 +00:00
|
|
|
if (!partOfDefaults) {
|
|
|
|
// merge that key
|
2019-09-16 03:32:40 +00:00
|
|
|
deepMerge(opts, { [objKey]: args[0] });
|
2019-07-22 20:15:55 +00:00
|
|
|
// remove the obj key from the keys
|
|
|
|
keys.splice(keys.indexOf(objKey), 1);
|
|
|
|
// shift the first argument off
|
|
|
|
args.shift();
|
|
|
|
}
|
|
|
|
}
|
2019-05-22 04:14:43 +00:00
|
|
|
if (args.length === 1 && isObject(args[0])) {
|
|
|
|
deepMerge(opts, args[0]);
|
|
|
|
} else {
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
|
|
if (isDefined(args[i])) {
|
|
|
|
opts[keys[i]] = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return deepMerge(defaults, opts);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return this instances default values by calling Constructor.getDefaults()
|
|
|
|
*/
|
|
|
|
export function getDefaultsFromInstance<T>(instance: T): BaseToneOptions {
|
|
|
|
type ToneClass = {
|
|
|
|
constructor: ToneClass;
|
|
|
|
getDefaults: () => BaseToneOptions;
|
|
|
|
} & T;
|
|
|
|
|
|
|
|
return (instance as ToneClass).constructor.getDefaults();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-06-18 01:52:02 +00:00
|
|
|
* Returns the fallback if the given object is undefined.
|
2019-05-22 04:14:43 +00:00
|
|
|
* Take an array of arguments and return a formatted options object.
|
2024-04-29 14:48:28 +00:00
|
|
|
* @internal
|
2019-05-22 04:14:43 +00:00
|
|
|
*/
|
2019-06-18 01:52:02 +00:00
|
|
|
export function defaultArg<T>(given: T, fallback: T): T {
|
|
|
|
if (isUndef(given)) {
|
|
|
|
return fallback;
|
|
|
|
} else {
|
|
|
|
return given;
|
|
|
|
}
|
|
|
|
}
|
2019-07-18 18:06:10 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove all of the properties belonging to omit from obj.
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
export function omitFromObject<T extends object, O extends string[]>(
|
|
|
|
obj: T,
|
|
|
|
omit: O
|
|
|
|
): Omit<T, keyof O> {
|
|
|
|
omit.forEach((prop) => {
|
2019-07-18 18:06:10 +00:00
|
|
|
if (Reflect.has(obj, prop)) {
|
|
|
|
delete obj[prop];
|
|
|
|
}
|
2019-07-19 16:32:42 +00:00
|
|
|
});
|
2019-07-18 18:06:10 +00:00
|
|
|
return obj;
|
|
|
|
}
|