UNPKG

immer

Version:

Create your next immutable state by mutating the current one

1,503 lines (1,490 loc) 45.6 kB
"use strict"; var __defProp = Object.defineProperty; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropNames = Object.getOwnPropertyNames; var __hasOwnProp = Object.prototype.hasOwnProperty; var __export = (target, all) => { for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __copyProps = (to, from, except, desc) => { if (from && typeof from === "object" || typeof from === "function") { for (let key of __getOwnPropNames(from)) if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); } return to; }; var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); // src/immer.ts var immer_exports = {}; __export(immer_exports, { Immer: () => Immer2, applyPatches: () => applyPatches, castDraft: () => castDraft, castImmutable: () => castImmutable, createDraft: () => createDraft, current: () => current, enableMapSet: () => enableMapSet, enablePatches: () => enablePatches, finishDraft: () => finishDraft, freeze: () => freeze, immerable: () => DRAFTABLE, isDraft: () => isDraft, isDraftable: () => isDraftable, nothing: () => NOTHING, original: () => original, produce: () => produce, produceWithPatches: () => produceWithPatches, setAutoFreeze: () => setAutoFreeze, setUseStrictIteration: () => setUseStrictIteration, setUseStrictShallowCopy: () => setUseStrictShallowCopy }); module.exports = __toCommonJS(immer_exports); // src/utils/env.ts var NOTHING = Symbol.for("immer-nothing"); var DRAFTABLE = Symbol.for("immer-draftable"); var DRAFT_STATE = Symbol.for("immer-state"); // src/utils/errors.ts var errors = process.env.NODE_ENV !== "production" ? [ // All error codes, starting by 0: function(plugin) { return `The plugin for '${plugin}' has not been loaded into Immer. To enable the plugin, import and call \`enable${plugin}()\` when initializing your application.`; }, function(thing) { return `produce can only be called on things that are draftable: plain objects, arrays, Map, Set or classes that are marked with '[immerable]: true'. Got '${thing}'`; }, "This object has been frozen and should not be mutated", function(data) { return "Cannot use a proxy that has been revoked. Did you pass an object from inside an immer function to an async process? " + data; }, "An immer producer returned a new value *and* modified its draft. Either return a new value *or* modify the draft.", "Immer forbids circular references", "The first or second argument to `produce` must be a function", "The third argument to `produce` must be a function or undefined", "First argument to `createDraft` must be a plain object, an array, or an immerable object", "First argument to `finishDraft` must be a draft returned by `createDraft`", function(thing) { return `'current' expects a draft, got: ${thing}`; }, "Object.defineProperty() cannot be used on an Immer draft", "Object.setPrototypeOf() cannot be used on an Immer draft", "Immer only supports deleting array indices", "Immer only supports setting array indices and the 'length' property", function(thing) { return `'original' expects a draft, got: ${thing}`; } // Note: if more errors are added, the errorOffset in Patches.ts should be increased // See Patches.ts for additional errors ] : []; function die(error, ...args) { if (process.env.NODE_ENV !== "production") { const e = errors[error]; const msg = isFunction(e) ? e.apply(null, args) : e; throw new Error(`[Immer] ${msg}`); } throw new Error( `[Immer] minified error nr: ${error}. Full error at: https://bit.ly/3cXEKWf` ); } // src/utils/common.ts var O = Object; var getPrototypeOf = O.getPrototypeOf; var CONSTRUCTOR = "constructor"; var PROTOTYPE = "prototype"; var CONFIGURABLE = "configurable"; var ENUMERABLE = "enumerable"; var WRITABLE = "writable"; var VALUE = "value"; var isDraft = (value) => !!value && !!value[DRAFT_STATE]; function isDraftable(value) { if (!value) return false; return isPlainObject(value) || isArray(value) || !!value[DRAFTABLE] || !!value[CONSTRUCTOR]?.[DRAFTABLE] || isMap(value) || isSet(value); } var objectCtorString = O[PROTOTYPE][CONSTRUCTOR].toString(); var cachedCtorStrings = /* @__PURE__ */ new WeakMap(); function isPlainObject(value) { if (!value || !isObjectish(value)) return false; const proto = getPrototypeOf(value); if (proto === null || proto === O[PROTOTYPE]) return true; const Ctor = O.hasOwnProperty.call(proto, CONSTRUCTOR) && proto[CONSTRUCTOR]; if (Ctor === Object) return true; if (!isFunction(Ctor)) return false; let ctorString = cachedCtorStrings.get(Ctor); if (ctorString === void 0) { ctorString = Function.toString.call(Ctor); cachedCtorStrings.set(Ctor, ctorString); } return ctorString === objectCtorString; } function original(value) { if (!isDraft(value)) die(15, value); return value[DRAFT_STATE].base_; } function each(obj, iter, strict = true) { if (getArchtype(obj) === 0 /* Object */) { const keys = strict ? Reflect.ownKeys(obj) : O.keys(obj); keys.forEach((key) => { iter(key, obj[key], obj); }); } else { obj.forEach((entry, index) => iter(index, entry, obj)); } } function getArchtype(thing) { const state = thing[DRAFT_STATE]; return state ? state.type_ : isArray(thing) ? 1 /* Array */ : isMap(thing) ? 2 /* Map */ : isSet(thing) ? 3 /* Set */ : 0 /* Object */; } var has = (thing, prop, type = getArchtype(thing)) => type === 2 /* Map */ ? thing.has(prop) : O[PROTOTYPE].hasOwnProperty.call(thing, prop); var get = (thing, prop, type = getArchtype(thing)) => ( // @ts-ignore type === 2 /* Map */ ? thing.get(prop) : thing[prop] ); var set = (thing, propOrOldValue, value, type = getArchtype(thing)) => { if (type === 2 /* Map */) thing.set(propOrOldValue, value); else if (type === 3 /* Set */) { thing.add(value); } else thing[propOrOldValue] = value; }; function is(x, y) { if (x === y) { return x !== 0 || 1 / x === 1 / y; } else { return x !== x && y !== y; } } var isArray = Array.isArray; var isMap = (target) => target instanceof Map; var isSet = (target) => target instanceof Set; var isObjectish = (target) => typeof target === "object"; var isFunction = (target) => typeof target === "function"; var isBoolean = (target) => typeof target === "boolean"; var getProxyDraft = (value) => { if (!isObjectish(value)) return null; return value?.[DRAFT_STATE]; }; var latest = (state) => state.copy_ || state.base_; var getValue = (value) => { const proxyDraft = getProxyDraft(value); return proxyDraft ? proxyDraft.copy_ ?? proxyDraft.base_ : value; }; var getFinalValue = (state) => state.modified_ ? state.copy_ : state.base_; function shallowCopy(base, strict) { if (isMap(base)) { return new Map(base); } if (isSet(base)) { return new Set(base); } if (isArray(base)) return Array[PROTOTYPE].slice.call(base); const isPlain = isPlainObject(base); if (strict === true || strict === "class_only" && !isPlain) { const descriptors = O.getOwnPropertyDescriptors(base); delete descriptors[DRAFT_STATE]; let keys = Reflect.ownKeys(descriptors); for (let i = 0; i < keys.length; i++) { const key = keys[i]; const desc = descriptors[key]; if (desc[WRITABLE] === false) { desc[WRITABLE] = true; desc[CONFIGURABLE] = true; } if (desc.get || desc.set) descriptors[key] = { [CONFIGURABLE]: true, [WRITABLE]: true, // could live with !!desc.set as well here... [ENUMERABLE]: desc[ENUMERABLE], [VALUE]: base[key] }; } return O.create(getPrototypeOf(base), descriptors); } else { const proto = getPrototypeOf(base); if (proto !== null && isPlain) { return { ...base }; } const obj = O.create(proto); return O.assign(obj, base); } } function freeze(obj, deep = false) { if (isFrozen(obj) || isDraft(obj) || !isDraftable(obj)) return obj; if (getArchtype(obj) > 1) { O.defineProperties(obj, { set: dontMutateMethodOverride, add: dontMutateMethodOverride, clear: dontMutateMethodOverride, delete: dontMutateMethodOverride }); } O.freeze(obj); if (deep) each( obj, (_key, value) => { freeze(value, true); }, false ); return obj; } function dontMutateFrozenCollections() { die(2); } var dontMutateMethodOverride = { [VALUE]: dontMutateFrozenCollections }; function isFrozen(obj) { if (obj === null || !isObjectish(obj)) return true; return O.isFrozen(obj); } // src/utils/plugins.ts var PluginMapSet = "MapSet"; var PluginPatches = "Patches"; var plugins = {}; function getPlugin(pluginKey) { const plugin = plugins[pluginKey]; if (!plugin) { die(0, pluginKey); } return plugin; } var isPluginLoaded = (pluginKey) => !!plugins[pluginKey]; function loadPlugin(pluginKey, implementation) { if (!plugins[pluginKey]) plugins[pluginKey] = implementation; } // src/core/scope.ts var currentScope; var getCurrentScope = () => currentScope; var createScope = (parent_, immer_) => ({ drafts_: [], parent_, immer_, // Whenever the modified draft contains a draft from another scope, we // need to prevent auto-freezing so the unowned draft can be finalized. canAutoFreeze_: true, unfinalizedDrafts_: 0, handledSet_: /* @__PURE__ */ new Set(), processedForPatches_: /* @__PURE__ */ new Set(), mapSetPlugin_: isPluginLoaded(PluginMapSet) ? getPlugin(PluginMapSet) : void 0 }); function usePatchesInScope(scope, patchListener) { if (patchListener) { scope.patchPlugin_ = getPlugin(PluginPatches); scope.patches_ = []; scope.inversePatches_ = []; scope.patchListener_ = patchListener; } } function revokeScope(scope) { leaveScope(scope); scope.drafts_.forEach(revokeDraft); scope.drafts_ = null; } function leaveScope(scope) { if (scope === currentScope) { currentScope = scope.parent_; } } var enterScope = (immer2) => currentScope = createScope(currentScope, immer2); function revokeDraft(draft) { const state = draft[DRAFT_STATE]; if (state.type_ === 0 /* Object */ || state.type_ === 1 /* Array */) state.revoke_(); else state.revoked_ = true; } // src/core/finalize.ts function processResult(result, scope) { scope.unfinalizedDrafts_ = scope.drafts_.length; const baseDraft = scope.drafts_[0]; const isReplaced = result !== void 0 && result !== baseDraft; if (isReplaced) { if (baseDraft[DRAFT_STATE].modified_) { revokeScope(scope); die(4); } if (isDraftable(result)) { result = finalize(scope, result); } const { patchPlugin_ } = scope; if (patchPlugin_) { patchPlugin_.generateReplacementPatches_( baseDraft[DRAFT_STATE].base_, result, scope ); } } else { result = finalize(scope, baseDraft); } maybeFreeze(scope, result, true); revokeScope(scope); if (scope.patches_) { scope.patchListener_(scope.patches_, scope.inversePatches_); } return result !== NOTHING ? result : void 0; } function finalize(rootScope, value) { if (isFrozen(value)) return value; const state = value[DRAFT_STATE]; if (!state) { const finalValue = handleValue(value, rootScope.handledSet_, rootScope); return finalValue; } if (!isSameScope(state, rootScope)) { return value; } if (!state.modified_) { return state.base_; } if (!state.finalized_) { const { callbacks_ } = state; if (callbacks_) { while (callbacks_.length > 0) { const callback = callbacks_.pop(); callback(rootScope); } } generatePatchesAndFinalize(state, rootScope); } return state.copy_; } function maybeFreeze(scope, value, deep = false) { if (!scope.parent_ && scope.immer_.autoFreeze_ && scope.canAutoFreeze_) { freeze(value, deep); } } function markStateFinalized(state) { state.finalized_ = true; state.scope_.unfinalizedDrafts_--; } var isSameScope = (state, rootScope) => state.scope_ === rootScope; var EMPTY_LOCATIONS_RESULT = []; function updateDraftInParent(parent, draftValue, finalizedValue, originalKey) { const parentCopy = latest(parent); const parentType = parent.type_; if (originalKey !== void 0) { const currentValue = get(parentCopy, originalKey, parentType); if (currentValue === draftValue) { set(parentCopy, originalKey, finalizedValue, parentType); return; } } if (!parent.draftLocations_) { const draftLocations = parent.draftLocations_ = /* @__PURE__ */ new Map(); each(parentCopy, (key, value) => { if (isDraft(value)) { const keys = draftLocations.get(value) || []; keys.push(key); draftLocations.set(value, keys); } }); } const locations = parent.draftLocations_.get(draftValue) ?? EMPTY_LOCATIONS_RESULT; for (const location of locations) { set(parentCopy, location, finalizedValue, parentType); } } function registerChildFinalizationCallback(parent, child, key) { parent.callbacks_.push(function childCleanup(rootScope) { const state = child; if (!state || !isSameScope(state, rootScope)) { return; } rootScope.mapSetPlugin_?.fixSetContents(state); const finalizedValue = getFinalValue(state); updateDraftInParent(parent, state.draft_ ?? state, finalizedValue, key); generatePatchesAndFinalize(state, rootScope); }); } function generatePatchesAndFinalize(state, rootScope) { const shouldFinalize = state.modified_ && !state.finalized_ && (state.type_ === 3 /* Set */ || (state.assigned_?.size ?? 0) > 0); if (shouldFinalize) { const { patchPlugin_ } = rootScope; if (patchPlugin_) { const basePath = patchPlugin_.getPath(state); if (basePath) { patchPlugin_.generatePatches_(state, basePath, rootScope); } } markStateFinalized(state); } } function handleCrossReference(target, key, value) { const { scope_ } = target; if (isDraft(value)) { const state = value[DRAFT_STATE]; if (isSameScope(state, scope_)) { state.callbacks_.push(function crossReferenceCleanup() { prepareCopy(target); const finalizedValue = getFinalValue(state); updateDraftInParent(target, value, finalizedValue, key); }); } } else if (isDraftable(value)) { target.callbacks_.push(function nestedDraftCleanup() { const targetCopy = latest(target); if (get(targetCopy, key, target.type_) === value) { if (scope_.drafts_.length > 1 && (target.assigned_.get(key) ?? false) === true && target.copy_) { handleValue( get(target.copy_, key, target.type_), scope_.handledSet_, scope_ ); } } }); } } function handleValue(target, handledSet, rootScope) { if (!rootScope.immer_.autoFreeze_ && rootScope.unfinalizedDrafts_ < 1) { return target; } if (isDraft(target) || handledSet.has(target) || !isDraftable(target) || isFrozen(target)) { return target; } handledSet.add(target); each(target, (key, value) => { if (isDraft(value)) { const state = value[DRAFT_STATE]; if (isSameScope(state, rootScope)) { const updatedValue = getFinalValue(state); set(target, key, updatedValue, target.type_); markStateFinalized(state); } } else if (isDraftable(value)) { handleValue(value, handledSet, rootScope); } }); return target; } // src/core/proxy.ts function createProxyProxy(base, parent) { const baseIsArray = isArray(base); const state = { type_: baseIsArray ? 1 /* Array */ : 0 /* Object */, // Track which produce call this is associated with. scope_: parent ? parent.scope_ : getCurrentScope(), // True for both shallow and deep changes. modified_: false, // Used during finalization. finalized_: false, // Track which properties have been assigned (true) or deleted (false). // actually instantiated in `prepareCopy()` assigned_: void 0, // The parent draft state. parent_: parent, // The base state. base_: base, // The base proxy. draft_: null, // set below // The base copy with any updated values. copy_: null, // Called by the `produce` function. revoke_: null, isManual_: false, // `callbacks` actually gets assigned in `createProxy` callbacks_: void 0 }; let target = state; let traps = objectTraps; if (baseIsArray) { target = [state]; traps = arrayTraps; } const { revoke, proxy } = Proxy.revocable(target, traps); state.draft_ = proxy; state.revoke_ = revoke; return [proxy, state]; } var objectTraps = { get(state, prop) { if (prop === DRAFT_STATE) return state; const source = latest(state); if (!has(source, prop, state.type_)) { return readPropFromProto(state, source, prop); } const value = source[prop]; if (state.finalized_ || !isDraftable(value)) { return value; } if (value === peek(state.base_, prop)) { prepareCopy(state); const childKey = state.type_ === 1 /* Array */ ? +prop : prop; const childDraft = createProxy(state.scope_, value, state, childKey); return state.copy_[childKey] = childDraft; } return value; }, has(state, prop) { return prop in latest(state); }, ownKeys(state) { return Reflect.ownKeys(latest(state)); }, set(state, prop, value) { const desc = getDescriptorFromProto(latest(state), prop); if (desc?.set) { desc.set.call(state.draft_, value); return true; } if (!state.modified_) { const current2 = peek(latest(state), prop); const currentState = current2?.[DRAFT_STATE]; if (currentState && currentState.base_ === value) { state.copy_[prop] = value; state.assigned_.set(prop, false); return true; } if (is(value, current2) && (value !== void 0 || has(state.base_, prop, state.type_))) return true; prepareCopy(state); markChanged(state); } if (state.copy_[prop] === value && // special case: handle new props with value 'undefined' (value !== void 0 || prop in state.copy_) || // special case: NaN Number.isNaN(value) && Number.isNaN(state.copy_[prop])) return true; state.copy_[prop] = value; state.assigned_.set(prop, true); handleCrossReference(state, prop, value); return true; }, deleteProperty(state, prop) { prepareCopy(state); if (peek(state.base_, prop) !== void 0 || prop in state.base_) { state.assigned_.set(prop, false); markChanged(state); } else { state.assigned_.delete(prop); } if (state.copy_) { delete state.copy_[prop]; } return true; }, // Note: We never coerce `desc.value` into an Immer draft, because we can't make // the same guarantee in ES5 mode. getOwnPropertyDescriptor(state, prop) { const owner = latest(state); const desc = Reflect.getOwnPropertyDescriptor(owner, prop); if (!desc) return desc; return { [WRITABLE]: true, [CONFIGURABLE]: state.type_ !== 1 /* Array */ || prop !== "length", [ENUMERABLE]: desc[ENUMERABLE], [VALUE]: owner[prop] }; }, defineProperty() { die(11); }, getPrototypeOf(state) { return getPrototypeOf(state.base_); }, setPrototypeOf() { die(12); } }; var arrayTraps = {}; each(objectTraps, (key, fn) => { arrayTraps[key] = function() { const args = arguments; args[0] = args[0][0]; return fn.apply(this, args); }; }); arrayTraps.deleteProperty = function(state, prop) { if (process.env.NODE_ENV !== "production" && isNaN(parseInt(prop))) die(13); return arrayTraps.set.call(this, state, prop, void 0); }; arrayTraps.set = function(state, prop, value) { if (process.env.NODE_ENV !== "production" && prop !== "length" && isNaN(parseInt(prop))) die(14); return objectTraps.set.call(this, state[0], prop, value, state[0]); }; function peek(draft, prop) { const state = draft[DRAFT_STATE]; const source = state ? latest(state) : draft; return source[prop]; } function readPropFromProto(state, source, prop) { const desc = getDescriptorFromProto(source, prop); return desc ? VALUE in desc ? desc[VALUE] : ( // This is a very special case, if the prop is a getter defined by the // prototype, we should invoke it with the draft as context! desc.get?.call(state.draft_) ) : void 0; } function getDescriptorFromProto(source, prop) { if (!(prop in source)) return void 0; let proto = getPrototypeOf(source); while (proto) { const desc = Object.getOwnPropertyDescriptor(proto, prop); if (desc) return desc; proto = getPrototypeOf(proto); } return void 0; } function markChanged(state) { if (!state.modified_) { state.modified_ = true; if (state.parent_) { markChanged(state.parent_); } } } function prepareCopy(state) { if (!state.copy_) { state.assigned_ = /* @__PURE__ */ new Map(); state.copy_ = shallowCopy( state.base_, state.scope_.immer_.useStrictShallowCopy_ ); } } // src/core/immerClass.ts var Immer2 = class { constructor(config) { this.autoFreeze_ = true; this.useStrictShallowCopy_ = false; this.useStrictIteration_ = false; /** * The `produce` function takes a value and a "recipe function" (whose * return value often depends on the base state). The recipe function is * free to mutate its first argument however it wants. All mutations are * only ever applied to a __copy__ of the base state. * * Pass only a function to create a "curried producer" which relieves you * from passing the recipe function every time. * * Only plain objects and arrays are made mutable. All other objects are * considered uncopyable. * * Note: This function is __bound__ to its `Immer` instance. * * @param {any} base - the initial state * @param {Function} recipe - function that receives a proxy of the base state as first argument and which can be freely modified * @param {Function} patchListener - optional function that will be called with all the patches produced here * @returns {any} a new state, or the initial state if nothing was modified */ this.produce = (base, recipe, patchListener) => { if (isFunction(base) && !isFunction(recipe)) { const defaultBase = recipe; recipe = base; const self = this; return function curriedProduce(base2 = defaultBase, ...args) { return self.produce(base2, (draft) => recipe.call(this, draft, ...args)); }; } if (!isFunction(recipe)) die(6); if (patchListener !== void 0 && !isFunction(patchListener)) die(7); let result; if (isDraftable(base)) { const scope = enterScope(this); const proxy = createProxy(scope, base, void 0); let hasError = true; try { result = recipe(proxy); hasError = false; } finally { if (hasError) revokeScope(scope); else leaveScope(scope); } usePatchesInScope(scope, patchListener); return processResult(result, scope); } else if (!base || !isObjectish(base)) { result = recipe(base); if (result === void 0) result = base; if (result === NOTHING) result = void 0; if (this.autoFreeze_) freeze(result, true); if (patchListener) { const p = []; const ip = []; getPlugin(PluginPatches).generateReplacementPatches_(base, result, { patches_: p, inversePatches_: ip }); patchListener(p, ip); } return result; } else die(1, base); }; this.produceWithPatches = (base, recipe) => { if (isFunction(base)) { return (state, ...args) => this.produceWithPatches(state, (draft) => base(draft, ...args)); } let patches, inversePatches; const result = this.produce(base, recipe, (p, ip) => { patches = p; inversePatches = ip; }); return [result, patches, inversePatches]; }; if (isBoolean(config?.autoFreeze)) this.setAutoFreeze(config.autoFreeze); if (isBoolean(config?.useStrictShallowCopy)) this.setUseStrictShallowCopy(config.useStrictShallowCopy); if (isBoolean(config?.useStrictIteration)) this.setUseStrictIteration(config.useStrictIteration); } createDraft(base) { if (!isDraftable(base)) die(8); if (isDraft(base)) base = current(base); const scope = enterScope(this); const proxy = createProxy(scope, base, void 0); proxy[DRAFT_STATE].isManual_ = true; leaveScope(scope); return proxy; } finishDraft(draft, patchListener) { const state = draft && draft[DRAFT_STATE]; if (!state || !state.isManual_) die(9); const { scope_: scope } = state; usePatchesInScope(scope, patchListener); return processResult(void 0, scope); } /** * Pass true to automatically freeze all copies created by Immer. * * By default, auto-freezing is enabled. */ setAutoFreeze(value) { this.autoFreeze_ = value; } /** * Pass true to enable strict shallow copy. * * By default, immer does not copy the object descriptors such as getter, setter and non-enumrable properties. */ setUseStrictShallowCopy(value) { this.useStrictShallowCopy_ = value; } /** * Pass false to use faster iteration that skips non-enumerable properties * but still handles symbols for compatibility. * * By default, strict iteration is enabled (includes all own properties). */ setUseStrictIteration(value) { this.useStrictIteration_ = value; } shouldUseStrictIteration() { return this.useStrictIteration_; } applyPatches(base, patches) { let i; for (i = patches.length - 1; i >= 0; i--) { const patch = patches[i]; if (patch.path.length === 0 && patch.op === "replace") { base = patch.value; break; } } if (i > -1) { patches = patches.slice(i + 1); } const applyPatchesImpl = getPlugin(PluginPatches).applyPatches_; if (isDraft(base)) { return applyPatchesImpl(base, patches); } return this.produce( base, (draft) => applyPatchesImpl(draft, patches) ); } }; function createProxy(rootScope, value, parent, key) { const [draft, state] = isMap(value) ? getPlugin(PluginMapSet).proxyMap_(value, parent) : isSet(value) ? getPlugin(PluginMapSet).proxySet_(value, parent) : createProxyProxy(value, parent); const scope = parent?.scope_ ?? getCurrentScope(); scope.drafts_.push(draft); state.callbacks_ = parent?.callbacks_ ?? []; state.key_ = key; if (parent && key !== void 0) { registerChildFinalizationCallback(parent, state, key); } else { state.callbacks_.push(function rootDraftCleanup(rootScope2) { rootScope2.mapSetPlugin_?.fixSetContents(state); const { patchPlugin_ } = rootScope2; if (state.modified_ && patchPlugin_) { patchPlugin_.generatePatches_(state, [], rootScope2); } }); } return draft; } // src/core/current.ts function current(value) { if (!isDraft(value)) die(10, value); return currentImpl(value); } function currentImpl(value) { if (!isDraftable(value) || isFrozen(value)) return value; const state = value[DRAFT_STATE]; let copy; let strict = true; if (state) { if (!state.modified_) return state.base_; state.finalized_ = true; copy = shallowCopy(value, state.scope_.immer_.useStrictShallowCopy_); strict = state.scope_.immer_.shouldUseStrictIteration(); } else { copy = shallowCopy(value, true); } each( copy, (key, childValue) => { set(copy, key, currentImpl(childValue)); }, strict ); if (state) { state.finalized_ = false; } return copy; } // src/plugins/patches.ts function enablePatches() { const errorOffset = 16; if (process.env.NODE_ENV !== "production") { errors.push( 'Sets cannot have "replace" patches.', function(op) { return "Unsupported patch operation: " + op; }, function(path) { return "Cannot apply patch, path doesn't resolve: " + path; }, "Patching reserved attributes like __proto__, prototype and constructor is not allowed" ); } function getPath(state, path = []) { if ("key_" in state && state.key_ !== void 0) { const parentCopy = state.parent_.copy_ ?? state.parent_.base_; const proxyDraft = getProxyDraft(get(parentCopy, state.key_)); const valueAtKey = get(parentCopy, state.key_); if (valueAtKey === void 0) { return null; } if (valueAtKey !== state.draft_ && valueAtKey !== state.base_ && valueAtKey !== state.copy_) { return null; } if (proxyDraft != null && proxyDraft.base_ !== state.base_) { return null; } const isSet2 = state.parent_.type_ === 3 /* Set */; let key; if (isSet2) { const setParent = state.parent_; key = Array.from(setParent.drafts_.keys()).indexOf(state.key_); } else { key = state.key_; } if (!(isSet2 && parentCopy.size > key || has(parentCopy, key))) { return null; } path.push(key); } if (state.parent_) { return getPath(state.parent_, path); } path.reverse(); try { resolvePath(state.copy_, path); } catch (e) { return null; } return path; } function resolvePath(base, path) { let current2 = base; for (let i = 0; i < path.length - 1; i++) { const key = path[i]; current2 = get(current2, key); if (!isObjectish(current2) || current2 === null) { throw new Error(`Cannot resolve path at '${path.join("/")}'`); } } return current2; } const REPLACE = "replace"; const ADD = "add"; const REMOVE = "remove"; function generatePatches_(state, basePath, scope) { if (state.scope_.processedForPatches_.has(state)) { return; } state.scope_.processedForPatches_.add(state); const { patches_, inversePatches_ } = scope; switch (state.type_) { case 0 /* Object */: case 2 /* Map */: return generatePatchesFromAssigned( state, basePath, patches_, inversePatches_ ); case 1 /* Array */: return generateArrayPatches( state, basePath, patches_, inversePatches_ ); case 3 /* Set */: return generateSetPatches( state, basePath, patches_, inversePatches_ ); } } function generateArrayPatches(state, basePath, patches, inversePatches) { let { base_, assigned_ } = state; let copy_ = state.copy_; if (copy_.length < base_.length) { ; [base_, copy_] = [copy_, base_]; [patches, inversePatches] = [inversePatches, patches]; } for (let i = 0; i < base_.length; i++) { const copiedItem = copy_[i]; const baseItem = base_[i]; if (assigned_?.get(i.toString()) && copiedItem !== baseItem) { const childState = copiedItem?.[DRAFT_STATE]; if (childState && childState.modified_) { continue; } const path = basePath.concat([i]); patches.push({ op: REPLACE, path, // Need to maybe clone it, as it can in fact be the original value // due to the base/copy inversion at the start of this function value: clonePatchValueIfNeeded(copiedItem) }); inversePatches.push({ op: REPLACE, path, value: clonePatchValueIfNeeded(baseItem) }); } } for (let i = base_.length; i < copy_.length; i++) { const path = basePath.concat([i]); patches.push({ op: ADD, path, // Need to maybe clone it, as it can in fact be the original value // due to the base/copy inversion at the start of this function value: clonePatchValueIfNeeded(copy_[i]) }); } for (let i = copy_.length - 1; base_.length <= i; --i) { const path = basePath.concat([i]); inversePatches.push({ op: REMOVE, path }); } } function generatePatchesFromAssigned(state, basePath, patches, inversePatches) { const { base_, copy_, type_ } = state; each(state.assigned_, (key, assignedValue) => { const origValue = get(base_, key, type_); const value = get(copy_, key, type_); const op = !assignedValue ? REMOVE : has(base_, key) ? REPLACE : ADD; if (origValue === value && op === REPLACE) return; const path = basePath.concat(key); patches.push( op === REMOVE ? { op, path } : { op, path, value: clonePatchValueIfNeeded(value) } ); inversePatches.push( op === ADD ? { op: REMOVE, path } : op === REMOVE ? { op: ADD, path, value: clonePatchValueIfNeeded(origValue) } : { op: REPLACE, path, value: clonePatchValueIfNeeded(origValue) } ); }); } function generateSetPatches(state, basePath, patches, inversePatches) { let { base_, copy_ } = state; let i = 0; base_.forEach((value) => { if (!copy_.has(value)) { const path = basePath.concat([i]); patches.push({ op: REMOVE, path, value }); inversePatches.unshift({ op: ADD, path, value }); } i++; }); i = 0; copy_.forEach((value) => { if (!base_.has(value)) { const path = basePath.concat([i]); patches.push({ op: ADD, path, value }); inversePatches.unshift({ op: REMOVE, path, value }); } i++; }); } function generateReplacementPatches_(baseValue, replacement, scope) { const { patches_, inversePatches_ } = scope; patches_.push({ op: REPLACE, path: [], value: replacement === NOTHING ? void 0 : replacement }); inversePatches_.push({ op: REPLACE, path: [], value: baseValue }); } function applyPatches_(draft, patches) { patches.forEach((patch) => { const { path, op } = patch; let base = draft; for (let i = 0; i < path.length - 1; i++) { const parentType = getArchtype(base); let p = path[i]; if (typeof p !== "string" && typeof p !== "number") { p = "" + p; } if ((parentType === 0 /* Object */ || parentType === 1 /* Array */) && (p === "__proto__" || p === CONSTRUCTOR)) die(errorOffset + 3); if (isFunction(base) && p === PROTOTYPE) die(errorOffset + 3); base = get(base, p); if (!isObjectish(base)) die(errorOffset + 2, path.join("/")); } const type = getArchtype(base); const value = deepClonePatchValue(patch.value); const key = path[path.length - 1]; switch (op) { case REPLACE: switch (type) { case 2 /* Map */: return base.set(key, value); case 3 /* Set */: die(errorOffset); default: return base[key] = value; } case ADD: switch (type) { case 1 /* Array */: return key === "-" ? base.push(value) : base.splice(key, 0, value); case 2 /* Map */: return base.set(key, value); case 3 /* Set */: return base.add(value); default: return base[key] = value; } case REMOVE: switch (type) { case 1 /* Array */: return base.splice(key, 1); case 2 /* Map */: return base.delete(key); case 3 /* Set */: return base.delete(patch.value); default: return delete base[key]; } default: die(errorOffset + 1, op); } }); return draft; } function deepClonePatchValue(obj) { if (!isDraftable(obj)) return obj; if (isArray(obj)) return obj.map(deepClonePatchValue); if (isMap(obj)) return new Map( Array.from(obj.entries()).map(([k, v]) => [k, deepClonePatchValue(v)]) ); if (isSet(obj)) return new Set(Array.from(obj).map(deepClonePatchValue)); const cloned = Object.create(getPrototypeOf(obj)); for (const key in obj) cloned[key] = deepClonePatchValue(obj[key]); if (has(obj, DRAFTABLE)) cloned[DRAFTABLE] = obj[DRAFTABLE]; return cloned; } function clonePatchValueIfNeeded(obj) { if (isDraft(obj)) { return deepClonePatchValue(obj); } else return obj; } loadPlugin(PluginPatches, { applyPatches_, generatePatches_, generateReplacementPatches_, getPath }); } // src/plugins/mapset.ts function enableMapSet() { class DraftMap extends Map { constructor(target, parent) { super(); this[DRAFT_STATE] = { type_: 2 /* Map */, parent_: parent, scope_: parent ? parent.scope_ : getCurrentScope(), modified_: false, finalized_: false, copy_: void 0, assigned_: void 0, base_: target, draft_: this, isManual_: false, revoked_: false, callbacks_: [] }; } get size() { return latest(this[DRAFT_STATE]).size; } has(key) { return latest(this[DRAFT_STATE]).has(key); } set(key, value) { const state = this[DRAFT_STATE]; assertUnrevoked(state); if (!latest(state).has(key) || latest(state).get(key) !== value) { prepareMapCopy(state); markChanged(state); state.assigned_.set(key, true); state.copy_.set(key, value); state.assigned_.set(key, true); } return this; } delete(key) { if (!this.has(key)) { return false; } const state = this[DRAFT_STATE]; assertUnrevoked(state); prepareMapCopy(state); markChanged(state); if (state.base_.has(key)) { state.assigned_.set(key, false); } else { state.assigned_.delete(key); } state.copy_.delete(key); return true; } clear() { const state = this[DRAFT_STATE]; assertUnrevoked(state); if (latest(state).size) { prepareMapCopy(state); markChanged(state); state.assigned_ = /* @__PURE__ */ new Map(); each(state.base_, (key) => { state.assigned_.set(key, false); }); state.copy_.clear(); } } forEach(cb, thisArg) { const state = this[DRAFT_STATE]; latest(state).forEach((_value, key, _map) => { cb.call(thisArg, this.get(key), key, this); }); } get(key) { const state = this[DRAFT_STATE]; assertUnrevoked(state); const value = latest(state).get(key); if (state.finalized_ || !isDraftable(value)) { return value; } if (value !== state.base_.get(key)) { return value; } const draft = createProxy(state.scope_, value, state, key); prepareMapCopy(state); state.copy_.set(key, draft); return draft; } keys() { return latest(this[DRAFT_STATE]).keys(); } values() { const iterator = this.keys(); return { [Symbol.iterator]: () => this.values(), next: () => { const r = iterator.next(); if (r.done) return r; const value = this.get(r.value); return { done: false, value }; } }; } entries() { const iterator = this.keys(); return { [Symbol.iterator]: () => this.entries(), next: () => { const r = iterator.next(); if (r.done) return r; const value = this.get(r.value); return { done: false, value: [r.value, value] }; } }; } [(DRAFT_STATE, Symbol.iterator)]() { return this.entries(); } } function proxyMap_(target, parent) { const map = new DraftMap(target, parent); return [map, map[DRAFT_STATE]]; } function prepareMapCopy(state) { if (!state.copy_) { state.assigned_ = /* @__PURE__ */ new Map(); state.copy_ = new Map(state.base_); } } class DraftSet extends Set { constructor(target, parent) { super(); this[DRAFT_STATE] = { type_: 3 /* Set */, parent_: parent, scope_: parent ? parent.scope_ : getCurrentScope(), modified_: false, finalized_: false, copy_: void 0, base_: target, draft_: this, drafts_: /* @__PURE__ */ new Map(), revoked_: false, isManual_: false, assigned_: void 0, callbacks_: [] }; } get size() { return latest(this[DRAFT_STATE]).size; } has(value) { const state = this[DRAFT_STATE]; assertUnrevoked(state); if (!state.copy_) { return state.base_.has(value); } if (state.copy_.has(value)) return true; if (state.drafts_.has(value) && state.copy_.has(state.drafts_.get(value))) return true; return false; } add(value) { const state = this[DRAFT_STATE]; assertUnrevoked(state); if (!this.has(value)) { prepareSetCopy(state); markChanged(state); state.copy_.add(value); } return this; } delete(value) { if (!this.has(value)) { return false; } const state = this[DRAFT_STATE]; assertUnrevoked(state); prepareSetCopy(state); markChanged(state); return state.copy_.delete(value) || (state.drafts_.has(value) ? state.copy_.delete(state.drafts_.get(value)) : ( /* istanbul ignore next */ false )); } clear() { const state = this[DRAFT_STATE]; assertUnrevoked(state); if (latest(state).size) { prepareSetCopy(state); markChanged(state); state.copy_.clear(); } } values() { const state = this[DRAFT_STATE]; assertUnrevoked(state); prepareSetCopy(state); return state.copy_.values(); } entries() { const state = this[DRAFT_STATE]; assertUnrevoked(state); prepareSetCopy(state); return state.copy_.entries(); } keys() { return this.values(); } [(DRAFT_STATE, Symbol.iterator)]() { return this.values(); } forEach(cb, thisArg) { const iterator = this.values(); let result = iterator.next(); while (!result.done) { cb.call(thisArg, result.value, result.value, this); result = iterator.next(); } } } function proxySet_(target, parent) { const set2 = new DraftSet(target, parent); return [set2, set2[DRAFT_STATE]]; } function prepareSetCopy(state) { if (!state.copy_) { state.copy_ = /* @__PURE__ */ new Set(); state.base_.forEach((value) => { if (isDraftable(value)) { const draft = createProxy(state.scope_, value, state, value); state.drafts_.set(value, draft); state.copy_.add(draft); } else { state.copy_.add(value); } }); } } function assertUnrevoked(state) { if (state.revoked_) die(3, JSON.stringify(latest(state))); } function fixSetContents(target) { if (target.type_ === 3 /* Set */ && target.copy_) { const copy = new Set(target.copy_); target.copy_.clear(); copy.forEach((value) => { target.copy_.add(getValue(value)); }); } } loadPlugin(PluginMapSet, { proxyMap_, proxySet_, fixSetContents }); } // src/immer.ts var immer = new Immer2(); var produce = immer.produce; var produceWithPatches = /* @__PURE__ */ immer.produceWithPatches.bind( immer ); var setAutoFreeze = /* @__PURE__ */ immer.setAutoFreeze.bind(immer); var setUseStrictShallowCopy = /* @__PURE__ */ immer.setUseStrictShallowCopy.bind( immer ); var setUseStrictIteration = /* @__PURE__ */ immer.setUseStrictIteration.bind( immer ); var applyPatches = /* @__PURE__ */ immer.applyPatches.bind(immer); var createDraft = /* @__PURE__ */ immer.createDraft.bind(immer); var finishDraft = /* @__PURE__ */ immer.finishDraft.bind(immer); var castDraft = (value) => value; var castImmutable = (value) => value; // Annotate the CommonJS export names for ESM import in node: 0 && (module.exports = { Immer, applyPatches, castDraft, castImmutable, createDraft, current, enableMapSet, enablePatches, finishDraft, freeze, immerable, isDraft, isDraftable, nothing, original, produce, produceWithPatches, setAutoFreeze, setUseStrictIteration, setUseStrictShallowCopy }); //# sourceMappingURL=immer.cjs.development.js.map