mirror of
https://github.com/LaCasemate/fab-manager.git
synced 2025-01-24 13:52:21 +01:00
60 lines
2.1 KiB
TypeScript
60 lines
2.1 KiB
TypeScript
|
import React, { PropsWithChildren, ReactNode, useEffect, useState } from 'react';
|
||
|
import { AbstractFormComponent } from '../../models/form-component';
|
||
|
import { FieldValues } from 'react-hook-form/dist/types/fields';
|
||
|
import { get as _get } from 'lodash';
|
||
|
|
||
|
export interface AbstractFormItemProps<TFieldValues> extends PropsWithChildren<AbstractFormComponent<TFieldValues>> {
|
||
|
id: string,
|
||
|
label?: string,
|
||
|
tooltip?: ReactNode,
|
||
|
className?: string,
|
||
|
disabled?: boolean,
|
||
|
readOnly?: boolean
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* This abstract component should not be used directly.
|
||
|
* Other forms components that are intended to be used with react-hook-form must extend this component.
|
||
|
*/
|
||
|
export const AbstractFormItem = <TFieldValues extends FieldValues>({ id, label, tooltip, className, disabled, readOnly, error, warning, rules, formState, children }: AbstractFormItemProps<TFieldValues>) => {
|
||
|
const [isDirty, setIsDirty] = useState(false);
|
||
|
const [fieldError, setFieldError] = useState(error);
|
||
|
|
||
|
useEffect(() => {
|
||
|
setIsDirty(_get(formState?.dirtyFields, id));
|
||
|
setFieldError(_get(formState?.errors, id));
|
||
|
}, [formState]);
|
||
|
|
||
|
useEffect(() => {
|
||
|
setFieldError(error);
|
||
|
}, [error]);
|
||
|
|
||
|
// Compose classnames from props
|
||
|
const classNames = [
|
||
|
'form-item',
|
||
|
`${className || ''}`,
|
||
|
`${isDirty && fieldError ? 'is-incorrect' : ''}`,
|
||
|
`${isDirty && warning ? 'is-warned' : ''}`,
|
||
|
`${rules && rules.required ? 'is-required' : ''}`,
|
||
|
`${readOnly ? 'is-readonly' : ''}`,
|
||
|
`${disabled ? 'is-disabled' : ''}`
|
||
|
].join(' ');
|
||
|
|
||
|
return (
|
||
|
<label className={classNames}>
|
||
|
{label && <div className='form-item-header'>
|
||
|
<p>{label}</p>
|
||
|
{tooltip && <div className="item-tooltip">
|
||
|
<span className="trigger"><i className="fa fa-question-circle" /></span>
|
||
|
<div className="content">{tooltip}</div>
|
||
|
</div>}
|
||
|
</div>}
|
||
|
<div className='form-item-field'>
|
||
|
{children}
|
||
|
</div>
|
||
|
{(isDirty && fieldError) && <div className="form-item-error">{fieldError.message}</div> }
|
||
|
{(isDirty && warning) && <div className="form-item-warning">{warning.message}</div> }
|
||
|
</label>
|
||
|
);
|
||
|
};
|