From c3d8d75ba3eaee56c436bce15cd3d568d81db9c4 Mon Sep 17 00:00:00 2001 From: ztimson Date: Wed, 30 Jul 2025 14:22:40 -0400 Subject: [PATCH] Added delta functions --- package.json | 2 +- src/objects.ts | 55 ++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 56 insertions(+), 1 deletion(-) diff --git a/package.json b/package.json index 13a510c..a16c2b6 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@ztimson/utils", - "version": "0.26.23", + "version": "0.26.24", "description": "Utility library", "author": "Zak Timson", "license": "MIT", diff --git a/src/objects.ts b/src/objects.ts index 3db50a9..29b3257 100644 --- a/src/objects.ts +++ b/src/objects.ts @@ -1,3 +1,58 @@ +type Delta = { [key: string]: any | Delta | null }; + +/** + * Applies deltas in order to modify `target`. + * @param target - Object to mutate + * @param deltas - List of deltas to apply + * @returns Mutated target + */ +function applyDeltas(target: any, deltas: Delta[]): any { + for(const delta of deltas) { + for(const [key, value] of Object.entries(delta)) { + if(value === null) { + delete target[key]; // remove + } else if(typeof value === 'object' && !Array.isArray(value)) { + if(typeof target[key] !== 'object' || Array.isArray(target[key])) { + target[key] = {}; // nested obj + } + applyDeltas(target[key], [value]); // recurse + } else { + target[key] = value; // restore + } + } + } + return target; +} + +/** + * Creates a nested delta that reverts `target` back to `old`. + * @param old - Original object + * @param target - Modified object + * @returns Delta to revert changes + */ +function calcDelta(old: any, target: any): Delta { + const delta: Delta = {}; + const keys = new Set([...Object.keys(old || {}), ...Object.keys(target || {})]); + for(const key of keys) { + const val1 = old?.[key]; + const val2 = target?.[key]; + if(!(key in target)) { + delta[key] = val1; // removed + } else if(!(key in old)) { + delta[key] = null; // added + } else if( + typeof val1 === 'object' && typeof val2 === 'object' && + val1 && val2 && !Array.isArray(val1) + ) { + const nested = calcDelta(val1, val2); + if(Object.keys(nested).length) delta[key] = nested; + } else if(val1 !== val2) { + delta[key] = val1; // changed + } + } + return delta; +} + /** * Removes any null values from an object in-place *