Reactable Forms API
RxFormActions
Actions available to trigger state changes on Form Reactable.
updateValues
Updates values of a FormControl
. For form group and form arrays, updates will only occur if the specified descendant controls exists. Otherwise it will throw an error.
type updateValues = <T>(payload: UpdateValuesPayload<T>) => void;
interface UpdateValuesPayload<T> {
value: T;
controlRef: ControlRef;
}
addControl
Adds a control to a form group.
type addControl = (payload: AddControlPayload) => void;
interface AddControlPayload {
config: AbstractControlConfig;
controlRef: ControlRef;
}
pushControl
Pushes a control to a form array.
type pushControl = (payload: PushControlPayload) => void;
interface PushControlPayload {
config: AbstractControlConfig;
controlRef: ControlRef;
}
removeControl
Removes a specified control from the form.
type removeControl = (payload: ControlRef) => void;
markControlAsPristine
Marks a control and all descendant controls as pristine.
type markControlAsPristine = (payload: ControlRef) => void;
markControlAsTouched
Marks a control and all ancestors as touched. Can set markAll
to true
to mark all descendants as touched as well (defaults to false
).
type markControlAsTouched = (payload: MarkTouchedPayload) => void;
interface MarkTouchedPayload {
controlRef: ControlRef;
markAll?: boolean;
}
markControlAsUntouched
Marks a control and all descendants as untouched. This will recheck ancestor controls and update the touched status.
type markControlAsUnTouched = (payload: ControlRef) => void;
resetControl
Resets a control by removing existing control and rebuilding it with the original configuration.
type resetControls = (payload: ControlRef) => void;
FormBuilders
build
Factory function for creating a form Reactable. Accepts a configuration object generated by one or more helper methods - control
, group
, array
. Also accepts an RxFormOptions
object.
type build = (config: AbstractControlConfig, options?: RxFormOptions) => Reactable<Form<unknown>, RxFormActions>
load
Function to create a reactable from a previously saved form state.
type load = <Value, Actions = RxFormActions>(
state: Form<Value>,
options: RxFormOptions = {},
) => Reactable<Form<Value>, Actions>
RxFormOptions
Options to customize RxForm behaviour.
export interface RxFormOptions extends EffectsAndSources {
reducers?: CustomReducers<unknown>;
providers?: RxFormProviders;
name?: string;
debug?: boolean;
storeValue?: boolean;
}
Property | Description |
---|---|
reducers (optional) | Dictionary of CustomReducer s to implement custom form behaviour. The CustomReducerFunc (#api-custom-reducers) provides built in FormReducers . Form state updates need to be made with FormReducers to maintain integrity of the form state tree (i.e validation states of parent and child controls). |
effects (optional) | Array of Effect s to be registered to the Reactable. |
sources (optional) | Additional Action Observables the Reactable is listening to. Can be an array or a dictionary where key is the action type and value is the Observable emitting the payload. |
providers (optional) | See RxFormProviders which provides the ValidatorFn s, ValidatorAsyncFn s and normalizer functions to be used in form |
RxFormProviders
The key in the dictionaries should match the name declared for the validator/normalizer in the configuration creation with control
, array
and group
.
type NormalizerFunction<T> = (value: T) => T;
export interface RxFormProviders {
normalizers?: { [key: string]: NormalizerFunction<unknown> };
validators?: { [key: string]: ValidatorFn };
asyncValidators?: { [key: string]: ValidatorAsyncFn };
}
control
Function to create a FormControlConfig
configuration object. Accepts a configuration object or a tuple.
type control = <T>(config: FormControlConfig<T> | FbControl<T>) => FormControlConfig<T>
interface FormControlConfig<T> {
initialValue: T;
validators?: ValidatorFn[];
asyncValidators?: ValidatorAsyncFn[];
}
type FbControl<T> = [T, (ValidatorFn | ValidatorFn[])?, (ValidatorAsyncFn | ValidatorAsyncFn[])?];
group
Function to create a FormGroupConfig
configuration object. Accepts a configuration object containing a controls
dictionary of additional configuration objects generated by control
, group
, or array
.
type group = (config: FormGroupConfig) => FormGroupConfig
interface FormGroupConfig{
validators?: ValidatorFn[];
asyncValidators?: ValidatorAsyncFn[];
controls: { [key: string]: AbstractControlConfig };
}
array
Function to create a FormArrayConfig
configuration object. Accepts a configuration object containing a controls
array of additional configuration objects generated by control
, group
, or array
.
type array = (config: FormArrayConfig) => FormArrayConfig
interface FormArrayConfig {
validators?: ValidatorFn[];
asyncValidators?: ValidatorAsyncFn[];
controls: AbstractControlConfig[];
}
Helpers
getArrayItems
Given a controlRef
for a form array and a Form
, returns all the controls for the form array control. If the controlRef
does not find a form array control an error is throw.
type getArrayItems = <T extends BaseForm<unknown> | Form<unknown>>(
controlRef: ControlRef,
form: T,
) => T extends BaseForm<unknown> ? BaseControl<unknown>[] : FormControl<unknown>[]
getAncestorControls
Given a controlRef
a Form
, returns all the ancestor controls including itself.
type getAncestorControls = <T extends BaseForm<unknown> | Form<unknown>>(
controlRef: ControlRef,
form: T,
excludeSelf = false,
) => (T extends Form<unknown> ? FormControl<unknown> : BaseControl<unknown>)[]
getDescendantControls
Given a controlRef
a Form
, returns all the descendant controls including itself.
type getDescendantControls = <T extends BaseForm<unknown> | Form<unknown>>(
controlRef: ControlRef,
form: T,
excludeSelf = false,
) => (T extends Form<unknown> ? FormControl<unknown> : BaseControl<unknown>)[]
getValueFromControlConfig
Reads a AbstractControlConfig
and returns its initial value for the form.
type getValueFromControlConfig = <T>(controlConfig: AbstractControlConfig) => T
Other Interfaces
Form
Form state. Dictionary of FormControl
(s) where the key is a period separated representation of the ControlRef
tuple.
interface Form<T> {
root?: FormControl<T>;
[key: string]: FormControl<unknown>;
}
FormControl
interface FormControl<T> {
pristineValue: T;
controlRef: ControlRef;
value: T;
dirty: boolean;
touched: boolean;
validatorErrors: FormErrors;
key: string;
asyncValidatorErrors: FormErrors;
asyncValidateInProgress: { [key: string | number]: boolean };
errors: FormErrors;
valid: boolean;
childrenValid: boolean;
pending?: boolean;
config: AbstractControlConfig;
}
Property | Description |
---|---|
pristineValue | Original value of control. Use to determine if control is dirty. |
controlRef | Controls ControlRef . |
value | Control value. |
touched | Touched status of control |
validatorErrors | FormErrors from validators (non-async) |
asyncValidatorErrors | FormErrors from async validators |
errors | FormErrors validatorErrors and asyncValidatorErrors merged. |
valid | Valid status of control. Also checks descendants. |
childrenValid | Valid status of direct child controls. |
config | Original config for form control |
ControlRef
Control Reference represented as a tuple for the FormControl
FormErrors
Dictionary of errors for the control.
interface FormErrors {
[key: string]: boolean;
}
ValidatorFn
Validator function that reads the value of the FormControl
and returns a FormErrors
object.
type ValidatorFn = (value: unknown) => FormErrors
ValidatorFnAsync
Validator function takes in an BaseControl
observable and returns an Observable<FormErrors>
.
type ValidatorAsyncFn = <T>(control$: Observable<BaseControl<T>>) => Observable<FormErrors>;
FormReducers
Built in reducers which can be used to update the state of the form tree. Payload and behaviour is the same and described in RxActions
;
interface FormReducers {
updateValues: <T>(state: BaseFormState<T>, payload: UpdateValuesPayload<unknown>,
) => BaseFormState<T>;
removeControl: <T>(state: BaseFormState<T>, payload: ControlRef) => BaseFormState<T>;
pushControl: <T>(state: BaseFormState<T>, payload: PushControlPayload) => BaseFormState<T>;
addControl: <T>(state: BaseFormState<T>, payload: AddControlPayload) => BaseFormState<T>;
markControlAsPristine: <T>(state: BaseFormState<T>, payload: ControlRef) => BaseFormState<T>;
markControlAsTouched: <T>(state: BaseFormState<T>, payload: MarkTouchedPayload) => BaseFormState<T>;
markControlAsUntouched: <T>(state: BaseFormState<T>, payload: ControlRef,
) => BaseFormState<T>;
resetControl: <T>(state: BaseFormState<T>, payload: ControlRef) => BaseFormState<T>;
}
CustomReducers
export type CustomReducerFunc = (
reducers: FormReducers,
state: BaseFormState<unknown>,
action: Action<unknown>,
) => BaseFormState<unknown>;
export type CustomReducer =
| CustomReducerFunc
| {
reducer: CustomReducerFunc;
effects?: Effect<unknown, unknown>[] | ((payload?: unknown) => ScopedEffects<unknown>);
};
export type CustomReducers<T> = {
[key in keyof (T & {
[key: string]: CustomReducer;
})]: CustomReducer;
};
BaseFormState
Form state before it is fully validated. This is accessible in CustomReducer
s so developer can read the current state and implement custom form behaviours.
interface BaseFormState<T> {
form: BaseForm<T>;
}
interface BaseForm<T> {
root?: BaseControl<T>;
[key: string]: BaseControl<unknown>;
}
BaseControl
BaseControl
contains some control information before a fully validated FormControl
is created.
interface BaseControl<T> {
pristineValue: T;
controlRef: ControlRef;
value: T;
dirty: boolean;
touched: boolean;
validatorErrors: FormErrors;
config: AbstractControlConfig;
key: string;
}
Configuration Interfaces
interface ValidatorConfigs {
validators?: string[];
asyncValidators?: string[];
}
export interface FormGroupConfig extends ValidatorConfigs {
controls: { [key: string]: AbstractControlConfig };
}
export interface FormArrayConfig extends ValidatorConfigs {
controls: AbstractControlConfig[];
}
export interface FormControlConfig<T> extends ValidatorConfigs {
initialValue: T;
normalizers?: string[];
}
export type AbstractControlConfig = (
| FormControlConfig<unknown>
| FormArrayConfig
| FormGroupConfig
) & {
controls?: AbstractControlConfig[] | { [key: string]: AbstractControlConfig };
};