@mmstack/form-core
Version:
[](https://www.npmjs.com/package/@mmstack/form-core) [](https://github.com/mihajm/mmstack/blob/master/packages/form/core/LICEN
1 lines • 92.7 kB
Source Map (JSON)
{"version":3,"file":"mmstack-form-core.mjs","sources":["../tmp-esm2022/lib/form-control.js","../tmp-esm2022/lib/form-array.js","../tmp-esm2022/lib/form-group.js","../tmp-esm2022/mmstack-form-core.js"],"sourcesContent":["import { computed, isSignal, signal, untracked, } from '@angular/core';\nimport { v7 } from 'uuid';\n/**\n * Creates a `FormControlSignal`, a reactive form control that holds a value and tracks its\n * validity, dirty state, touched state, and other metadata.\n *\n * @typeParam T - The type of the form control's value.\n * @typeParam TParent - The type of the parent form control's value (if this control is part of a group or array).\n * @typeParam TControlType - The type of the control. Defaults to `'control'`.\n * @typeParam TPartialValue - The type of value when patching\n * @param initial - The initial value of the control, or a `DerivedSignal` if this control is part of a `formGroup` or `formArray`.\n * @param options - Optional configuration options for the control.\n * @returns A `FormControlSignal` instance.\n *\n * @example\n * // Create a simple form control:\n * const name = formControl('Initial Name');\n *\n * // Create a form control with validation:\n * const age = formControl(0, {\n * validator: () => (value) => value >= 18 ? '' : 'Must be at least 18',\n * });\n *\n * // Create a derived form control (equivalent to the above, but more explicit):\n * const user = signal({ name: 'John Doe', age: 30 });\n * const name = formControl(derived(user, {\n * from: (u) => u.name,\n * onChange: (newName) => user.update(u => ({...u, name: newName}))\n * }));\n *\n * // Create a form group with nested controls:\n * const user = signal({ name: 'John Doe', age: 30 });\n * const form = formGroup(user, {\n * name: formControl(derived(user, 'name')),\n * age: formControl(derived(user, 'age')),\n * })\n */\nexport function formControl(initial, opt) {\n const value = isSignal(initial) ? initial : signal(initial, opt);\n const initialValue = signal(untracked(value));\n const eq = opt?.equal ?? Object.is;\n const dirtyEq = opt?.dirtyEquality ?? eq;\n const disabled = computed(() => opt?.disable?.() ?? false);\n const readonly = computed(() => opt?.readonly?.() ?? false);\n const dirty = computed(() => !dirtyEq(value(), initialValue()));\n const touched = signal(false);\n const validator = computed(() => opt?.validator?.() ?? (() => ''));\n const error = computed(() => {\n if (opt?.overrideValidation)\n return opt.overrideValidation();\n if (disabled() || readonly())\n return '';\n return validator()(value());\n });\n const markAsTouched = () => {\n touched.set(true);\n opt?.onTouched?.();\n };\n const markAllAsTouched = markAsTouched;\n const markAsPristine = () => touched.set(false);\n const markAllAsPristine = markAsPristine;\n const label = computed(() => opt?.label?.() ?? '');\n const partialValue = computed(() => (dirty() ? value() : undefined));\n const internalReconcile = (newValue, force = false) => {\n const isDirty = untracked(dirty);\n if (!isDirty || force) {\n // very dangerous use of untracked here, don't do this everywhere :)\n // thanks to u/synalx for the idea to use untracked here\n untracked(() => {\n initialValue.set(newValue);\n value.set(newValue);\n });\n }\n };\n const pending = computed(() => opt?.pending?.() ?? false);\n return {\n id: opt?.id?.() ?? v7(),\n value,\n dirty,\n touched,\n error,\n label,\n required: computed(() => opt?.required?.() ?? false),\n disabled,\n readonly,\n pending,\n valid: computed(() => !pending() && !error()),\n hint: computed(() => opt?.hint?.() ?? ''),\n markAsTouched,\n markAllAsTouched,\n markAsPristine,\n markAllAsPristine,\n from: (isSignal(initial) ? initial.from : undefined),\n reconcile: (newValue) => internalReconcile(newValue),\n forceReconcile: (newValue) => internalReconcile(newValue, true),\n reset: () => {\n opt?.onReset?.();\n value.set(untracked(initialValue));\n },\n resetWithInitial: (initial) => {\n opt?.onReset?.();\n initialValue.set(initial);\n value.set(initial);\n },\n equal: eq,\n controlType: (opt?.controlType ?? 'control'),\n partialValue: partialValue,\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-control.js","sourceRoot":"","sources":["../../../../../../packages/form/core/src/lib/form-control.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  computed,\r\n  isSignal,\r\n  signal,\r\n  untracked,\r\n  type CreateSignalOptions,\r\n  type Signal,\r\n  type ValueEqualityFn,\r\n  type WritableSignal,\r\n} from '@angular/core';\r\nimport { type DerivedSignal } from '@mmstack/primitives';\r\nimport { v7 } from 'uuid';\r\n\r\n/**\r\n * Represents the type of a form control.\r\n * - `control`: A single form control (e.g., an input field).\r\n * - `array`: An array of form controls (like Angular's `FormArray`).\r\n * - `group`: A group of form controls (like Angular's `FormGroup`).\r\n */\r\nexport type ControlType = 'control' | 'array' | 'group';\r\n\r\n/**\r\n * Represents a reactive form control.  It holds the value, validation status, and other\r\n * metadata for a form field.  This is the core building block for creating reactive forms\r\n * with signals.\r\n *\r\n * @typeParam T - The type of the form control's value.\r\n * @typeParam TParent - The type of the parent form control's value (if this control is part of a group or array). Defaults to `undefined`.\r\n * @typeParam TControlType - The type of the control ('control', 'array', or 'group'). Defaults to 'control'.\r\n * @typeParam TPartialValue -  The type of the partial value, used for patching.\r\n */\r\nexport type FormControlSignal<\r\n  T,\r\n  TParent = undefined,\r\n  TControlType extends ControlType = 'control',\r\n  TPartialValue = T | undefined,\r\n> = {\r\n  /** A unique identifier for the control. Used for tracking in `@for` loops. */\r\n  id: string;\r\n  /** The main value signal for the control. */\r\n  value: WritableSignal<T>;\r\n  /** A signal indicating whether the control's value has been changed. */\r\n  dirty: Signal<boolean>;\r\n  /** A signal indicating whether the control has been interacted with (e.g., blurred). */\r\n  touched: Signal<boolean>;\r\n  /** A signal containing the current validation error message (empty string if valid). */\r\n  error: Signal<string>;\r\n  /** A signal indicating whether the control is pending */\r\n  pending: Signal<boolean>;\r\n  /** A signal indicating whether the control is disabled. */\r\n  /** A signal indicating whether the control is in a valid state (without errors & not pending) */\r\n  valid: Signal<boolean>;\r\n  disabled: Signal<boolean>;\r\n  /** A signal indicating whether the control is read-only. */\r\n  readonly: Signal<boolean>;\r\n  /** A signal indicating whether the control is required. */\r\n  required: Signal<boolean>;\r\n  /** A signal containing the label for the control. */\r\n  label: Signal<string>;\r\n  /** A signal containing the hint text for the control. */\r\n  hint: Signal<string>;\r\n  /** Marks the control as touched. */\r\n  markAsTouched: () => void;\r\n  /** Marks the control and all its child controls (if any) as touched. */\r\n  markAllAsTouched: () => void;\r\n  /** Marks the control as pristine (not touched). */\r\n  markAsPristine: () => void;\r\n  /** Marks the control and all its child controls (if any) as pristine. */\r\n  markAllAsPristine: () => void;\r\n  /**\r\n   * Resets the control to a new value and sets a new initial value. This is intended for\r\n   * scenarios where the underlying data is updated externally (e.g., data coming from\r\n   * the server).  If the control is not dirty, the value is updated. If the control *is*\r\n   * dirty, the value is *not* updated (preserving user changes).\r\n   */\r\n  reconcile: (newValue: T) => void;\r\n  /**\r\n   * Similar to `reconcile`, but forces the update even if the control is dirty.\r\n   */\r\n  forceReconcile: (newValue: T) => void;\r\n  /** Resets the control's value to its initial value. */\r\n  reset: () => void;\r\n  /** Resets the control's value and initial value. */\r\n  resetWithInitial: (initial: T) => void;\r\n  /**\r\n   * The derivation function used to create this control if it's part of a `formGroup` or `formArray`.\r\n   * @internal\r\n   */\r\n  from?: DerivedSignal<TParent, T>['from'];\r\n  /** The equality function used to compare values. Defaults to `Object.is`. */\r\n  equal: (a: T, b: T) => boolean;\r\n  /** The type of the control ('control', 'array', or 'group'). */\r\n  controlType: TControlType;\r\n  /**\r\n   * A signal representing the partial value of the control, suitable for patching data on a server.\r\n   * It contains the changed value if `dirty` is `true`.\r\n   */\r\n  partialValue: Signal<TPartialValue>;\r\n};\r\n\r\nexport type CreateFormControlOptions<\r\n  T,\r\n  TControlType extends ControlType = ControlType,\r\n> = CreateSignalOptions<T> & {\r\n  validator?: () => (value: T) => string;\r\n  onTouched?: () => void;\r\n  disable?: () => boolean;\r\n  readonly?: () => boolean;\r\n  required?: () => boolean;\r\n  label?: () => string;\r\n  id?: () => string;\r\n  hint?: () => string;\r\n  dirtyEquality?: ValueEqualityFn<T>;\r\n  onReset?: () => void;\r\n  controlType?: TControlType;\r\n  overrideValidation?: () => string;\r\n  pending?: () => boolean;\r\n};\r\n\r\n/**\r\n * Creates a `FormControlSignal`, a reactive form control that holds a value and tracks its\r\n * validity, dirty state, touched state, and other metadata.\r\n *\r\n * @typeParam T - The type of the form control's value.\r\n * @typeParam TParent - The type of the parent form control's value (if this control is part of a group or array).\r\n * @typeParam TControlType - The type of the control. Defaults to `'control'`.\r\n * @typeParam TPartialValue - The type of value when patching\r\n * @param initial - The initial value of the control, or a `DerivedSignal` if this control is part of a `formGroup` or `formArray`.\r\n * @param options - Optional configuration options for the control.\r\n * @returns A `FormControlSignal` instance.\r\n *\r\n * @example\r\n * // Create a simple form control:\r\n * const name = formControl('Initial Name');\r\n *\r\n * // Create a form control with validation:\r\n * const age = formControl(0, {\r\n *   validator: () => (value) => value >= 18 ? '' : 'Must be at least 18',\r\n * });\r\n *\r\n * // Create a derived form control (equivalent to the above, but more explicit):\r\n *  const user = signal({ name: 'John Doe', age: 30 });\r\n *  const name = formControl(derived(user, {\r\n *    from: (u) => u.name,\r\n *    onChange: (newName) => user.update(u => ({...u, name: newName}))\r\n *  }));\r\n *\r\n * // Create a form group with nested controls:\r\n * const user = signal({ name: 'John Doe', age: 30 });\r\n * const form = formGroup(user, {\r\n *  name: formControl(derived(user, 'name')),\r\n *  age: formControl(derived(user, 'age')),\r\n * })\r\n */\r\nexport function formControl<\r\n  T,\r\n  TParent = undefined,\r\n  TControlType extends ControlType = 'control',\r\n  TPartialValue = T | undefined,\r\n>(\r\n  initial: DerivedSignal<TParent, T> | T,\r\n  opt?: CreateFormControlOptions<T, TControlType>,\r\n): FormControlSignal<T, TParent, TControlType, TPartialValue> {\r\n  const value = isSignal(initial) ? initial : signal(initial, opt);\r\n  const initialValue = signal(untracked(value));\r\n  const eq = opt?.equal ?? Object.is;\r\n\r\n  const dirtyEq = opt?.dirtyEquality ?? eq;\r\n\r\n  const disabled = computed(() => opt?.disable?.() ?? false);\r\n  const readonly = computed(() => opt?.readonly?.() ?? false);\r\n\r\n  const dirty = computed(() => !dirtyEq(value(), initialValue()));\r\n\r\n  const touched = signal(false);\r\n\r\n  const validator = computed(() => opt?.validator?.() ?? (() => ''));\r\n\r\n  const error = computed(() => {\r\n    if (opt?.overrideValidation) return opt.overrideValidation();\r\n    if (disabled() || readonly()) return '';\r\n    return validator()(value());\r\n  });\r\n\r\n  const markAsTouched = () => {\r\n    touched.set(true);\r\n    opt?.onTouched?.();\r\n  };\r\n  const markAllAsTouched = markAsTouched;\r\n\r\n  const markAsPristine = () => touched.set(false);\r\n  const markAllAsPristine = markAsPristine;\r\n\r\n  const label = computed(() => opt?.label?.() ?? '');\r\n\r\n  const partialValue = computed(() => (dirty() ? value() : undefined));\r\n\r\n  const internalReconcile = (newValue: T, force = false) => {\r\n    const isDirty = untracked(dirty);\r\n\r\n    if (!isDirty || force) {\r\n      // very dangerous use of untracked here, don't do this everywhere :)\r\n      // thanks to  u/synalx for the idea to use untracked here\r\n\r\n      untracked(() => {\r\n        initialValue.set(newValue);\r\n        value.set(newValue);\r\n      });\r\n    }\r\n  };\r\n\r\n  const pending = computed(() => opt?.pending?.() ?? false);\r\n\r\n  return {\r\n    id: opt?.id?.() ?? v7(),\r\n    value,\r\n    dirty,\r\n    touched,\r\n    error,\r\n    label,\r\n    required: computed(() => opt?.required?.() ?? false),\r\n    disabled,\r\n    readonly,\r\n    pending,\r\n    valid: computed(() => !pending() && !error()),\r\n    hint: computed(() => opt?.hint?.() ?? ''),\r\n    markAsTouched,\r\n    markAllAsTouched,\r\n    markAsPristine,\r\n    markAllAsPristine,\r\n    from: (isSignal(initial) ? initial.from : undefined) as FormControlSignal<\r\n      T,\r\n      TParent,\r\n      TControlType\r\n    >['from'],\r\n    reconcile: (newValue: T) => internalReconcile(newValue),\r\n    forceReconcile: (newValue: T) => internalReconcile(newValue, true),\r\n    reset: () => {\r\n      opt?.onReset?.();\r\n      value.set(untracked(initialValue));\r\n    },\r\n    resetWithInitial: (initial: T) => {\r\n      opt?.onReset?.();\r\n      initialValue.set(initial);\r\n      value.set(initial);\r\n    },\r\n    equal: eq,\r\n    controlType: (opt?.controlType ?? 'control') as TControlType,\r\n    partialValue: partialValue as Signal<TPartialValue>,\r\n  };\r\n}\r\n"]}","import { computed, linkedSignal, untracked, } from '@angular/core';\nimport { mergeArray } from '@mmstack/object';\nimport { derived } from '@mmstack/primitives';\nimport { formControl, } from './form-control';\nfunction createReconcileChildren(factory, opt) {\n return (length, source, prev) => {\n if (!prev) {\n const nextControls = [];\n for (let i = 0; i < length; i++) {\n nextControls.push(factory(derived(source, i, { equal: opt?.equal }), i));\n }\n return nextControls;\n }\n if (length === prev.length)\n return prev;\n const next = [...prev];\n if (length < prev.length) {\n next.splice(length);\n }\n else if (length > prev.length) {\n for (let i = prev.length; i < length; i++) {\n next.push(factory(derived(source, i, { equal: opt?.equal }), i));\n }\n }\n return next;\n };\n}\nexport function formArray(initial, factory, opt) {\n const eq = opt?.equal ?? Object.is;\n const arrayEqual = (a, b) => {\n if (a.length !== b.length)\n return false;\n if (!a.length)\n return true;\n return a.every((v, i) => eq(v, b[i]));\n };\n const min = computed(() => opt?.min?.() ?? 0);\n const max = computed(() => opt?.max?.() ?? Number.MAX_SAFE_INTEGER);\n const arrayOptions = {\n ...opt,\n equal: arrayEqual,\n dirtyEquality: (a, b) => a.length === b.length,\n controlType: 'array',\n };\n const ctrl = formControl(initial, arrayOptions);\n const length = computed(() => ctrl.value().length);\n const reconcileChildren = createReconcileChildren(factory, { equal: eq });\n // linkedSignal used to re-use previous value so that only length changes are affected and existing controls are kept, but updated\n const children = linkedSignal({\n source: () => length(),\n computation: (len, prev) => reconcileChildren(len, ctrl.value, prev?.value),\n });\n const ownError = computed(() => ctrl.error());\n const error = computed(() => {\n const own = ownError();\n if (own)\n return own;\n if (!children().length)\n return '';\n return children()\n .map((c, idx) => (c.error() ? `${idx}: ${c.error()}` : ''))\n .filter(Boolean)\n .join('\\n');\n });\n const dirty = computed(() => {\n if (ctrl.dirty())\n return true;\n if (!children().length)\n return false;\n return children().some((c) => c.dirty());\n });\n const markAllAsTouched = () => {\n ctrl.markAllAsTouched();\n for (const c of untracked(children)) {\n c.markAllAsTouched();\n }\n };\n const markAllAsPristine = () => {\n ctrl.markAllAsPristine();\n for (const c of untracked(children)) {\n c.markAllAsPristine();\n }\n };\n const toPartialValue = opt?.toPartialValue ?? ((v) => v);\n const partialValue = computed(() => {\n if (!dirty())\n return undefined;\n return children().map((c) => {\n const pv = c.partialValue();\n if (pv)\n return pv;\n if (c.controlType === 'control')\n return undefined;\n // return full value for child objects/arrays as this cannot be partially patched without idx\n return toPartialValue(c.value());\n });\n });\n const touched = computed(() => ctrl.touched() ||\n !!(children().length && children().some((c) => c.touched())));\n const reconcile = (newValue) => {\n const ctrls = untracked(children);\n for (let i = 0; i < newValue.length; i++) {\n ctrls.at(i)?.reconcile(newValue[i]); // reconcile existing controls that are relevant addition/removal will be handled after ctrl.reconcile through linkedSignal\n }\n ctrl.reconcile(mergeArray(newValue, untracked(ctrl.value)));\n };\n const forceReconcile = (newValue) => {\n const ctrls = untracked(children);\n for (let i = 0; i < newValue.length; i++) {\n ctrls.at(i)?.forceReconcile(newValue[i]);\n }\n ctrl.forceReconcile(newValue);\n };\n const childrenValid = computed(() => {\n if (!children().length)\n return true;\n return children().every((d) => d.valid());\n });\n const childrenPending = computed(() => !!children().length && children().some((d) => d.pending()));\n return {\n ...ctrl,\n ownError,\n error,\n valid: computed(() => ctrl.valid() && childrenValid()),\n pending: computed(() => ctrl.pending() || childrenPending()),\n touched,\n children,\n dirty,\n markAllAsTouched,\n markAllAsPristine,\n min,\n max,\n partialValue,\n canAdd: computed(() => !ctrl.disabled() && !ctrl.readonly() && length() < max()),\n canRemove: computed(() => !ctrl.disabled() && !ctrl.readonly() && length() > min()),\n reconcile,\n forceReconcile,\n reset: () => {\n for (const c of untracked(children)) {\n c.reset();\n }\n ctrl.reset();\n },\n resetWithInitial: (initial) => {\n const ctrls = untracked(children);\n for (let i = 0; i < initial.length; i++) {\n ctrls.at(i)?.resetWithInitial(initial[i]);\n }\n ctrl.resetWithInitial(initial);\n },\n push: (next) => ctrl.value.update((cur) => [...cur, next]),\n remove: (idx) => ctrl.value.update((cur) => cur.filter((_, i) => i !== idx)),\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-array.js","sourceRoot":"","sources":["../../../../../../packages/form/core/src/lib/form-array.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  computed,\r\n  linkedSignal,\r\n  untracked,\r\n  type Signal,\r\n  type WritableSignal,\r\n} from '@angular/core';\r\nimport { mergeArray } from '@mmstack/object';\r\nimport { derived, type DerivedSignal } from '@mmstack/primitives';\r\nimport {\r\n  formControl,\r\n  type CreateFormControlOptions,\r\n  type FormControlSignal,\r\n} from './form-control';\r\nimport { type SignalValue } from './signal-value.type';\r\n\r\nexport type FormArraySignal<\r\n  T,\r\n  TIndividualState extends FormControlSignal<\r\n    T,\r\n    any,\r\n    any,\r\n    any\r\n  > = FormControlSignal<T, any, any, any>,\r\n  TParent = undefined,\r\n> = FormControlSignal<\r\n  T[],\r\n  TParent,\r\n  'array',\r\n  | Exclude<SignalValue<TIndividualState['partialValue']>, null | undefined>[]\r\n  | undefined\r\n> & {\r\n  ownError: Signal<string>;\r\n  children: Signal<TIndividualState[]>;\r\n  push: (value: T) => void; // add new control with value\r\n  remove: (index: number) => void; // remove at index\r\n  min: Signal<number>; // for display purposes\r\n  max: Signal<number>; // for display purposes\r\n  canAdd: Signal<boolean>; // disable add button if false\r\n  canRemove: Signal<boolean>; // disable remove buttons if false\r\n};\r\n\r\nexport type CreateFormArraySignalOptions<\r\n  T,\r\n  TIndividualState extends FormControlSignal<T, any, any, any>,\r\n> = Omit<CreateFormControlOptions<T[]>, 'equal'> & {\r\n  min?: () => number;\r\n  max?: () => number;\r\n  equal?: (a: T, b: T) => boolean;\r\n  toPartialValue?: (\r\n    v: T,\r\n  ) => Exclude<SignalValue<TIndividualState['partialValue']>, null | undefined>;\r\n};\r\n\r\nfunction createReconcileChildren<\r\n  T,\r\n  TIndividualState extends FormControlSignal<T, any, any, any>,\r\n>(\r\n  factory: (val: DerivedSignal<T[], T>, idx: number) => TIndividualState,\r\n  opt: { equal: (a: T, b: T) => boolean },\r\n) {\r\n  return (\r\n    length: number,\r\n    source: WritableSignal<T[]>,\r\n    prev?: TIndividualState[],\r\n  ): TIndividualState[] => {\r\n    if (!prev) {\r\n      const nextControls = [];\r\n\r\n      for (let i = 0; i < length; i++) {\r\n        nextControls.push(\r\n          factory(derived(source, i, { equal: opt?.equal }), i),\r\n        );\r\n      }\r\n\r\n      return nextControls;\r\n    }\r\n\r\n    if (length === prev.length) return prev;\r\n\r\n    const next = [...prev];\r\n\r\n    if (length < prev.length) {\r\n      next.splice(length);\r\n    } else if (length > prev.length) {\r\n      for (let i = prev.length; i < length; i++) {\r\n        next.push(factory(derived(source, i, { equal: opt?.equal }), i));\r\n      }\r\n    }\r\n\r\n    return next;\r\n  };\r\n}\r\n\r\nexport function formArray<\r\n  T,\r\n  TIndividualState extends FormControlSignal<\r\n    T,\r\n    any,\r\n    any,\r\n    any\r\n  > = FormControlSignal<T, any, any, any>,\r\n  TParent = undefined,\r\n>(\r\n  initial: T[] | DerivedSignal<TParent, T[]>,\r\n  factory: (val: DerivedSignal<T[], T>, idx: number) => TIndividualState,\r\n  opt?: CreateFormArraySignalOptions<T, TIndividualState>,\r\n): FormArraySignal<T, TIndividualState, TParent> {\r\n  const eq = opt?.equal ?? Object.is;\r\n\r\n  const arrayEqual = (a: T[], b: T[]) => {\r\n    if (a.length !== b.length) return false;\r\n    if (!a.length) return true;\r\n\r\n    return a.every((v, i) => eq(v, b[i]));\r\n  };\r\n\r\n  const min = computed(() => opt?.min?.() ?? 0);\r\n  const max = computed(() => opt?.max?.() ?? Number.MAX_SAFE_INTEGER);\r\n\r\n  const arrayOptions: CreateFormControlOptions<T[], 'array'> = {\r\n    ...opt,\r\n    equal: arrayEqual,\r\n    dirtyEquality: (a, b) => a.length === b.length,\r\n    controlType: 'array',\r\n  };\r\n\r\n  const ctrl = formControl<T[], TParent, 'array'>(\r\n    initial,\r\n    arrayOptions,\r\n  ) satisfies FormControlSignal<T[], TParent, 'array'>;\r\n\r\n  const length = computed(() => ctrl.value().length);\r\n\r\n  const reconcileChildren = createReconcileChildren<T, TIndividualState>(\r\n    factory,\r\n    { equal: eq },\r\n  );\r\n\r\n  // linkedSignal used to re-use previous value so that only length changes are affected and existing controls are kept, but updated\r\n  const children = linkedSignal<number, TIndividualState[]>({\r\n    source: () => length(),\r\n    computation: (len, prev) => reconcileChildren(len, ctrl.value, prev?.value),\r\n  });\r\n\r\n  const ownError = computed(() => ctrl.error());\r\n\r\n  const error = computed((): string => {\r\n    const own = ownError();\r\n    if (own) return own;\r\n    if (!children().length) return '';\r\n    return children()\r\n      .map((c, idx) => (c.error() ? `${idx}: ${c.error()}` : ''))\r\n      .filter(Boolean)\r\n      .join('\\n');\r\n  });\r\n\r\n  const dirty = computed(() => {\r\n    if (ctrl.dirty()) return true;\r\n    if (!children().length) return false;\r\n    return children().some((c) => c.dirty());\r\n  });\r\n\r\n  const markAllAsTouched = () => {\r\n    ctrl.markAllAsTouched();\r\n    for (const c of untracked(children)) {\r\n      c.markAllAsTouched();\r\n    }\r\n  };\r\n  const markAllAsPristine = () => {\r\n    ctrl.markAllAsPristine();\r\n    for (const c of untracked(children)) {\r\n      c.markAllAsPristine();\r\n    }\r\n  };\r\n\r\n  const toPartialValue = opt?.toPartialValue ?? ((v: T) => v);\r\n  const partialValue = computed(() => {\r\n    if (!dirty()) return undefined;\r\n    return children().map((c) => {\r\n      const pv = c.partialValue();\r\n      if (pv) return pv;\r\n      if (c.controlType === 'control') return undefined;\r\n\r\n      // return full value for  child objects/arrays as this cannot be partially patched without idx\r\n      return toPartialValue(c.value());\r\n    });\r\n  });\r\n\r\n  const touched = computed(\r\n    () =>\r\n      ctrl.touched() ||\r\n      !!(children().length && children().some((c) => c.touched())),\r\n  );\r\n\r\n  const reconcile = (newValue: T[]) => {\r\n    const ctrls = untracked(children);\r\n\r\n    for (let i = 0; i < newValue.length; i++) {\r\n      ctrls.at(i)?.reconcile(newValue[i]); // reconcile existing controls that are relevant addition/removal will be handled after ctrl.reconcile through linkedSignal\r\n    }\r\n\r\n    ctrl.reconcile(mergeArray(newValue, untracked(ctrl.value)));\r\n  };\r\n\r\n  const forceReconcile = (newValue: T[]) => {\r\n    const ctrls = untracked(children);\r\n\r\n    for (let i = 0; i < newValue.length; i++) {\r\n      ctrls.at(i)?.forceReconcile(newValue[i]);\r\n    }\r\n\r\n    ctrl.forceReconcile(newValue);\r\n  };\r\n\r\n  const childrenValid = computed(() => {\r\n    if (!children().length) return true;\r\n    return children().every((d) => d.valid());\r\n  });\r\n\r\n  const childrenPending = computed(\r\n    () => !!children().length && children().some((d) => d.pending()),\r\n  );\r\n\r\n  return {\r\n    ...ctrl,\r\n    ownError,\r\n    error,\r\n    valid: computed(() => ctrl.valid() && childrenValid()),\r\n    pending: computed(() => ctrl.pending() || childrenPending()),\r\n    touched,\r\n    children,\r\n    dirty,\r\n    markAllAsTouched,\r\n    markAllAsPristine,\r\n    min,\r\n    max,\r\n    partialValue,\r\n    canAdd: computed(\r\n      () => !ctrl.disabled() && !ctrl.readonly() && length() < max(),\r\n    ),\r\n    canRemove: computed(\r\n      () => !ctrl.disabled() && !ctrl.readonly() && length() > min(),\r\n    ),\r\n    reconcile,\r\n    forceReconcile,\r\n    reset: () => {\r\n      for (const c of untracked(children)) {\r\n        c.reset();\r\n      }\r\n      ctrl.reset();\r\n    },\r\n    resetWithInitial: (initial: T[]) => {\r\n      const ctrls = untracked(children);\r\n      for (let i = 0; i < initial.length; i++) {\r\n        ctrls.at(i)?.resetWithInitial(initial[i]);\r\n      }\r\n      ctrl.resetWithInitial(initial);\r\n    },\r\n    push: (next) => ctrl.value.update((cur) => [...cur, next]),\r\n    remove: (idx) =>\r\n      ctrl.value.update((cur) => cur.filter((_, i) => i !== idx)),\r\n  };\r\n}\r\n"]}","import { computed, isSignal, signal, untracked, } from '@angular/core';\nimport { entries, mergeIfObject, values } from '@mmstack/object';\nimport { isDerivation, toFakeSignalDerivation, } from '@mmstack/primitives';\nimport { formControl, } from './form-control';\n/**\n * Creates a `FormGroupSignal`, which aggregates a set of child form controls into a single object.\n *\n * @typeParam T - The type of the form group's value (an object).\n * @typeParam TDerivations - A record where keys are the names of the child controls and values are the `FormControlSignal` instances.\n * @typeParam TParent - The type of the parent form control's value (if this group is nested within another group or array).\n * @param initial - The initial value of the form group (or a `WritableSignal` or `DerivedSignal` if the group is nested).\n * @param providedChildren - An object containing the child `FormControlSignal` instances, or a function that returns such an object.\n * Using a function allows for dynamic creation of child controls (e.g., in response to changes in other signals).\n * @param options - Optional configuration options for the form group.\n * @returns A `FormGroupSignal` instance.\n *\n * @example\n * // Create a simple form group:\n * const user = signal({ name: 'John Doe', age: 30 });\n * const form = formGroup(user, {\n * name: formControl(derived(user, 'name')),\n * age: formControl(derived(user, 'age')),\n * })\n *\n * // Create a nested form group:\n * const user = signal({ name: 'John', age: 30, address: {street: \"Some street\"} });\n *\n * const address = derived(user, 'address');\n * const userForm = formGroup(user, {\n * name: formControl(derived(user, 'name')),\n * age: formControl(derived(user, 'age')),\n * address: formGroup(address, {\n * street: formControl(derived(address, (address) => address.street), {\n * validator: () => (value) => value ? \"\" : \"required!\"\n * }) // you can create deeply nested structures.\n * })\n * });\n *\n * // Create a form group with dynamically created children replaced rare FormRecord requirements.\n * const showAddress = signal(false);\n * type Characteristic = {\n * valueType: 'string';\n * value: string;\n * } | {\n * valueType: 'number';\n * value: number;\n * }\n * const char = signal<Characteristic>({ valueType: 'string', value: '' });\n * const charForm = formGroup(char, () => {\n * if (char().valueType === 'string) retur