gform-react 2.5.0 → 2.5.2

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.
@@ -1 +1 @@
1
- {"version":3,"file":"gform-react-native.production.js","sources":["../../../src/helpers.ts","../../../src/useFormHandlers.ts","../../../src/form-context.tsx","../../../src/selectors.ts","../../../src/RNGForm.tsx","../../../src/fields/RNGInput.tsx"],"sourcesContent":["import type { GInputInitialState, GInputProps, GInputState, GInputStateMutable } from './fields';\r\nimport type { GChangeEvent, GDOMElement, IForm } from './form';\r\nimport type {GFormState, InitialState, RawData, ToFormDataOptions, ToRawDataOptions, ToURLSearchParamsOptions} from './state';\r\nimport type {ReactElement, ReactNode} from \"react\";\r\n\r\nexport const isObject = (o: any): o is object => (o && typeof o === 'object' && !Array.isArray(o));\r\n\r\nconst defaultFieldProps: { [key: string]: { value: string | number | boolean } } = {\r\n text: { value: '' },\r\n checkbox: { value: false },\r\n number: { value: 0 }\r\n};\r\n\r\nconst typeValueDict: { [key: string]: keyof HTMLFormElement | GDOMElement } = {\r\n checkbox: 'checked',\r\n number: 'valueAsNumber',\r\n};\r\n\r\nconst generateId = () => (+new Date()).toString(36) + (1 - Math.random()).toString(36).substring(2, 16);\r\n\r\nexport const _buildFormInitialValues = <T>(rows: ReactNode | ReactNode[] = []): InitialState<T> => {\r\n const fields: { [key: string]: GInputInitialState } = {};\r\n\r\n if (!Array.isArray(rows)) rows = [rows];\r\n\r\n if (__DEBUG__) {\r\n console.log('[buildFormInitialValues] -', 'building initial values for ', rows);\r\n }\r\n\r\n for (const row of rows) {\r\n const inputConfigs = _findInputs(row as any);\r\n\r\n inputConfigs.forEach(config => {\r\n if (__DEBUG__) {\r\n console.log('[buildFormInitialValues] -', 'building input', `(${config.formKey})`, config);\r\n }\r\n\r\n if (__DEV__ && fields[config.formKey]) {\r\n console.warn(`[Duplicate Keys] - field with key '${config.formKey}' has already been defined.`);\r\n }\r\n\r\n const { required = false, max, maxLength, min, minLength, step, pattern, type = 'text', defaultValue, value, checked, defaultChecked, formKey, debounce, validatorKey } = config;\r\n const defaultProps = defaultFieldProps[type] || defaultFieldProps.text;\r\n const inputValue = value || defaultValue || checked || defaultChecked || defaultProps.value;\r\n\r\n fields[config.formKey] = {\r\n formKey,\r\n type,\r\n required,\r\n max,\r\n maxLength,\r\n min,\r\n minLength,\r\n step,\r\n pattern,\r\n value: inputValue,\r\n validatorKey,\r\n debounce,\r\n dirty: false,\r\n touched: false,\r\n gid: generateId()\r\n };\r\n\r\n Object.keys(fields[config.formKey]).forEach(key => {\r\n if (typeof fields[config.formKey][key] === 'undefined') delete fields[config.formKey][key];\r\n });\r\n });\r\n }\r\n\r\n return { fields: fields as IForm<T>, key: generateId() };\r\n};\r\n\r\nconst _findInputs = (root?: ReactElement<any> | ReactElement<any>[] | undefined[], total: (GInputProps & GInputStateMutable)[] = []): (GInputProps & GInputStateMutable)[] => {\r\n if (!root) return total;\r\n\r\n if (Array.isArray(root)) {\r\n root.forEach(element => _findInputs(element, total));\r\n return total;\r\n }\r\n\r\n\r\n if (root.props?.formKey) {\r\n if (__DEBUG__) {\r\n console.log('[findInputs] -', 'input config found', `(${root.props.formKey})`);\r\n }\r\n total.push(root.props);\r\n return total;\r\n }\r\n\r\n return _findInputs(root.props?.children, total);\r\n};\r\n\r\nexport const _findValidityKey = (validity: Partial<ValidityState>): keyof ValidityState | undefined => {\r\n for (const key in validity) {\r\n if (key !== 'valid' && validity[key as keyof ValidityState]) {\r\n if (__DEBUG__) {\r\n console.log('[findValidityKey] -', 'found validity key:', key);\r\n }\r\n return key as keyof ValidityState;\r\n }\r\n }\r\n};\r\n\r\nexport const _checkTypeMismatch = (input: GInputState<any>) => {\r\n const value = input.value?.toString().trim();\r\n if (!value) return false;\r\n\r\n switch (input.type) {\r\n case 'email':\r\n return !/^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(value); // basic email pattern\r\n case 'url':\r\n try {\r\n new URL(value);\r\n return false;\r\n } catch {\r\n return true;\r\n }\r\n case 'tel':\r\n return !/^\\+?[0-9\\s\\-().]{7,}$/.test(value); // basic phone pattern\r\n default:\r\n return false;\r\n }\r\n};\r\n\r\n\r\nexport const hasSubmitter = (form?: HTMLFormElement | null): boolean => {\r\n if (!form) return false;\r\n\r\n for (const element of form) {\r\n if ((element as HTMLInputElement).type === 'submit') return true;\r\n }\r\n\r\n return false;\r\n};\r\n\r\nexport const _checkIfFormIsValid = <T>(fields: IForm<T>): boolean => {\r\n for (const f in fields) {\r\n if (fields[f].error) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n};\r\n\r\nexport const _toRawData = <T>(fields: IForm<T> & { [key: string]: GInputState<any> }, options: ToRawDataOptions<T> = {}): RawData<T> => {\r\n const data: { [key: string]: unknown } = {};\r\n\r\n const { include, exclude, transform } = options;\r\n\r\n if (include) {\r\n include.forEach(key => data[key as string] = fields[key]?.value);\r\n } else for (const f in fields) {\r\n data[f] = fields[f].value;\r\n }\r\n\r\n exclude?.forEach(key => delete data[key as string]);\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n data[key] = set(fields[key]?.value || fields[key]);\r\n }\r\n }\r\n\r\n return data as RawData<T>;\r\n};\r\n\r\nexport const _toFormData = <T>(form: HTMLFormElement | null, options?: ToFormDataOptions<T>): FormData => {\r\n if (!form) return new FormData();\r\n\r\n if (options) {\r\n const { exclude, include, transform } = options;\r\n let formData: FormData;\r\n\r\n if (include) {\r\n formData = new FormData();\r\n include.forEach(key => formData.set(key as string, form[key as string]?.value));\r\n } else {\r\n formData = new FormData(form);\r\n exclude?.forEach(key => formData.delete(key as string));\r\n }\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n formData.set(key, set(form[key]?.value));\r\n }\r\n }\r\n return formData;\r\n\r\n }\r\n\r\n return new FormData(form);\r\n};\r\n\r\nexport function _toURLSearchParams<T>(this: GFormState<T>, options?: ToURLSearchParamsOptions<T>): URLSearchParams {\r\n let data: Record<keyof T, any>;\r\n if (options) {\r\n const { exclude, include, transform } = options;\r\n if (include) {\r\n data = {} as Record<keyof T, any>;\r\n include.forEach(key => (data[key] = this[key]?.value));\r\n } else {\r\n data = this.toRawData();\r\n exclude?.forEach(key => delete data[key]);\r\n }\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n (data[key] = set(this[key]?.value));\r\n }\r\n }\r\n }\r\n else data = this.toRawData();\r\n\r\n return new URLSearchParams(data); // this is ok because URLSearchParams will stringify the values (boolean/number)\r\n}\r\n\r\nfunction __debounce(this: { [key: string]: { timerId: NodeJS.Timeout } }, timeout: number, id: string): Promise<void> {\r\n return new Promise(resolve => {\r\n if (this[id]?.timerId)\r\n clearTimeout(this[id].timerId);\r\n\r\n const timerId = setTimeout(() => resolve(), timeout);\r\n\r\n if (this[id]) {\r\n this[id].timerId = timerId;\r\n } else this[id] = { timerId };\r\n });\r\n}\r\n\r\nexport const _debounce = __debounce.bind({});\r\n\r\nexport const _extractValue = <T>(e?: GChangeEvent<GDOMElement | HTMLFormElement>, unknown?: { value: T } | string | number): undefined | string | number | boolean | T => {\r\n if (e?.target) {\r\n if (Object.hasOwn(typeValueDict, e.target.type)) return e.target[typeValueDict[e.target.type] as 'value'];\r\n return e.target.value;\r\n }\r\n return (e?.value as T) || (isObject(unknown) ? unknown.value : unknown);\r\n};\r\n\r\nexport const _checkResult = (handlerResult: boolean | RegExp | string, value: string | number | boolean): boolean => typeof handlerResult === 'boolean' ? handlerResult : typeof value === 'string' ? typeof handlerResult === 'string' ? !new RegExp(handlerResult).test(value) : !handlerResult.test(value) : false;\r\n\r\nexport const _merge = <T extends object>(target: { [key: string]: any } = {}, ...nodes: ({ [key: string]: any } | undefined | void)[]): T => {\r\n if (!nodes.length) return target as T;\r\n\r\n const next = nodes.shift();\r\n if (isObject(next)) {\r\n for (const key in next) {\r\n target[key] = target[key] ? { ...target[key], ...next[key] } : next[key];\r\n }\r\n }\r\n\r\n return _merge(target, ...nodes);\r\n};","import {_checkResult, _checkTypeMismatch, _debounce, _extractValue, _findValidityKey} from \"./helpers\";\r\nimport {type GInputValidator, type GValidators} from \"./validations\";\r\nimport type {GInputState} from \"./fields\";\r\nimport type {GChangeEvent, GDOMElement, GFocusEvent, GFormEvent, GInvalidEvent} from \"./form\";\r\nimport type {InitialState, Store} from \"./state\";\r\n\r\nexport const useFormHandlers = (getState: Store['getState'], setState: Store['setState'], validators: GValidators = {}, optimized = false) => {\r\n /**\r\n * handler for validating a form input\r\n * @param input the input to be validated\r\n * @param e the event object\r\n */\r\n const _viHandler = (input: GInputState, e?: GFocusEvent<GDOMElement | HTMLFormElement> | GInvalidEvent<GDOMElement | HTMLFormElement> | GFormEvent<GDOMElement | HTMLFormElement> | GFormEvent): void => {\r\n if (!input) return;\r\n const element = e && e.target;\r\n\r\n if (typeof document !== 'undefined' && (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement || element instanceof HTMLSelectElement)) {\r\n if (!input.checkValidity) input.checkValidity = () => element.checkValidity();\r\n\r\n //if the field has initial value\r\n if (!input.dirty && input.value && !input.touched) {\r\n /**\r\n * for inputs with initial value we have to manually check for validations.\r\n * validity.tooShort is false even though initial value is smaller than minLength, because its required to be filled in by user (native dirty flag is true).\r\n * it only works for validity.valueMissing.\r\n * If an element has a minimum allowed value length, its dirty value flag is true, its value was last changed by a user edit (as opposed to a change made by a script), its value is not the empty string, and the length of the element's API value is less than the element's minimum allowed value length, then the element is suffering from being too short.\r\n * @see https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#setting-minimum-input-length-requirements:-the-minlength-attribute\r\n */\r\n _checkInputManually(input);\r\n _dispatchChanges(input, input.formKey);\r\n return;\r\n }\r\n element.setCustomValidity(''); //reset any previous error (custom)\r\n\r\n const validityKey = _findValidityKey(element.validity);\r\n _validateInput(input, validityKey, (v: string) => element.setCustomValidity(v));\r\n\r\n if (!validityKey && input.error) {\r\n element.setCustomValidity(input.errorText || 'error');\r\n }\r\n\r\n _dispatchChanges(input, input.formKey);\r\n } else {\r\n if (__DEBUG__) {\r\n console.log('[validateInputHandler] -', `the input '${input.formKey}' is not a native web element\\nevent:`, e);\r\n }\r\n\r\n //fallback - validate the input for validations manually\r\n input.checkValidity = () => _checkInputManually(input);\r\n input.checkValidity();\r\n\r\n _dispatchChanges(input, input.formKey);\r\n }\r\n };\r\n\r\n const _checkInputManually = (input: GInputState) => {\r\n let validityKey = _findValidityKey({\r\n valueMissing: input.required && !input.value || false,\r\n typeMismatch: _checkTypeMismatch(input),\r\n tooShort: input.minLength && input.value.toString().length < input.minLength || false,\r\n tooLong: input.maxLength && input.value.toString().length > input.maxLength || false,\r\n patternMismatch: input.pattern && _checkResult(input.pattern, input.value) || false,\r\n rangeUnderflow: input.min && Number(input.value) < Number(input.min) || false,\r\n rangeOverflow: input.max && Number(input.value) > Number(input.max) || false\r\n });\r\n\r\n if (!validityKey && input.error) {\r\n validityKey = 'customError';\r\n }\r\n _validateInput(input, validityKey);\r\n return !input.error;\r\n };\r\n\r\n /**\r\n * handler for updating and validating a form input\r\n * @param input\r\n * @param e the event object\r\n * @param unknown\r\n */\r\n const _updateInputHandler = (input: GInputState, e?: GChangeEvent<GDOMElement | HTMLFormElement>, unknown?: {\r\n value: unknown\r\n } | string | number): void => {\r\n input.value = _extractValue(e, unknown) as GInputState['value'];\r\n _viHandler(input, e);\r\n };\r\n\r\n /**\r\n * Validates the input and updates the state with the result\r\n * @param input the input to be validated\r\n * @param validityKey the `Constraint Validation` key\r\n * @param setValidity\r\n */\r\n const _validateInput = (input: GInputState, validityKey?: keyof ValidityState, setValidity?: (e: string) => void): void => {\r\n const inputValidator = validators[input.validatorKey || input.formKey] || validators['*'];\r\n if (__DEBUG__) {\r\n console.log('[validateInput] -', 'validating input:', input.formKey, `(${validityKey ? validityKey : 'custom'})`);\r\n }\r\n\r\n if (inputValidator) {\r\n __validateInput(input, inputValidator, validityKey, setValidity);\r\n }\r\n input.touched = true;\r\n };\r\n\r\n const _dispatchChanges = (changes: Partial<InitialState> | Partial<GInputState>, key?: string) => setState(prev => {\r\n if (key) {\r\n return {...prev, fields: {...prev.fields, [key]: {...prev.fields[key], ...changes}}};\r\n }\r\n return {...prev, ...changes};\r\n });\r\n\r\n /**\r\n * @internal\r\n */\r\n const __validateInput = (input: GInputState, inputValidator: GInputValidator<any>, validityKey?: keyof ValidityState, setValidity?: (e: string) => void): void => {\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validating input (${input.formKey}) with handlers:`, inputValidator.handlers);\r\n }\r\n const fields = getState().fields;\r\n\r\n for (const index in inputValidator.constraintHandlers) {\r\n const result = inputValidator.constraintHandlers[index](input, validityKey);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for constraint handler (${index}):\\n`, inputValidator.constraintHandlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) return;\r\n }\r\n\r\n for (const index in inputValidator.handlers) {\r\n const result = inputValidator.handlers[index](input, fields);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for custom handler (${index}):\\n`, inputValidator.handlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) return;\r\n }\r\n\r\n input.errorText = '';\r\n\r\n if (inputValidator.asyncHandlers.length) {\r\n input.error = true;\r\n _debounce(input.debounce || 300, `${input.gid}-async`).then(() => {\r\n const validateAsync = async () => {\r\n for (const index in inputValidator.asyncHandlers) {\r\n const result = await inputValidator.asyncHandlers[index](input, fields);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for custom async handler (${index}):\\n`, inputValidator.asyncHandlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) break;\r\n }\r\n if (!input.error) input.errorText = '';\r\n\r\n _dispatchChanges({error: input.error, errorText: input.errorText}, input.formKey);\r\n if (setValidity) {\r\n setValidity(input.errorText);\r\n }\r\n };\r\n\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validating input (${input.formKey}) with async handlers:`, inputValidator.asyncHandlers);\r\n }\r\n validateAsync();\r\n });\r\n }\r\n };\r\n\r\n return {_updateInputHandler, _viHandler, _dispatchChanges, optimized, _createInputChecker: _checkInputManually};\r\n};","import React, {createContext, useCallback, useContext, useEffect, useRef, useSyncExternalStore} from 'react';\r\nimport type {FC, PropsWithChildren} from 'react';\r\n\r\nimport {useFormHandlers} from \"./useFormHandlers\";\r\nimport type {InitialState, Store} from \"./state\";\r\nimport type {GValidators} from \"./validations\";\r\nimport type {GInputState} from './fields';\r\n\r\nconst GFormContext = createContext<Store>({} as Store);\r\n\r\ntype GFormContextProviderProps = PropsWithChildren & {\r\n initialState: InitialState;\r\n validators?: GValidators;\r\n optimized?: boolean;\r\n}\r\n\r\nexport const GFormContextProvider: FC<GFormContextProviderProps> = ({ children, initialState, validators, optimized }) => {\r\n const stateRef = useRef(initialState);\r\n const listeners = useRef(new Set<() => void>());\r\n\r\n const setState = useCallback((updater: InitialState | ((state: InitialState) => InitialState)) => {\r\n stateRef.current = typeof updater === 'function' ? updater(stateRef.current) : updater;\r\n listeners.current.forEach((l) => l());\r\n }, []);\r\n\r\n const handlers = useFormHandlers(() => stateRef.current, setState, validators, optimized);\r\n\r\n const getState = useCallback(() => stateRef.current, []);\r\n\r\n const subscribe = useCallback((listener: () => void) => {\r\n listeners.current.add(listener);\r\n return () => listeners.current.delete(listener);\r\n }, []);\r\n\r\n useEffect(() => {\r\n for (const fieldKey in initialState.fields) {\r\n initialState.fields[fieldKey].dispatchChanges = (changes: Partial<GInputState>) => handlers._dispatchChanges(changes, fieldKey);\r\n }\r\n }, []);\r\n\r\n const store = useRef<Store>({ getState, setState, subscribe, handlers });\r\n\r\n return <GFormContext.Provider value={store.current}>{children}</GFormContext.Provider>;\r\n};\r\n\r\nexport const useFormStore = () => {\r\n const store = useContext(GFormContext);\r\n if (!store) throw new Error('useGFormStore must be used within `GForm` component');\r\n return store;\r\n};\r\n\r\nexport const useFormSelector = <T extends any>(selector: (state: InitialState) => T): T => {\r\n const store = useFormStore();\r\n\r\n return useSyncExternalStore(\r\n store.subscribe,\r\n () => selector(store.getState()),\r\n () => selector(store.getState()) // for SSR\r\n );\r\n};\r\n\r\nexport function createSelector<\r\n State=InitialState,\r\n Selectors extends Array<(state: State) => any> = [],\r\n Result = any\r\n>(\r\n selectors: Selectors,\r\n combiner: (...args: {\r\n [K in keyof Selectors]: Selectors[K] extends (state: State) => infer R ? R : never;\r\n }) => Result\r\n): (state: State) => Result {\r\n let lastArgs: any[] = [];\r\n let lastResult: Result;\r\n\r\n return (state: State): Result => {\r\n const args = selectors.map(fn => fn(state));\r\n if (\r\n lastArgs.length === args.length &&\r\n args.every((val, i) => val === lastArgs[i])\r\n ) {\r\n return lastResult;\r\n }\r\n lastArgs = args;\r\n lastResult = combiner(...args as any);\r\n return lastResult;\r\n };\r\n}","import {createSelector} from \"./form-context\";\r\nimport {InitialState} from \"./state\";\r\n\r\nconst selectFields = [(state: InitialState) => state.fields];\r\n\r\nexport const selectFirstInvalidField =\r\n createSelector(\r\n selectFields,\r\n (fields) => {\r\n for (const f in fields) {\r\n if (fields[f].error) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n );\r\n\r\nexport const makeSelectFields = (keys: string[] = []) =>\r\n createSelector(\r\n selectFields,\r\n (fields) => {\r\n const selected = keys.map((key) => fields[key]).filter(Boolean);\r\n return selected.length ? selected : null;\r\n }\r\n );","import React, {useMemo, useEffect, forwardRef, useCallback} from \"react\";\r\nimport type { ReactNode, FC, ReactElement, Ref, RefObject } from \"react\";\r\n\r\nimport {_toRawData, _toURLSearchParams, _merge, _buildFormInitialValues} from \"./helpers\";\r\nimport {GFormContextProvider, useFormSelector, useFormStore} from \"./form-context\";\r\nimport type {RNGFormState, ToRawDataOptions} from \"./state\";\r\nimport type { GValidators } from \"./validations\";\r\nimport type { IForm, PartialForm } from \"./form\";\r\nimport type { GInputState } from \"./fields\";\r\nimport {selectFirstInvalidField} from \"./selectors\";\r\n\r\nconst FormRenderer = forwardRef<any, RNGFormProps<any>>(\r\n <T, >({\r\n stateRef,\r\n children,\r\n onInit,\r\n el: El,\r\n ...rest\r\n }: RNGFormProps<T>, ref: React.Ref<any>) => {\r\n const {getState, handlers} = useFormStore();\r\n const isFormInvalid = useFormSelector(selectFirstInvalidField);\r\n\r\n const getFormState = useCallback(() => {\r\n const fields = getState<T>().fields;\r\n\r\n const formState: RNGFormState<T> = {\r\n ...fields,\r\n isValid: !isFormInvalid,\r\n isInvalid: isFormInvalid,\r\n toRawData: (options?: ToRawDataOptions<T>) => _toRawData(fields, options),\r\n toURLSearchParams: _toURLSearchParams,\r\n checkValidity: () => {\r\n for (const i in fields) {\r\n const valid = fields[i].checkValidity();\r\n if (!valid) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n },\r\n dispatchChanges: (changes: PartialForm<T> & {\r\n [key: string]: Partial<GInputState<any>>\r\n }) => handlers._dispatchChanges({\r\n fields: _merge<IForm<T> & {\r\n [key: string]: GInputState;\r\n }>({}, fields, changes)\r\n })\r\n };\r\n\r\n if (stateRef) stateRef.current = formState;\r\n\r\n return formState;\r\n }, [isFormInvalid]);\r\n\r\n const formComponent = useMemo(() => {\r\n const state = getFormState();\r\n\r\n const formChildren = typeof children === 'function' ? children(state) : children;\r\n\r\n return <El {...rest} ref={ref}>\r\n {formChildren}\r\n </El>;\r\n }, [getFormState, children]);\r\n\r\n useEffect(() => {\r\n const state = getFormState();\r\n\r\n if (onInit) {\r\n const changes = onInit(state);\r\n if (changes) {\r\n const _handler = (_c: void | PartialForm<T>) => handlers._dispatchChanges({\r\n fields: _merge<IForm<T> & {\r\n [key: string]: GInputState;\r\n }>({}, state, _c)\r\n });\r\n if (changes instanceof Promise) {\r\n changes.then(_handler);\r\n } else _handler(changes);\r\n }\r\n }\r\n\r\n const dispatchers: Record<string, Record<string, (changes: Partial<GInputState>) => void>> = {};\r\n \r\n if (__DEBUG__) {\r\n console.log('checking for initial values');\r\n }\r\n const fields = getState().fields;\r\n\r\n for (const fieldKey in fields) {\r\n dispatchers[fieldKey] = {\r\n dispatchChanges: (changes: Partial<GInputState>) => handlers._dispatchChanges(changes, fieldKey),\r\n checkValidity: () => {\r\n const result = handlers._createInputChecker(state[fieldKey]);\r\n handlers._dispatchChanges(state[fieldKey], fieldKey);\r\n return result;\r\n }\r\n };\r\n\r\n const field = fields[fieldKey];\r\n\r\n //we don't want to apply validation on empty fields so skip it.\r\n if (!field.value) continue;\r\n\r\n if (__DEBUG__) {\r\n console.log(`found input '${fieldKey}', applying validation(s)`);\r\n }\r\n /**\r\n * We have to manually check for validations (checkValidity() will not result with validity.tooShort = true).\r\n * If an element has a minimum allowed value length, its dirty value flag is true, its value was last changed by a user edit (as opposed to a change made by a script), its value is not the empty string, and the length of the element's API value is less than the element's minimum allowed value length, then the element is suffering from being too short.\r\n * @see https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#setting-minimum-input-length-requirements:-the-minlength-attribute\r\n */\r\n handlers._viHandler(field);\r\n }\r\n handlers._dispatchChanges({fields: _merge(dispatchers, state)});\r\n }, [getFormState]);\r\n\r\n return formComponent;\r\n }\r\n) as <T>(props: RNGFormProps<T> & { ref?: Ref<any> }) => ReactElement | null;\r\n\r\nexport type RNGFormProps<T> = {\r\n children?: ReactNode | ReactNode[] | ((state: RNGFormState<T>) => ReactNode | ReactNode[]);\r\n /** @param stateRef - pass a ref which will points to the current state of the form (optional). */\r\n stateRef?: RefObject<RNGFormState<T> | undefined>;\r\n /** @param validators - an object for handling validations (optional). */\r\n validators?: GValidators<T>;\r\n /** @param onInit - execute a handler once the form has initialized (optional). */\r\n onInit?: (state: RNGFormState<T>) => void | PartialForm<T> | Promise<void | PartialForm<T>>;\r\n /** @param el - the element to use as the form container. */\r\n el: FC<any>;\r\n};\r\n\r\n/**\r\n * build dynamic forms with validations.\r\n * @link Docs - https://gform-react.onrender.com\r\n * @link Npm - https://www.npmjs.com/package/gform-react\r\n */\r\nexport const RNGForm = forwardRef<any, RNGFormProps<any>>(\r\n <T, >({children, validators, ...props}: RNGFormProps<T>, ref: React.Ref<any>) => {\r\n const initialState = useMemo(() => {\r\n return _buildFormInitialValues(\r\n typeof children === 'function'\r\n ? children({} as RNGFormState<T>)\r\n : children\r\n );\r\n }, [children]);\r\n\r\n return (\r\n <GFormContextProvider key={initialState.key} initialState={initialState} validators={validators}>\r\n <FormRenderer ref={ref} {...props}>\r\n {children}\r\n </FormRenderer>\r\n </GFormContextProvider>\r\n );\r\n }\r\n) as <T>(props: RNGFormProps<T> & { ref?: React.Ref<HTMLFormElement> }) => React.ReactElement | null;","import React, {forwardRef, memo, type ReactNode, useEffect, useMemo} from 'react';\r\nimport {TextInput} from 'react-native';\r\n\r\nimport {_debounce} from '../helpers';\r\nimport type {GInputState, RNGInputProps} from '.';\r\nimport {useFormSelector, useFormStore} from \"../form-context\";\r\nimport {makeSelectFields} from \"../selectors\";\r\n\r\nconst _RNGInput = forwardRef<any, RNGInputProps>(({\r\n formKey,\r\n element,\r\n type,\r\n fetch,\r\n fetchDeps,\r\n debounce = 300,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n defaultValue,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n validatorKey,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n value,\r\n ...rest\r\n}, ref) => {\r\n const inputState = useFormSelector(state => state.fields[formKey]);\r\n const store = useFormStore();\r\n\r\n const _element = useMemo(() => {\r\n const value = inputState.value || '';\r\n\r\n const _props = {\r\n ...rest,\r\n value,\r\n inputMode: type,\r\n ref\r\n };\r\n\r\n _props.onEndEditing = rest.onEndEditing ?\r\n (e) => {\r\n store.handlers._viHandler(inputState);\r\n rest.onEndEditing!(e);\r\n }\r\n :\r\n () => {\r\n store.handlers._viHandler(inputState);\r\n };\r\n _props.onChangeText = rest.onChangeText ?\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n rest.onChangeText!(e);\r\n }\r\n :\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n };\r\n\r\n if (element) {\r\n return (element as (input: GInputState, props: any) => ReactNode)(inputState, _props);\r\n }\r\n\r\n return <TextInput {..._props}/>;\r\n }, [inputState, element]);\r\n\r\n const _fetchDeps = useFormSelector(makeSelectFields(fetchDeps));\r\n const stableFetchDeps = useMemo(() => JSON.stringify(_fetchDeps), [_fetchDeps]);\r\n\r\n useEffect(() => {\r\n if (fetch) {\r\n _debounce(debounce, `${inputState.gid}-fetch`).then(() => {\r\n const res = fetch(inputState, store.getState().fields);\r\n if (res instanceof Promise) {\r\n res.then((state) => state && store.handlers._dispatchChanges(state, formKey));\r\n } else if (res) {\r\n store.handlers._dispatchChanges(res, formKey);\r\n }\r\n });\r\n }\r\n }, [stableFetchDeps]);\r\n\r\n return _element;\r\n});\r\n\r\nexport const RNGInput = memo(_RNGInput);"],"names":["isObject","o","Array","isArray","defaultFieldProps","text","value","checkbox","number","typeValueDict","generateId","Date","toString","Math","random","substring","_findInputs","root","total","_root$props","_root$props2","forEach","element","props","formKey","push","children","_findValidityKey","validity","key","_checkTypeMismatch","input","_input$value","trim","type","test","URL","_toURLSearchParams","options","data","exclude","include","transform","_this$key","this","toRawData","_this$key2","set","URLSearchParams","_debounce","timeout","id","Promise","resolve","_this$id","timerId","clearTimeout","setTimeout","bind","_checkResult","handlerResult","RegExp","_merge","target","nodes","length","next","shift","useFormHandlers","getState","setState","validators","optimized","_viHandler","e","document","HTMLInputElement","HTMLTextAreaElement","HTMLSelectElement","checkValidity","dirty","touched","_checkInputManually","_dispatchChanges","setCustomValidity","validityKey","_validateInput","v","error","errorText","valueMissing","required","typeMismatch","tooShort","minLength","tooLong","maxLength","patternMismatch","pattern","rangeUnderflow","min","Number","rangeOverflow","max","setValidity","inputValidator","validatorKey","__validateInput","changes","prev","fields","index","constraintHandlers","result","handlers","asyncHandlers","debounce","gid","then","async","validateAsync","_updateInputHandler","unknown","_extractValue","Object","hasOwn","_createInputChecker","GFormContext","createContext","GFormContextProvider","initialState","stateRef","useRef","listeners","Set","useCallback","updater","current","l","subscribe","listener","add","delete","useEffect","fieldKey","dispatchChanges","store","React","createElement","Provider","useFormStore","useContext","Error","useFormSelector","selector","useSyncExternalStore","createSelector","selectors","combiner","lastResult","lastArgs","state","args","map","fn","every","val","i","selectFields","selectFirstInvalidField","f","FormRenderer","forwardRef","onInit","el","El","rest","ref","isFormInvalid","getFormState","formState","isValid","isInvalid","_toRawData","_fields$key","_fields$key2","toURLSearchParams","formComponent","useMemo","formChildren","_extends","_handler","_c","dispatchers","field","RNGForm","_buildFormInitialValues","rows","row","config","step","defaultValue","checked","defaultChecked","inputValue","keys","_RNGInput","fetch","fetchDeps","inputState","_element","_props","inputMode","onEndEditing","onChangeText","undefined","TextInput","_fetchDeps","makeSelectFields","selected","filter","Boolean","stableFetchDeps","JSON","stringify","res","RNGInput","memo"],"mappings":"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"}
1
+ {"version":3,"file":"gform-react-native.production.js","sources":["../../../src/helpers.ts","../../../src/useFormHandlers.ts","../../../src/form-context.tsx","../../../src/selectors.ts","../../../src/RNGForm.tsx","../../../src/fields/RNGInput.tsx"],"sourcesContent":["import type { GInputInitialState, GInputProps, GInputState, GInputStateMutable } from './fields';\r\nimport type { GChangeEvent, GDOMElement, IForm } from './form';\r\nimport type {GFormState, InitialState, RawData, ToFormDataOptions, ToRawDataOptions, ToURLSearchParamsOptions} from './state';\r\nimport type {ReactElement, ReactNode} from \"react\";\r\n\r\nexport const isObject = (o: any): o is object => (o && typeof o === 'object' && !Array.isArray(o));\r\n\r\nconst defaultFieldProps: { [key: string]: { value: string | number | boolean } } = {\r\n text: { value: '' },\r\n checkbox: { value: false },\r\n number: { value: 0 }\r\n};\r\n\r\nconst typeValueDict: { [key: string]: keyof HTMLFormElement | GDOMElement } = {\r\n checkbox: 'checked',\r\n number: 'valueAsNumber',\r\n};\r\n\r\nconst generateId = () => (+new Date()).toString(36) + (1 - Math.random()).toString(36).substring(2, 16);\r\n\r\nexport const _buildFormInitialValues = <T>(rows: ReactNode | ReactNode[] = []): InitialState<T> => {\r\n const fields: { [key: string]: GInputInitialState } = {};\r\n\r\n if (!Array.isArray(rows)) rows = [rows];\r\n\r\n if (__DEBUG__) {\r\n console.log('[buildFormInitialValues] -', 'building initial values for ', rows);\r\n }\r\n\r\n for (const row of rows) {\r\n const inputConfigs = _findInputs(row as any);\r\n\r\n inputConfigs.forEach(config => {\r\n if (__DEBUG__) {\r\n console.log('[buildFormInitialValues] -', 'building input', `(${config.formKey})`, config);\r\n }\r\n\r\n if (__DEV__ && fields[config.formKey]) {\r\n console.warn(`DEV ONLY - [Duplicate Keys] - field with key '${config.formKey}' has already been defined.`);\r\n }\r\n\r\n const { required = false, max, maxLength, min, minLength, step, pattern, type = 'text', defaultValue, value, checked, defaultChecked, formKey, debounce, validatorKey } = config;\r\n const defaultProps = defaultFieldProps[type] || defaultFieldProps.text;\r\n const inputValue = value || defaultValue || checked || defaultChecked || defaultProps.value;\r\n\r\n fields[config.formKey] = {\r\n formKey,\r\n type,\r\n required,\r\n max,\r\n maxLength,\r\n min,\r\n minLength,\r\n step,\r\n pattern,\r\n value: inputValue,\r\n validatorKey,\r\n debounce,\r\n dirty: false,\r\n touched: false,\r\n gid: generateId()\r\n };\r\n\r\n Object.keys(fields[config.formKey]).forEach(key => {\r\n if (typeof fields[config.formKey][key] === 'undefined') delete fields[config.formKey][key];\r\n });\r\n });\r\n }\r\n\r\n return { fields: fields as IForm<T>, key: generateId() };\r\n};\r\n\r\nconst _findInputs = (root?: ReactElement<any> | ReactElement<any>[] | undefined[], total: (GInputProps & GInputStateMutable)[] = []): (GInputProps & GInputStateMutable)[] => {\r\n if (!root) return total;\r\n\r\n if (Array.isArray(root)) {\r\n root.forEach(element => _findInputs(element, total));\r\n return total;\r\n }\r\n\r\n\r\n if (root.props?.formKey) {\r\n if (__DEBUG__) {\r\n console.log('[findInputs] -', 'input config found', `(${root.props.formKey})`);\r\n }\r\n total.push(root.props);\r\n return total;\r\n }\r\n\r\n return _findInputs(root.props?.children, total);\r\n};\r\n\r\nexport const _findValidityKey = (validity: Partial<ValidityState>, exclude: (keyof ValidityState)[] = []): keyof ValidityState | undefined => {\r\n for (const key in validity) {\r\n if (exclude.includes(key as keyof ValidityState)) continue;\r\n if (key !== 'valid' && validity[key as keyof ValidityState]) {\r\n if (__DEBUG__) {\r\n console.log('[findValidityKey] -', 'found validity key:', key);\r\n }\r\n return key as keyof ValidityState;\r\n }\r\n }\r\n};\r\n\r\nexport const _checkTypeMismatch = (input: GInputState<any>) => {\r\n const value = input.value?.toString().trim();\r\n if (!value) return false;\r\n\r\n switch (input.type) {\r\n case 'email':\r\n return !/^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(value); // basic email pattern\r\n case 'url':\r\n try {\r\n new URL(value);\r\n return false;\r\n } catch {\r\n return true;\r\n }\r\n case 'tel':\r\n return !/^\\+?[0-9\\s\\-().]{7,}$/.test(value); // basic phone pattern\r\n default:\r\n return false;\r\n }\r\n};\r\n\r\n\r\nexport const hasSubmitter = (form?: HTMLFormElement | null): boolean => {\r\n if (!form) return false;\r\n\r\n for (const element of form) {\r\n if ((element as HTMLInputElement).type === 'submit') return true;\r\n }\r\n\r\n return false;\r\n};\r\n\r\nexport const _checkIfFormIsValid = <T>(fields: IForm<T>): boolean => {\r\n for (const f in fields) {\r\n if (fields[f].error) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n};\r\n\r\nexport const _toRawData = <T>(fields: IForm<T> & { [key: string]: GInputState<any> }, options: ToRawDataOptions<T> = {}): RawData<T> => {\r\n const data: { [key: string]: unknown } = {};\r\n\r\n const { include, exclude, transform } = options;\r\n\r\n if (include) {\r\n include.forEach(key => data[key as string] = fields[key]?.value);\r\n } else for (const f in fields) {\r\n data[f] = fields[f].value;\r\n }\r\n\r\n exclude?.forEach(key => delete data[key as string]);\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n data[key] = set(fields[key]?.value || fields[key]);\r\n }\r\n }\r\n\r\n return data as RawData<T>;\r\n};\r\n\r\nexport const _toFormData = <T>(form: HTMLFormElement | null, options?: ToFormDataOptions<T>): FormData => {\r\n if (!form) return new FormData();\r\n\r\n if (options) {\r\n const { exclude, include, transform } = options;\r\n let formData: FormData;\r\n\r\n if (include) {\r\n formData = new FormData();\r\n include.forEach(key => formData.set(key as string, form[key as string]?.value));\r\n } else {\r\n formData = new FormData(form);\r\n exclude?.forEach(key => formData.delete(key as string));\r\n }\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n formData.set(key, set(form[key]?.value));\r\n }\r\n }\r\n return formData;\r\n\r\n }\r\n\r\n return new FormData(form);\r\n};\r\n\r\nexport function _toURLSearchParams<T>(this: GFormState<T>, options?: ToURLSearchParamsOptions<T>): URLSearchParams {\r\n let data: Record<keyof T, any>;\r\n if (options) {\r\n const { exclude, include, transform } = options;\r\n if (include) {\r\n data = {} as Record<keyof T, any>;\r\n include.forEach(key => (data[key] = this[key]?.value));\r\n } else {\r\n data = this.toRawData();\r\n exclude?.forEach(key => delete data[key]);\r\n }\r\n\r\n if (transform) {\r\n for (const key in transform) {\r\n const set = transform[key] as (value: GFormState<T>[typeof key]['value']) => any;\r\n (data[key] = set(this[key]?.value));\r\n }\r\n }\r\n }\r\n else data = this.toRawData();\r\n\r\n return new URLSearchParams(data); // this is ok because URLSearchParams will stringify the values (boolean/number)\r\n}\r\n\r\nfunction __debounce(this: { [key: string]: { timerId: NodeJS.Timeout } }, timeout: number, id: string): Promise<void> {\r\n return new Promise(resolve => {\r\n if (this[id]?.timerId)\r\n clearTimeout(this[id].timerId);\r\n\r\n const timerId = setTimeout(() => resolve(), timeout);\r\n\r\n if (this[id]) {\r\n this[id].timerId = timerId;\r\n } else this[id] = { timerId };\r\n });\r\n}\r\n\r\nexport const _debounce = __debounce.bind({});\r\n\r\nexport const _extractValue = <T>(e?: GChangeEvent<GDOMElement | HTMLFormElement>, unknown?: { value: T } | string | number): undefined | string | number | boolean | T => {\r\n if (e?.target) {\r\n if (Object.hasOwn(typeValueDict, e.target.type)) return e.target[typeValueDict[e.target.type] as 'value'];\r\n return e.target.value;\r\n }\r\n return (e?.value as T) || (isObject(unknown) ? unknown.value : unknown);\r\n};\r\n\r\nexport const _checkResult = (handlerResult: boolean | RegExp | string, value: string | number | boolean): boolean => typeof handlerResult === 'boolean' ? handlerResult : typeof value === 'string' ? typeof handlerResult === 'string' ? !new RegExp(handlerResult).test(value) : !handlerResult.test(value) : false;\r\n\r\nexport const _merge = <T extends object>(target: { [key: string]: any } = {}, ...nodes: ({ [key: string]: any } | undefined | void)[]): T => {\r\n if (!nodes.length) return target as T;\r\n\r\n const next = nodes.shift();\r\n if (isObject(next)) {\r\n for (const key in next) {\r\n target[key] = target[key] ? { ...target[key], ...next[key] } : next[key];\r\n }\r\n }\r\n\r\n return _merge(target, ...nodes);\r\n};\r\n","import {_checkResult, _checkTypeMismatch, _debounce, _extractValue, _findValidityKey} from \"./helpers\";\r\nimport {type GInputValidator, type GValidators} from \"./validations\";\r\nimport type {GInputState} from \"./fields\";\r\nimport type {GChangeEvent, GDOMElement, GFocusEvent, GFormEvent, GInvalidEvent} from \"./form\";\r\nimport type {InitialState, Store} from \"./state\";\r\nimport {handlersMap, validityMap} from \"./validations/GValidator\";\r\n\r\nexport const useFormHandlers = (getState: Store['getState'], setState: Store['setState'], validators: GValidators = {}, optimized = false) => {\r\n /**\r\n * handler for validating a form input\r\n * @param input the input to be validated\r\n * @param e the event object\r\n */\r\n const _viHandler = (input: GInputState, e?: GFocusEvent<GDOMElement | HTMLFormElement> | GInvalidEvent<GDOMElement | HTMLFormElement> | GFormEvent<GDOMElement | HTMLFormElement> | GFormEvent): void => {\r\n if (!input) return;\r\n\r\n const element = e && e.target;\r\n const hasInitialValue = !input.dirty && input.value && !input.touched;\r\n\r\n if (!element && !hasInitialValue) return;\r\n\r\n if (typeof document !== 'undefined' && (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement || element instanceof HTMLSelectElement)) {\r\n if (!input.checkValidity) input.checkValidity = () => element.checkValidity();\r\n\r\n if (hasInitialValue) { //if the field has initial value\r\n /**\r\n * for inputs with initial value we have to manually check for validations.\r\n * validity.tooShort is false even though initial value is smaller than minLength, because its required to be filled in by user (native dirty flag is true).\r\n * it only works for validity.valueMissing.\r\n * If an element has a minimum allowed value length, its dirty value flag is true, its value was last changed by a user edit (as opposed to a change made by a script), its value is not the empty string, and the length of the element's API value is less than the element's minimum allowed value length, then the element is suffering from being too short.\r\n * @see https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#setting-minimum-input-length-requirements:-the-minlength-attribute\r\n */\r\n _checkInputManually(input);\r\n _dispatchChanges(input, input.formKey);\r\n return;\r\n }\r\n element.setCustomValidity(''); //reset any previous error (custom)\r\n\r\n const exclude: (keyof ValidityState)[] = input.type && (input.pattern || hasCustomValidation(input)) ? ['typeMismatch'] : [];\r\n const validityKey = _findValidityKey(element.validity, exclude);\r\n _validateInput(input, validityKey, (v: string) => element.setCustomValidity(v));\r\n\r\n if (!validityKey && input.error) {\r\n element.setCustomValidity(input.errorText || 'error');\r\n }\r\n\r\n _dispatchChanges(input, input.formKey);\r\n } else {\r\n if (__DEBUG__) {\r\n console.log('[validateInputHandler] -', `the input '${input.formKey}' is not a native web element\\nevent:`, e);\r\n }\r\n\r\n //fallback - validate the input for validations manually\r\n input.checkValidity = () => _checkInputManually(input);\r\n input.checkValidity();\r\n\r\n _dispatchChanges(input, input.formKey);\r\n }\r\n };\r\n\r\n const _checkInputManually = (input: GInputState) => {\r\n const exclude: (keyof ValidityState)[] = input.type && (input.pattern || hasCustomValidation(input)) ? ['typeMismatch'] : [];\r\n\r\n let validityKey = _findValidityKey({\r\n valueMissing: input.required && !input.value || false,\r\n typeMismatch: _checkTypeMismatch(input),\r\n tooShort: input.minLength && input.value.toString().length < input.minLength || false,\r\n tooLong: input.maxLength && input.value.toString().length > input.maxLength || false,\r\n patternMismatch: input.pattern && _checkResult(input.pattern, input.value) || false,\r\n rangeUnderflow: input.min && Number(input.value) < Number(input.min) || false,\r\n rangeOverflow: input.max && Number(input.value) > Number(input.max) || false\r\n }, exclude);\r\n\r\n if (!validityKey && input.error) {\r\n validityKey = 'customError';\r\n }\r\n _validateInput(input, validityKey);\r\n return !input.error;\r\n };\r\n\r\n /**\r\n * handler for updating and validating a form input\r\n * @param input\r\n * @param e the event object\r\n * @param unknown\r\n */\r\n const _updateInputHandler = (input: GInputState, e?: GChangeEvent<GDOMElement | HTMLFormElement>, unknown?: {\r\n value: unknown\r\n } | string | number): void => {\r\n input.value = _extractValue(e, unknown) as GInputState['value'];\r\n _viHandler(input, e);\r\n };\r\n\r\n /**\r\n * Validates the input and updates the state with the result\r\n * @param input the input to be validated\r\n * @param validityKey the `Constraint Validation` key\r\n * @param setValidity\r\n */\r\n const _validateInput = (input: GInputState, validityKey?: keyof ValidityState, setValidity?: (e: string) => void): void => {\r\n const inputValidator = validators[input.validatorKey || input.formKey] || validators['*'];\r\n if (__DEBUG__) {\r\n console.log('[validateInput] -', 'validating input:', input.formKey, `(${validityKey ? validityKey : 'custom'})`);\r\n }\r\n\r\n if (__DEV__) {\r\n if (validityKey && !inputValidator?.hasConstraint(validityKey)) {\r\n if (validityKey === 'typeMismatch') console.warn(`DEV ONLY - [Missing Validator] - the input '${input.formKey}' has described the constraint '${validityMap[validityKey]}' however, a correspond validator / custom validation / pattern validator are missing.\\nadd '${handlersMap[validityMap[validityKey]]}' or 'withCustomValidation' or '${handlersMap[validityMap.patternMismatch]}' to the input validator.\\nexample:\\nconst validators: GValidators = {\\n\\temail: new GValidator().withPatternMismatchMessage('pattern mismatch'),\\n\\t...\\n}\\n\\nor either remove the constraint '${validityMap[validityKey]}' from the input props`);\r\n else console.warn(`DEV ONLY - [Missing Validator] - the input '${input.formKey}' has described the constraint '${validityMap[validityKey]}' however, a correspond validator is missing.\\nadd '${handlersMap[validityMap[validityKey]]}' to the input validator.\\nexample:\\nconst validators: GValidators = {\\n\\temail: new GValidator().withPatternMismatchMessage('pattern mismatch'),\\n\\t...\\n}\\n\\nor either remove the constraint '${validityMap[validityKey]}' from the input props`);\r\n }\r\n }\r\n\r\n if (inputValidator) {\r\n __validateInput(input, inputValidator, validityKey, setValidity);\r\n }\r\n input.touched = true;\r\n };\r\n\r\n const _dispatchChanges = (changes: Partial<InitialState> | Partial<GInputState>, key?: string) => setState(prev => {\r\n if (key) {\r\n return {...prev, fields: {...prev.fields, [key]: {...prev.fields[key], ...changes}}};\r\n }\r\n return {...prev, ...changes};\r\n });\r\n\r\n /**\r\n * @internal\r\n */\r\n const __validateInput = (input: GInputState, inputValidator: GInputValidator<any>, validityKey?: keyof ValidityState, setValidity?: (e: string) => void): void => {\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validating input (${input.formKey}) with handlers:`, inputValidator.handlers);\r\n }\r\n const fields = getState().fields;\r\n\r\n for (const index in inputValidator.constraintHandlers) {\r\n const result = inputValidator.constraintHandlers[index](input, validityKey);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for constraint handler (${index}):\\n`, inputValidator.constraintHandlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) return;\r\n }\r\n\r\n for (const index in inputValidator.handlers) {\r\n const result = inputValidator.handlers[index](input, fields);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for custom handler (${index}):\\n`, inputValidator.handlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) return;\r\n }\r\n\r\n input.errorText = '';\r\n\r\n if (inputValidator.asyncHandlers.length) {\r\n input.error = true;\r\n _debounce(input.debounce || 300, `${input.gid}-async`).then(() => {\r\n const validateAsync = async () => {\r\n for (const index in inputValidator.asyncHandlers) {\r\n const result = await inputValidator.asyncHandlers[index](input, fields);\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validation results for custom async handler (${index}):\\n`, inputValidator.asyncHandlers[index], '\\n\\nvalidator result:', result, '\\nviolation:', input.error, `(${input.error ? 'failed' : 'passed'})`);\r\n }\r\n\r\n input.error = _checkResult(result, input.value);\r\n if (input.error) break;\r\n }\r\n if (!input.error) input.errorText = '';\r\n\r\n _dispatchChanges({error: input.error, errorText: input.errorText}, input.formKey);\r\n if (setValidity) {\r\n setValidity(input.errorText);\r\n }\r\n };\r\n\r\n if (__DEBUG__) {\r\n console.log('[_validateInput] -', `validating input (${input.formKey}) with async handlers:`, inputValidator.asyncHandlers);\r\n }\r\n validateAsync();\r\n });\r\n }\r\n };\r\n\r\n const hasCustomValidation = (input: GInputState) => {\r\n const validator = validators[input.validatorKey || input.formKey] || validators['*'];\r\n return validator && (validator.asyncHandlers.length > 0 || validator.handlers.length > 0);\r\n };\r\n\r\n return {_updateInputHandler, _viHandler, _dispatchChanges, optimized, _createInputChecker: _checkInputManually};\r\n};","import React, {createContext, useCallback, useContext, useEffect, useRef, useSyncExternalStore} from 'react';\r\nimport type {FC, PropsWithChildren} from 'react';\r\n\r\nimport {useFormHandlers} from \"./useFormHandlers\";\r\nimport type {InitialState, Store} from \"./state\";\r\nimport type {GValidators} from \"./validations\";\r\nimport type {GInputState} from './fields';\r\n\r\nconst GFormContext = createContext<Store>({} as Store);\r\n\r\ntype GFormContextProviderProps = PropsWithChildren & {\r\n initialState: InitialState;\r\n validators?: GValidators;\r\n optimized?: boolean;\r\n}\r\n\r\nexport const GFormContextProvider: FC<GFormContextProviderProps> = ({ children, initialState, validators, optimized }) => {\r\n const stateRef = useRef(initialState);\r\n const listeners = useRef(new Set<() => void>());\r\n\r\n const setState = useCallback((updater: InitialState | ((state: InitialState) => InitialState)) => {\r\n stateRef.current = typeof updater === 'function' ? updater(stateRef.current) : updater;\r\n listeners.current.forEach((l) => l());\r\n }, []);\r\n\r\n const handlers = useFormHandlers(() => stateRef.current, setState, validators, optimized);\r\n\r\n const getState = useCallback(() => stateRef.current, []);\r\n\r\n const subscribe = useCallback((listener: () => void) => {\r\n listeners.current.add(listener);\r\n return () => listeners.current.delete(listener);\r\n }, []);\r\n\r\n useEffect(() => {\r\n for (const fieldKey in initialState.fields) {\r\n initialState.fields[fieldKey].dispatchChanges = (changes: Partial<GInputState>) => handlers._dispatchChanges(changes, fieldKey);\r\n }\r\n }, []);\r\n\r\n const store = useRef<Store>({ getState, setState, subscribe, handlers });\r\n\r\n return <GFormContext.Provider value={store.current}>{children}</GFormContext.Provider>;\r\n};\r\n\r\nexport const useFormStore = () => {\r\n const store = useContext(GFormContext);\r\n if (!store) throw new Error('useGFormStore must be used within `GForm` component');\r\n return store;\r\n};\r\n\r\nexport const useFormSelector = <T extends any>(selector: (state: InitialState) => T): T => {\r\n const store = useFormStore();\r\n\r\n return useSyncExternalStore(\r\n store.subscribe,\r\n () => selector(store.getState()),\r\n () => selector(store.getState()) // for SSR\r\n );\r\n};\r\n\r\nexport function createSelector<\r\n State=InitialState,\r\n Selectors extends Array<(state: State) => any> = [],\r\n Result = any\r\n>(\r\n selectors: Selectors,\r\n combiner: (...args: {\r\n [K in keyof Selectors]: Selectors[K] extends (state: State) => infer R ? R : never;\r\n }) => Result\r\n): (state: State) => Result {\r\n let lastArgs: any[] = [];\r\n let lastResult: Result;\r\n\r\n return (state: State): Result => {\r\n const args = selectors.map(fn => fn(state));\r\n if (\r\n lastArgs.length === args.length &&\r\n args.every((val, i) => val === lastArgs[i])\r\n ) {\r\n return lastResult;\r\n }\r\n lastArgs = args;\r\n lastResult = combiner(...args as any);\r\n return lastResult;\r\n };\r\n}","import {createSelector} from \"./form-context\";\r\nimport {InitialState} from \"./state\";\r\n\r\nconst selectFields = [(state: InitialState) => state.fields];\r\n\r\nexport const selectFirstInvalidField =\r\n createSelector(\r\n selectFields,\r\n (fields) => {\r\n for (const f in fields) {\r\n if (fields[f].error) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n );\r\n\r\nexport const makeSelectFields = (keys: string[] = []) =>\r\n createSelector(\r\n selectFields,\r\n (fields) => {\r\n const selected = keys.map((key) => fields[key]).filter(Boolean);\r\n return selected.length ? selected : null;\r\n }\r\n );","import React, {useMemo, useEffect, forwardRef, useCallback} from \"react\";\r\nimport type { ReactNode, FC, ReactElement, Ref, RefObject } from \"react\";\r\n\r\nimport {_toRawData, _toURLSearchParams, _merge, _buildFormInitialValues} from \"./helpers\";\r\nimport {GFormContextProvider, useFormSelector, useFormStore} from \"./form-context\";\r\nimport type {RNGFormState, ToRawDataOptions} from \"./state\";\r\nimport type { GValidators } from \"./validations\";\r\nimport type { IForm, PartialForm } from \"./form\";\r\nimport type { GInputState } from \"./fields\";\r\nimport {selectFirstInvalidField} from \"./selectors\";\r\n\r\nconst FormRenderer = forwardRef<any, RNGFormProps<any>>(\r\n <T, >({\r\n stateRef,\r\n children,\r\n onInit,\r\n el: El,\r\n ...rest\r\n }: RNGFormProps<T>, ref: React.Ref<any>) => {\r\n const {getState, handlers} = useFormStore();\r\n const isFormInvalid = useFormSelector(selectFirstInvalidField);\r\n\r\n const getFormState = useCallback(() => {\r\n const fields = getState<T>().fields;\r\n\r\n const formState: RNGFormState<T> = {\r\n ...fields,\r\n isValid: !isFormInvalid,\r\n isInvalid: isFormInvalid,\r\n toRawData: (options?: ToRawDataOptions<T>) => _toRawData(fields, options),\r\n toURLSearchParams: _toURLSearchParams,\r\n checkValidity: () => {\r\n for (const i in fields) {\r\n const valid = fields[i].checkValidity();\r\n if (!valid) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n },\r\n dispatchChanges: (changes: PartialForm<T> & {\r\n [key: string]: Partial<GInputState<any>>\r\n }) => handlers._dispatchChanges({\r\n fields: _merge<IForm<T> & {\r\n [key: string]: GInputState;\r\n }>({}, fields, changes)\r\n })\r\n };\r\n\r\n if (stateRef) stateRef.current = formState;\r\n\r\n return formState;\r\n }, [isFormInvalid]);\r\n\r\n const formComponent = useMemo(() => {\r\n const state = getFormState();\r\n\r\n const formChildren = typeof children === 'function' ? children(state) : children;\r\n\r\n return <El {...rest} ref={ref}>\r\n {formChildren}\r\n </El>;\r\n }, [getFormState, children]);\r\n\r\n useEffect(() => {\r\n const state = getFormState();\r\n\r\n if (onInit) {\r\n const changes = onInit(state);\r\n if (changes) {\r\n const _handler = (_c: void | PartialForm<T>) => handlers._dispatchChanges({\r\n fields: _merge<IForm<T> & {\r\n [key: string]: GInputState;\r\n }>({}, state, _c)\r\n });\r\n if (changes instanceof Promise) {\r\n changes.then(_handler);\r\n } else _handler(changes);\r\n }\r\n }\r\n\r\n const dispatchers: Record<string, Record<string, (changes: Partial<GInputState>) => void>> = {};\r\n \r\n if (__DEBUG__) {\r\n console.log('checking for initial values');\r\n }\r\n const fields = getState().fields;\r\n\r\n for (const fieldKey in fields) {\r\n dispatchers[fieldKey] = {\r\n dispatchChanges: (changes: Partial<GInputState>) => handlers._dispatchChanges(changes, fieldKey),\r\n checkValidity: () => {\r\n const result = handlers._createInputChecker(state[fieldKey]);\r\n handlers._dispatchChanges(state[fieldKey], fieldKey);\r\n return result;\r\n }\r\n };\r\n\r\n const field = fields[fieldKey];\r\n\r\n //we don't want to apply validation on empty fields so skip it.\r\n if (!field.value) continue;\r\n\r\n if (__DEBUG__) {\r\n console.log(`found input '${fieldKey}', applying validation(s)`);\r\n }\r\n /**\r\n * We have to manually check for validations (checkValidity() will not result with validity.tooShort = true).\r\n * If an element has a minimum allowed value length, its dirty value flag is true, its value was last changed by a user edit (as opposed to a change made by a script), its value is not the empty string, and the length of the element's API value is less than the element's minimum allowed value length, then the element is suffering from being too short.\r\n * @see https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#setting-minimum-input-length-requirements:-the-minlength-attribute\r\n */\r\n handlers._viHandler(field);\r\n }\r\n handlers._dispatchChanges({fields: _merge(dispatchers, state)});\r\n }, [getFormState]);\r\n\r\n return formComponent;\r\n }\r\n) as <T>(props: RNGFormProps<T> & { ref?: Ref<any> }) => ReactElement | null;\r\n\r\nexport type RNGFormProps<T> = {\r\n children?: ReactNode | ReactNode[] | ((state: RNGFormState<T>) => ReactNode | ReactNode[]);\r\n /** @param stateRef - pass a ref which will points to the current state of the form (optional). */\r\n stateRef?: RefObject<RNGFormState<T> | undefined>;\r\n /** @param validators - an object for handling validations (optional). */\r\n validators?: GValidators<T>;\r\n /** @param onInit - execute a handler once the form has initialized (optional). */\r\n onInit?: (state: RNGFormState<T>) => void | PartialForm<T> | Promise<void | PartialForm<T>>;\r\n /** @param el - the element to use as the form container. */\r\n el: FC<any>;\r\n};\r\n\r\n/**\r\n * build dynamic forms with validations.\r\n * @link Docs - https://gform-react.onrender.com\r\n * @link Npm - https://www.npmjs.com/package/gform-react\r\n */\r\nexport const RNGForm = forwardRef<any, RNGFormProps<any>>(\r\n <T, >({children, validators, ...props}: RNGFormProps<T>, ref: React.Ref<any>) => {\r\n const initialState = useMemo(() => {\r\n return _buildFormInitialValues(\r\n typeof children === 'function'\r\n ? children({} as RNGFormState<T>)\r\n : children\r\n );\r\n }, [children]);\r\n\r\n return (\r\n <GFormContextProvider key={initialState.key} initialState={initialState} validators={validators}>\r\n <FormRenderer ref={ref} {...props}>\r\n {children}\r\n </FormRenderer>\r\n </GFormContextProvider>\r\n );\r\n }\r\n) as <T>(props: RNGFormProps<T> & { ref?: React.Ref<HTMLFormElement> }) => React.ReactElement | null;","import React, {forwardRef, memo, type ReactNode, useEffect, useMemo} from 'react';\r\nimport {TextInput} from 'react-native';\r\n\r\nimport {_debounce} from '../helpers';\r\nimport type {GInputState, RNGInputProps} from '.';\r\nimport {useFormSelector, useFormStore} from \"../form-context\";\r\nimport {makeSelectFields} from \"../selectors\";\r\n\r\nconst _RNGInput = forwardRef<any, RNGInputProps>(({\r\n formKey,\r\n element,\r\n type,\r\n fetch,\r\n fetchDeps,\r\n debounce = 300,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n defaultValue,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n validatorKey,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n value,\r\n ...rest\r\n}, ref) => {\r\n const inputState = useFormSelector(state => state.fields[formKey]);\r\n const store = useFormStore();\r\n\r\n const _element = useMemo(() => {\r\n const value = inputState.value || '';\r\n\r\n const _props = {\r\n ...rest,\r\n value,\r\n inputMode: type,\r\n ref\r\n };\r\n\r\n _props.onEndEditing = rest.onEndEditing ?\r\n (e) => {\r\n store.handlers._viHandler(inputState);\r\n rest.onEndEditing!(e);\r\n }\r\n :\r\n () => {\r\n store.handlers._viHandler(inputState);\r\n };\r\n _props.onChangeText = rest.onChangeText ?\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n rest.onChangeText!(e);\r\n }\r\n :\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n };\r\n\r\n if (!inputState.touched) {\r\n _props.onFocus = rest.onFocus ?\r\n (e) => {\r\n rest.onFocus!(e);\r\n inputState.dispatchChanges({touched: true});\r\n }\r\n :\r\n () => {\r\n inputState.dispatchChanges({touched: true});\r\n };\r\n }\r\n\r\n if (element) {\r\n return (element as (input: GInputState, props: any) => ReactNode)(inputState, _props);\r\n }\r\n\r\n return <TextInput {..._props}/>;\r\n }, [inputState, element]);\r\n\r\n const _fetchDeps = useFormSelector(makeSelectFields(fetchDeps));\r\n const stableFetchDeps = useMemo(() => JSON.stringify(_fetchDeps), [_fetchDeps]);\r\n\r\n useEffect(() => {\r\n if (fetch) {\r\n _debounce(debounce, `${inputState.gid}-fetch`).then(() => {\r\n const res = fetch(inputState, store.getState().fields);\r\n if (res instanceof Promise) {\r\n res.then((state) => state && store.handlers._dispatchChanges(state, formKey));\r\n } else if (res) {\r\n store.handlers._dispatchChanges(res, formKey);\r\n }\r\n });\r\n }\r\n }, [stableFetchDeps]);\r\n\r\n return _element;\r\n});\r\n\r\nexport const RNGInput = memo(_RNGInput);"],"names":["isObject","o","Array","isArray","defaultFieldProps","text","value","checkbox","number","typeValueDict","generateId","Date","toString","Math","random","substring","_findInputs","root","total","_root$props","_root$props2","forEach","element","props","formKey","push","children","_findValidityKey","validity","exclude","key","includes","_checkTypeMismatch","input","_input$value","trim","type","test","URL","_toURLSearchParams","options","data","include","transform","_this$key","this","toRawData","_this$key2","set","URLSearchParams","_debounce","timeout","id","Promise","resolve","_this$id","timerId","clearTimeout","setTimeout","bind","_checkResult","handlerResult","RegExp","_merge","target","nodes","length","next","shift","useFormHandlers","getState","setState","validators","optimized","_viHandler","e","hasInitialValue","dirty","touched","document","HTMLInputElement","HTMLTextAreaElement","HTMLSelectElement","checkValidity","_checkInputManually","_dispatchChanges","setCustomValidity","pattern","hasCustomValidation","validityKey","_validateInput","v","error","errorText","valueMissing","required","typeMismatch","tooShort","minLength","tooLong","maxLength","patternMismatch","rangeUnderflow","min","Number","rangeOverflow","max","setValidity","inputValidator","validatorKey","__validateInput","changes","prev","fields","index","constraintHandlers","result","handlers","asyncHandlers","debounce","gid","then","async","validateAsync","validator","_updateInputHandler","unknown","_extractValue","Object","hasOwn","_createInputChecker","GFormContext","createContext","GFormContextProvider","initialState","stateRef","useRef","listeners","Set","useCallback","updater","current","l","subscribe","listener","add","delete","useEffect","fieldKey","dispatchChanges","store","React","createElement","Provider","useFormStore","useContext","Error","useFormSelector","selector","useSyncExternalStore","createSelector","selectors","combiner","lastResult","lastArgs","state","args","map","fn","every","val","i","selectFields","selectFirstInvalidField","f","FormRenderer","forwardRef","onInit","el","El","rest","ref","isFormInvalid","getFormState","formState","isValid","isInvalid","_toRawData","_fields$key","_fields$key2","toURLSearchParams","formComponent","useMemo","formChildren","_extends","_handler","_c","dispatchers","field","RNGForm","_buildFormInitialValues","rows","row","config","step","defaultValue","checked","defaultChecked","inputValue","keys","_RNGInput","fetch","fetchDeps","inputState","_element","_props","inputMode","onEndEditing","onChangeText","undefined","onFocus","TextInput","_fetchDeps","makeSelectFields","selected","filter","Boolean","stableFetchDeps","JSON","stringify","res","RNGInput","memo"],"mappings":"0GAKO,MAAMA,EAAYC,GAAyBA,GAAkB,iBAANA,IAAmBC,MAAMC,QAAQF,GAEzFG,EAA6E,CAC/EC,KAAM,CAAEC,MAAO,IACfC,SAAU,CAAED,OAAO,GACnBE,OAAQ,CAAEF,MAAO,IAGfG,EAAwE,CAC1EF,SAAU,UACVC,OAAQ,iBAGNE,EAAaA,MAAQ,IAAIC,MAAQC,SAAS,KAAO,EAAIC,KAAKC,UAAUF,SAAS,IAAIG,UAAU,EAAG,IAsD9FC,EAAcA,CAACC,EAA8DC,EAA8C,MAA6C,IAAAC,EAAAC,EAC1K,OAAKH,EAEDf,MAAMC,QAAQc,IACdA,EAAKI,SAAQC,GAAWN,EAAYM,EAASJ,KACtCA,GAIG,QAAdC,EAAIF,EAAKM,aAAK,IAAAJ,GAAVA,EAAYK,SAIZN,EAAMO,KAAKR,EAAKM,OACTL,GAGJF,EAAsB,QAAXI,EAACH,EAAKM,iBAAKH,SAAVA,EAAYM,SAAUR,GAhBvBA,CAgB6B,EAGtCS,EAAmBA,CAACC,EAAkCC,EAAmC,MAClG,IAAK,MAAMC,KAAOF,EACd,IAAIC,EAAQE,SAASD,IACT,UAARA,GAAmBF,EAASE,GAI5B,OAAOA,CAEf,EAGSE,EAAsBC,IAA4B,IAAAC,EAC3D,MAAM5B,EAAmB,QAAd4B,EAAGD,EAAM3B,aAAK,IAAA4B,OAAA,EAAXA,EAAatB,WAAWuB,OACtC,IAAK7B,EAAO,OAAO,EAEnB,OAAQ2B,EAAMG,MACV,IAAK,QACD,OAAQ,6BAA6BC,KAAK/B,GAC9C,IAAK,MACD,IAEI,OADA,IAAIgC,IAAIhC,IACD,CACX,CAAE,MACE,OAAO,CACX,CACJ,IAAK,MACD,OAAQ,wBAAwB+B,KAAK/B,GACzC,QACI,OAAO,EACf,EA0EG,SAASiC,EAA2CC,GACvD,IAAIC,EACJ,GAAID,EAAS,CACT,MAAMX,QAAEA,EAAOa,QAAEA,EAAOC,UAAEA,GAAcH,EASxC,GARIE,GACAD,EAAO,CAAA,EACPC,EAAQrB,SAAQS,IAAG,IAAAc,EAAA,OAAKH,EAAKX,GAAgB,QAAZc,EAAGC,KAAKf,UAAI,IAAAc,OAAA,EAATA,EAAWtC,KAAK,MAEpDmC,EAAOI,KAAKC,YACZjB,SAAAA,EAASR,SAAQS,UAAcW,EAAKX,MAGpCa,EACA,IAAK,MAAMb,KAAOa,EAAW,CAAA,IAAAI,EACzB,MAAMC,EAAML,EAAUb,GACrBW,EAAKX,GAAOkB,EAAa,QAAVD,EAACF,KAAKf,cAAIiB,SAATA,EAAWzC,MAChC,CAER,MACKmC,EAAOI,KAAKC,YAEjB,OAAO,IAAIG,gBAAgBR,EAC/B,CAeO,MAAMS,EAbb,SAA0EC,EAAiBC,GACvF,OAAO,IAAIC,SAAQC,IAAW,IAAAC,EACd,QAAZA,EAAIV,KAAKO,UAAG,IAAAG,GAARA,EAAUC,SACVC,aAAaZ,KAAKO,GAAII,SAE1B,MAAMA,EAAUE,YAAW,IAAMJ,KAAWH,GAExCN,KAAKO,GACLP,KAAKO,GAAII,QAAUA,EAChBX,KAAKO,GAAM,CAAEI,UAAS,GAErC,EAEoCG,KAAK,IAU5BC,EAAeA,CAACC,EAA0CvD,IAAuE,kBAAlBuD,EAA8BA,EAAiC,iBAAVvD,IAA8C,iBAAlBuD,GAA8B,IAAIC,OAAOD,GAAexB,KAAK/B,IAAUuD,EAAcxB,KAAK/B,IAE1RyD,EAASA,CAAmBC,EAAiC,CAAA,KAAOC,KAC7E,IAAKA,EAAMC,OAAQ,OAAOF,EAE1B,MAAMG,EAAOF,EAAMG,QACnB,GAAIpE,EAASmE,GACT,IAAK,MAAMrC,KAAOqC,EACdH,EAAOlC,GAAOkC,EAAOlC,GAAO,IAAKkC,EAAOlC,MAASqC,EAAKrC,IAASqC,EAAKrC,GAI5E,OAAOiC,EAAOC,KAAWC,EAAM,ECxPtBI,EAAkBA,CAACC,EAA6BC,EAA6BC,EAA0B,CAAA,EAAIC,GAAY,KAMhI,MAAMC,EAAaA,CAACzC,EAAoB0C,KACpC,IAAK1C,EAAO,OAEZ,MAAMX,EAAUqD,GAAKA,EAAEX,OACjBY,GAAmB3C,EAAM4C,OAAS5C,EAAM3B,QAAU2B,EAAM6C,QAE9D,GAAKxD,GAAYsD,EAEjB,GAAwB,oBAAbG,WAA6BzD,aAAmB0D,kBAAoB1D,aAAmB2D,qBAAuB3D,aAAmB4D,mBAAoB,CAG5J,GAFKjD,EAAMkD,gBAAelD,EAAMkD,cAAgB,IAAM7D,EAAQ6D,iBAE1DP,EAUA,OAFAQ,EAAoBnD,QACpBoD,EAAiBpD,EAAOA,EAAMT,SAGlCF,EAAQgE,kBAAkB,IAE1B,MAAMzD,EAAmCI,EAAMG,OAASH,EAAMsD,SAAWC,EAAoBvD,IAAW,CAAC,gBAAkB,GACrHwD,EAAc9D,EAAiBL,EAAQM,SAAUC,GACvD6D,EAAezD,EAAOwD,GAAcE,GAAcrE,EAAQgE,kBAAkBK,MAEvEF,GAAexD,EAAM2D,OACtBtE,EAAQgE,kBAAkBrD,EAAM4D,WAAa,SAGjDR,EAAiBpD,EAAOA,EAAMT,QAClC,MAMIS,EAAMkD,cAAgB,IAAMC,EAAoBnD,GAChDA,EAAMkD,gBAENE,EAAiBpD,EAAOA,EAAMT,QAClC,EAGE4D,EAAuBnD,IACzB,MAAMJ,EAAmCI,EAAMG,OAASH,EAAMsD,SAAWC,EAAoBvD,IAAW,CAAC,gBAAkB,GAE3H,IAAIwD,EAAc9D,EAAiB,CAC/BmE,aAAc7D,EAAM8D,WAAa9D,EAAM3B,QAAS,EAChD0F,aAAchE,EAAmBC,GACjCgE,SAAUhE,EAAMiE,WAAajE,EAAM3B,MAAMM,WAAWsD,OAASjC,EAAMiE,YAAa,EAChFC,QAASlE,EAAMmE,WAAanE,EAAM3B,MAAMM,WAAWsD,OAASjC,EAAMmE,YAAa,EAC/EC,gBAAiBpE,EAAMsD,SAAW3B,EAAa3B,EAAMsD,QAAStD,EAAM3B,SAAU,EAC9EgG,eAAgBrE,EAAMsE,KAAOC,OAAOvE,EAAM3B,OAASkG,OAAOvE,EAAMsE,OAAQ,EACxEE,cAAexE,EAAMyE,KAAOF,OAAOvE,EAAM3B,OAASkG,OAAOvE,EAAMyE,OAAQ,GACxE7E,GAMH,OAJK4D,GAAexD,EAAM2D,QACtBH,EAAc,eAElBC,EAAezD,EAAOwD,IACdxD,EAAM2D,KAAK,EAsBjBF,EAAiBA,CAACzD,EAAoBwD,EAAmCkB,KAC3E,MAAMC,EAAiBpC,EAAWvC,EAAM4E,cAAgB5E,EAAMT,UAAYgD,EAAW,KAYjFoC,GACAE,EAAgB7E,EAAO2E,EAAgBnB,EAAakB,GAExD1E,EAAM6C,SAAU,CAAI,EAGlBO,EAAmBA,CAAC0B,EAAuDjF,IAAiByC,GAASyC,GACnGlF,EACO,IAAIkF,EAAMC,OAAQ,IAAID,EAAKC,OAAQnF,CAACA,GAAM,IAAIkF,EAAKC,OAAOnF,MAASiF,KAEvE,IAAIC,KAASD,KAMlBD,EAAkBA,CAAC7E,EAAoB2E,EAAsCnB,EAAmCkB,KAIlH,MAAMM,EAAS3C,IAAW2C,OAE1B,IAAK,MAAMC,KAASN,EAAeO,mBAAoB,CACnD,MAAMC,EAASR,EAAeO,mBAAmBD,GAAOjF,EAAOwD,GAM/D,GADAxD,EAAM2D,MAAQhC,EAAawD,EAAQnF,EAAM3B,OACrC2B,EAAM2D,MAAO,MACrB,CAEA,IAAK,MAAMsB,KAASN,EAAeS,SAAU,CACzC,MAAMD,EAASR,EAAeS,SAASH,GAAOjF,EAAOgF,GAMrD,GADAhF,EAAM2D,MAAQhC,EAAawD,EAAQnF,EAAM3B,OACrC2B,EAAM2D,MAAO,MACrB,CAEA3D,EAAM4D,UAAY,GAEde,EAAeU,cAAcpD,SAC7BjC,EAAM2D,OAAQ,EACd1C,EAAUjB,EAAMsF,UAAY,IAAK,GAAGtF,EAAMuF,aAAaC,MAAK,KAClCC,WAClB,IAAK,MAAMR,KAASN,EAAeU,cAAe,CAC9C,MAAMF,QAAeR,EAAeU,cAAcJ,GAAOjF,EAAOgF,GAMhE,GADAhF,EAAM2D,MAAQhC,EAAawD,EAAQnF,EAAM3B,OACrC2B,EAAM2D,MAAO,KACrB,CACK3D,EAAM2D,QAAO3D,EAAM4D,UAAY,IAEpCR,EAAiB,CAACO,MAAO3D,EAAM2D,MAAOC,UAAW5D,EAAM4D,WAAY5D,EAAMT,SACrEmF,GACAA,EAAY1E,EAAM4D,UACtB,EAMJ8B,EAAe,IAEvB,EAGEnC,EAAuBvD,IACzB,MAAM2F,EAAYpD,EAAWvC,EAAM4E,cAAgB5E,EAAMT,UAAYgD,EAAW,KAChF,OAAOoD,IAAcA,EAAUN,cAAcpD,OAAS,GAAK0D,EAAUP,SAASnD,OAAS,EAAE,EAG7F,MAAO,CAAC2D,EAxGoBA,CAAC5F,EAAoB0C,EAAiDmD,KAG9F7F,EAAM3B,MDkJeyH,EAAIpD,EAAiDmD,IAC1EnD,SAAAA,EAAGX,OACCgE,OAAOC,OAAOxH,EAAekE,EAAEX,OAAO5B,MAAcuC,EAAEX,OAAOvD,EAAckE,EAAEX,OAAO5B,OACjFuC,EAAEX,OAAO1D,OAEZqE,aAAC,EAADA,EAAGrE,SAAgBN,EAAS8H,GAAWA,EAAQxH,MAAQwH,GCvJ7CC,CAAcpD,EAAGmD,GAC/BpD,EAAWzC,EAAO0C,EAAE,EAoGKD,IAAYW,IAAkBZ,YAAWyD,EAAqB9C,EAAoB,ECtL7G+C,EAAeC,EAAAA,cAAqB,IAQ7BC,EAAsDA,EAAG3G,WAAU4G,eAAc9D,aAAYC,gBACtG,MAAM8D,EAAWC,EAAAA,OAAOF,GAClBG,EAAYD,EAAAA,OAAO,IAAIE,KAEvBnE,EAAWoE,EAAAA,aAAaC,IAC1BL,EAASM,QAA6B,mBAAZD,EAAyBA,EAAQL,EAASM,SAAWD,EAC/EH,EAAUI,QAAQxH,SAASyH,GAAMA,KAAI,GACtC,IAEGzB,EAAWhD,GAAgB,IAAMkE,EAASM,SAAStE,EAAUC,EAAYC,GAEzEH,EAAWqE,EAAAA,aAAY,IAAMJ,EAASM,SAAS,IAE/CE,EAAYJ,EAAAA,aAAaK,IAC3BP,EAAUI,QAAQI,IAAID,GACf,IAAMP,EAAUI,QAAQK,OAAOF,KACvC,IAEHG,EAAAA,WAAU,KACN,IAAK,MAAMC,KAAYd,EAAarB,OAChCqB,EAAarB,OAAOmC,GAAUC,gBAAmBtC,GAAkCM,EAAShC,EAAiB0B,EAASqC,EAC1H,GACD,IAEH,MAAME,EAAQd,EAAAA,OAAc,CAAElE,WAAUC,WAAUwE,YAAW1B,aAE7D,OAAOkC,EAAAC,cAACrB,EAAasB,SAAQ,CAACnJ,MAAOgJ,EAAMT,SAAUnH,EAAiC,EAG7EgI,EAAeA,KACxB,MAAMJ,EAAQK,EAAAA,WAAWxB,GACzB,IAAKmB,EAAO,MAAM,IAAIM,MAAM,uDAC5B,OAAON,CAAK,EAGHO,EAAkCC,IAC3C,MAAMR,EAAQI,IAEd,OAAOK,EAAAA,qBACHT,EAAMP,WACN,IAAMe,EAASR,EAAMhF,cACrB,IAAMwF,EAASR,EAAMhF,aACxB,EAGE,SAAS0F,EAKZC,EACAC,GAIA,IACIC,EADAC,EAAkB,GAGtB,OAAQC,IACJ,MAAMC,EAAOL,EAAUM,KAAIC,GAAMA,EAAGH,KACpC,OACID,EAASlG,SAAWoG,EAAKpG,QACzBoG,EAAKG,OAAM,CAACC,EAAKC,IAAMD,IAAQN,EAASO,OAI5CP,EAAWE,EACXH,EAAaD,KAAYI,IAHdH,CAIM,CAEzB,CCnFA,MAAMS,EAAe,CAAEP,GAAwBA,EAAMpD,QAExC4D,EACTb,EACIY,GACC3D,IACG,IAAK,MAAM6D,KAAK7D,EACZ,GAAIA,EAAO6D,GAAGlF,MACV,OAAO,EAGf,OAAO,CAAK,ICHlBmF,EAAeC,EAAAA,YACjB,EACIzC,WACA7G,WACAuJ,SACAC,GAAIC,KACDC,GACaC,KAChB,MAAM/G,SAACA,EAAQ+C,SAAEA,GAAYqC,IACvB4B,EAAgBzB,EAAgBgB,GAEhCU,EAAe5C,EAAAA,aAAY,KAC7B,MAAM1B,EAAS3C,IAAc2C,OAEvBuE,EAA6B,IAC5BvE,EACHwE,SAAUH,EACVI,UAAWJ,EACXxI,UAAYN,GJoHFmJ,EAAI1E,EAAwDzE,EAA+B,MACjH,MAAMC,EAAmC,CAAA,GAEnCC,QAAEA,EAAOb,QAAEA,EAAOc,UAAEA,GAAcH,EAExC,GAAIE,EACAA,EAAQrB,SAAQS,IAAG,IAAA8J,EAAA,OAAInJ,EAAKX,GAA4B,QAAd8J,EAAG3E,EAAOnF,UAAI,IAAA8J,OAAA,EAAXA,EAAatL,KAAK,SAC5D,IAAK,MAAMwK,KAAK7D,EACnBxE,EAAKqI,GAAK7D,EAAO6D,GAAGxK,MAKxB,GAFAuB,SAAAA,EAASR,SAAQS,UAAcW,EAAKX,KAEhCa,EACA,IAAK,MAAMb,KAAOa,EAAW,CAAA,IAAAkJ,EACzB,MAAM7I,EAAML,EAAUb,GACtBW,EAAKX,GAAOkB,GAAe,QAAX6I,EAAA5E,EAAOnF,UAAI,IAAA+J,OAAA,EAAXA,EAAavL,QAAS2G,EAAOnF,GACjD,CAGJ,OAAOW,CAAI,EIxI+CkJ,CAAW1E,EAAQzE,GACjEsJ,kBAAmBvJ,EACnB4C,cAAeA,KACX,IAAK,MAAMwF,KAAK1D,EAAQ,CAEpB,IADcA,EAAO0D,GAAGxF,gBAEpB,OAAO,CAEf,CACA,OAAO,CAAI,EAEfkE,gBAAkBtC,GAEZM,EAAShC,EAAiB,CAC5B4B,OAAQlD,EAEL,GAAIkD,EAAQF,MAMvB,OAFIwB,IAAUA,EAASM,QAAU2C,GAE1BA,CAAS,GACjB,CAACF,IAEES,EAAgBC,EAAAA,SAAQ,KAC1B,MAAM3B,EAAQkB,IAERU,EAAmC,mBAAbvK,EAA0BA,EAAS2I,GAAS3I,EAExE,OAAO6H,EAAAC,cAAC2B,EAAEe,KAAKd,EAAI,CAAEC,IAAKA,IACrBY,EACA,GACN,CAACV,EAAc7J,IAsDlB,OApDAyH,EAAAA,WAAU,KACN,MAAMkB,EAAQkB,IAEd,GAAIN,EAAQ,CACR,MAAMlE,EAAUkE,EAAOZ,GACvB,GAAItD,EAAS,CACT,MAAMoF,EAAYC,GAA8B/E,EAAShC,EAAiB,CACtE4B,OAAQlD,EAEL,GAAIsG,EAAO+B,KAEdrF,aAAmB1D,QACnB0D,EAAQU,KAAK0E,GACVA,EAASpF,EACpB,CACJ,CAEA,MAAMsF,EAAuF,CAAA,EAKvFpF,EAAS3C,IAAW2C,OAE1B,IAAK,MAAMmC,KAAYnC,EAAQ,CAC3BoF,EAAYjD,GAAY,CACpBC,gBAAkBtC,GAAkCM,EAAShC,EAAiB0B,EAASqC,GACvFjE,cAAeA,KACX,MAAMiC,EAASC,EAASa,EAAoBmC,EAAMjB,IAElD,OADA/B,EAAShC,EAAiBgF,EAAMjB,GAAWA,GACpChC,CAAM,GAIrB,MAAMkF,EAAQrF,EAAOmC,GAGhBkD,EAAMhM,OAUX+G,EAAS3C,EAAW4H,EACxB,CACAjF,EAAShC,EAAiB,CAAC4B,OAAQlD,EAAOsI,EAAahC,IAAQ,GAChE,CAACkB,IAEGQ,CAAa,IAqBfQ,EAAUvB,EAAAA,YACnB,EAAOtJ,WAAU8C,gBAAejD,GAAyB8J,KACrD,MAAM/C,EAAe0D,EAAAA,SAAQ,IJvHEQ,EAAIC,EAAgC,MACvE,MAAMxF,EAAgD,CAAA,EAEjD/G,MAAMC,QAAQsM,KAAOA,EAAO,CAACA,IAMlC,IAAK,MAAMC,KAAOD,EACOzL,EAAY0L,GAEpBrL,SAAQsL,IASjB,MAAM5G,SAAEA,GAAW,EAAKW,IAAEA,EAAGN,UAAEA,EAASG,IAAEA,EAAGL,UAAEA,EAAS0G,KAAEA,EAAIrH,QAAEA,EAAOnD,KAAEA,EAAO,OAAMyK,aAAEA,EAAYvM,MAAEA,EAAKwM,QAAEA,EAAOC,eAAEA,EAAcvL,QAAEA,EAAO+F,SAAEA,EAAQV,aAAEA,GAAiB8F,EAEpKK,EAAa1M,GAASuM,GAAgBC,GAAWC,IADlC3M,EAAkBgC,IAAShC,EAAkBC,MACoBC,MAEtF2G,EAAO0F,EAAOnL,SAAW,CACrBA,UACAY,OACA2D,WACAW,MACAN,YACAG,MACAL,YACA0G,OACArH,UACAjF,MAAO0M,EACPnG,eACAU,WACA1C,OAAO,EACPC,SAAS,EACT0C,IAAK9G,KAGTsH,OAAOiF,KAAKhG,EAAO0F,EAAOnL,UAAUH,SAAQS,SACG,IAAhCmF,EAAO0F,EAAOnL,SAASM,WAA6BmF,EAAO0F,EAAOnL,SAASM,EAAI,GAC5F,IAIV,MAAO,CAAEmF,OAAQA,EAAoBnF,IAAKpB,IAAc,EIuEzC8L,CACiB,mBAAb9K,EACDA,EAAS,CAAA,GACTA,IAEX,CAACA,IAEJ,OACI6H,EAAAC,cAACnB,EAAoB,CAACvG,IAAKwG,EAAaxG,IAAKwG,aAAcA,EAAc9D,WAAYA,GACjF+E,EAAAC,cAACuB,EAAYmB,EAAA,CAACb,IAAKA,GAAS9J,GACvBG,GAEc,IChJ7BwL,EAAYlC,EAAAA,YAA+B,EAC7CxJ,UACAF,UACAc,OACA+K,QACAC,YACA7F,WAAW,IAEXsF,eAEAhG,eAEAvG,WACG8K,GACJC,KACC,MAAMgC,EAAaxD,GAAgBQ,GAASA,EAAMpD,OAAOzF,KACnD8H,EAAQI,IAER4D,EAAWtB,EAAAA,SAAQ,KACrB,MAAM1L,EAAQ+M,EAAW/M,OAAS,GAE5BiN,EAAS,IACRnC,EACH9K,QACAkN,UAAWpL,EACXiJ,OAkCJ,OA/BAkC,EAAOE,aAAerC,EAAKqC,aACtB9I,IACG2E,EAAMjC,SAAS3C,EAAW2I,GAC1BjC,EAAKqC,aAAc9I,EAAE,EAGzB,KACI2E,EAAMjC,SAAS3C,EAAW2I,EAAW,EAE7CE,EAAOG,aAAetC,EAAKsC,aACtB/I,IACG2E,EAAMjC,SAASQ,EAAoBwF,OAAYM,EAAW,CAACrN,MAAOqE,IAClEyG,EAAKsC,aAAc/I,EAAE,EAGxBA,IACG2E,EAAMjC,SAASQ,EAAoBwF,OAAYM,EAAW,CAACrN,MAAOqE,GAAG,EAGxE0I,EAAWvI,UACZyI,EAAOK,QAAUxC,EAAKwC,QACjBjJ,IACGyG,EAAKwC,QAASjJ,GACd0I,EAAWhE,gBAAgB,CAACvE,SAAS,GAAM,EAG/C,KACIuI,EAAWhE,gBAAgB,CAACvE,SAAS,GAAM,GAInDxD,EACQA,EAA0D+L,EAAYE,GAG3EhE,EAAAC,cAACqE,EAAAA,UAAcN,EAAS,GAChC,CAACF,EAAY/L,IAEVwM,EAAajE,EFxDSkE,EAACd,EAAiB,KAC9CjD,EACIY,GACC3D,IACG,MAAM+G,EAAWf,EAAK1C,KAAKzI,GAAQmF,EAAOnF,KAAMmM,OAAOC,SACvD,OAAOF,EAAS9J,OAAS8J,EAAW,IAAI,IEmDbD,CAAiBX,IAC9Ce,EAAkBnC,EAAAA,SAAQ,IAAMoC,KAAKC,UAAUP,IAAa,CAACA,IAenE,OAbA3E,EAAAA,WAAU,KACFgE,GACAjK,EAAUqE,EAAU,GAAG8F,EAAW7F,aAAaC,MAAK,KAChD,MAAM6G,EAAMnB,EAAME,EAAY/D,EAAMhF,WAAW2C,QAC3CqH,aAAejL,QACfiL,EAAI7G,MAAM4C,GAAUA,GAASf,EAAMjC,SAAShC,EAAiBgF,EAAO7I,KAC7D8M,GACPhF,EAAMjC,SAAShC,EAAiBiJ,EAAK9M,EACzC,GAER,GACD,CAAC2M,IAEGb,CAAQ,IAGNiB,EAAWC,EAAAA,KAAKtB"}
@@ -1,2 +1,2 @@
1
- import e from"@babel/runtime/helpers/esm/objectSpread2";import t from"@babel/runtime/helpers/esm/objectWithoutProperties";import o,{forwardRef as r,useMemo as a,useEffect as m,memo as s}from"react";import{TextInput as u}from"react-native";import{a as i,u as l,m as c,e as n}from"./shared.production.js";const f=["formKey","element","type","fetch","fetchDeps","debounce","defaultValue","validatorKey","value"],p=s(r(((r,s)=>{let{formKey:p,element:d,type:v,fetch:h,fetchDeps:b,debounce:y=300,defaultValue:K,validatorKey:j,value:D}=r,P=t(r,f);const S=i((e=>e.fields[p])),V=l(),x=a((()=>{const t=S.value||"",r=e(e({},P),{},{value:t,inputMode:v,ref:s});return r.onEndEditing=P.onEndEditing?e=>{V.handlers.t(S),P.onEndEditing(e)}:()=>{V.handlers.t(S)},r.onChangeText=P.onChangeText?e=>{V.handlers.o(S,void 0,{value:e}),P.onChangeText(e)}:e=>{V.handlers.o(S,void 0,{value:e})},d?d(S,r):o.createElement(u,r)}),[S,d]),J=i(c(b)),M=a((()=>JSON.stringify(J)),[J]);return m((()=>{h&&n(y,`${S.gid}-fetch`).then((()=>{const e=h(S,V.getState().fields);e instanceof Promise?e.then((e=>e&&V.handlers.m(e,p))):e&&V.handlers.m(e,p)}))}),[M]),x})));export{p as RNGInput};
1
+ import e from"@babel/runtime/helpers/esm/objectSpread2";import t from"@babel/runtime/helpers/esm/objectWithoutProperties";import o,{forwardRef as r,useMemo as a,useEffect as u,memo as m}from"react";import{TextInput as c}from"react-native";import{a as s,u as i,m as l,e as n}from"./shared.production.js";const f=["formKey","element","type","fetch","fetchDeps","debounce","defaultValue","validatorKey","value"],p=m(r(((r,m)=>{let{formKey:p,element:d,type:h,fetch:v,fetchDeps:b,debounce:y=300,defaultValue:K,validatorKey:j,value:D}=r,P=t(r,f);const S=s((e=>e.fields[p])),V=i(),x=a((()=>{const t=S.value||"",r=e(e({},P),{},{value:t,inputMode:h,ref:m});return r.onEndEditing=P.onEndEditing?e=>{V.handlers.t(S),P.onEndEditing(e)}:()=>{V.handlers.t(S)},r.onChangeText=P.onChangeText?e=>{V.handlers.o(S,void 0,{value:e}),P.onChangeText(e)}:e=>{V.handlers.o(S,void 0,{value:e})},S.touched||(r.onFocus=P.onFocus?e=>{P.onFocus(e),S.dispatchChanges({touched:!0})}:()=>{S.dispatchChanges({touched:!0})}),d?d(S,r):o.createElement(c,r)}),[S,d]),J=s(l(b)),M=a((()=>JSON.stringify(J)),[J]);return u((()=>{v&&n(y,`${S.gid}-fetch`).then((()=>{const e=v(S,V.getState().fields);e instanceof Promise?e.then((e=>e&&V.handlers.u(e,p))):e&&V.handlers.u(e,p)}))}),[M]),x})));export{p as RNGInput};
2
2
  //# sourceMappingURL=RNGInput.production.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"RNGInput.production.js","sources":["../../../src/fields/RNGInput.tsx"],"sourcesContent":["import React, {forwardRef, memo, type ReactNode, useEffect, useMemo} from 'react';\r\nimport {TextInput} from 'react-native';\r\n\r\nimport {_debounce} from '../helpers';\r\nimport type {GInputState, RNGInputProps} from '.';\r\nimport {useFormSelector, useFormStore} from \"../form-context\";\r\nimport {makeSelectFields} from \"../selectors\";\r\n\r\nconst _RNGInput = forwardRef<any, RNGInputProps>(({\r\n formKey,\r\n element,\r\n type,\r\n fetch,\r\n fetchDeps,\r\n debounce = 300,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n defaultValue,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n validatorKey,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n value,\r\n ...rest\r\n}, ref) => {\r\n const inputState = useFormSelector(state => state.fields[formKey]);\r\n const store = useFormStore();\r\n\r\n const _element = useMemo(() => {\r\n const value = inputState.value || '';\r\n\r\n const _props = {\r\n ...rest,\r\n value,\r\n inputMode: type,\r\n ref\r\n };\r\n\r\n _props.onEndEditing = rest.onEndEditing ?\r\n (e) => {\r\n store.handlers._viHandler(inputState);\r\n rest.onEndEditing!(e);\r\n }\r\n :\r\n () => {\r\n store.handlers._viHandler(inputState);\r\n };\r\n _props.onChangeText = rest.onChangeText ?\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n rest.onChangeText!(e);\r\n }\r\n :\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n };\r\n\r\n if (element) {\r\n return (element as (input: GInputState, props: any) => ReactNode)(inputState, _props);\r\n }\r\n\r\n return <TextInput {..._props}/>;\r\n }, [inputState, element]);\r\n\r\n const _fetchDeps = useFormSelector(makeSelectFields(fetchDeps));\r\n const stableFetchDeps = useMemo(() => JSON.stringify(_fetchDeps), [_fetchDeps]);\r\n\r\n useEffect(() => {\r\n if (fetch) {\r\n _debounce(debounce, `${inputState.gid}-fetch`).then(() => {\r\n const res = fetch(inputState, store.getState().fields);\r\n if (res instanceof Promise) {\r\n res.then((state) => state && store.handlers._dispatchChanges(state, formKey));\r\n } else if (res) {\r\n store.handlers._dispatchChanges(res, formKey);\r\n }\r\n });\r\n }\r\n }, [stableFetchDeps]);\r\n\r\n return _element;\r\n});\r\n\r\nexport const RNGInput = memo(_RNGInput);"],"names":["RNGInput","memo","forwardRef","_ref","ref","formKey","element","type","fetch","fetchDeps","debounce","defaultValue","validatorKey","value","rest","_objectWithoutProperties","_excluded","inputState","useFormSelector","state","fields","store","useFormStore","_element","useMemo","_props","_objectSpread","inputMode","onEndEditing","e","handlers","_viHandler","onChangeText","_updateInputHandler","undefined","React","createElement","TextInput","_fetchDeps","makeSelectFields","stableFetchDeps","JSON","stringify","useEffect","_debounce","gid","then","res","getState","Promise","_dispatchChanges"],"mappings":"yZAiFaA,EAAWC,EAzENC,GAA+B,CAAAC,EAc9CC,KAAQ,IAduCC,QAC9CA,EAAOC,QACPA,EAAOC,KACPA,EAAIC,MACJA,EAAKC,UACLA,EAASC,SACTA,EAAW,IAAGC,aAEdA,EAAYC,aAEZA,EAAYC,MAEZA,GAEHV,EADMW,EAAIC,EAAAZ,EAAAa,GAEP,MAAMC,EAAaC,GAAgBC,GAASA,EAAMC,OAAOf,KACnDgB,EAAQC,IAERC,EAAWC,GAAQ,KACrB,MAAMX,EAAQI,EAAWJ,OAAS,GAE5BY,EAAMC,EAAAA,KACLZ,GAAI,GAAA,CACPD,QACAc,UAAWpB,EACXH,QAsBJ,OAnBAqB,EAAOG,aAAed,EAAKc,aACtBC,IACGR,EAAMS,SAASC,EAAWd,GAC1BH,EAAKc,aAAcC,EAAE,EAGzB,KACIR,EAAMS,SAASC,EAAWd,EAAW,EAE7CQ,EAAOO,aAAelB,EAAKkB,aACtBH,IACGR,EAAMS,SAASG,EAAoBhB,OAAYiB,EAAW,CAACrB,MAAOgB,IAClEf,EAAKkB,aAAcH,EAAE,EAGxBA,IACGR,EAAMS,SAASG,EAAoBhB,OAAYiB,EAAW,CAACrB,MAAOgB,GAAG,EAGzEvB,EACQA,EAA0DW,EAAYQ,GAG3EU,EAAAC,cAACC,EAAcZ,EAAS,GAChC,CAACR,EAAYX,IAEVgC,EAAapB,EAAgBqB,EAAiB9B,IAC9C+B,EAAkBhB,GAAQ,IAAMiB,KAAKC,UAAUJ,IAAa,CAACA,IAenE,OAbAK,GAAU,KACFnC,GACAoC,EAAUlC,EAAU,GAAGO,EAAW4B,aAAaC,MAAK,KAChD,MAAMC,EAAMvC,EAAMS,EAAYI,EAAM2B,WAAW5B,QAC3C2B,aAAeE,QACfF,EAAID,MAAM3B,GAAUA,GAASE,EAAMS,SAASoB,EAAiB/B,EAAOd,KAC7D0C,GACP1B,EAAMS,SAASoB,EAAiBH,EAAK1C,EACzC,GAER,GACD,CAACmC,IAEGjB,CAAQ"}
1
+ {"version":3,"file":"RNGInput.production.js","sources":["../../../src/fields/RNGInput.tsx"],"sourcesContent":["import React, {forwardRef, memo, type ReactNode, useEffect, useMemo} from 'react';\r\nimport {TextInput} from 'react-native';\r\n\r\nimport {_debounce} from '../helpers';\r\nimport type {GInputState, RNGInputProps} from '.';\r\nimport {useFormSelector, useFormStore} from \"../form-context\";\r\nimport {makeSelectFields} from \"../selectors\";\r\n\r\nconst _RNGInput = forwardRef<any, RNGInputProps>(({\r\n formKey,\r\n element,\r\n type,\r\n fetch,\r\n fetchDeps,\r\n debounce = 300,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n defaultValue,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n validatorKey,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n value,\r\n ...rest\r\n}, ref) => {\r\n const inputState = useFormSelector(state => state.fields[formKey]);\r\n const store = useFormStore();\r\n\r\n const _element = useMemo(() => {\r\n const value = inputState.value || '';\r\n\r\n const _props = {\r\n ...rest,\r\n value,\r\n inputMode: type,\r\n ref\r\n };\r\n\r\n _props.onEndEditing = rest.onEndEditing ?\r\n (e) => {\r\n store.handlers._viHandler(inputState);\r\n rest.onEndEditing!(e);\r\n }\r\n :\r\n () => {\r\n store.handlers._viHandler(inputState);\r\n };\r\n _props.onChangeText = rest.onChangeText ?\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n rest.onChangeText!(e);\r\n }\r\n :\r\n (e) => {\r\n store.handlers._updateInputHandler(inputState, undefined, {value: e});\r\n };\r\n\r\n if (!inputState.touched) {\r\n _props.onFocus = rest.onFocus ?\r\n (e) => {\r\n rest.onFocus!(e);\r\n inputState.dispatchChanges({touched: true});\r\n }\r\n :\r\n () => {\r\n inputState.dispatchChanges({touched: true});\r\n };\r\n }\r\n\r\n if (element) {\r\n return (element as (input: GInputState, props: any) => ReactNode)(inputState, _props);\r\n }\r\n\r\n return <TextInput {..._props}/>;\r\n }, [inputState, element]);\r\n\r\n const _fetchDeps = useFormSelector(makeSelectFields(fetchDeps));\r\n const stableFetchDeps = useMemo(() => JSON.stringify(_fetchDeps), [_fetchDeps]);\r\n\r\n useEffect(() => {\r\n if (fetch) {\r\n _debounce(debounce, `${inputState.gid}-fetch`).then(() => {\r\n const res = fetch(inputState, store.getState().fields);\r\n if (res instanceof Promise) {\r\n res.then((state) => state && store.handlers._dispatchChanges(state, formKey));\r\n } else if (res) {\r\n store.handlers._dispatchChanges(res, formKey);\r\n }\r\n });\r\n }\r\n }, [stableFetchDeps]);\r\n\r\n return _element;\r\n});\r\n\r\nexport const RNGInput = memo(_RNGInput);"],"names":["RNGInput","memo","forwardRef","_ref","ref","formKey","element","type","fetch","fetchDeps","debounce","defaultValue","validatorKey","value","rest","_objectWithoutProperties","_excluded","inputState","useFormSelector","state","fields","store","useFormStore","_element","useMemo","_props","_objectSpread","inputMode","onEndEditing","e","handlers","_viHandler","onChangeText","_updateInputHandler","undefined","touched","onFocus","dispatchChanges","React","createElement","TextInput","_fetchDeps","makeSelectFields","stableFetchDeps","JSON","stringify","useEffect","_debounce","gid","then","res","getState","Promise","_dispatchChanges"],"mappings":"yZA6FaA,EAAWC,EArFNC,GAA+B,CAAAC,EAc9CC,KAAQ,IAduCC,QAC9CA,EAAOC,QACPA,EAAOC,KACPA,EAAIC,MACJA,EAAKC,UACLA,EAASC,SACTA,EAAW,IAAGC,aAEdA,EAAYC,aAEZA,EAAYC,MAEZA,GAEHV,EADMW,EAAIC,EAAAZ,EAAAa,GAEP,MAAMC,EAAaC,GAAgBC,GAASA,EAAMC,OAAOf,KACnDgB,EAAQC,IAERC,EAAWC,GAAQ,KACrB,MAAMX,EAAQI,EAAWJ,OAAS,GAE5BY,EAAMC,EAAAA,KACLZ,GAAI,GAAA,CACPD,QACAc,UAAWpB,EACXH,QAkCJ,OA/BAqB,EAAOG,aAAed,EAAKc,aACtBC,IACGR,EAAMS,SAASC,EAAWd,GAC1BH,EAAKc,aAAcC,EAAE,EAGzB,KACIR,EAAMS,SAASC,EAAWd,EAAW,EAE7CQ,EAAOO,aAAelB,EAAKkB,aACtBH,IACGR,EAAMS,SAASG,EAAoBhB,OAAYiB,EAAW,CAACrB,MAAOgB,IAClEf,EAAKkB,aAAcH,EAAE,EAGxBA,IACGR,EAAMS,SAASG,EAAoBhB,OAAYiB,EAAW,CAACrB,MAAOgB,GAAG,EAGxEZ,EAAWkB,UACZV,EAAOW,QAAUtB,EAAKsB,QACjBP,IACGf,EAAKsB,QAASP,GACdZ,EAAWoB,gBAAgB,CAACF,SAAS,GAAM,EAG/C,KACIlB,EAAWoB,gBAAgB,CAACF,SAAS,GAAM,GAInD7B,EACQA,EAA0DW,EAAYQ,GAG3Ea,EAAAC,cAACC,EAAcf,EAAS,GAChC,CAACR,EAAYX,IAEVmC,EAAavB,EAAgBwB,EAAiBjC,IAC9CkC,EAAkBnB,GAAQ,IAAMoB,KAAKC,UAAUJ,IAAa,CAACA,IAenE,OAbAK,GAAU,KACFtC,GACAuC,EAAUrC,EAAU,GAAGO,EAAW+B,aAAaC,MAAK,KAChD,MAAMC,EAAM1C,EAAMS,EAAYI,EAAM8B,WAAW/B,QAC3C8B,aAAeE,QACfF,EAAID,MAAM9B,GAAUA,GAASE,EAAMS,SAASuB,EAAiBlC,EAAOd,KAC7D6C,GACP7B,EAAMS,SAASuB,EAAiBH,EAAK7C,EACzC,GAER,GACD,CAACsC,IAEGpB,CAAQ"}
@@ -2,6 +2,7 @@ import _extends from '@babel/runtime/helpers/esm/extends';
2
2
  import _objectSpread from '@babel/runtime/helpers/esm/objectSpread2';
3
3
  import _objectWithoutProperties from '@babel/runtime/helpers/esm/objectWithoutProperties';
4
4
  import React, { useContext, createContext, useSyncExternalStore, useRef, useCallback, useEffect, forwardRef, useMemo, memo } from 'react';
5
+ import '@babel/runtime/helpers/esm/defineProperty';
5
6
  import { TextInput } from 'react-native';
6
7
 
7
8
  const isObject = o => o && typeof o === 'object' && !Array.isArray(o);
@@ -28,7 +29,7 @@ const _buildFormInitialValues = (rows = []) => {
28
29
  const inputConfigs = _findInputs(row);
29
30
  inputConfigs.forEach(config => {
30
31
  if (fields[config.formKey]) {
31
- console.warn(`[Duplicate Keys] - field with key '${config.formKey}' has already been defined.`);
32
+ console.warn(`DEV ONLY - [Duplicate Keys] - field with key '${config.formKey}' has already been defined.`);
32
33
  }
33
34
  const {
34
35
  required = false,
@@ -89,8 +90,9 @@ const _findInputs = (root, total = []) => {
89
90
  }
90
91
  return _findInputs((_root$props2 = root.props) === null || _root$props2 === void 0 ? void 0 : _root$props2.children, total);
91
92
  };
92
- const _findValidityKey = validity => {
93
+ const _findValidityKey = (validity, exclude = []) => {
93
94
  for (const key in validity) {
95
+ if (exclude.includes(key)) continue;
94
96
  if (key !== 'valid' && validity[key]) {
95
97
  return key;
96
98
  }
@@ -201,19 +203,47 @@ const _merge = (target = {}, ...nodes) => {
201
203
  return _merge(target, ...nodes);
202
204
  };
203
205
 
206
+ let handlersMap;
207
+ let validityMap;
208
+ {
209
+ handlersMap = {
210
+ minLength: 'withMinLengthMessage',
211
+ maxLength: 'withMaxLengthMessage',
212
+ required: 'withRequiredMessage',
213
+ pattern: 'withPatternMismatchMessage',
214
+ min: 'withRangeUnderflowMessage',
215
+ max: 'withRangeOverflowMessage',
216
+ step: 'withStepMismatchMessage',
217
+ type: 'withTypeMismatchMessage'
218
+ };
219
+ validityMap = {
220
+ tooShort: 'minLength',
221
+ valueMissing: 'required',
222
+ tooLong: 'maxLength',
223
+ patternMismatch: 'pattern',
224
+ rangeOverflow: 'max',
225
+ rangeUnderflow: 'min',
226
+ stepMismatch: 'step',
227
+ typeMismatch: 'type'
228
+ };
229
+ }
230
+
204
231
  const useFormHandlers = (getState, setState, validators = {}, optimized = false) => {
205
232
  const _viHandler = (input, e) => {
206
233
  if (!input) return;
207
234
  const element = e && e.target;
235
+ const hasInitialValue = !input.dirty && input.value && !input.touched;
236
+ if (!element && !hasInitialValue) return;
208
237
  if (typeof document !== 'undefined' && (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement || element instanceof HTMLSelectElement)) {
209
238
  if (!input.checkValidity) input.checkValidity = () => element.checkValidity();
210
- if (!input.dirty && input.value && !input.touched) {
239
+ if (hasInitialValue) {
211
240
  _checkInputManually(input);
212
241
  _dispatchChanges(input, input.formKey);
213
242
  return;
214
243
  }
215
244
  element.setCustomValidity('');
216
- const validityKey = _findValidityKey(element.validity);
245
+ const exclude = input.type && (input.pattern || hasCustomValidation(input)) ? ['typeMismatch'] : [];
246
+ const validityKey = _findValidityKey(element.validity, exclude);
217
247
  _validateInput(input, validityKey, v => element.setCustomValidity(v));
218
248
  if (!validityKey && input.error) {
219
249
  element.setCustomValidity(input.errorText || 'error');
@@ -226,6 +256,7 @@ const useFormHandlers = (getState, setState, validators = {}, optimized = false)
226
256
  }
227
257
  };
228
258
  const _checkInputManually = input => {
259
+ const exclude = input.type && (input.pattern || hasCustomValidation(input)) ? ['typeMismatch'] : [];
229
260
  let validityKey = _findValidityKey({
230
261
  valueMissing: input.required && !input.value || false,
231
262
  typeMismatch: _checkTypeMismatch(input),
@@ -234,7 +265,7 @@ const useFormHandlers = (getState, setState, validators = {}, optimized = false)
234
265
  patternMismatch: input.pattern && _checkResult(input.pattern, input.value) || false,
235
266
  rangeUnderflow: input.min && Number(input.value) < Number(input.min) || false,
236
267
  rangeOverflow: input.max && Number(input.value) > Number(input.max) || false
237
- });
268
+ }, exclude);
238
269
  if (!validityKey && input.error) {
239
270
  validityKey = 'customError';
240
271
  }
@@ -247,6 +278,11 @@ const useFormHandlers = (getState, setState, validators = {}, optimized = false)
247
278
  };
248
279
  const _validateInput = (input, validityKey, setValidity) => {
249
280
  const inputValidator = validators[input.validatorKey || input.formKey] || validators['*'];
281
+ {
282
+ if (validityKey && !(inputValidator !== null && inputValidator !== void 0 && inputValidator.hasConstraint(validityKey))) {
283
+ if (validityKey === 'typeMismatch') console.warn(`DEV ONLY - [Missing Validator] - the input '${input.formKey}' has described the constraint '${validityMap[validityKey]}' however, a correspond validator / custom validation / pattern validator are missing.\nadd '${handlersMap[validityMap[validityKey]]}' or 'withCustomValidation' or '${handlersMap[validityMap.patternMismatch]}' to the input validator.\nexample:\nconst validators: GValidators = {\n\temail: new GValidator().withPatternMismatchMessage('pattern mismatch'),\n\t...\n}\n\nor either remove the constraint '${validityMap[validityKey]}' from the input props`);else console.warn(`DEV ONLY - [Missing Validator] - the input '${input.formKey}' has described the constraint '${validityMap[validityKey]}' however, a correspond validator is missing.\nadd '${handlersMap[validityMap[validityKey]]}' to the input validator.\nexample:\nconst validators: GValidators = {\n\temail: new GValidator().withPatternMismatchMessage('pattern mismatch'),\n\t...\n}\n\nor either remove the constraint '${validityMap[validityKey]}' from the input props`);
284
+ }
285
+ }
250
286
  if (inputValidator) {
251
287
  __validateInput(input, inputValidator, validityKey, setValidity);
252
288
  }
@@ -297,6 +333,10 @@ const useFormHandlers = (getState, setState, validators = {}, optimized = false)
297
333
  });
298
334
  }
299
335
  };
336
+ const hasCustomValidation = input => {
337
+ const validator = validators[input.validatorKey || input.formKey] || validators['*'];
338
+ return validator && (validator.asyncHandlers.length > 0 || validator.handlers.length > 0);
339
+ };
300
340
  return {
301
341
  _updateInputHandler,
302
342
  _viHandler,
@@ -513,6 +553,18 @@ const _RNGInput = forwardRef((_ref, ref) => {
513
553
  value: e
514
554
  });
515
555
  };
556
+ if (!inputState.touched) {
557
+ _props.onFocus = rest.onFocus ? e => {
558
+ rest.onFocus(e);
559
+ inputState.dispatchChanges({
560
+ touched: true
561
+ });
562
+ } : () => {
563
+ inputState.dispatchChanges({
564
+ touched: true
565
+ });
566
+ };
567
+ }
516
568
  if (element) {
517
569
  return element(inputState, _props);
518
570
  }