cross-state 1.8.5 → 1.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{diff-BZm7wZ6G.d.cts → diff-Bcr0h7-x.d.cts} +2 -1
- package/dist/{diff-DxjoO6qz.d.ts → diff-CwiTYAVK.d.ts} +2 -1
- package/dist/extendedJson-DA3RQ_ly.js +138 -0
- package/dist/extendedJson-DA3RQ_ly.js.map +1 -0
- package/dist/extendedJson-DzXMFE1P.cjs +167 -0
- package/dist/extendedJson-DzXMFE1P.cjs.map +1 -0
- package/dist/index.cjs +4 -4
- package/dist/index.d.cts +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +3 -3
- package/dist/{patchMethods-CxI_x-zv.d.ts → patchMethods-C21iQqML.d.ts} +2 -2
- package/dist/{patchMethods-BfyJt9me.cjs → patchMethods-CZje49Dk.cjs} +4 -117
- package/dist/patchMethods-CZje49Dk.cjs.map +1 -0
- package/dist/{patchMethods-D-Vmx93m.js → patchMethods-Cui462lc.js} +4 -111
- package/dist/patchMethods-Cui462lc.js.map +1 -0
- package/dist/{patchMethods-DjHq-wbw.d.cts → patchMethods-Cyb6MEOG.d.cts} +2 -2
- package/dist/patches/index.cjs +1 -1
- package/dist/patches/index.d.cts +1 -1
- package/dist/patches/index.d.ts +1 -1
- package/dist/patches/index.js +1 -1
- package/dist/patches/register.cjs +1 -1
- package/dist/patches/register.d.cts +1 -1
- package/dist/patches/register.d.ts +1 -1
- package/dist/patches/register.js +1 -1
- package/dist/persist/register.cjs +1 -1
- package/dist/persist/register.js +1 -1
- package/dist/{persist-CwKzFRM7.cjs → persist-CG1WHP6D.cjs} +3 -3
- package/dist/{persist-CwKzFRM7.cjs.map → persist-CG1WHP6D.cjs.map} +1 -1
- package/dist/{persist-CZhDrupc.js → persist-CNA2sYtB.js} +3 -3
- package/dist/{persist-CZhDrupc.js.map → persist-CNA2sYtB.js.map} +1 -1
- package/dist/react/index.cjs +41 -3
- package/dist/react/index.cjs.map +1 -1
- package/dist/react/index.d.cts +9 -0
- package/dist/react/index.d.ts +9 -0
- package/dist/react/index.js +41 -3
- package/dist/react/index.js.map +1 -1
- package/package.json +1 -1
- package/dist/extendedJson-B1OEqZtF.js +0 -29
- package/dist/extendedJson-B1OEqZtF.js.map +0 -1
- package/dist/extendedJson-BwYSvpOA.cjs +0 -53
- package/dist/extendedJson-BwYSvpOA.cjs.map +0 -1
- package/dist/patchMethods-BfyJt9me.cjs.map +0 -1
- package/dist/patchMethods-D-Vmx93m.js.map +0 -1
package/dist/react/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":["matches: Record<KeyType, any>","GeneralFormContext: Context<Form<any, any> | null>","value","form","form","keys: (string | number)[]","name","FormContainer","options: FormOptions<TDraft, TOriginal>","form","lastDraft: TDraft | undefined","context: FormContext<TDraft, TOriginal>","reportValidity","options","transform","Form","Component","urlStore: Store<string>","urlOptions: Required<UrlOptions<T>>","path: any","update: Update<any>","update","value","delayedUpdate: (value: T) => void"],"sources":["../../src/react/form/customInput.tsx","../../src/lib/wildcardMatch.ts","../../src/react/form/closestFormContext.tsx","../../src/react/form/legacyFormField.tsx","../../src/react/form/formField.tsx","../../src/react/form/formForEach.tsx","../../src/react/form/useFormAutosave.ts","../../src/react/form/form.tsx","../../src/lib/castArray.ts","../../src/react/url/urlHelpers.ts","../../src/react/url/urlOptions.ts","../../src/react/url/urlParamStore.ts","../../src/react/useDecoupledState.ts"],"sourcesContent":["import type { ReactNode } from 'react';\n\nexport interface CustomInputProps extends React.HTMLAttributes<HTMLDivElement> {\n name: string;\n children?: ReactNode;\n}\n\nexport function CustomInput({ name, children, ...props }: CustomInputProps): React.JSX.Element {\n return (\n <div\n {...props}\n style={{\n position: 'relative',\n ...props.style,\n }}\n >\n {children}\n\n <input\n name={name}\n style={{\n position: 'absolute',\n top: 0,\n left: 0,\n opacity: 0,\n width: '100%',\n height: '100%',\n pointerEvents: 'none',\n }}\n />\n </div>\n );\n}\n","import { isObject } from '@lib/helpers';\nimport { type KeyType } from './path';\nimport { castArrayPath } from './propAccess';\n\nexport function wildcardMatch(s: KeyType[] | string, w: KeyType[] | string): boolean {\n if (typeof s === 'string') {\n s = castArrayPath(s);\n }\n\n if (typeof w === 'string') {\n w = castArrayPath(w);\n }\n\n return s.length === w.length && s.every((s, i) => w[i] === '*' || s === w[i]);\n}\n\nexport function getWildCardMatches(\n object: any,\n path: [KeyType, ...KeyType[]] | string,\n): Record<KeyType, any> {\n const matches: Record<KeyType, any> = {};\n const [first, second, ...rest] = castArrayPath(path);\n\n if (first === undefined) {\n throw new Error('Path is empty');\n }\n\n if (!Array.isArray(object) && !isObject(object)) {\n object = {};\n }\n\n for (const [key, value] of first !== '*' ? [[first, object[first]]] : Object.entries(object)) {\n if (first !== '*' && first !== key) {\n continue;\n }\n\n if (second === undefined) {\n matches[key] = value;\n continue;\n }\n\n for (const [subKey, subValue] of Object.entries(getWildCardMatches(value, [second, ...rest]))) {\n matches[`${key}.${subKey}`] = subValue;\n }\n }\n\n return matches;\n}\n","import { type Form } from '@react/form/form';\nimport { createContext, useContext, type Context } from 'react';\n\nexport const GeneralFormContext: Context<Form<any, any> | null> = createContext<Form<\n any,\n any\n> | null>(null);\n\nexport function useClosestForm<TDraft, TOriginal extends TDraft = TDraft>(): Form<\n TDraft,\n TOriginal\n> | null {\n return useContext(GeneralFormContext) as Form<TDraft, TOriginal> | null;\n}\n","import { calcDuration } from '@lib/duration';\nimport type { Value } from '@lib/path';\nimport { getDerivedState, type Form, type FormInstance } from '@react/form/form';\nimport type {\n FormFieldComponent,\n FormFieldComponentProps,\n FormFieldProps,\n} from '@react/form/formField';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { createElement, useEffect, useState, type ComponentPropsWithoutRef } from 'react';\n\ntype FieldValue<T extends FormFieldComponent> = ComponentPropsWithoutRef<T>['value'];\n\ntype FieldChangeValue<T extends FormFieldComponent> =\n ComponentPropsWithoutRef<T> extends {\n onChange?: (update: infer U) => void;\n }\n ? U extends { target: { value: infer V } }\n ? V\n : U\n : never;\n\ntype MakeOptional<T, Keys extends string> = Omit<T, Keys> & Partial<Pick<T, Keys & keyof T>>;\n\ntype Serialize<TDraft, TOriginal, TPath, TComponent extends FormFieldComponent> = (\n value: Value<TDraft, TPath>,\n formState: FormInstance<TDraft, TOriginal>,\n) => FieldValue<TComponent>;\n\ntype Deserialize<TDraft, TOriginal, TPath, TComponent extends FormFieldComponent> = (\n value: FieldChangeValue<TComponent>,\n formState: FormInstance<TDraft, TOriginal>,\n) => Value<TDraft, TPath>;\n\nexport type FormFieldPropsWithComponent<\n TDraft,\n TOriginal,\n TPath extends string,\n TComponent extends FormFieldComponent,\n> = FormFieldProps<TPath, TDraft> & {\n component: TComponent;\n render?: undefined;\n} & NoInfer<\n {\n inputFilter?: (value: FieldChangeValue<TComponent>) => boolean;\n } & MakeOptional<\n Omit<ComponentPropsWithoutRef<TComponent>, 'id' | 'name' | 'value' | 'defaultValue'>,\n 'onChange' | 'onBlur'\n > &\n (Value<TDraft, TPath> extends Exclude<FieldValue<TComponent>, undefined>\n ? {\n defaultValue?: FieldValue<TComponent>;\n serialize?: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : Value<TDraft, TPath> extends FieldValue<TComponent>\n ?\n | {\n defaultValue: FieldValue<TComponent>;\n serialize?: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n | {\n defaultValue?: FieldValue<TComponent>;\n serialize: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : {\n serialize: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }) &\n (FieldChangeValue<TComponent> extends Value<TDraft, TPath>\n ? {\n deserialize?: Deserialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : {\n deserialize: Deserialize<TDraft, TOriginal, TPath, TComponent>;\n })\n >;\n\nexport function LegacyFormField<\n TDraft,\n TOriginal,\n TPath extends string,\n TComponent extends FormFieldComponent,\n>(\n this: Form<TDraft, any>,\n {\n // id,\n name = '' as any,\n component,\n commitOnBlur,\n commitDebounce,\n inputFilter,\n defaultValue,\n serialize,\n deserialize = (x) => x as Value<TDraft, TPath>,\n onChange,\n onBlur,\n ...restProps\n }: FormFieldPropsWithComponent<TDraft, TOriginal, TPath, TComponent>,\n): React.JSX.Element | null {\n type T = FieldChangeValue<TComponent>;\n\n const form = this.useForm();\n const getFormState = () => ({ ...form, ...getDerivedState(form) });\n const [localValue, setLocalValue] = useState<T>();\n\n const value = this.useFormState((form) => {\n const value = form.getField(name as any).value;\n if (serialize) {\n return serialize(value as any, getFormState());\n }\n if (value !== undefined) {\n return value;\n }\n return defaultValue;\n });\n\n const setValue = useLatestFunction((x: FieldChangeValue<TComponent>) =>\n form.getField(name as any).setValue(deserialize(x, getFormState())),\n );\n\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n\n const commitDebounceMs = commitDebounce !== undefined ? calcDuration(commitDebounce) : undefined;\n useEffect(() => {\n if (localValue === undefined || commitDebounceMs === undefined || commitDebounceMs <= 0) {\n return;\n }\n\n const timeout = setTimeout(() => {\n setValue(localValue);\n setLocalValue(undefined);\n }, commitDebounceMs);\n\n return () => clearTimeout(timeout);\n }, [localValue, commitDebounceMs, setValue]);\n\n let props = {\n name,\n value: localValue ?? value,\n onChange: (event: { target: { value: T } } | T, ...moreArgs: any[]) => {\n const value =\n typeof event === 'object' && event !== null && 'target' in event\n ? event.target.value\n : event;\n\n if (inputFilter && !inputFilter(value)) {\n return;\n }\n\n if (commitOnBlur || commitDebounce) {\n setLocalValue(value);\n } else {\n setValue(value);\n }\n\n onChange?.(event, ...moreArgs);\n },\n onBlur(...args: any[]) {\n if (localValue !== undefined) {\n setValue(localValue);\n setLocalValue(undefined);\n }\n\n onBlur?.(...args);\n },\n } as FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n\n if (this.options.transformFieldProps) {\n props = this.options.transformFieldProps(\n props,\n { ...form.getField(name as any), hasTriggeredValidations } as any,\n form,\n );\n }\n\n if (component) {\n return createElement(component, { ...restProps, ...props });\n }\n\n return null;\n}\n","import type { Duration } from '@core';\nimport { calcDuration } from '@lib/duration';\nimport { type PathAsString, type Value } from '@lib/path';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { useEffect, useState, type Component, type ReactNode } from 'react';\nimport { Form, type Field, type FieldOptions, type FormContext } from './form';\n\nexport interface FormFieldComponentProps<TValue, TPath> {\n name: TPath;\n value: TValue;\n onChange: (event: { target: { value: TValue } } | TValue | undefined, ...args: any[]) => void;\n onBlur: (...args: any[]) => void;\n 'data-invalid'?: boolean;\n}\n\nexport type FormFieldInfos<TDraft, TOriginal, TPath extends string> = Field<\n TDraft,\n TOriginal,\n TPath\n> & {\n hasTriggeredValidations: boolean;\n};\n\ntype NativeInputType = 'input' | 'select' | 'textarea';\n\ntype PartialComponentType<P> =\n | (new (props: P, context?: any) => Component<P, any>)\n | ((props: P, context?: any) => ReactNode);\n\nexport type FormFieldComponent = NativeInputType | PartialComponentType<any>;\n\nexport interface FormFieldProps<TPath, TDraft> extends FieldOptions {\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>;\n commitOnBlur?: boolean;\n commitDebounce?: Duration;\n}\n\nexport interface FormFieldPropsWithRender<\n TDraft,\n TOriginal,\n TPath extends string,\n> extends FormFieldProps<TPath, TDraft> {\n render: NoInfer<\n (\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => ReactNode\n >;\n children?: undefined;\n}\n\nexport interface FormFieldPropsWithChildren<\n TDraft,\n TOriginal,\n TPath extends string,\n> extends FormFieldProps<TPath, TDraft> {\n render?: undefined;\n children: NoInfer<\n (\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => ReactNode\n >;\n}\n\nexport function FormField<TDraft, TOriginal extends TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name = '' as any,\n commitOnBlur,\n commitDebounce,\n children,\n render = children,\n includeNestedErrors,\n }:\n | FormFieldPropsWithRender<TDraft, TOriginal, TPath>\n | FormFieldPropsWithChildren<TDraft, TOriginal, TPath>,\n): React.JSX.Element | null {\n const form = this.useForm();\n const field = this.useField(name);\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n\n const renderProps = useFormFieldProps.call<\n Form<TDraft, TOriginal>,\n [FormFieldProps<TPath, TDraft>],\n FormFieldComponentProps<Value<TDraft, TPath>, TPath>\n >(this, {\n name,\n commitOnBlur,\n commitDebounce,\n includeNestedErrors,\n });\n\n if (render) {\n return <>{render(renderProps, { ...field, hasTriggeredValidations } as any, form)}</>;\n }\n\n return null;\n}\n\nexport function useFormFieldProps<TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name = '' as any,\n commitOnBlur,\n commitDebounce,\n includeNestedErrors,\n }: FormFieldProps<TPath, TDraft>,\n): FormFieldComponentProps<Value<TDraft, TPath>, TPath> {\n type T = Value<TDraft, TPath>;\n\n const form = this.useForm();\n const field = this.useField(name, { includeNestedErrors });\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n const [localValue, setLocalValue] = useState<{ v: T }>();\n\n const commitDebounceMs = commitDebounce !== undefined ? calcDuration(commitDebounce) : undefined;\n\n const commitLocalValue = useLatestFunction(() => {\n if (localValue) {\n field.setValue(localValue.v);\n setLocalValue(undefined);\n }\n });\n\n useEffect(() => {\n if (!localValue || commitDebounceMs === undefined || commitDebounceMs <= 0) {\n return;\n }\n\n const timeout = setTimeout(commitLocalValue, commitDebounceMs);\n return () => clearTimeout(timeout);\n }, [localValue, commitDebounceMs, commitLocalValue]);\n\n let props = {\n name,\n value: localValue ? localValue.v : field.value,\n onChange: useLatestFunction((event: { target: { value: T } } | T) => {\n const value =\n typeof event === 'object' && event !== null && 'target' in event\n ? event.target.value\n : event;\n\n if (commitOnBlur || (commitDebounceMs !== undefined && commitDebounceMs > 0)) {\n setLocalValue({ v: value });\n } else {\n field.setValue(value as any);\n }\n }),\n onBlur: useLatestFunction(() => {\n if (localValue) {\n commitLocalValue();\n }\n }),\n 'data-invalid': field.errors.length > 0 ? true : undefined,\n } as FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n\n if (this.options.transformFieldProps) {\n props = this.options.transformFieldProps(\n props,\n { ...field, hasTriggeredValidations } as any,\n form,\n );\n }\n\n return props;\n}\n","import { isObject } from '@lib/helpers';\nimport { type GetKeys, type Join, type PathAsString, type Value } from '@lib/path';\nimport { join } from '@lib/propAccess';\nimport { Fragment, useCallback, type ReactNode } from 'react';\nimport { type FieldHelperMethods, type Form } from './form';\n\nexport type ElementName<TDraft, TPath extends string> = keyof {\n [Path in TPath as Join<Path, GetKeys<NonNullable<Value<TDraft, Path>>> & (string | number)>]: 1;\n};\n\ntype ItemValue<T> = T extends readonly (infer U)[] ? U : T[keyof T];\n\nexport interface FormForEachProps<TDraft, TPath extends string> {\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>;\n renderElement?: (props: {\n name: ElementName<TDraft, TPath>;\n key: `${GetKeys<NonNullable<Value<TDraft, TPath>>> & (string | number)}`;\n index: number;\n remove: () => void;\n count: number;\n }) => ReactNode;\n renderAdditionalElement?: NonNullable<Value<TDraft, TPath>> extends readonly any[]\n ? boolean\n : never;\n filter?: (\n item: ItemValue<NonNullable<Value<TDraft, TPath>>>,\n key: GetKeys<NonNullable<Value<TDraft, TPath>>>,\n parent: NonNullable<Value<TDraft, TPath>>,\n ) => boolean;\n children?: (\n props: {\n setValue: (\n value: Value<TDraft, TPath> | ((value: Value<TDraft, TPath>) => Value<TDraft, TPath>),\n ) => void;\n } & FieldHelperMethods<TDraft, TPath>,\n ) => ReactNode;\n}\n\nexport function FormForEach<TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name,\n renderElement,\n renderAdditionalElement,\n filter,\n children,\n }: FormForEachProps<TDraft, TPath>,\n): React.JSX.Element {\n const form = this.useForm();\n\n const items = this.useFormState(() => {\n const field = form.getField(name);\n\n let keys: (string | number)[] = isObject(field.value) ? Object.keys(field.value) : [];\n const count = keys.length;\n\n if (Array.isArray(field.value)) {\n keys = keys.map(Number);\n }\n\n if (filter) {\n keys = keys.filter((key, index) =>\n filter((field.value as any)[index], key as any, field.value as any),\n );\n }\n\n if (renderAdditionalElement) {\n keys.push(count);\n }\n\n return keys.map((key) => ({\n key,\n name: join(name, String(key)),\n }));\n });\n\n const add = useCallback(\n (...args: any[]) => {\n const field = form.getField(name as any) as any;\n field.add(...args);\n },\n [form, name],\n );\n\n const remove = useCallback(\n (key: any) => {\n const field = form.getField(name as any) as any;\n field.remove(key);\n },\n [form, name],\n );\n\n const setValue = useCallback(\n (value: Value<TDraft, TPath> | ((value: Value<TDraft, TPath>) => Value<TDraft, TPath>)) => {\n const field = form.getField(name as any) as any;\n field.setValue(value);\n },\n [form, name],\n );\n\n return (\n <>\n {renderElement &&\n items.map(({ key, name }, index) => {\n return (\n <Fragment key={key}>\n {renderElement({\n name: name as any,\n key: key as any,\n index,\n remove: () => remove(key),\n count: items.length,\n })}\n </Fragment>\n );\n })}\n\n {children?.({\n names: items.map((item) => item.name),\n add,\n remove,\n setValue,\n } as any)}\n </>\n );\n}\n","import type { Duration } from '@core';\nimport { debounce } from '@lib/debounce';\nimport { calcDuration } from '@lib/duration';\nimport { deepEqual } from '@lib/equals';\nimport type { MaybePromise } from '@lib/maybePromise';\nimport { queue } from '@lib/queue';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { useEffect, useMemo, useRef } from 'react';\nimport type { FormContext } from './form';\n\nexport interface FormAutosaveOptions<TDraft, TOriginal> {\n enabled?: boolean;\n validateBeforeSave?: boolean;\n save?: (draft: TDraft, prev: TDraft, form: FormContext<TDraft, TOriginal>) => MaybePromise<void>;\n debounce?: Duration;\n resetAfterSave?: boolean;\n equals?: (a: any, b: any) => boolean;\n}\n\nexport function useFormAutosave<TDraft, TOriginal extends TDraft>(\n form: FormContext<TDraft, TOriginal>,\n): {\n flush(): Promise<void>;\n cancel(): Promise<void>;\n} {\n const { enabled = true, validateBeforeSave = true } = form.options.autoSave ?? {};\n\n const isActive = enabled && !!form.options.autoSave?.save;\n const debounceTime = calcDuration(form.options.autoSave?.debounce ?? 2_000);\n const prev = useRef(form.getDraft());\n const q = useMemo(() => queue(), []);\n\n const latestSave = useLatestFunction(async () => {\n if (validateBeforeSave && !form.validate()) {\n return;\n }\n\n const draft = form.getDraft();\n const equals =\n form.options.autoSave?.equals ?? ((a, b) => deepEqual(a, b, { undefinedEqualsAbsent: true }));\n\n if (!isActive || equals(draft, prev.current) || equals(draft, form.options.original)) {\n return;\n }\n\n form.formState.set('saveInProgress', true);\n\n try {\n await form.options.autoSave?.save?.(draft, prev.current, form);\n prev.current = draft;\n\n if (q.size === 0 && form.options.autoSave?.resetAfterSave) {\n form.reset();\n }\n } catch (error) {\n console.error('Unhandled error during form autosave:', error);\n } finally {\n form.formState.set('saveInProgress', false);\n }\n });\n\n const scheduleSave = useMemo(\n () =>\n debounce(() => {\n q.clear();\n q(latestSave);\n }, debounceTime),\n [q, latestSave, debounceTime],\n );\n\n useEffect(() => {\n if (!isActive) {\n return;\n }\n\n return form.formState\n .map((state) => state.draft)\n .subscribe(\n (draft) => {\n if (draft === undefined) {\n return;\n }\n\n scheduleSave();\n },\n {\n runNow: false,\n },\n );\n }, [isActive, form.formState, scheduleSave]);\n\n useEffect(() => {\n return () => {\n scheduleSave.flush();\n };\n }, [scheduleSave]);\n\n return {\n flush() {\n scheduleSave.flush();\n return q.whenDone();\n },\n cancel() {\n scheduleSave.cancel();\n return q.whenDone();\n },\n };\n}\n","import { createStore, type Store, type Update } from '@core';\nimport { autobind } from '@lib/autobind';\nimport { deepEqual } from '@lib/equals';\nimport { isObject } from '@lib/helpers';\nimport {\n type PathAsString,\n type Value,\n type WildcardPathAsString,\n type WildcardValue,\n} from '@lib/path';\nimport { get, join, set } from '@lib/propAccess';\nimport type { Object_ } from '@lib/typeHelpers';\nimport { getWildCardMatches } from '@lib/wildcardMatch';\nimport { GeneralFormContext } from '@react/form/closestFormContext';\nimport { LegacyFormField, type FormFieldPropsWithComponent } from '@react/form/legacyFormField';\nimport { create, type Draft } from 'mutative';\nimport {\n createContext,\n forwardRef,\n useContext,\n useEffect,\n useMemo,\n useRef,\n type Context,\n type FormEvent,\n type ForwardedRef,\n type FunctionComponent,\n type HTMLProps,\n type ReactNode,\n} from 'react';\nimport { useStore, type UseStoreOptions } from '../useStore';\nimport {\n FormField,\n useFormFieldProps,\n type FormFieldComponent,\n type FormFieldComponentProps,\n type FormFieldInfos,\n type FormFieldProps,\n type FormFieldPropsWithChildren,\n type FormFieldPropsWithRender,\n} from './formField';\nimport { FormForEach, type ElementName, type FormForEachProps } from './formForEach';\nimport { useFormAutosave, type FormAutosaveOptions } from './useFormAutosave';\n\n/// /////////////////////////////////////////////////////////////////////////////\n// Form types\n/// /////////////////////////////////////////////////////////////////////////////\n\nexport interface Transform<TDraft, TOriginal> {\n (value: Draft<TDraft>, form: FormContext<TDraft, TOriginal>): void | TDraft;\n}\n\nexport interface FormOptions<TDraft, TOriginal> {\n id?: string;\n defaultValue: TDraft;\n validations?: Validations<TDraft, TOriginal>;\n initiallyTriggerValidations?: boolean;\n localizeError?: (error: string, field: string) => string | undefined;\n autoSave?: FormAutosaveOptions<TDraft, TOriginal>;\n transform?: Transform<TDraft, TOriginal>;\n validatedClass?: string;\n original?: TOriginal;\n onSubmit?: (event: FormEvent<HTMLFormElement>, form: FormInstance<TDraft, TOriginal>) => void;\n reportValidity?: boolean | 'browser' | 'scrollTo';\n transformFieldProps?: <TPath extends string>(\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n}\n\nexport type Validations<TDraft, TOriginal> =\n | ((context: { draft: TDraft; original: TOriginal | undefined }) => Iterable<{\n name: string;\n error: string;\n }>)\n | ({\n [TPath in WildcardPathAsString<TDraft>]?: Record<\n string,\n Validation<TDraft, TOriginal, TPath>\n >;\n } & Record<string, Record<string, Validation<TDraft, TOriginal, any>>>);\n\nexport type Validation<TDraft, TOriginal, TPath> = (\n value: WildcardValue<TDraft, TPath>,\n context: {\n draft: TDraft;\n original: TOriginal;\n field: PathAsString<TDraft> | '';\n },\n) => boolean;\n\nexport type Field<TDraft, TOriginal, TPath extends string> = {\n originalValue: Value<TOriginal, TPath> | undefined;\n value: Value<TDraft, TPath>;\n setValue: (value: Update<Value<TDraft, TPath>>) => void;\n removeValue: () => void;\n hasChange: boolean;\n errors: string[];\n} & (Value<TDraft, TPath> extends Object_ ? FieldHelperMethods<TDraft, TPath> : {});\n\nexport type FieldHelperMethods<TDraft, TPath extends string> = {\n names: ElementName<TDraft, TPath>[];\n add: NonNullable<Value<TDraft, TPath>> extends readonly (infer T)[]\n ? (element: T) => void\n : NonNullable<Value<TDraft, TPath>> extends Record<infer K, infer V>\n ? (key: K, value: V) => void\n : never;\n remove: NonNullable<Value<TDraft, TPath>> extends readonly any[]\n ? (index: number) => void\n : (key: string) => void;\n};\n\nexport interface FormState<TDraft> {\n draft: TDraft | undefined;\n hasTriggeredValidations: boolean;\n saveInProgress: boolean;\n}\n\nexport interface FormDerivedState<TDraft> {\n draft: TDraft;\n hasTriggeredValidations: boolean;\n saveInProgress: boolean;\n hasChanges: boolean;\n errors: Map<string, string[]>;\n isValid: boolean;\n}\n\nexport interface FormContext<TDraft, TOriginal> {\n formState: Store<FormState<TDraft>>;\n options: FormOptions<TDraft, TOriginal>;\n original: TOriginal | undefined;\n getField: <TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n options?: FieldOptions,\n ) => Field<TDraft, TOriginal, TPath>;\n getDraft: () => TDraft;\n hasTriggeredValidations: () => boolean;\n saveInProgress: () => boolean;\n flushAutosave: () => Promise<void>;\n cancelAutosave: () => void;\n hasChanges: () => boolean;\n getErrors: () => Map<string, string[]>;\n isValid: () => boolean;\n validate: (options?: ValidateOptions) => boolean;\n reset: () => void;\n}\n\nexport interface FieldOptions {\n includeNestedErrors?: boolean;\n}\n\nexport interface ValidateOptions {\n reportValidity?: boolean | 'browser' | 'scrollTo';\n button?: HTMLButtonElement;\n}\n\nexport interface FormInstance<TDraft, TOriginal>\n extends\n FormDerivedState<TDraft>,\n Pick<\n FormContext<TDraft, TOriginal>,\n 'options' | 'original' | 'getField' | 'validate' | 'reset'\n > {}\n\n/// /////////////////////////////////////////////////////////////////////////////\n// Implementation\n/// /////////////////////////////////////////////////////////////////////////////\n\nconst FormContainer = forwardRef(function FormContainer(\n {\n form,\n ...formProps\n }: {\n form: Form<any, any>;\n onSubmit?: (\n event: FormEvent<HTMLFormElement>,\n form: FormInstance<any, any>,\n ) => void | Promise<void>;\n } & Omit<HTMLProps<HTMLFormElement>, 'form' | 'onSubmit'>,\n ref: ForwardedRef<HTMLFormElement>,\n) {\n const formInstance = form.useForm();\n const hasTriggeredValidations = form.useFormState((state) => state.hasTriggeredValidations);\n const hasErrors = form.useFormState((state) => hasTriggeredValidations && state.errors.size > 0);\n\n return (\n <form\n ref={ref}\n noValidate\n {...formProps}\n className={[\n formProps.className,\n hasTriggeredValidations ? (formInstance.options.validatedClass ?? 'validated') : undefined,\n ]\n .filter(Boolean)\n .join(' ')}\n data-validated={hasTriggeredValidations || undefined}\n data-valid={hasErrors ? 'false' : hasTriggeredValidations ? 'true' : undefined}\n onSubmit={async (event) => {\n if (formInstance.saveInProgress()) {\n return;\n }\n\n try {\n formInstance.formState.set('saveInProgress', true);\n event.preventDefault();\n\n const button =\n event.nativeEvent instanceof SubmitEvent &&\n event.nativeEvent.submitter instanceof HTMLButtonElement\n ? event.nativeEvent.submitter\n : undefined;\n\n const isValid = formInstance.validate({ button });\n if (isValid) {\n await formProps.onSubmit?.(event, {\n ...formInstance,\n ...getDerivedState(formInstance),\n });\n }\n } finally {\n formInstance.formState.set('saveInProgress', false);\n }\n }}\n />\n );\n});\n\nfunction getField<TDraft, TOriginal extends TDraft, TPath extends string>(\n form: FormContext<TDraft, TOriginal>,\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n { includeNestedErrors }: FieldOptions = {},\n): Field<TDraft, TOriginal, TPath> {\n const field = {\n get originalValue() {\n return form.original !== undefined ? get(form.original as any, name as any) : undefined;\n },\n\n get value() {\n const draft = form.getDraft();\n return get(draft ?? form.original ?? form.options.defaultValue, name as any);\n },\n\n setValue(update: Update<Value<TDraft, TPath>>) {\n form.formState.set('draft', (draft = form.original ?? form.options.defaultValue) => {\n if (update instanceof Function) {\n update = update(get(draft, name as any) as Value<TDraft, TPath>);\n }\n\n return set(draft, name as any, update as any);\n });\n },\n\n get hasChange() {\n return !deepEqual(this.originalValue, this.value, { undefinedEqualsAbsent: true });\n },\n\n get errors() {\n const errors = form.getErrors();\n\n if (includeNestedErrors) {\n return Array.from(errors.entries())\n .filter(([key]) => key === name || key.startsWith(`${name}.`))\n .flatMap(([, value]) => value);\n } else {\n return errors.get(name) ?? [];\n }\n },\n\n get names(): any {\n const { value } = this;\n\n if (isObject(value)) {\n return Object.keys(value).map((key) => join(name, key));\n }\n\n return [];\n },\n\n add(...args: any[]) {\n this.setValue((value): any => {\n if (!value) {\n throw new Error(`Cannot add element to ${JSON.stringify(value)}`);\n }\n\n if (Array.isArray(value)) {\n return [...(value ?? []), args[0]];\n }\n\n if (isObject(value)) {\n return {\n ...value,\n [args[0]]: args[1],\n };\n }\n\n throw new Error(`Cannot add element to ${JSON.stringify(value)}`);\n });\n },\n\n remove(key: string | number) {\n this.setValue((value): any => {\n if (!value) {\n throw new Error(`Cannot remove element from ${JSON.stringify(value)}`);\n }\n\n if (Array.isArray(value)) {\n return value.filter((_, index) => index !== Number(key));\n }\n\n if (isObject(value)) {\n const { [key]: _, ...rest } = value as Record<string | number, unknown>;\n return rest;\n }\n\n throw new Error(`Cannot remove element from ${JSON.stringify(value)}`);\n });\n },\n };\n\n return field as any;\n}\n\nfunction getErrors<TDraft, TOriginal>(\n draft: TDraft,\n { original, validations, localizeError }: FormOptions<TDraft, TOriginal>,\n) {\n const errors = new Map<string, string[]>();\n\n if (typeof validations === 'function') {\n const issues = validations({ draft, original });\n\n for (const { name, error } of issues) {\n const fieldErrors = errors.get(name) ?? [];\n fieldErrors.push(error);\n errors.set(name, fieldErrors);\n }\n } else {\n for (const [path, block] of Object.entries(validations ?? {})) {\n for (const [validationName, validate] of Object.entries(\n block as Record<string, Validation<any, any, any>>,\n )) {\n let matched = false;\n\n for (const [field, value] of Object.entries(getWildCardMatches(draft, path))) {\n matched = true;\n if (!validate(value, { draft, original, field })) {\n const fieldErrors = errors.get(field) ?? [];\n fieldErrors.push(validationName);\n errors.set(field, fieldErrors);\n }\n }\n\n if (!matched && !path.includes('*')) {\n if (!validate(undefined, { draft, original, field: path })) {\n const fieldErrors = errors.get(path) ?? [];\n fieldErrors.push(validationName);\n errors.set(path, fieldErrors);\n }\n }\n }\n }\n }\n\n if (localizeError) {\n for (const [field, fieldErrors] of errors.entries()) {\n errors.set(\n field,\n fieldErrors.map((error) => localizeError(error, field) ?? error),\n );\n }\n }\n\n return errors;\n}\n\nexport function getDerivedState<TDraft>(\n instance: FormContext<TDraft, any>,\n): FormDerivedState<TDraft> {\n return {\n draft: instance.getDraft(),\n hasTriggeredValidations: instance.hasTriggeredValidations(),\n saveInProgress: instance.saveInProgress(),\n hasChanges: instance.hasChanges(),\n errors: instance.getErrors(),\n isValid: instance.isValid(),\n };\n}\n\nexport class Form<TDraft, TOriginal extends TDraft = TDraft> {\n context: Context<FormContext<TDraft, TOriginal> | null> = createContext<FormContext<\n TDraft,\n TOriginal\n > | null>(null);\n\n constructor(public readonly options: FormOptions<TDraft, TOriginal>) {\n autobind(Form);\n }\n\n useForm(): FormContext<TDraft, TOriginal> {\n const context = useContext(this.context);\n\n if (!context) {\n throw new Error('Form context not found');\n }\n\n return context;\n }\n\n useFormState<S>(\n selector: (state: FormInstance<TDraft, TOriginal>) => S,\n useStoreOptions?: UseStoreOptions<S>,\n ): S {\n const form = this.useForm();\n\n return useStore(\n form.formState,\n () =>\n selector({\n ...form,\n ...getDerivedState(form),\n }),\n\n useStoreOptions,\n );\n }\n\n useField<TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n { includeNestedErrors, ...useStoreOptions }: FieldOptions & UseStoreOptions<unknown> = {},\n ): Field<TDraft, TOriginal, TPath> {\n const form = this.useForm();\n this.useFormState((form) => [form.getField(name).value, form.original], useStoreOptions);\n\n return form.getField(name, { includeNestedErrors });\n }\n\n useFieldProps<TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n options?: Omit<FormFieldProps<TPath, TDraft>, 'name'>,\n ): FormFieldComponentProps<Value<TDraft, TPath>, TPath> {\n return useFormFieldProps.call<\n Form<TDraft, TOriginal>,\n [FormFieldProps<TPath, TDraft>],\n FormFieldComponentProps<Value<TDraft, TPath>, TPath>\n >(this, { name, ...options } as any);\n }\n\n // ///////////////////////////////////////////////////////////////////////////\n // React Components\n // ///////////////////////////////////////////////////////////////////////////\n\n Form({\n defaultValue,\n validations,\n initiallyTriggerValidations,\n localizeError,\n autoSave,\n transform,\n validatedClass,\n original,\n onSubmit,\n reportValidity,\n transformFieldProps,\n ...formProps\n }: Partial<FormOptions<TDraft, TOriginal>> &\n Omit<HTMLProps<HTMLFormElement>, 'defaultValue' | 'autoSave' | 'onSubmit'>): React.JSX.Element {\n const options: FormOptions<TDraft, TOriginal> = {\n defaultValue: { ...this.options.defaultValue, ...defaultValue },\n validations:\n typeof validations === 'function'\n ? validations\n : validations\n ? ({ ...this.options.validations, ...validations } as Validations<TDraft, TOriginal>)\n : this.options.validations,\n localizeError: localizeError ?? this.options.localizeError,\n autoSave:\n this.options.autoSave || autoSave\n ? ({ ...this.options.autoSave, ...autoSave } as FormAutosaveOptions<TDraft, TOriginal>)\n : undefined,\n transform: transform ?? this.options.transform,\n validatedClass: validatedClass ?? this.options.validatedClass,\n original: original ?? this.options.original,\n onSubmit: onSubmit ?? this.options.onSubmit,\n reportValidity: reportValidity ?? this.options.reportValidity ?? 'browser',\n transformFieldProps: transformFieldProps ?? this.options.transformFieldProps,\n };\n\n const formState = useMemo(() => {\n return createStore<FormState<TDraft>>({\n draft: undefined,\n hasTriggeredValidations: initiallyTriggerValidations ?? false,\n saveInProgress: false,\n });\n // oxlint-disable-next-line exhaustive-deps\n }, []);\n\n const formRef = useRef<HTMLFormElement>(null);\n\n let lastDraft: TDraft | undefined;\n const cache = new Map<string, unknown>();\n function lazy<T>(key: string, fn: () => T): T {\n if (lastDraft !== formState.get().draft) {\n cache.clear();\n lastDraft = formState.get().draft;\n }\n\n let value = cache.get(key);\n if (!cache.has(key)) {\n value = fn();\n cache.set(key, value);\n }\n\n return value as T;\n }\n\n const context: FormContext<TDraft, TOriginal> = {\n formState,\n options,\n original: options.original,\n\n getField() {\n throw new Error('Not implemented');\n },\n\n getDraft() {\n return formState.get().draft ?? options.original ?? options.defaultValue;\n },\n\n hasTriggeredValidations() {\n return formState.get().hasTriggeredValidations;\n },\n\n saveInProgress() {\n return formState.get().saveInProgress;\n },\n\n flushAutosave() {\n return autosave.flush();\n },\n\n cancelAutosave() {\n return autosave.cancel();\n },\n\n hasChanges() {\n return lazy(\n 'hasChanges',\n () =>\n !deepEqual(this.getDraft(), options.original ?? options.defaultValue, {\n undefinedEqualsAbsent: true,\n }),\n );\n },\n\n getErrors() {\n return lazy('getErrors', () => getErrors(this.getDraft(), options));\n },\n\n isValid() {\n return lazy('isValid', () => this.getErrors().size === 0);\n },\n\n validate({ reportValidity = options.reportValidity, button }: ValidateOptions = {}) {\n formState.set('hasTriggeredValidations', true);\n\n updateValidity(this.getErrors(), button);\n\n switch (reportValidity) {\n case 'browser':\n formRef.current?.reportValidity();\n break;\n\n case true:\n case 'scrollTo':\n {\n const invalidElement = document.querySelector(':invalid, [data-invalid=\"true\"]');\n if (invalidElement && invalidElement instanceof HTMLElement) {\n invalidElement.scrollIntoView({ behavior: 'smooth', block: 'center' });\n invalidElement.focus({ preventScroll: true });\n }\n }\n break;\n }\n\n return this.isValid();\n },\n\n reset() {\n formState.set('draft', undefined);\n formState.set('hasTriggeredValidations', false);\n },\n };\n\n context.getField = (path, options) =>\n lazy(`${path}:${options?.includeNestedErrors}`, () => getField(context, path, options));\n\n useEffect(() => {\n const transform = options.transform;\n if (!transform) {\n return;\n }\n\n return context.formState.subscribe((state) => {\n const value = state.draft ?? options.original ?? options.defaultValue;\n const result = create(value, (draft) => transform(draft, context)) as TDraft;\n\n if (!deepEqual(result, value)) {\n context.formState.set('draft', result);\n }\n });\n });\n\n function updateValidity(errors: Map<string, string[]>, buttonElement?: HTMLButtonElement) {\n const formElement = formRef.current;\n if (!formElement) {\n return;\n }\n\n for (const element of Array.from(formElement.elements)) {\n if ('name' in element && 'setCustomValidity' in element) {\n (element as HTMLObjectElement).setCustomValidity(\n errors.get((element as HTMLObjectElement).name)?.join('\\n') ?? '',\n );\n }\n }\n\n if (buttonElement && 'setCustomValidity' in buttonElement) {\n const errorString = [...errors.values()].flat().join('\\n');\n\n buttonElement.setCustomValidity(errorString);\n }\n }\n\n useEffect(() => {\n return formState.map(() => context.getErrors()).subscribe((errors) => updateValidity(errors));\n });\n\n const autosave = useFormAutosave(context);\n\n return (\n <GeneralFormContext.Provider value={this}>\n <this.context.Provider value={context}>\n <FormContainer {...formProps} form={this} onSubmit={options.onSubmit} />\n </this.context.Provider>\n </GeneralFormContext.Provider>\n );\n }\n\n FormState<S>({\n selector,\n children,\n }: {\n selector: (form: FormInstance<TDraft, TOriginal>) => S;\n children: (selectedState: S) => ReactNode;\n }): React.JSX.Element {\n const selectedState = this.useFormState(selector);\n return <>{children(selectedState)}</>;\n }\n\n Field<const TPath extends string>(\n props: FormFieldPropsWithRender<TDraft, TOriginal, TPath>,\n ): React.JSX.Element;\n\n Field<const TPath extends string>(\n props: FormFieldPropsWithChildren<TDraft, TOriginal, TPath>,\n ): React.JSX.Element;\n\n /** @deprecated */\n Field<const TPath extends string, const TComponent extends FormFieldComponent = 'input'>(\n props: FormFieldPropsWithComponent<TDraft, TOriginal, TPath, TComponent>,\n ): React.JSX.Element;\n\n Field(props: any): React.JSX.Element {\n if (props.component) {\n return Reflect.apply(LegacyFormField, this, [props]);\n }\n\n return Reflect.apply(FormField, this, [props]);\n }\n\n ForEach<const TPath extends string>(props: FormForEachProps<TDraft, TPath>): React.JSX.Element {\n return Reflect.apply(FormForEach, this, [props]);\n }\n\n withForm<TProps extends Record<string, unknown>>(\n Component: React.ComponentType<TProps>,\n formProps?: Parameters<this['Form']>[0],\n ): FunctionComponent<TProps> {\n const { Form } = this;\n return function FormWrapper(props: TProps) {\n return (\n <Form {...formProps}>\n <Component {...props} />\n </Form>\n );\n };\n }\n}\n\nexport function createForm<TDraft, TOriginal extends TDraft = TDraft>(\n options: FormOptions<TDraft, TOriginal>,\n): Form<TDraft, TOriginal> {\n return new Form(options);\n}\n","export function castArray<T>(value: T | T[]): T[] {\n return Array.isArray(value) ? value : [value];\n}\n","import { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\n\nexport function defaultDeserializer<T>(value: string): T {\n if (value === undefined) {\n return undefined as T;\n }\n\n try {\n return fromExtendedJsonString(value) as T;\n } catch {\n return undefined as T;\n }\n}\n\nexport function defaultSerializer<T>(value: T): string {\n return toExtendedJsonString(value);\n}\n\nexport function normalizePath<T>(path: string | T): string | T {\n if (typeof path === 'string') {\n return path.replace(/^\\//g, '').replace(/\\/$/g, '');\n }\n return path;\n}\n","import { castArray } from '@lib/castArray';\nimport { defaultDeserializer, defaultSerializer, normalizePath } from '@react/url/urlHelpers';\n\nexport interface UrlOptions<T> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue: T;\n writeDefaultValue?: boolean;\n onCommit?: (value: T) => void;\n persist?: { id: string } | null;\n path?: string | RegExp | (string | RegExp)[] | null;\n}\n\nexport interface UrlOptionsWithoutDefaults<T> extends Omit<\n UrlOptions<T | undefined>,\n 'defaultValue'\n> {\n defaultValue?: T | undefined;\n}\n\nexport function createUrlOptions<T>(options: UrlOptions<T>): Required<UrlOptions<T>>;\nexport function createUrlOptions<T>(\n options: UrlOptionsWithoutDefaults<T>,\n): Required<UrlOptions<T | undefined>>;\nexport function createUrlOptions<T>({\n key,\n type = 'hash',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n writeDefaultValue = false,\n onCommit = () => undefined,\n persist = null,\n path = null,\n}: UrlOptionsWithoutDefaults<T>): Required<UrlOptionsWithoutDefaults<T>> {\n return {\n key,\n type,\n serialize,\n deserialize,\n defaultValue,\n writeDefaultValue,\n onCommit,\n persist,\n path: path === null ? null : castArray(path).map(normalizePath),\n };\n}\n","import { createStore, Store, type Update } from '@core';\nimport { autobind } from '@lib/autobind';\nimport { castArray } from '@lib/castArray';\nimport type { Constrain } from '@lib/constrain';\nimport type { Path, Value } from '@lib/path';\nimport { get, set } from '@lib/propAccess';\nimport { normalizePath } from '@react/url/urlHelpers';\nimport {\n createUrlOptions,\n type UrlOptions,\n type UrlOptionsWithoutDefaults,\n} from '@react/url/urlOptions';\n\nexport const urlStore: Store<string> = createStore(() => window.location.href, {\n cacheValue: false,\n effect() {\n const update = () => {\n if (window.location.href !== this.calculatedValue?.value) {\n this.invalidate();\n }\n };\n\n const interval = setInterval(update, 1);\n window.addEventListener('popstate', update);\n\n return () => {\n clearInterval(interval);\n window.removeEventListener('popstate', update);\n };\n },\n});\n\nexport class UrlParamStore<T> extends Store<T> {\n readonly storageKey: string | null;\n private lastHref?: string;\n private lastStorageValue?: string | null;\n private lastValue?: T;\n\n constructor(public readonly urlOptions: Required<UrlOptions<T>>) {\n super(() => this.calc(), { cacheValue: false });\n autobind(UrlParamStore);\n\n this.storageKey =\n urlOptions.persist && `cross-state:url:${urlOptions.persist.id}:${urlOptions.key}`;\n this.addEffect(this.watch);\n }\n\n private watch() {\n let isActive = false;\n let urlValue = this.getUrlValue();\n let storageValue = this.getStorageValue();\n\n const update = () => {\n const oldIsActive = isActive;\n isActive = this.isPathActive();\n const oldUrlValue = urlValue;\n urlValue = this.getUrlValue();\n const oldStorageValue = storageValue;\n storageValue = this.getStorageValue();\n\n // If inactive => ignore changes\n if (!isActive) {\n return;\n }\n\n // No changes => ignore\n if (\n isActive === oldIsActive &&\n urlValue === oldUrlValue &&\n storageValue === oldStorageValue\n ) {\n return;\n }\n\n if (!oldIsActive) {\n // Became active =>\n // - if url has value => update storage\n // - else if storage has value or writeDefaultValue => update url\n if (urlValue !== null) {\n this.updateStorage(this.urlOptions.deserialize(urlValue));\n } else if (storageValue !== null) {\n this.updateUrl(this.urlOptions.deserialize(storageValue));\n } else if (this.urlOptions.writeDefaultValue) {\n this.updateUrl(this.urlOptions.defaultValue);\n }\n } else if (urlValue !== oldUrlValue) {\n // Url change while active =>\n // - if url has no value and writeDefaultValue => update url\n // - update storage\n if (urlValue === null && this.urlOptions.writeDefaultValue) {\n this.updateUrl(this.urlOptions.defaultValue);\n }\n\n this.updateStorage(\n urlValue !== null ? this.urlOptions.deserialize(urlValue) : this.urlOptions.defaultValue,\n );\n }\n\n this.invalidate();\n };\n\n const cancel = urlStore.subscribe(update);\n window.addEventListener('storage', update);\n\n return () => {\n cancel();\n window.removeEventListener('storage', update);\n };\n }\n\n private getUrlValue() {\n const href = urlStore.get();\n const url = new URL(href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1) || '');\n return params.get(this.urlOptions.key);\n }\n\n private getStorageValue() {\n return this.storageKey !== null ? localStorage.getItem(this.storageKey) : null;\n }\n\n private isPathActive() {\n if (this.urlOptions.path === null) {\n return true;\n }\n\n const path = normalizePath(window.location.pathname);\n\n return castArray(this.urlOptions.path).some((p) => {\n if (typeof p === 'string') {\n return !p || p === path || path.startsWith(p + '/');\n }\n\n return p.test(path);\n });\n }\n\n private calc() {\n let href = window.location.href;\n const storageValue = this.storageKey !== null ? localStorage.getItem(this.storageKey) : null;\n\n if (!this.isPathActive() && this.lastHref !== undefined) {\n href = this.lastHref;\n }\n\n if (this.lastHref === href && this.lastStorageValue === storageValue) {\n return this.lastValue as T;\n }\n\n const url = new URL(href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1));\n const urlValue = params.get(this.urlOptions.key);\n\n const value =\n urlValue !== null\n ? this.urlOptions.deserialize(urlValue)\n : this.storageKey !== null && storageValue !== null\n ? this.urlOptions.deserialize(storageValue)\n : this.urlOptions.defaultValue;\n\n this.lastHref = href;\n this.lastStorageValue = storageValue;\n this.lastValue = value;\n return value;\n }\n\n private updateUrl(value: T) {\n const serializedValue = this.urlOptions.serialize(value);\n\n const url = new URL(window.location.href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1));\n\n if (\n !this.urlOptions.writeDefaultValue &&\n serializedValue === this.urlOptions.serialize(this.urlOptions.defaultValue)\n ) {\n params.delete(this.urlOptions.key);\n } else {\n params.set(this.urlOptions.key, serializedValue);\n }\n\n url[this.urlOptions.type] = params.toString();\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate', { state: window.history.state }));\n }\n\n private updateStorage(value: T) {\n if (this.storageKey === null) {\n return;\n }\n\n const serializedValue = this.urlOptions.serialize(value);\n localStorage.setItem(this.storageKey, serializedValue);\n }\n\n set(update: Update<T>): void;\n set<const P>(path: Constrain<P, Path<T>>, update: Update<Value<T, P>>): void;\n set(...args: any[]): void {\n const path: any = args.length > 1 ? args[0] : [];\n let update: Update<any> = args.length > 1 ? args[1] : args[0];\n\n if (update instanceof Function) {\n const before = this.get();\n const valueBefore = get(before, path);\n const valueAfter = update(valueBefore);\n update = set(before, path, valueAfter);\n } else if (path.length > 0) {\n update = set(this.get(), path, update);\n }\n\n if (this.isPathActive()) {\n this.updateUrl(update);\n } else {\n this.updateStorage(update);\n }\n }\n\n parse(path: string): T {\n const url = new URL(path, window.location.href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1) || '');\n const urlValue = params.get(this.urlOptions.key);\n return urlValue !== null ? this.urlOptions.deserialize(urlValue) : this.urlOptions.defaultValue;\n }\n}\n\nexport function createUrlParam<T>(options: UrlOptions<T>): UrlParamStore<T>;\nexport function createUrlParam<T>(\n options: UrlOptionsWithoutDefaults<T>,\n): UrlParamStore<T | undefined>;\nexport function createUrlParam<T>(options: UrlOptionsWithoutDefaults<T>) {\n return new UrlParamStore(createUrlOptions(options));\n}\n","import { type Duration } from '@core';\nimport { debounce } from '@lib/debounce';\nimport isPromise from '@lib/isPromise';\nimport type { MaybePromise } from '@lib/maybePromise';\nimport { throttle } from '@lib/throttle';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport useMemoEquals from '@react/lib/useMemoEquals';\nimport { startTransition, useMemo, useRef, useState } from 'react';\n\nexport interface UseDecoupledStateOptions<T> {\n debounce?: Duration;\n throttle?: Duration;\n onCommit?: (value: T) => void;\n}\n\nexport function useDecoupledState<T>(\n value: T,\n onChange: (value: T) => MaybePromise<void>,\n options: UseDecoupledStateOptions<T> = {},\n): [state: T, setState: (value: T) => void] {\n const [dirty, setDirty] = useState<{ v: T }>();\n const onChangeAC = useRef<AbortController>(undefined);\n\n const latestOnChange = useLatestFunction(onChange);\n const latestOnCommit = useLatestFunction(options.onCommit ?? (() => {}));\n const debounceOptions = useMemoEquals(options.debounce);\n const throttleOptions = useMemoEquals(options.throttle);\n\n const update = useMemo(() => {\n const update = async (value: T) => {\n const result = latestOnChange(value);\n\n if (isPromise(result)) {\n const ac = (onChangeAC.current = new AbortController());\n\n await result;\n\n if (ac.signal.aborted) {\n return;\n }\n }\n\n latestOnCommit(value);\n setDirty(undefined);\n };\n\n let delayedUpdate: (value: T) => void;\n\n if (debounceOptions) {\n delayedUpdate = debounce(update, debounceOptions);\n } else if (throttleOptions) {\n delayedUpdate = throttle(update, throttleOptions);\n } else {\n delayedUpdate = (value) => startTransition(() => update(value));\n }\n\n return (value: T) => {\n onChangeAC.current?.abort();\n setDirty({ v: value });\n delayedUpdate(value);\n };\n }, [latestOnChange, latestOnCommit, debounceOptions, throttleOptions]);\n\n return [dirty ? dirty.v : value, update];\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":["matches: Record<KeyType, any>","GeneralFormContext: Context<Form<any, any> | null>","value","form","form","keys: (string | number)[]","name","FormContainer","options: FormOptions<TDraft, TOriginal>","form","lastDraft: TDraft | undefined","context: FormContext<TDraft, TOriginal>","reportValidity","options","transform","Form","Component","urlStore: Store<string>","urlOptions: Required<UrlOptions<T>>","path: any","update: Update<any>","update","value","delayedUpdate: (value: T) => void"],"sources":["../../src/react/form/customInput.tsx","../../src/lib/wildcardMatch.ts","../../src/react/form/closestFormContext.tsx","../../src/react/form/legacyFormField.tsx","../../src/react/form/formField.tsx","../../src/react/form/formForEach.tsx","../../src/react/form/formOnOriginalChange.tsx","../../src/react/form/useFormAutosave.ts","../../src/react/form/form.tsx","../../src/lib/castArray.ts","../../src/react/url/urlHelpers.ts","../../src/react/url/urlOptions.ts","../../src/react/url/urlParamStore.ts","../../src/react/useDecoupledState.ts"],"sourcesContent":["import type { ReactNode } from 'react';\n\nexport interface CustomInputProps extends React.HTMLAttributes<HTMLDivElement> {\n name: string;\n children?: ReactNode;\n}\n\nexport function CustomInput({ name, children, ...props }: CustomInputProps): React.JSX.Element {\n return (\n <div\n {...props}\n style={{\n position: 'relative',\n ...props.style,\n }}\n >\n {children}\n\n <input\n name={name}\n style={{\n position: 'absolute',\n top: 0,\n left: 0,\n opacity: 0,\n width: '100%',\n height: '100%',\n pointerEvents: 'none',\n }}\n />\n </div>\n );\n}\n","import { isObject } from '@lib/helpers';\nimport { type KeyType } from './path';\nimport { castArrayPath } from './propAccess';\n\nexport function wildcardMatch(s: KeyType[] | string, w: KeyType[] | string): boolean {\n if (typeof s === 'string') {\n s = castArrayPath(s);\n }\n\n if (typeof w === 'string') {\n w = castArrayPath(w);\n }\n\n return s.length === w.length && s.every((s, i) => w[i] === '*' || s === w[i]);\n}\n\nexport function getWildCardMatches(\n object: any,\n path: [KeyType, ...KeyType[]] | string,\n): Record<KeyType, any> {\n const matches: Record<KeyType, any> = {};\n const [first, second, ...rest] = castArrayPath(path);\n\n if (first === undefined) {\n throw new Error('Path is empty');\n }\n\n if (!Array.isArray(object) && !isObject(object)) {\n object = {};\n }\n\n for (const [key, value] of first !== '*' ? [[first, object[first]]] : Object.entries(object)) {\n if (first !== '*' && first !== key) {\n continue;\n }\n\n if (second === undefined) {\n matches[key] = value;\n continue;\n }\n\n for (const [subKey, subValue] of Object.entries(getWildCardMatches(value, [second, ...rest]))) {\n matches[`${key}.${subKey}`] = subValue;\n }\n }\n\n return matches;\n}\n","import { type Form } from '@react/form/form';\nimport { createContext, useContext, type Context } from 'react';\n\nexport const GeneralFormContext: Context<Form<any, any> | null> = createContext<Form<\n any,\n any\n> | null>(null);\n\nexport function useClosestForm<TDraft, TOriginal extends TDraft = TDraft>(): Form<\n TDraft,\n TOriginal\n> | null {\n return useContext(GeneralFormContext) as Form<TDraft, TOriginal> | null;\n}\n","import { calcDuration } from '@lib/duration';\nimport type { Value } from '@lib/path';\nimport { getDerivedState, type Form, type FormInstance } from '@react/form/form';\nimport type {\n FormFieldComponent,\n FormFieldComponentProps,\n FormFieldProps,\n} from '@react/form/formField';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { createElement, useEffect, useState, type ComponentPropsWithoutRef } from 'react';\n\ntype FieldValue<T extends FormFieldComponent> = ComponentPropsWithoutRef<T>['value'];\n\ntype FieldChangeValue<T extends FormFieldComponent> =\n ComponentPropsWithoutRef<T> extends {\n onChange?: (update: infer U) => void;\n }\n ? U extends { target: { value: infer V } }\n ? V\n : U\n : never;\n\ntype MakeOptional<T, Keys extends string> = Omit<T, Keys> & Partial<Pick<T, Keys & keyof T>>;\n\ntype Serialize<TDraft, TOriginal, TPath, TComponent extends FormFieldComponent> = (\n value: Value<TDraft, TPath>,\n formState: FormInstance<TDraft, TOriginal>,\n) => FieldValue<TComponent>;\n\ntype Deserialize<TDraft, TOriginal, TPath, TComponent extends FormFieldComponent> = (\n value: FieldChangeValue<TComponent>,\n formState: FormInstance<TDraft, TOriginal>,\n) => Value<TDraft, TPath>;\n\nexport type FormFieldPropsWithComponent<\n TDraft,\n TOriginal,\n TPath extends string,\n TComponent extends FormFieldComponent,\n> = FormFieldProps<TPath, TDraft> & {\n component: TComponent;\n render?: undefined;\n} & NoInfer<\n {\n inputFilter?: (value: FieldChangeValue<TComponent>) => boolean;\n } & MakeOptional<\n Omit<ComponentPropsWithoutRef<TComponent>, 'id' | 'name' | 'value' | 'defaultValue'>,\n 'onChange' | 'onBlur'\n > &\n (Value<TDraft, TPath> extends Exclude<FieldValue<TComponent>, undefined>\n ? {\n defaultValue?: FieldValue<TComponent>;\n serialize?: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : Value<TDraft, TPath> extends FieldValue<TComponent>\n ?\n | {\n defaultValue: FieldValue<TComponent>;\n serialize?: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n | {\n defaultValue?: FieldValue<TComponent>;\n serialize: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : {\n serialize: Serialize<TDraft, TOriginal, TPath, TComponent>;\n }) &\n (FieldChangeValue<TComponent> extends Value<TDraft, TPath>\n ? {\n deserialize?: Deserialize<TDraft, TOriginal, TPath, TComponent>;\n }\n : {\n deserialize: Deserialize<TDraft, TOriginal, TPath, TComponent>;\n })\n >;\n\nexport function LegacyFormField<\n TDraft,\n TOriginal,\n TPath extends string,\n TComponent extends FormFieldComponent,\n>(\n this: Form<TDraft, any>,\n {\n // id,\n name = '' as any,\n component,\n commitOnBlur,\n commitDebounce,\n inputFilter,\n defaultValue,\n serialize,\n deserialize = (x) => x as Value<TDraft, TPath>,\n onChange,\n onBlur,\n ...restProps\n }: FormFieldPropsWithComponent<TDraft, TOriginal, TPath, TComponent>,\n): React.JSX.Element | null {\n type T = FieldChangeValue<TComponent>;\n\n const form = this.useForm();\n const getFormState = () => ({ ...form, ...getDerivedState(form) });\n const [localValue, setLocalValue] = useState<T>();\n\n const value = this.useFormState((form) => {\n const value = form.getField(name as any).value;\n if (serialize) {\n return serialize(value as any, getFormState());\n }\n if (value !== undefined) {\n return value;\n }\n return defaultValue;\n });\n\n const setValue = useLatestFunction((x: FieldChangeValue<TComponent>) =>\n form.getField(name as any).setValue(deserialize(x, getFormState())),\n );\n\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n\n const commitDebounceMs = commitDebounce !== undefined ? calcDuration(commitDebounce) : undefined;\n useEffect(() => {\n if (localValue === undefined || commitDebounceMs === undefined || commitDebounceMs <= 0) {\n return;\n }\n\n const timeout = setTimeout(() => {\n setValue(localValue);\n setLocalValue(undefined);\n }, commitDebounceMs);\n\n return () => clearTimeout(timeout);\n }, [localValue, commitDebounceMs, setValue]);\n\n let props = {\n name,\n value: localValue ?? value,\n onChange: (event: { target: { value: T } } | T, ...moreArgs: any[]) => {\n const value =\n typeof event === 'object' && event !== null && 'target' in event\n ? event.target.value\n : event;\n\n if (inputFilter && !inputFilter(value)) {\n return;\n }\n\n if (commitOnBlur || commitDebounce) {\n setLocalValue(value);\n } else {\n setValue(value);\n }\n\n onChange?.(event, ...moreArgs);\n },\n onBlur(...args: any[]) {\n if (localValue !== undefined) {\n setValue(localValue);\n setLocalValue(undefined);\n }\n\n onBlur?.(...args);\n },\n } as FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n\n if (this.options.transformFieldProps) {\n props = this.options.transformFieldProps(\n props,\n { ...form.getField(name as any), hasTriggeredValidations } as any,\n form,\n );\n }\n\n if (component) {\n return createElement(component, { ...restProps, ...props });\n }\n\n return null;\n}\n","import type { Duration } from '@core';\nimport { calcDuration } from '@lib/duration';\nimport { type PathAsString, type Value } from '@lib/path';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { useEffect, useState, type Component, type ReactNode } from 'react';\nimport { Form, type Field, type FieldOptions, type FormContext } from './form';\n\nexport interface FormFieldComponentProps<TValue, TPath> {\n name: TPath;\n value: TValue;\n onChange: (event: { target: { value: TValue } } | TValue | undefined, ...args: any[]) => void;\n onBlur: (...args: any[]) => void;\n 'data-invalid'?: boolean;\n}\n\nexport type FormFieldInfos<TDraft, TOriginal, TPath extends string> = Field<\n TDraft,\n TOriginal,\n TPath\n> & {\n hasTriggeredValidations: boolean;\n};\n\ntype NativeInputType = 'input' | 'select' | 'textarea';\n\ntype PartialComponentType<P> =\n | (new (props: P, context?: any) => Component<P, any>)\n | ((props: P, context?: any) => ReactNode);\n\nexport type FormFieldComponent = NativeInputType | PartialComponentType<any>;\n\nexport interface FormFieldProps<TPath, TDraft> extends FieldOptions {\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>;\n commitOnBlur?: boolean;\n commitDebounce?: Duration;\n}\n\nexport interface FormFieldPropsWithRender<\n TDraft,\n TOriginal,\n TPath extends string,\n> extends FormFieldProps<TPath, TDraft> {\n render: NoInfer<\n (\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => ReactNode\n >;\n children?: undefined;\n}\n\nexport interface FormFieldPropsWithChildren<\n TDraft,\n TOriginal,\n TPath extends string,\n> extends FormFieldProps<TPath, TDraft> {\n render?: undefined;\n children: NoInfer<\n (\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => ReactNode\n >;\n}\n\nexport function FormField<TDraft, TOriginal extends TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name = '' as any,\n commitOnBlur,\n commitDebounce,\n children,\n render = children,\n includeNestedErrors,\n }:\n | FormFieldPropsWithRender<TDraft, TOriginal, TPath>\n | FormFieldPropsWithChildren<TDraft, TOriginal, TPath>,\n): React.JSX.Element | null {\n const form = this.useForm();\n const field = this.useField(name);\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n\n const renderProps = useFormFieldProps.call<\n Form<TDraft, TOriginal>,\n [FormFieldProps<TPath, TDraft>],\n FormFieldComponentProps<Value<TDraft, TPath>, TPath>\n >(this, {\n name,\n commitOnBlur,\n commitDebounce,\n includeNestedErrors,\n });\n\n if (render) {\n return <>{render(renderProps, { ...field, hasTriggeredValidations } as any, form)}</>;\n }\n\n return null;\n}\n\nexport function useFormFieldProps<TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name = '' as any,\n commitOnBlur,\n commitDebounce,\n includeNestedErrors,\n }: FormFieldProps<TPath, TDraft>,\n): FormFieldComponentProps<Value<TDraft, TPath>, TPath> {\n type T = Value<TDraft, TPath>;\n\n const form = this.useForm();\n const field = this.useField(name, { includeNestedErrors });\n const hasTriggeredValidations = this.useFormState((form) => form.hasTriggeredValidations);\n const [localValue, setLocalValue] = useState<{ v: T }>();\n\n const commitDebounceMs = commitDebounce !== undefined ? calcDuration(commitDebounce) : undefined;\n\n const commitLocalValue = useLatestFunction(() => {\n if (localValue) {\n field.setValue(localValue.v);\n setLocalValue(undefined);\n }\n });\n\n useEffect(() => {\n if (!localValue || commitDebounceMs === undefined || commitDebounceMs <= 0) {\n return;\n }\n\n const timeout = setTimeout(commitLocalValue, commitDebounceMs);\n return () => clearTimeout(timeout);\n }, [localValue, commitDebounceMs, commitLocalValue]);\n\n let props = {\n name,\n value: localValue ? localValue.v : field.value,\n onChange: useLatestFunction((event: { target: { value: T } } | T) => {\n const value =\n typeof event === 'object' && event !== null && 'target' in event\n ? event.target.value\n : event;\n\n if (commitOnBlur || (commitDebounceMs !== undefined && commitDebounceMs > 0)) {\n setLocalValue({ v: value });\n } else {\n field.setValue(value as any);\n }\n }),\n onBlur: useLatestFunction(() => {\n if (localValue) {\n commitLocalValue();\n }\n }),\n 'data-invalid': field.errors.length > 0 ? true : undefined,\n } as FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n\n if (this.options.transformFieldProps) {\n props = this.options.transformFieldProps(\n props,\n { ...field, hasTriggeredValidations } as any,\n form,\n );\n }\n\n return props;\n}\n","import { isObject } from '@lib/helpers';\nimport { type GetKeys, type Join, type PathAsString, type Value } from '@lib/path';\nimport { join } from '@lib/propAccess';\nimport { Fragment, useCallback, type ReactNode } from 'react';\nimport { type FieldHelperMethods, type Form } from './form';\n\nexport type ElementName<TDraft, TPath extends string> = keyof {\n [Path in TPath as Join<Path, GetKeys<NonNullable<Value<TDraft, Path>>> & (string | number)>]: 1;\n};\n\ntype ItemValue<T> = T extends readonly (infer U)[] ? U : T[keyof T];\n\nexport interface FormForEachProps<TDraft, TPath extends string> {\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>;\n renderElement?: (props: {\n name: ElementName<TDraft, TPath>;\n key: `${GetKeys<NonNullable<Value<TDraft, TPath>>> & (string | number)}`;\n index: number;\n remove: () => void;\n count: number;\n }) => ReactNode;\n renderAdditionalElement?: NonNullable<Value<TDraft, TPath>> extends readonly any[]\n ? boolean\n : never;\n filter?: (\n item: ItemValue<NonNullable<Value<TDraft, TPath>>>,\n key: GetKeys<NonNullable<Value<TDraft, TPath>>>,\n parent: NonNullable<Value<TDraft, TPath>>,\n ) => boolean;\n children?: (\n props: {\n setValue: (\n value: Value<TDraft, TPath> | ((value: Value<TDraft, TPath>) => Value<TDraft, TPath>),\n ) => void;\n } & FieldHelperMethods<TDraft, TPath>,\n ) => ReactNode;\n}\n\nexport function FormForEach<TDraft, TPath extends string>(\n this: Form<TDraft, any>,\n {\n name,\n renderElement,\n renderAdditionalElement,\n filter,\n children,\n }: FormForEachProps<TDraft, TPath>,\n): React.JSX.Element {\n const form = this.useForm();\n\n const items = this.useFormState(() => {\n const field = form.getField(name);\n\n let keys: (string | number)[] = isObject(field.value) ? Object.keys(field.value) : [];\n const count = keys.length;\n\n if (Array.isArray(field.value)) {\n keys = keys.map(Number);\n }\n\n if (filter) {\n keys = keys.filter((key, index) =>\n filter((field.value as any)[index], key as any, field.value as any),\n );\n }\n\n if (renderAdditionalElement) {\n keys.push(count);\n }\n\n return keys.map((key) => ({\n key,\n name: join(name, String(key)),\n }));\n });\n\n const add = useCallback(\n (...args: any[]) => {\n const field = form.getField(name as any) as any;\n field.add(...args);\n },\n [form, name],\n );\n\n const remove = useCallback(\n (key: any) => {\n const field = form.getField(name as any) as any;\n field.remove(key);\n },\n [form, name],\n );\n\n const setValue = useCallback(\n (value: Value<TDraft, TPath> | ((value: Value<TDraft, TPath>) => Value<TDraft, TPath>)) => {\n const field = form.getField(name as any) as any;\n field.setValue(value);\n },\n [form, name],\n );\n\n return (\n <>\n {renderElement &&\n items.map(({ key, name }, index) => {\n return (\n <Fragment key={key}>\n {renderElement({\n name: name as any,\n key: key as any,\n index,\n remove: () => remove(key),\n count: items.length,\n })}\n </Fragment>\n );\n })}\n\n {children?.({\n names: items.map((item) => item.name),\n add,\n remove,\n setValue,\n } as any)}\n </>\n );\n}\n","import { diff } from '@lib/diff';\nimport type { FormContext } from './form';\nimport { get, set } from '@lib/propAccess';\nimport { deepEqual } from '@lib/equals';\n\nexport interface OnOriginalChangeHandler<TDraft, TOriginal> {\n (\n oldOriginal: TOriginal | undefined,\n newOriginal: TOriginal | undefined,\n draft: TDraft,\n form: FormContext<TDraft, TOriginal>,\n ): TDraft | void;\n}\n\nexport type OnOriginalChangeBuiltin = 'default' | 'merge' | 'overwrite';\nexport type OnOriginalChange<TDraft, TOriginal> =\n | OnOriginalChangeHandler<TDraft, TOriginal>\n | OnOriginalChangeBuiltin;\n\nexport function onOriginalChangeDefault<TDraft, TOriginal>(\n _oldOriginal: TOriginal | undefined,\n _newOriginal: TOriginal | undefined,\n draft: TDraft,\n _form: FormContext<TDraft, TOriginal>,\n): TDraft | void {\n return draft;\n}\n\nexport function onOriginalChangeMerge<TDraft, TOriginal>(\n oldOriginal: TOriginal | undefined,\n newOriginal: TOriginal | undefined,\n draft: TDraft,\n _form: FormContext<TDraft, TOriginal>,\n): TDraft | void {\n const [patches] = diff(oldOriginal, newOriginal, { diffArrays: true });\n\n for (const p of patches) {\n const draftValue = get(draft, p.path as any);\n const oldValue = get(oldOriginal, p.path as any);\n\n if (deepEqual(draftValue, oldValue)) {\n const newValue = p.op === 'remove' ? undefined : p.value;\n draft = set(draft, p.path as any, newValue);\n }\n }\n\n return draft;\n}\n\nexport function onOriginalChangeOverwrite<TDraft, TOriginal extends TDraft>(\n _oldOriginal: TOriginal | undefined,\n newOriginal: TOriginal | undefined,\n _draft: TDraft,\n _form: FormContext<TDraft, TOriginal>,\n): TDraft | void {\n return newOriginal;\n}\n\nconst builtinHandlers = {\n default: onOriginalChangeDefault,\n merge: onOriginalChangeMerge,\n overwrite: onOriginalChangeOverwrite,\n};\n\nexport function resolveOnOriginalChange<TDraft, TOriginal>(\n onOriginalChange: OnOriginalChange<TDraft, TOriginal> | undefined,\n): OnOriginalChangeHandler<TDraft, TOriginal> {\n if (typeof onOriginalChange === 'function') {\n return onOriginalChange;\n }\n\n if (typeof onOriginalChange === 'string' && onOriginalChange in builtinHandlers) {\n return builtinHandlers[onOriginalChange as OnOriginalChangeBuiltin] as OnOriginalChangeHandler<\n TDraft,\n TOriginal\n >;\n }\n\n return onOriginalChangeDefault;\n}\n","import type { Duration } from '@core';\nimport { debounce } from '@lib/debounce';\nimport { calcDuration } from '@lib/duration';\nimport { deepEqual } from '@lib/equals';\nimport type { MaybePromise } from '@lib/maybePromise';\nimport { queue } from '@lib/queue';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport { useEffect, useMemo, useRef } from 'react';\nimport type { FormContext } from './form';\n\nexport interface FormAutosaveOptions<TDraft, TOriginal> {\n enabled?: boolean;\n validateBeforeSave?: boolean;\n save?: (draft: TDraft, prev: TDraft, form: FormContext<TDraft, TOriginal>) => MaybePromise<void>;\n debounce?: Duration;\n resetAfterSave?: boolean;\n equals?: (a: any, b: any) => boolean;\n}\n\nexport function useFormAutosave<TDraft, TOriginal extends TDraft>(\n form: FormContext<TDraft, TOriginal>,\n): {\n flush(): Promise<void>;\n cancel(): Promise<void>;\n} {\n const { enabled = true, validateBeforeSave = true } = form.options.autoSave ?? {};\n\n const isActive = enabled && !!form.options.autoSave?.save;\n const debounceTime = calcDuration(form.options.autoSave?.debounce ?? 2_000);\n const prev = useRef(form.getDraft());\n const q = useMemo(() => queue(), []);\n\n const latestSave = useLatestFunction(async () => {\n if (validateBeforeSave && !form.validate()) {\n return;\n }\n\n const draft = form.getDraft();\n const equals =\n form.options.autoSave?.equals ?? ((a, b) => deepEqual(a, b, { undefinedEqualsAbsent: true }));\n\n if (!isActive || equals(draft, prev.current) || equals(draft, form.options.original)) {\n return;\n }\n\n form.formState.set('saveInProgress', true);\n\n try {\n await form.options.autoSave?.save?.(draft, prev.current, form);\n prev.current = draft;\n\n if (q.size === 0 && form.options.autoSave?.resetAfterSave) {\n form.reset();\n }\n } catch (error) {\n console.error('Unhandled error during form autosave:', error);\n } finally {\n form.formState.set('saveInProgress', false);\n }\n });\n\n const scheduleSave = useMemo(\n () =>\n debounce(() => {\n q.clear();\n q(latestSave);\n }, debounceTime),\n [q, latestSave, debounceTime],\n );\n\n useEffect(() => {\n if (!isActive) {\n return;\n }\n\n return form.formState\n .map((state) => state.draft)\n .subscribe(\n (draft) => {\n if (draft === undefined) {\n return;\n }\n\n scheduleSave();\n },\n {\n runNow: false,\n },\n );\n }, [isActive, form.formState, scheduleSave]);\n\n useEffect(() => {\n return () => {\n scheduleSave.flush();\n };\n }, [scheduleSave]);\n\n return {\n flush() {\n scheduleSave.flush();\n return q.whenDone();\n },\n cancel() {\n scheduleSave.cancel();\n return q.whenDone();\n },\n };\n}\n","import { createStore, type Store, type Update } from '@core';\nimport { autobind } from '@lib/autobind';\nimport { deepEqual } from '@lib/equals';\nimport { isObject } from '@lib/helpers';\nimport {\n type PathAsString,\n type Value,\n type WildcardPathAsString,\n type WildcardValue,\n} from '@lib/path';\nimport { get, join, set } from '@lib/propAccess';\nimport type { Object_ } from '@lib/typeHelpers';\nimport { getWildCardMatches } from '@lib/wildcardMatch';\nimport { GeneralFormContext } from '@react/form/closestFormContext';\nimport { LegacyFormField, type FormFieldPropsWithComponent } from '@react/form/legacyFormField';\nimport { create, type Draft } from 'mutative';\nimport {\n createContext,\n forwardRef,\n useContext,\n useEffect,\n useEffectEvent,\n useMemo,\n useRef,\n type Context,\n type FormEvent,\n type ForwardedRef,\n type FunctionComponent,\n type HTMLProps,\n type ReactNode,\n} from 'react';\nimport { useStore, type UseStoreOptions } from '../useStore';\nimport {\n FormField,\n useFormFieldProps,\n type FormFieldComponent,\n type FormFieldComponentProps,\n type FormFieldInfos,\n type FormFieldProps,\n type FormFieldPropsWithChildren,\n type FormFieldPropsWithRender,\n} from './formField';\nimport { FormForEach, type ElementName, type FormForEachProps } from './formForEach';\nimport { resolveOnOriginalChange, type OnOriginalChange } from './formOnOriginalChange';\nimport { useFormAutosave, type FormAutosaveOptions } from './useFormAutosave';\n\n/// /////////////////////////////////////////////////////////////////////////////\n// Form types\n/// /////////////////////////////////////////////////////////////////////////////\n\nexport interface Transform<TDraft, TOriginal> {\n (value: Draft<TDraft>, form: FormContext<TDraft, TOriginal>): void | TDraft;\n}\n\nexport interface FormOptions<TDraft, TOriginal> {\n id?: string;\n defaultValue: TDraft;\n validations?: Validations<TDraft, TOriginal>;\n initiallyTriggerValidations?: boolean;\n localizeError?: (error: string, field: string) => string | undefined;\n autoSave?: FormAutosaveOptions<TDraft, TOriginal>;\n transform?: Transform<TDraft, TOriginal>;\n validatedClass?: string;\n original?: TOriginal;\n onSubmit?: (event: FormEvent<HTMLFormElement>, form: FormInstance<TDraft, TOriginal>) => void;\n reportValidity?: boolean | 'browser' | 'scrollTo';\n transformFieldProps?: <TPath extends string>(\n props: FormFieldComponentProps<Value<TDraft, TPath>, TPath>,\n info: FormFieldInfos<TDraft, TOriginal, TPath>,\n form: FormContext<TDraft, TOriginal>,\n ) => FormFieldComponentProps<Value<TDraft, TPath>, TPath>;\n onOriginalChange?: OnOriginalChange<TDraft, TOriginal>;\n}\n\nexport type Validations<TDraft, TOriginal> =\n | ((context: { draft: TDraft; original: TOriginal | undefined }) => Iterable<{\n name: string;\n error: string;\n }>)\n | ({\n [TPath in WildcardPathAsString<TDraft>]?: Record<\n string,\n Validation<TDraft, TOriginal, TPath>\n >;\n } & Record<string, Record<string, Validation<TDraft, TOriginal, any>>>);\n\nexport type Validation<TDraft, TOriginal, TPath> = (\n value: WildcardValue<TDraft, TPath>,\n context: {\n draft: TDraft;\n original: TOriginal;\n field: PathAsString<TDraft> | '';\n },\n) => boolean;\n\nexport type Field<TDraft, TOriginal, TPath extends string> = {\n originalValue: Value<TOriginal, TPath> | undefined;\n value: Value<TDraft, TPath>;\n setValue: (value: Update<Value<TDraft, TPath>>) => void;\n removeValue: () => void;\n hasChange: boolean;\n errors: string[];\n} & (Value<TDraft, TPath> extends Object_ ? FieldHelperMethods<TDraft, TPath> : {});\n\nexport type FieldHelperMethods<TDraft, TPath extends string> = {\n names: ElementName<TDraft, TPath>[];\n add: NonNullable<Value<TDraft, TPath>> extends readonly (infer T)[]\n ? (element: T) => void\n : NonNullable<Value<TDraft, TPath>> extends Record<infer K, infer V>\n ? (key: K, value: V) => void\n : never;\n remove: NonNullable<Value<TDraft, TPath>> extends readonly any[]\n ? (index: number) => void\n : (key: string) => void;\n};\n\nexport interface FormState<TDraft> {\n draft: TDraft | undefined;\n hasTriggeredValidations: boolean;\n saveInProgress: boolean;\n}\n\nexport interface FormDerivedState<TDraft> {\n draft: TDraft;\n hasTriggeredValidations: boolean;\n saveInProgress: boolean;\n hasChanges: boolean;\n errors: Map<string, string[]>;\n isValid: boolean;\n}\n\nexport interface FormContext<TDraft, TOriginal> {\n formState: Store<FormState<TDraft>>;\n options: FormOptions<TDraft, TOriginal>;\n original: TOriginal | undefined;\n getField: <TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n options?: FieldOptions,\n ) => Field<TDraft, TOriginal, TPath>;\n getDraft: () => TDraft;\n hasTriggeredValidations: () => boolean;\n saveInProgress: () => boolean;\n flushAutosave: () => Promise<void>;\n cancelAutosave: () => void;\n hasChanges: () => boolean;\n getErrors: () => Map<string, string[]>;\n isValid: () => boolean;\n validate: (options?: ValidateOptions) => boolean;\n reset: () => void;\n}\n\nexport interface FieldOptions {\n includeNestedErrors?: boolean;\n}\n\nexport interface ValidateOptions {\n reportValidity?: boolean | 'browser' | 'scrollTo';\n button?: HTMLButtonElement;\n}\n\nexport interface FormInstance<TDraft, TOriginal>\n extends\n FormDerivedState<TDraft>,\n Pick<\n FormContext<TDraft, TOriginal>,\n 'options' | 'original' | 'getField' | 'validate' | 'reset'\n > {}\n\n/// /////////////////////////////////////////////////////////////////////////////\n// Implementation\n/// /////////////////////////////////////////////////////////////////////////////\n\nconst FormContainer = forwardRef(function FormContainer(\n {\n form,\n ...formProps\n }: {\n form: Form<any, any>;\n onSubmit?: (\n event: FormEvent<HTMLFormElement>,\n form: FormInstance<any, any>,\n ) => void | Promise<void>;\n } & Omit<HTMLProps<HTMLFormElement>, 'form' | 'onSubmit'>,\n ref: ForwardedRef<HTMLFormElement>,\n) {\n const formInstance = form.useForm();\n const hasTriggeredValidations = form.useFormState((state) => state.hasTriggeredValidations);\n const hasErrors = form.useFormState((state) => hasTriggeredValidations && state.errors.size > 0);\n\n return (\n <form\n ref={ref}\n noValidate\n {...formProps}\n className={[\n formProps.className,\n hasTriggeredValidations ? (formInstance.options.validatedClass ?? 'validated') : undefined,\n ]\n .filter(Boolean)\n .join(' ')}\n data-validated={hasTriggeredValidations || undefined}\n data-valid={hasErrors ? 'false' : hasTriggeredValidations ? 'true' : undefined}\n onSubmit={async (event) => {\n if (formInstance.saveInProgress()) {\n return;\n }\n\n try {\n formInstance.formState.set('saveInProgress', true);\n event.preventDefault();\n\n const button =\n event.nativeEvent instanceof SubmitEvent &&\n event.nativeEvent.submitter instanceof HTMLButtonElement\n ? event.nativeEvent.submitter\n : undefined;\n\n const isValid = formInstance.validate({ button });\n if (isValid) {\n await formProps.onSubmit?.(event, {\n ...formInstance,\n ...getDerivedState(formInstance),\n });\n }\n } finally {\n formInstance.formState.set('saveInProgress', false);\n }\n }}\n />\n );\n});\n\nfunction getField<TDraft, TOriginal extends TDraft, TPath extends string>(\n form: FormContext<TDraft, TOriginal>,\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n { includeNestedErrors }: FieldOptions = {},\n): Field<TDraft, TOriginal, TPath> {\n const field = {\n get originalValue() {\n return form.original !== undefined ? get(form.original as any, name as any) : undefined;\n },\n\n get value() {\n const draft = form.getDraft();\n return get(draft ?? form.original ?? form.options.defaultValue, name as any);\n },\n\n setValue(update: Update<Value<TDraft, TPath>>) {\n form.formState.set('draft', (draft = form.original ?? form.options.defaultValue) => {\n if (update instanceof Function) {\n update = update(get(draft, name as any) as Value<TDraft, TPath>);\n }\n\n return set(draft, name as any, update as any);\n });\n },\n\n get hasChange() {\n return !deepEqual(this.originalValue, this.value, { undefinedEqualsAbsent: true });\n },\n\n get errors() {\n const errors = form.getErrors();\n\n if (includeNestedErrors) {\n return Array.from(errors.entries())\n .filter(([key]) => key === name || key.startsWith(`${name}.`))\n .flatMap(([, value]) => value);\n } else {\n return errors.get(name) ?? [];\n }\n },\n\n get names(): any {\n const { value } = this;\n\n if (isObject(value)) {\n return Object.keys(value).map((key) => join(name, key));\n }\n\n return [];\n },\n\n add(...args: any[]) {\n this.setValue((value): any => {\n if (!value) {\n throw new Error(`Cannot add element to ${JSON.stringify(value)}`);\n }\n\n if (Array.isArray(value)) {\n return [...(value ?? []), args[0]];\n }\n\n if (isObject(value)) {\n return {\n ...value,\n [args[0]]: args[1],\n };\n }\n\n throw new Error(`Cannot add element to ${JSON.stringify(value)}`);\n });\n },\n\n remove(key: string | number) {\n this.setValue((value): any => {\n if (!value) {\n throw new Error(`Cannot remove element from ${JSON.stringify(value)}`);\n }\n\n if (Array.isArray(value)) {\n return value.filter((_, index) => index !== Number(key));\n }\n\n if (isObject(value)) {\n const { [key]: _, ...rest } = value as Record<string | number, unknown>;\n return rest;\n }\n\n throw new Error(`Cannot remove element from ${JSON.stringify(value)}`);\n });\n },\n };\n\n return field as any;\n}\n\nfunction getErrors<TDraft, TOriginal>(\n draft: TDraft,\n { original, validations, localizeError }: FormOptions<TDraft, TOriginal>,\n) {\n const errors = new Map<string, string[]>();\n\n if (typeof validations === 'function') {\n const issues = validations({ draft, original });\n\n for (const { name, error } of issues) {\n const fieldErrors = errors.get(name) ?? [];\n fieldErrors.push(error);\n errors.set(name, fieldErrors);\n }\n } else {\n for (const [path, block] of Object.entries(validations ?? {})) {\n for (const [validationName, validate] of Object.entries(\n block as Record<string, Validation<any, any, any>>,\n )) {\n let matched = false;\n\n for (const [field, value] of Object.entries(getWildCardMatches(draft, path))) {\n matched = true;\n if (!validate(value, { draft, original, field })) {\n const fieldErrors = errors.get(field) ?? [];\n fieldErrors.push(validationName);\n errors.set(field, fieldErrors);\n }\n }\n\n if (!matched && !path.includes('*')) {\n if (!validate(undefined, { draft, original, field: path })) {\n const fieldErrors = errors.get(path) ?? [];\n fieldErrors.push(validationName);\n errors.set(path, fieldErrors);\n }\n }\n }\n }\n }\n\n if (localizeError) {\n for (const [field, fieldErrors] of errors.entries()) {\n errors.set(\n field,\n fieldErrors.map((error) => localizeError(error, field) ?? error),\n );\n }\n }\n\n return errors;\n}\n\nexport function getDerivedState<TDraft>(\n instance: FormContext<TDraft, any>,\n): FormDerivedState<TDraft> {\n return {\n draft: instance.getDraft(),\n hasTriggeredValidations: instance.hasTriggeredValidations(),\n saveInProgress: instance.saveInProgress(),\n hasChanges: instance.hasChanges(),\n errors: instance.getErrors(),\n isValid: instance.isValid(),\n };\n}\n\nexport class Form<TDraft, TOriginal extends TDraft = TDraft> {\n context: Context<FormContext<TDraft, TOriginal> | null> = createContext<FormContext<\n TDraft,\n TOriginal\n > | null>(null);\n\n constructor(public readonly options: FormOptions<TDraft, TOriginal>) {\n autobind(Form);\n }\n\n useForm(): FormContext<TDraft, TOriginal> {\n const context = useContext(this.context);\n\n if (!context) {\n throw new Error('Form context not found');\n }\n\n return context;\n }\n\n useFormState<S>(\n selector: (state: FormInstance<TDraft, TOriginal>) => S,\n useStoreOptions?: UseStoreOptions<S>,\n ): S {\n const form = this.useForm();\n\n return useStore(\n form.formState,\n () =>\n selector({\n ...form,\n ...getDerivedState(form),\n }),\n\n useStoreOptions,\n );\n }\n\n useField<TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n { includeNestedErrors, ...useStoreOptions }: FieldOptions & UseStoreOptions<unknown> = {},\n ): Field<TDraft, TOriginal, TPath> {\n const form = this.useForm();\n this.useFormState((form) => [form.getField(name).value, form.original], useStoreOptions);\n\n return form.getField(name, { includeNestedErrors });\n }\n\n useFieldProps<TPath extends string>(\n name: TPath extends PathAsString<TDraft> ? TPath : PathAsString<TDraft>,\n options?: Omit<FormFieldProps<TPath, TDraft>, 'name'>,\n ): FormFieldComponentProps<Value<TDraft, TPath>, TPath> {\n return useFormFieldProps.call<\n Form<TDraft, TOriginal>,\n [FormFieldProps<TPath, TDraft>],\n FormFieldComponentProps<Value<TDraft, TPath>, TPath>\n >(this, { name, ...options } as any);\n }\n\n // ///////////////////////////////////////////////////////////////////////////\n // React Components\n // ///////////////////////////////////////////////////////////////////////////\n\n Form({\n defaultValue,\n validations,\n initiallyTriggerValidations,\n localizeError,\n autoSave,\n transform,\n validatedClass,\n original,\n onSubmit,\n reportValidity,\n transformFieldProps,\n onOriginalChange,\n ...formProps\n }: Partial<FormOptions<TDraft, TOriginal>> &\n Omit<HTMLProps<HTMLFormElement>, 'defaultValue' | 'autoSave' | 'onSubmit'>): React.JSX.Element {\n const options: FormOptions<TDraft, TOriginal> = {\n defaultValue: { ...this.options.defaultValue, ...defaultValue },\n validations:\n typeof validations === 'function'\n ? validations\n : validations\n ? ({ ...this.options.validations, ...validations } as Validations<TDraft, TOriginal>)\n : this.options.validations,\n localizeError: localizeError ?? this.options.localizeError,\n autoSave:\n this.options.autoSave || autoSave\n ? ({ ...this.options.autoSave, ...autoSave } as FormAutosaveOptions<TDraft, TOriginal>)\n : undefined,\n transform: transform ?? this.options.transform,\n validatedClass: validatedClass ?? this.options.validatedClass,\n original: original ?? this.options.original,\n onSubmit: onSubmit ?? this.options.onSubmit,\n reportValidity: reportValidity ?? this.options.reportValidity ?? 'browser',\n transformFieldProps: transformFieldProps ?? this.options.transformFieldProps,\n onOriginalChange: onOriginalChange ?? this.options.onOriginalChange ?? 'default',\n };\n\n const formState = useMemo(() => {\n return createStore<FormState<TDraft>>({\n draft: undefined,\n hasTriggeredValidations: initiallyTriggerValidations ?? false,\n saveInProgress: false,\n });\n // oxlint-disable-next-line exhaustive-deps\n }, []);\n\n const formRef = useRef<HTMLFormElement>(null);\n\n let lastDraft: TDraft | undefined;\n const cache = new Map<string, unknown>();\n function lazy<T>(key: string, fn: () => T): T {\n if (lastDraft !== formState.get().draft) {\n cache.clear();\n lastDraft = formState.get().draft;\n }\n\n let value = cache.get(key);\n if (!cache.has(key)) {\n value = fn();\n cache.set(key, value);\n }\n\n return value as T;\n }\n\n const context: FormContext<TDraft, TOriginal> = {\n formState,\n options,\n original: options.original,\n\n getField() {\n throw new Error('Not implemented');\n },\n\n getDraft() {\n return formState.get().draft ?? options.original ?? options.defaultValue;\n },\n\n hasTriggeredValidations() {\n return formState.get().hasTriggeredValidations;\n },\n\n saveInProgress() {\n return formState.get().saveInProgress;\n },\n\n flushAutosave() {\n return autosave.flush();\n },\n\n cancelAutosave() {\n return autosave.cancel();\n },\n\n hasChanges() {\n return lazy(\n 'hasChanges',\n () =>\n !deepEqual(this.getDraft(), options.original ?? options.defaultValue, {\n undefinedEqualsAbsent: true,\n }),\n );\n },\n\n getErrors() {\n return lazy('getErrors', () => getErrors(this.getDraft(), options));\n },\n\n isValid() {\n return lazy('isValid', () => this.getErrors().size === 0);\n },\n\n validate({ reportValidity = options.reportValidity, button }: ValidateOptions = {}) {\n formState.set('hasTriggeredValidations', true);\n\n updateValidity(this.getErrors(), button);\n\n switch (reportValidity) {\n case 'browser':\n formRef.current?.reportValidity();\n break;\n\n case true:\n case 'scrollTo':\n {\n const invalidElement = document.querySelector(':invalid, [data-invalid=\"true\"]');\n if (invalidElement && invalidElement instanceof HTMLElement) {\n invalidElement.scrollIntoView({ behavior: 'smooth', block: 'center' });\n invalidElement.focus({ preventScroll: true });\n }\n }\n break;\n }\n\n return this.isValid();\n },\n\n reset() {\n formState.set('draft', undefined);\n formState.set('hasTriggeredValidations', false);\n },\n };\n\n context.getField = (path, options) =>\n lazy(`${path}:${options?.includeNestedErrors}`, () => getField(context, path, options));\n\n useEffect(() => {\n const transform = options.transform;\n if (!transform) {\n return;\n }\n\n return context.formState.subscribe((state) => {\n const value = state.draft ?? options.original ?? options.defaultValue;\n const result = create(value, (draft) => transform(draft, context)) as TDraft;\n\n if (!deepEqual(result, value)) {\n context.formState.set('draft', result);\n }\n });\n });\n\n const lastOriginal = useRef(options.original);\n\n useEffect(() => {\n const draft = formState.get().draft;\n\n if (draft !== undefined && !deepEqual(options.original, lastOriginal.current)) {\n const handler = resolveOnOriginalChange(options.onOriginalChange);\n const result = handler(lastOriginal.current, options.original, draft, context);\n\n if (result !== undefined && !deepEqual(result, draft)) {\n formState.set('draft', result);\n }\n }\n\n lastOriginal.current = options.original;\n // oxlint-disable-next-line exhaustive-deps\n }, [options.original]);\n\n function updateValidity(errors: Map<string, string[]>, buttonElement?: HTMLButtonElement) {\n const formElement = formRef.current;\n if (!formElement) {\n return;\n }\n\n for (const element of Array.from(formElement.elements)) {\n if ('name' in element && 'setCustomValidity' in element) {\n (element as HTMLObjectElement).setCustomValidity(\n errors.get((element as HTMLObjectElement).name)?.join('\\n') ?? '',\n );\n }\n }\n\n if (buttonElement && 'setCustomValidity' in buttonElement) {\n const errorString = [...errors.values()].flat().join('\\n');\n\n buttonElement.setCustomValidity(errorString);\n }\n }\n\n useEffect(() => {\n return formState.map(() => context.getErrors()).subscribe((errors) => updateValidity(errors));\n });\n\n const autosave = useFormAutosave(context);\n\n return (\n <GeneralFormContext.Provider value={this}>\n <this.context.Provider value={context}>\n <FormContainer {...formProps} ref={formRef} form={this} onSubmit={options.onSubmit} />\n </this.context.Provider>\n </GeneralFormContext.Provider>\n );\n }\n\n FormState<S>({\n selector,\n children,\n }: {\n selector: (form: FormInstance<TDraft, TOriginal>) => S;\n children: (selectedState: S) => ReactNode;\n }): React.JSX.Element {\n const selectedState = this.useFormState(selector);\n return <>{children(selectedState)}</>;\n }\n\n Field<const TPath extends string>(\n props: FormFieldPropsWithRender<TDraft, TOriginal, TPath>,\n ): React.JSX.Element;\n\n Field<const TPath extends string>(\n props: FormFieldPropsWithChildren<TDraft, TOriginal, TPath>,\n ): React.JSX.Element;\n\n /** @deprecated */\n Field<const TPath extends string, const TComponent extends FormFieldComponent = 'input'>(\n props: FormFieldPropsWithComponent<TDraft, TOriginal, TPath, TComponent>,\n ): React.JSX.Element;\n\n Field(props: any): React.JSX.Element {\n if (props.component) {\n return Reflect.apply(LegacyFormField, this, [props]);\n }\n\n return Reflect.apply(FormField, this, [props]);\n }\n\n ForEach<const TPath extends string>(props: FormForEachProps<TDraft, TPath>): React.JSX.Element {\n return Reflect.apply(FormForEach, this, [props]);\n }\n\n withForm<TProps extends Record<string, unknown>>(\n Component: React.ComponentType<TProps>,\n formProps?: Parameters<this['Form']>[0],\n ): FunctionComponent<TProps> {\n const { Form } = this;\n return function FormWrapper(props: TProps) {\n return (\n <Form {...formProps}>\n <Component {...props} />\n </Form>\n );\n };\n }\n}\n\nexport function createForm<TDraft, TOriginal extends TDraft = TDraft>(\n options: FormOptions<TDraft, TOriginal>,\n): Form<TDraft, TOriginal> {\n return new Form(options);\n}\n","export function castArray<T>(value: T | T[]): T[] {\n return Array.isArray(value) ? value : [value];\n}\n","import { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\n\nexport function defaultDeserializer<T>(value: string): T {\n if (value === undefined) {\n return undefined as T;\n }\n\n try {\n return fromExtendedJsonString(value) as T;\n } catch {\n return undefined as T;\n }\n}\n\nexport function defaultSerializer<T>(value: T): string {\n return toExtendedJsonString(value);\n}\n\nexport function normalizePath<T>(path: string | T): string | T {\n if (typeof path === 'string') {\n return path.replace(/^\\//g, '').replace(/\\/$/g, '');\n }\n return path;\n}\n","import { castArray } from '@lib/castArray';\nimport { defaultDeserializer, defaultSerializer, normalizePath } from '@react/url/urlHelpers';\n\nexport interface UrlOptions<T> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue: T;\n writeDefaultValue?: boolean;\n onCommit?: (value: T) => void;\n persist?: { id: string } | null;\n path?: string | RegExp | (string | RegExp)[] | null;\n}\n\nexport interface UrlOptionsWithoutDefaults<T> extends Omit<\n UrlOptions<T | undefined>,\n 'defaultValue'\n> {\n defaultValue?: T | undefined;\n}\n\nexport function createUrlOptions<T>(options: UrlOptions<T>): Required<UrlOptions<T>>;\nexport function createUrlOptions<T>(\n options: UrlOptionsWithoutDefaults<T>,\n): Required<UrlOptions<T | undefined>>;\nexport function createUrlOptions<T>({\n key,\n type = 'hash',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n writeDefaultValue = false,\n onCommit = () => undefined,\n persist = null,\n path = null,\n}: UrlOptionsWithoutDefaults<T>): Required<UrlOptionsWithoutDefaults<T>> {\n return {\n key,\n type,\n serialize,\n deserialize,\n defaultValue,\n writeDefaultValue,\n onCommit,\n persist,\n path: path === null ? null : castArray(path).map(normalizePath),\n };\n}\n","import { createStore, Store, type Update } from '@core';\nimport { autobind } from '@lib/autobind';\nimport { castArray } from '@lib/castArray';\nimport type { Constrain } from '@lib/constrain';\nimport type { Path, Value } from '@lib/path';\nimport { get, set } from '@lib/propAccess';\nimport { normalizePath } from '@react/url/urlHelpers';\nimport {\n createUrlOptions,\n type UrlOptions,\n type UrlOptionsWithoutDefaults,\n} from '@react/url/urlOptions';\n\nexport const urlStore: Store<string> = createStore(() => window.location.href, {\n cacheValue: false,\n effect() {\n const update = () => {\n if (window.location.href !== this.calculatedValue?.value) {\n this.invalidate();\n }\n };\n\n const interval = setInterval(update, 1);\n window.addEventListener('popstate', update);\n\n return () => {\n clearInterval(interval);\n window.removeEventListener('popstate', update);\n };\n },\n});\n\nexport class UrlParamStore<T> extends Store<T> {\n readonly storageKey: string | null;\n private lastHref?: string;\n private lastStorageValue?: string | null;\n private lastValue?: T;\n\n constructor(public readonly urlOptions: Required<UrlOptions<T>>) {\n super(() => this.calc(), { cacheValue: false });\n autobind(UrlParamStore);\n\n this.storageKey =\n urlOptions.persist && `cross-state:url:${urlOptions.persist.id}:${urlOptions.key}`;\n this.addEffect(this.watch);\n }\n\n private watch() {\n let isActive = false;\n let urlValue = this.getUrlValue();\n let storageValue = this.getStorageValue();\n\n const update = () => {\n const oldIsActive = isActive;\n isActive = this.isPathActive();\n const oldUrlValue = urlValue;\n urlValue = this.getUrlValue();\n const oldStorageValue = storageValue;\n storageValue = this.getStorageValue();\n\n // If inactive => ignore changes\n if (!isActive) {\n return;\n }\n\n // No changes => ignore\n if (\n isActive === oldIsActive &&\n urlValue === oldUrlValue &&\n storageValue === oldStorageValue\n ) {\n return;\n }\n\n if (!oldIsActive) {\n // Became active =>\n // - if url has value => update storage\n // - else if storage has value or writeDefaultValue => update url\n if (urlValue !== null) {\n this.updateStorage(this.urlOptions.deserialize(urlValue));\n } else if (storageValue !== null) {\n this.updateUrl(this.urlOptions.deserialize(storageValue));\n } else if (this.urlOptions.writeDefaultValue) {\n this.updateUrl(this.urlOptions.defaultValue);\n }\n } else if (urlValue !== oldUrlValue) {\n // Url change while active =>\n // - if url has no value and writeDefaultValue => update url\n // - update storage\n if (urlValue === null && this.urlOptions.writeDefaultValue) {\n this.updateUrl(this.urlOptions.defaultValue);\n }\n\n this.updateStorage(\n urlValue !== null ? this.urlOptions.deserialize(urlValue) : this.urlOptions.defaultValue,\n );\n }\n\n this.invalidate();\n };\n\n const cancel = urlStore.subscribe(update);\n window.addEventListener('storage', update);\n\n return () => {\n cancel();\n window.removeEventListener('storage', update);\n };\n }\n\n private getUrlValue() {\n const href = urlStore.get();\n const url = new URL(href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1) || '');\n return params.get(this.urlOptions.key);\n }\n\n private getStorageValue() {\n return this.storageKey !== null ? localStorage.getItem(this.storageKey) : null;\n }\n\n private isPathActive() {\n if (this.urlOptions.path === null) {\n return true;\n }\n\n const path = normalizePath(window.location.pathname);\n\n return castArray(this.urlOptions.path).some((p) => {\n if (typeof p === 'string') {\n return !p || p === path || path.startsWith(p + '/');\n }\n\n return p.test(path);\n });\n }\n\n private calc() {\n let href = window.location.href;\n const storageValue = this.storageKey !== null ? localStorage.getItem(this.storageKey) : null;\n\n if (!this.isPathActive() && this.lastHref !== undefined) {\n href = this.lastHref;\n }\n\n if (this.lastHref === href && this.lastStorageValue === storageValue) {\n return this.lastValue as T;\n }\n\n const url = new URL(href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1));\n const urlValue = params.get(this.urlOptions.key);\n\n const value =\n urlValue !== null\n ? this.urlOptions.deserialize(urlValue)\n : this.storageKey !== null && storageValue !== null\n ? this.urlOptions.deserialize(storageValue)\n : this.urlOptions.defaultValue;\n\n this.lastHref = href;\n this.lastStorageValue = storageValue;\n this.lastValue = value;\n return value;\n }\n\n private updateUrl(value: T) {\n const serializedValue = this.urlOptions.serialize(value);\n\n const url = new URL(window.location.href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1));\n\n if (\n !this.urlOptions.writeDefaultValue &&\n serializedValue === this.urlOptions.serialize(this.urlOptions.defaultValue)\n ) {\n params.delete(this.urlOptions.key);\n } else {\n params.set(this.urlOptions.key, serializedValue);\n }\n\n url[this.urlOptions.type] = params.toString();\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate', { state: window.history.state }));\n }\n\n private updateStorage(value: T) {\n if (this.storageKey === null) {\n return;\n }\n\n const serializedValue = this.urlOptions.serialize(value);\n localStorage.setItem(this.storageKey, serializedValue);\n }\n\n set(update: Update<T>): void;\n set<const P>(path: Constrain<P, Path<T>>, update: Update<Value<T, P>>): void;\n set(...args: any[]): void {\n const path: any = args.length > 1 ? args[0] : [];\n let update: Update<any> = args.length > 1 ? args[1] : args[0];\n\n if (update instanceof Function) {\n const before = this.get();\n const valueBefore = get(before, path);\n const valueAfter = update(valueBefore);\n update = set(before, path, valueAfter);\n } else if (path.length > 0) {\n update = set(this.get(), path, update);\n }\n\n if (this.isPathActive()) {\n this.updateUrl(update);\n } else {\n this.updateStorage(update);\n }\n }\n\n parse(path: string): T {\n const url = new URL(path, window.location.href);\n const params = new URLSearchParams(url[this.urlOptions.type].slice(1) || '');\n const urlValue = params.get(this.urlOptions.key);\n return urlValue !== null ? this.urlOptions.deserialize(urlValue) : this.urlOptions.defaultValue;\n }\n}\n\nexport function createUrlParam<T>(options: UrlOptions<T>): UrlParamStore<T>;\nexport function createUrlParam<T>(\n options: UrlOptionsWithoutDefaults<T>,\n): UrlParamStore<T | undefined>;\nexport function createUrlParam<T>(options: UrlOptionsWithoutDefaults<T>) {\n return new UrlParamStore(createUrlOptions(options));\n}\n","import { type Duration } from '@core';\nimport { debounce } from '@lib/debounce';\nimport isPromise from '@lib/isPromise';\nimport type { MaybePromise } from '@lib/maybePromise';\nimport { throttle } from '@lib/throttle';\nimport useLatestFunction from '@react/lib/useLatestFunction';\nimport useMemoEquals from '@react/lib/useMemoEquals';\nimport { startTransition, useMemo, useRef, useState } from 'react';\n\nexport interface UseDecoupledStateOptions<T> {\n debounce?: Duration;\n throttle?: Duration;\n onCommit?: (value: T) => void;\n}\n\nexport function useDecoupledState<T>(\n value: T,\n onChange: (value: T) => MaybePromise<void>,\n options: UseDecoupledStateOptions<T> = {},\n): [state: T, setState: (value: T) => void] {\n const [dirty, setDirty] = useState<{ v: T }>();\n const onChangeAC = useRef<AbortController>(undefined);\n\n const latestOnChange = useLatestFunction(onChange);\n const latestOnCommit = useLatestFunction(options.onCommit ?? (() => {}));\n const debounceOptions = useMemoEquals(options.debounce);\n const throttleOptions = useMemoEquals(options.throttle);\n\n const update = useMemo(() => {\n const update = async (value: T) => {\n const result = latestOnChange(value);\n\n if (isPromise(result)) {\n const ac = (onChangeAC.current = new AbortController());\n\n await result;\n\n if (ac.signal.aborted) {\n return;\n }\n }\n\n latestOnCommit(value);\n setDirty(undefined);\n };\n\n let delayedUpdate: (value: T) => void;\n\n if (debounceOptions) {\n delayedUpdate = debounce(update, debounceOptions);\n } else if (throttleOptions) {\n delayedUpdate = throttle(update, throttleOptions);\n } else {\n delayedUpdate = (value) => startTransition(() => update(value));\n }\n\n return (value: T) => {\n onChangeAC.current?.abort();\n setDirty({ v: value });\n delayedUpdate(value);\n };\n }, [latestOnChange, latestOnCommit, debounceOptions, throttleOptions]);\n\n return [dirty ? dirty.v : value, update];\n}\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
//#region src/lib/extendedJson.ts
|
|
2
|
-
function toExtendedJson(value) {
|
|
3
|
-
if (value instanceof Map) return { __map: [...value].map(([k, v]) => [toExtendedJson(k), toExtendedJson(v)]) };
|
|
4
|
-
if (value instanceof Set) return { __set: [...value].map(toExtendedJson) };
|
|
5
|
-
if (value instanceof Date) return { __date: value.toISOString() };
|
|
6
|
-
if (typeof value === "bigint") return { __bigint: value.toString() };
|
|
7
|
-
if (Array.isArray(value)) return value.map(toExtendedJson);
|
|
8
|
-
if (typeof value === "object" && value !== null) return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, toExtendedJson(v)]));
|
|
9
|
-
return value;
|
|
10
|
-
}
|
|
11
|
-
function toExtendedJsonString(value, ...args) {
|
|
12
|
-
return JSON.stringify(toExtendedJson(value), ...args);
|
|
13
|
-
}
|
|
14
|
-
function fromExtendedJson(value) {
|
|
15
|
-
if (typeof value !== "object" || value === null) return value;
|
|
16
|
-
if ("__map" in value) return new Map(value.__map.map(([k, v]) => [fromExtendedJson(k), fromExtendedJson(v)]));
|
|
17
|
-
if ("__set" in value) return new Set(value.__set.map(fromExtendedJson));
|
|
18
|
-
if ("__date" in value) return new Date(value.__date);
|
|
19
|
-
if ("__bigint" in value) return BigInt(value.__bigint);
|
|
20
|
-
if (Array.isArray(value)) return value.map(fromExtendedJson);
|
|
21
|
-
return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, fromExtendedJson(v)]));
|
|
22
|
-
}
|
|
23
|
-
function fromExtendedJsonString(value, reviver) {
|
|
24
|
-
return fromExtendedJson(JSON.parse(value, reviver));
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
//#endregion
|
|
28
|
-
export { toExtendedJsonString as i, fromExtendedJsonString as n, toExtendedJson as r, fromExtendedJson as t };
|
|
29
|
-
//# sourceMappingURL=extendedJson-B1OEqZtF.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"extendedJson-B1OEqZtF.js","names":[],"sources":["../src/lib/extendedJson.ts"],"sourcesContent":["export function toExtendedJson(value: unknown): unknown {\n if (value instanceof Map) {\n return {\n __map: [...value].map(([k, v]) => [toExtendedJson(k), toExtendedJson(v)]),\n };\n }\n\n if (value instanceof Set) {\n return {\n __set: [...value].map(toExtendedJson),\n };\n }\n\n if (value instanceof Date) {\n return {\n __date: value.toISOString(),\n };\n }\n\n if (typeof value === 'bigint') {\n return {\n __bigint: value.toString(),\n };\n }\n\n if (Array.isArray(value)) {\n return value.map(toExtendedJson);\n }\n\n if (typeof value === 'object' && value !== null) {\n return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, toExtendedJson(v)]));\n }\n\n return value;\n}\n\nexport function toExtendedJsonString(\n value: any,\n replacer?: (this: any, key: string, value: any) => any,\n space?: string | number,\n): string;\nexport function toExtendedJsonString(\n value: any,\n replacer?: (number | string)[] | null,\n space?: string | number,\n): string;\nexport function toExtendedJsonString(value: unknown, ...args: any[]): string {\n return JSON.stringify(toExtendedJson(value), ...args);\n}\n\nexport function fromExtendedJson(value: unknown): unknown {\n if (typeof value !== 'object' || value === null) {\n return value;\n }\n\n if ('__map' in value) {\n return new Map(\n (value.__map as [unknown, unknown][]).map(([k, v]) => [\n fromExtendedJson(k),\n fromExtendedJson(v),\n ]),\n );\n }\n\n if ('__set' in value) {\n return new Set((value.__set as unknown[]).map(fromExtendedJson));\n }\n\n if ('__date' in value) {\n return new Date(value.__date as string);\n }\n\n if ('__bigint' in value) {\n return BigInt(value.__bigint as string);\n }\n\n if (Array.isArray(value)) {\n return value.map(fromExtendedJson);\n }\n\n return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, fromExtendedJson(v)]));\n}\n\nexport function fromExtendedJsonString(\n value: string,\n reviver?: (this: any, key: string, value: any) => any,\n): unknown {\n return fromExtendedJson(JSON.parse(value, reviver));\n}\n"],"mappings":";AAAA,SAAgB,eAAe,OAAyB;AACtD,KAAI,iBAAiB,IACnB,QAAO,EACL,OAAO,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,eAAe,EAAE,EAAE,eAAe,EAAE,CAAC,CAAC,EAC1E;AAGH,KAAI,iBAAiB,IACnB,QAAO,EACL,OAAO,CAAC,GAAG,MAAM,CAAC,IAAI,eAAe,EACtC;AAGH,KAAI,iBAAiB,KACnB,QAAO,EACL,QAAQ,MAAM,aAAa,EAC5B;AAGH,KAAI,OAAO,UAAU,SACnB,QAAO,EACL,UAAU,MAAM,UAAU,EAC3B;AAGH,KAAI,MAAM,QAAQ,MAAM,CACtB,QAAO,MAAM,IAAI,eAAe;AAGlC,KAAI,OAAO,UAAU,YAAY,UAAU,KACzC,QAAO,OAAO,YAAY,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,GAAG,eAAe,EAAE,CAAC,CAAC,CAAC;AAG1F,QAAO;;AAaT,SAAgB,qBAAqB,OAAgB,GAAG,MAAqB;AAC3E,QAAO,KAAK,UAAU,eAAe,MAAM,EAAE,GAAG,KAAK;;AAGvD,SAAgB,iBAAiB,OAAyB;AACxD,KAAI,OAAO,UAAU,YAAY,UAAU,KACzC,QAAO;AAGT,KAAI,WAAW,MACb,QAAO,IAAI,IACR,MAAM,MAA+B,KAAK,CAAC,GAAG,OAAO,CACpD,iBAAiB,EAAE,EACnB,iBAAiB,EAAE,CACpB,CAAC,CACH;AAGH,KAAI,WAAW,MACb,QAAO,IAAI,IAAK,MAAM,MAAoB,IAAI,iBAAiB,CAAC;AAGlE,KAAI,YAAY,MACd,QAAO,IAAI,KAAK,MAAM,OAAiB;AAGzC,KAAI,cAAc,MAChB,QAAO,OAAO,MAAM,SAAmB;AAGzC,KAAI,MAAM,QAAQ,MAAM,CACtB,QAAO,MAAM,IAAI,iBAAiB;AAGpC,QAAO,OAAO,YAAY,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,GAAG,iBAAiB,EAAE,CAAC,CAAC,CAAC;;AAG5F,SAAgB,uBACd,OACA,SACS;AACT,QAAO,iBAAiB,KAAK,MAAM,OAAO,QAAQ,CAAC"}
|
|
@@ -1,53 +0,0 @@
|
|
|
1
|
-
|
|
2
|
-
//#region src/lib/extendedJson.ts
|
|
3
|
-
function toExtendedJson(value) {
|
|
4
|
-
if (value instanceof Map) return { __map: [...value].map(([k, v]) => [toExtendedJson(k), toExtendedJson(v)]) };
|
|
5
|
-
if (value instanceof Set) return { __set: [...value].map(toExtendedJson) };
|
|
6
|
-
if (value instanceof Date) return { __date: value.toISOString() };
|
|
7
|
-
if (typeof value === "bigint") return { __bigint: value.toString() };
|
|
8
|
-
if (Array.isArray(value)) return value.map(toExtendedJson);
|
|
9
|
-
if (typeof value === "object" && value !== null) return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, toExtendedJson(v)]));
|
|
10
|
-
return value;
|
|
11
|
-
}
|
|
12
|
-
function toExtendedJsonString(value, ...args) {
|
|
13
|
-
return JSON.stringify(toExtendedJson(value), ...args);
|
|
14
|
-
}
|
|
15
|
-
function fromExtendedJson(value) {
|
|
16
|
-
if (typeof value !== "object" || value === null) return value;
|
|
17
|
-
if ("__map" in value) return new Map(value.__map.map(([k, v]) => [fromExtendedJson(k), fromExtendedJson(v)]));
|
|
18
|
-
if ("__set" in value) return new Set(value.__set.map(fromExtendedJson));
|
|
19
|
-
if ("__date" in value) return new Date(value.__date);
|
|
20
|
-
if ("__bigint" in value) return BigInt(value.__bigint);
|
|
21
|
-
if (Array.isArray(value)) return value.map(fromExtendedJson);
|
|
22
|
-
return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, fromExtendedJson(v)]));
|
|
23
|
-
}
|
|
24
|
-
function fromExtendedJsonString(value, reviver) {
|
|
25
|
-
return fromExtendedJson(JSON.parse(value, reviver));
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
//#endregion
|
|
29
|
-
Object.defineProperty(exports, 'fromExtendedJson', {
|
|
30
|
-
enumerable: true,
|
|
31
|
-
get: function () {
|
|
32
|
-
return fromExtendedJson;
|
|
33
|
-
}
|
|
34
|
-
});
|
|
35
|
-
Object.defineProperty(exports, 'fromExtendedJsonString', {
|
|
36
|
-
enumerable: true,
|
|
37
|
-
get: function () {
|
|
38
|
-
return fromExtendedJsonString;
|
|
39
|
-
}
|
|
40
|
-
});
|
|
41
|
-
Object.defineProperty(exports, 'toExtendedJson', {
|
|
42
|
-
enumerable: true,
|
|
43
|
-
get: function () {
|
|
44
|
-
return toExtendedJson;
|
|
45
|
-
}
|
|
46
|
-
});
|
|
47
|
-
Object.defineProperty(exports, 'toExtendedJsonString', {
|
|
48
|
-
enumerable: true,
|
|
49
|
-
get: function () {
|
|
50
|
-
return toExtendedJsonString;
|
|
51
|
-
}
|
|
52
|
-
});
|
|
53
|
-
//# sourceMappingURL=extendedJson-BwYSvpOA.cjs.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"extendedJson-BwYSvpOA.cjs","names":[],"sources":["../src/lib/extendedJson.ts"],"sourcesContent":["export function toExtendedJson(value: unknown): unknown {\n if (value instanceof Map) {\n return {\n __map: [...value].map(([k, v]) => [toExtendedJson(k), toExtendedJson(v)]),\n };\n }\n\n if (value instanceof Set) {\n return {\n __set: [...value].map(toExtendedJson),\n };\n }\n\n if (value instanceof Date) {\n return {\n __date: value.toISOString(),\n };\n }\n\n if (typeof value === 'bigint') {\n return {\n __bigint: value.toString(),\n };\n }\n\n if (Array.isArray(value)) {\n return value.map(toExtendedJson);\n }\n\n if (typeof value === 'object' && value !== null) {\n return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, toExtendedJson(v)]));\n }\n\n return value;\n}\n\nexport function toExtendedJsonString(\n value: any,\n replacer?: (this: any, key: string, value: any) => any,\n space?: string | number,\n): string;\nexport function toExtendedJsonString(\n value: any,\n replacer?: (number | string)[] | null,\n space?: string | number,\n): string;\nexport function toExtendedJsonString(value: unknown, ...args: any[]): string {\n return JSON.stringify(toExtendedJson(value), ...args);\n}\n\nexport function fromExtendedJson(value: unknown): unknown {\n if (typeof value !== 'object' || value === null) {\n return value;\n }\n\n if ('__map' in value) {\n return new Map(\n (value.__map as [unknown, unknown][]).map(([k, v]) => [\n fromExtendedJson(k),\n fromExtendedJson(v),\n ]),\n );\n }\n\n if ('__set' in value) {\n return new Set((value.__set as unknown[]).map(fromExtendedJson));\n }\n\n if ('__date' in value) {\n return new Date(value.__date as string);\n }\n\n if ('__bigint' in value) {\n return BigInt(value.__bigint as string);\n }\n\n if (Array.isArray(value)) {\n return value.map(fromExtendedJson);\n }\n\n return Object.fromEntries(Object.entries(value).map(([k, v]) => [k, fromExtendedJson(v)]));\n}\n\nexport function fromExtendedJsonString(\n value: string,\n reviver?: (this: any, key: string, value: any) => any,\n): unknown {\n return fromExtendedJson(JSON.parse(value, reviver));\n}\n"],"mappings":";;AAAA,SAAgB,eAAe,OAAyB;AACtD,KAAI,iBAAiB,IACnB,QAAO,EACL,OAAO,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,eAAe,EAAE,EAAE,eAAe,EAAE,CAAC,CAAC,EAC1E;AAGH,KAAI,iBAAiB,IACnB,QAAO,EACL,OAAO,CAAC,GAAG,MAAM,CAAC,IAAI,eAAe,EACtC;AAGH,KAAI,iBAAiB,KACnB,QAAO,EACL,QAAQ,MAAM,aAAa,EAC5B;AAGH,KAAI,OAAO,UAAU,SACnB,QAAO,EACL,UAAU,MAAM,UAAU,EAC3B;AAGH,KAAI,MAAM,QAAQ,MAAM,CACtB,QAAO,MAAM,IAAI,eAAe;AAGlC,KAAI,OAAO,UAAU,YAAY,UAAU,KACzC,QAAO,OAAO,YAAY,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,GAAG,eAAe,EAAE,CAAC,CAAC,CAAC;AAG1F,QAAO;;AAaT,SAAgB,qBAAqB,OAAgB,GAAG,MAAqB;AAC3E,QAAO,KAAK,UAAU,eAAe,MAAM,EAAE,GAAG,KAAK;;AAGvD,SAAgB,iBAAiB,OAAyB;AACxD,KAAI,OAAO,UAAU,YAAY,UAAU,KACzC,QAAO;AAGT,KAAI,WAAW,MACb,QAAO,IAAI,IACR,MAAM,MAA+B,KAAK,CAAC,GAAG,OAAO,CACpD,iBAAiB,EAAE,EACnB,iBAAiB,EAAE,CACpB,CAAC,CACH;AAGH,KAAI,WAAW,MACb,QAAO,IAAI,IAAK,MAAM,MAAoB,IAAI,iBAAiB,CAAC;AAGlE,KAAI,YAAY,MACd,QAAO,IAAI,KAAK,MAAM,OAAiB;AAGzC,KAAI,cAAc,MAChB,QAAO,OAAO,MAAM,SAAmB;AAGzC,KAAI,MAAM,QAAQ,MAAM,CACtB,QAAO,MAAM,IAAI,iBAAiB;AAGpC,QAAO,OAAO,YAAY,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,GAAG,iBAAiB,EAAE,CAAC,CAAC,CAAC;;AAG5F,SAAgB,uBACd,OACA,SACS;AACT,QAAO,iBAAiB,KAAK,MAAM,OAAO,QAAQ,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"patchMethods-BfyJt9me.cjs","names":["remove","set","deepEqual","isObject","applyPatches","_applyPatches","toExtendedJson","fromExtendedJson","patchMethods: {\n subscribePatches: typeof subscribePatches;\n applyPatches: typeof applyPatches;\n sync: typeof sync;\n acceptSync: typeof acceptSync;\n}"],"sources":["../src/lib/applyPatches.ts","../src/lib/diff.ts","../src/lib/trie.ts","../src/patches/patchMethods.ts"],"sourcesContent":["import type { Patch } from './diff';\nimport { remove, set } from './propAccess';\n\nfunction applySinglePatch<T>(target: T, patch: Patch): T {\n if (patch.op === 'remove') {\n return remove(target, patch.path as any);\n }\n\n return set(target, patch.path as any, patch.value);\n}\n\nexport function applyPatches<T>(target: T, ...patches: Patch[]): T {\n for (const patch of patches) {\n target = applySinglePatch(target, patch);\n }\n\n return target;\n}\n","import { isObject } from '@lib/helpers';\nimport { deepEqual } from './equals';\nimport type { KeyType } from './path';\n\nexport type Patch =\n | { op: 'add'; path: KeyType[]; value: any }\n | { op: 'remove'; path: KeyType[] }\n | { op: 'replace'; path: KeyType[]; value: any };\n\nexport interface DiffOptions {\n stopAt?: number | ((path: KeyType[]) => boolean);\n}\n\nexport function diff(\n a: any,\n b: any,\n options: DiffOptions = {},\n): [patches: Patch[], reversePatches: Patch[]] {\n const result = [..._diff(a, b, options)];\n const patches = result.map(([patch]) => patch);\n const reversePatches = result.map(([, reversePatch]) => reversePatch);\n\n return [patches, reversePatches];\n}\n\nfunction* _diff(\n a: any,\n b: any,\n options: DiffOptions,\n prefix: KeyType[] = [],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n if (a === b) {\n return;\n }\n\n if (\n (typeof options.stopAt === 'number' && prefix.length >= options.stopAt) ||\n (typeof options.stopAt === 'function' && options.stopAt(prefix))\n ) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (a instanceof Date && b instanceof Date) {\n if (a.getTime() === b.getTime()) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (a instanceof Map && b instanceof Map) {\n return yield* mapDiff(a, b, options, prefix);\n }\n\n if (a instanceof Set && b instanceof Set) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (Array.isArray(a) && Array.isArray(b)) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (isObject(a) && isObject(b) && !Array.isArray(a) && !Array.isArray(b)) {\n return yield* objectDiff(a, b, options, prefix);\n }\n\n yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n}\n\nfunction* mapDiff(\n a: Map<any, any>,\n b: Map<any, any>,\n options: { stopAt?: number | ((path: KeyType[]) => boolean) },\n prefix: KeyType[],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n for (const [key, value] of a) {\n if (!b.has(key)) {\n yield [\n { op: 'remove', path: [...prefix, key] },\n { op: 'add', path: [...prefix, key], value },\n ];\n } else {\n yield* _diff(value, b.get(key), options, [...prefix, key]);\n }\n }\n\n for (const [key, value] of b) {\n if (!a.has(key)) {\n yield [\n { op: 'add', path: [...prefix, key], value },\n { op: 'remove', path: [...prefix, key] },\n ];\n }\n }\n}\n\nfunction* objectDiff(\n a: any,\n b: any,\n options: { stopAt?: number | ((path: KeyType[]) => boolean) },\n prefix: KeyType[],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n for (const [key, value] of Object.entries(a)) {\n if (!(key in b)) {\n yield [\n { op: 'remove', path: [...prefix, key] },\n { op: 'add', path: [...prefix, key], value },\n ];\n } else {\n yield* _diff(value, b[key], options, [...prefix, key]);\n }\n }\n\n for (const [key, value] of Object.entries(b)) {\n if (!(key in a)) {\n yield [\n { op: 'add', path: [...prefix, key], value },\n { op: 'remove', path: [...prefix, key] },\n ];\n }\n }\n}\n","import type { KeyType } from '@lib/path';\n\nclass TrieNode {\n children: Map<KeyType, TrieNode> = new Map();\n isLeaf = false;\n}\n\nexport class Trie {\n root: TrieNode = new TrieNode();\n\n add(path: KeyType[]): void {\n let node = this.root;\n for (const key of path) {\n let next = node.children.get(key);\n if (!next) node.children.set(key, (next = new TrieNode()));\n node = next;\n }\n node.isLeaf = true;\n }\n\n hasSubPath(path: KeyType[]): boolean {\n let node = this.root;\n for (const key of path) {\n const next = node.children.get(key);\n if (!next) return false;\n node = next;\n }\n return node.isLeaf;\n }\n}\n","import type { Cancel, DisposableCancel, SubscribeOptions } from '@core/commonTypes';\nimport type { Store } from '@core/store';\nimport { applyPatches as _applyPatches } from '@lib/applyPatches';\nimport { diff, type DiffOptions, type Patch } from '@lib/diff';\nimport { fromExtendedJson, toExtendedJson } from '@lib/extendedJson';\nimport { Trie } from '@lib/trie';\n\nexport interface SyncMessage {\n fromVersion?: string;\n toVersion: string;\n patches: Patch[];\n}\n\nexport interface HistoryEntry extends SyncMessage {\n reversePatches: Patch[];\n}\n\ndeclare module '..' {\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n interface Store<T> {\n __patches?: {\n value: T;\n version: string;\n history: HistoryEntry[];\n };\n }\n}\n\nexport interface SubscribePatchOptions extends SubscribeOptions, DiffOptions {\n /** @default false */\n runNow?: boolean;\n /** try to start from a specific version and only receive patches after that.\n * If the id is not found, it will start from the beginning */\n startAt?: string;\n}\n\nexport type InteropPatch = Patch | { op: 'add' | 'replace' | 'remove'; value?: any };\n\nconst genId = () => Math.random().toString(36).slice(2);\n\nexport function subscribePatches<T>(\n this: Store<T>,\n listener: (\n this: { cancel: Cancel },\n patches: Patch[],\n reversePatches: Patch[],\n version: string,\n previousVersion: string | undefined,\n ) => void,\n options: SubscribePatchOptions = {},\n): DisposableCancel {\n const patches = (this.__patches ??= {\n value: this.get(),\n version: genId(),\n history: [],\n });\n\n options = { ...options };\n options.runNow ??= false;\n let cursor = options.startAt ?? (options.runNow ? undefined : this.__patches.version);\n\n return this.subscribe(function (value) {\n if (patches.value !== value) {\n const result = diff(patches.value, value, options);\n patches.value = value;\n\n if (result[0].length > 0) {\n const newVersion = genId();\n\n patches.history = patches.history\n .concat({\n fromVersion: patches.version,\n toVersion: newVersion,\n patches: result[0],\n reversePatches: result[1],\n })\n .slice(-1000);\n\n patches.version = newVersion;\n }\n }\n\n if (cursor === patches.version) return;\n const index = patches.history.findIndex((h) => h.fromVersion === cursor);\n let forward, backward, previousVersion;\n\n if (index === -1) {\n [forward, backward] = diff(undefined, value, options);\n previousVersion = undefined;\n } else {\n forward = patches.history.slice(index).flatMap((h) => h.patches);\n backward = patches.history.slice(index).flatMap((h) => h.reversePatches);\n previousVersion = cursor;\n }\n\n cursor = patches.version;\n listener.apply(this, [forward, backward, cursor, previousVersion]);\n }, options);\n}\n\nexport function applyPatches<T>(this: Store<T>, patches: InteropPatch[]): void;\nexport function applyPatches<T>(this: Store<T>, ...patches: InteropPatch[]): void;\nexport function applyPatches<T>(\n this: Store<T>,\n ...patches: (InteropPatch | InteropPatch[])[]\n): void {\n this.set((value) => _applyPatches(value, ...(patches.flat() as Patch[])));\n}\n\nexport function sync<T>(\n this: Store<T>,\n listener: (syncMessage: SyncMessage) => void,\n options?: Omit<SubscribePatchOptions, 'runNow'>,\n): DisposableCancel {\n const debounce =\n typeof options?.debounce === 'object' && 'wait' in options.debounce\n ? { ...options.debounce }\n : options?.debounce !== undefined\n ? { wait: options.debounce }\n : undefined;\n\n if (debounce) {\n debounce.waitOnRunNow ??= false;\n }\n\n return subscribePatches.apply<\n Store<T>,\n Parameters<typeof subscribePatches<T>>,\n ReturnType<typeof subscribePatches<T>>\n >(this, [\n (patches, _, version, previousVersion) => {\n const trie = new Trie();\n\n patches = [...patches]\n .reverse()\n .filter((patch) => {\n if (trie.hasSubPath(patch.path)) {\n return false;\n }\n\n trie.add(patch.path);\n return true;\n })\n .reverse();\n\n listener({\n fromVersion: previousVersion,\n toVersion: version,\n patches: toExtendedJson(patches) as Patch[],\n });\n },\n { ...options, debounce, runNow: true },\n ]);\n}\n\nexport function acceptSync<T>(this: Store<T>, message: SyncMessage): void {\n if (message.fromVersion && message.fromVersion !== this.version) {\n throw new Error(\n `version mismatch! version=${this.version}, fromVersion=${message.fromVersion}`,\n );\n }\n\n const patches = fromExtendedJson(message.patches) as Patch[];\n\n this.version = message.toVersion;\n applyPatches.apply<Store<T>, Patch[], void>(this, patches);\n}\n\nexport const patchMethods: {\n subscribePatches: typeof subscribePatches;\n applyPatches: typeof applyPatches;\n sync: typeof sync;\n acceptSync: typeof acceptSync;\n} = {\n subscribePatches,\n applyPatches,\n sync,\n acceptSync,\n};\n"],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"patchMethods-D-Vmx93m.js","names":["applyPatches","_applyPatches","patchMethods: {\n subscribePatches: typeof subscribePatches;\n applyPatches: typeof applyPatches;\n sync: typeof sync;\n acceptSync: typeof acceptSync;\n}"],"sources":["../src/lib/applyPatches.ts","../src/lib/diff.ts","../src/lib/trie.ts","../src/patches/patchMethods.ts"],"sourcesContent":["import type { Patch } from './diff';\nimport { remove, set } from './propAccess';\n\nfunction applySinglePatch<T>(target: T, patch: Patch): T {\n if (patch.op === 'remove') {\n return remove(target, patch.path as any);\n }\n\n return set(target, patch.path as any, patch.value);\n}\n\nexport function applyPatches<T>(target: T, ...patches: Patch[]): T {\n for (const patch of patches) {\n target = applySinglePatch(target, patch);\n }\n\n return target;\n}\n","import { isObject } from '@lib/helpers';\nimport { deepEqual } from './equals';\nimport type { KeyType } from './path';\n\nexport type Patch =\n | { op: 'add'; path: KeyType[]; value: any }\n | { op: 'remove'; path: KeyType[] }\n | { op: 'replace'; path: KeyType[]; value: any };\n\nexport interface DiffOptions {\n stopAt?: number | ((path: KeyType[]) => boolean);\n}\n\nexport function diff(\n a: any,\n b: any,\n options: DiffOptions = {},\n): [patches: Patch[], reversePatches: Patch[]] {\n const result = [..._diff(a, b, options)];\n const patches = result.map(([patch]) => patch);\n const reversePatches = result.map(([, reversePatch]) => reversePatch);\n\n return [patches, reversePatches];\n}\n\nfunction* _diff(\n a: any,\n b: any,\n options: DiffOptions,\n prefix: KeyType[] = [],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n if (a === b) {\n return;\n }\n\n if (\n (typeof options.stopAt === 'number' && prefix.length >= options.stopAt) ||\n (typeof options.stopAt === 'function' && options.stopAt(prefix))\n ) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (a instanceof Date && b instanceof Date) {\n if (a.getTime() === b.getTime()) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (a instanceof Map && b instanceof Map) {\n return yield* mapDiff(a, b, options, prefix);\n }\n\n if (a instanceof Set && b instanceof Set) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (Array.isArray(a) && Array.isArray(b)) {\n if (deepEqual(a, b)) {\n return;\n }\n\n return yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n }\n\n if (isObject(a) && isObject(b) && !Array.isArray(a) && !Array.isArray(b)) {\n return yield* objectDiff(a, b, options, prefix);\n }\n\n yield [\n { op: 'replace', path: prefix, value: b },\n { op: 'replace', path: prefix, value: a },\n ];\n}\n\nfunction* mapDiff(\n a: Map<any, any>,\n b: Map<any, any>,\n options: { stopAt?: number | ((path: KeyType[]) => boolean) },\n prefix: KeyType[],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n for (const [key, value] of a) {\n if (!b.has(key)) {\n yield [\n { op: 'remove', path: [...prefix, key] },\n { op: 'add', path: [...prefix, key], value },\n ];\n } else {\n yield* _diff(value, b.get(key), options, [...prefix, key]);\n }\n }\n\n for (const [key, value] of b) {\n if (!a.has(key)) {\n yield [\n { op: 'add', path: [...prefix, key], value },\n { op: 'remove', path: [...prefix, key] },\n ];\n }\n }\n}\n\nfunction* objectDiff(\n a: any,\n b: any,\n options: { stopAt?: number | ((path: KeyType[]) => boolean) },\n prefix: KeyType[],\n): Iterable<[patch: Patch, reversePatch: Patch]> {\n for (const [key, value] of Object.entries(a)) {\n if (!(key in b)) {\n yield [\n { op: 'remove', path: [...prefix, key] },\n { op: 'add', path: [...prefix, key], value },\n ];\n } else {\n yield* _diff(value, b[key], options, [...prefix, key]);\n }\n }\n\n for (const [key, value] of Object.entries(b)) {\n if (!(key in a)) {\n yield [\n { op: 'add', path: [...prefix, key], value },\n { op: 'remove', path: [...prefix, key] },\n ];\n }\n }\n}\n","import type { KeyType } from '@lib/path';\n\nclass TrieNode {\n children: Map<KeyType, TrieNode> = new Map();\n isLeaf = false;\n}\n\nexport class Trie {\n root: TrieNode = new TrieNode();\n\n add(path: KeyType[]): void {\n let node = this.root;\n for (const key of path) {\n let next = node.children.get(key);\n if (!next) node.children.set(key, (next = new TrieNode()));\n node = next;\n }\n node.isLeaf = true;\n }\n\n hasSubPath(path: KeyType[]): boolean {\n let node = this.root;\n for (const key of path) {\n const next = node.children.get(key);\n if (!next) return false;\n node = next;\n }\n return node.isLeaf;\n }\n}\n","import type { Cancel, DisposableCancel, SubscribeOptions } from '@core/commonTypes';\nimport type { Store } from '@core/store';\nimport { applyPatches as _applyPatches } from '@lib/applyPatches';\nimport { diff, type DiffOptions, type Patch } from '@lib/diff';\nimport { fromExtendedJson, toExtendedJson } from '@lib/extendedJson';\nimport { Trie } from '@lib/trie';\n\nexport interface SyncMessage {\n fromVersion?: string;\n toVersion: string;\n patches: Patch[];\n}\n\nexport interface HistoryEntry extends SyncMessage {\n reversePatches: Patch[];\n}\n\ndeclare module '..' {\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n interface Store<T> {\n __patches?: {\n value: T;\n version: string;\n history: HistoryEntry[];\n };\n }\n}\n\nexport interface SubscribePatchOptions extends SubscribeOptions, DiffOptions {\n /** @default false */\n runNow?: boolean;\n /** try to start from a specific version and only receive patches after that.\n * If the id is not found, it will start from the beginning */\n startAt?: string;\n}\n\nexport type InteropPatch = Patch | { op: 'add' | 'replace' | 'remove'; value?: any };\n\nconst genId = () => Math.random().toString(36).slice(2);\n\nexport function subscribePatches<T>(\n this: Store<T>,\n listener: (\n this: { cancel: Cancel },\n patches: Patch[],\n reversePatches: Patch[],\n version: string,\n previousVersion: string | undefined,\n ) => void,\n options: SubscribePatchOptions = {},\n): DisposableCancel {\n const patches = (this.__patches ??= {\n value: this.get(),\n version: genId(),\n history: [],\n });\n\n options = { ...options };\n options.runNow ??= false;\n let cursor = options.startAt ?? (options.runNow ? undefined : this.__patches.version);\n\n return this.subscribe(function (value) {\n if (patches.value !== value) {\n const result = diff(patches.value, value, options);\n patches.value = value;\n\n if (result[0].length > 0) {\n const newVersion = genId();\n\n patches.history = patches.history\n .concat({\n fromVersion: patches.version,\n toVersion: newVersion,\n patches: result[0],\n reversePatches: result[1],\n })\n .slice(-1000);\n\n patches.version = newVersion;\n }\n }\n\n if (cursor === patches.version) return;\n const index = patches.history.findIndex((h) => h.fromVersion === cursor);\n let forward, backward, previousVersion;\n\n if (index === -1) {\n [forward, backward] = diff(undefined, value, options);\n previousVersion = undefined;\n } else {\n forward = patches.history.slice(index).flatMap((h) => h.patches);\n backward = patches.history.slice(index).flatMap((h) => h.reversePatches);\n previousVersion = cursor;\n }\n\n cursor = patches.version;\n listener.apply(this, [forward, backward, cursor, previousVersion]);\n }, options);\n}\n\nexport function applyPatches<T>(this: Store<T>, patches: InteropPatch[]): void;\nexport function applyPatches<T>(this: Store<T>, ...patches: InteropPatch[]): void;\nexport function applyPatches<T>(\n this: Store<T>,\n ...patches: (InteropPatch | InteropPatch[])[]\n): void {\n this.set((value) => _applyPatches(value, ...(patches.flat() as Patch[])));\n}\n\nexport function sync<T>(\n this: Store<T>,\n listener: (syncMessage: SyncMessage) => void,\n options?: Omit<SubscribePatchOptions, 'runNow'>,\n): DisposableCancel {\n const debounce =\n typeof options?.debounce === 'object' && 'wait' in options.debounce\n ? { ...options.debounce }\n : options?.debounce !== undefined\n ? { wait: options.debounce }\n : undefined;\n\n if (debounce) {\n debounce.waitOnRunNow ??= false;\n }\n\n return subscribePatches.apply<\n Store<T>,\n Parameters<typeof subscribePatches<T>>,\n ReturnType<typeof subscribePatches<T>>\n >(this, [\n (patches, _, version, previousVersion) => {\n const trie = new Trie();\n\n patches = [...patches]\n .reverse()\n .filter((patch) => {\n if (trie.hasSubPath(patch.path)) {\n return false;\n }\n\n trie.add(patch.path);\n return true;\n })\n .reverse();\n\n listener({\n fromVersion: previousVersion,\n toVersion: version,\n patches: toExtendedJson(patches) as Patch[],\n });\n },\n { ...options, debounce, runNow: true },\n ]);\n}\n\nexport function acceptSync<T>(this: Store<T>, message: SyncMessage): void {\n if (message.fromVersion && message.fromVersion !== this.version) {\n throw new Error(\n `version mismatch! version=${this.version}, fromVersion=${message.fromVersion}`,\n );\n }\n\n const patches = fromExtendedJson(message.patches) as Patch[];\n\n this.version = message.toVersion;\n applyPatches.apply<Store<T>, Patch[], void>(this, patches);\n}\n\nexport const patchMethods: {\n subscribePatches: typeof subscribePatches;\n applyPatches: typeof applyPatches;\n sync: typeof sync;\n acceptSync: typeof acceptSync;\n} = {\n subscribePatches,\n applyPatches,\n sync,\n acceptSync,\n};\n"],"mappings":"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"}
|