/** * This component is a template for an input component that wraps the application style */ import React, { BaseSyntheticEvent, ReactNode, useCallback, useEffect, useState } from 'react'; import { debounce as _debounce } from 'lodash'; interface FabInputProps { id: string, onChange?: (value: any) => void, defaultValue: any, icon?: ReactNode, addOn?: ReactNode, addOnClassName?: string, className?: string, disabled?: boolean, required?: boolean, debounce?: number, readOnly?: boolean, type?: 'text' | 'date' | 'password' | 'url' | 'time' | 'tel' | 'search' | 'number' | 'month' | 'email' | 'datetime-local' | 'week', } export const FabInput: React.FC = ({ id, onChange, defaultValue, icon, className, disabled, type, required, debounce, addOn, addOnClassName, readOnly }) => { const [inputValue, setInputValue] = useState(defaultValue); useEffect(() => { if (!inputValue) { setInputValue(defaultValue); if (typeof onChange === 'function') { onChange(defaultValue); } } }, [defaultValue]); /** * Check if the current component was provided an icon to display */ const hasIcon = (): boolean => { return !!icon; } /** * Check if the current component was provided an add-on element to display, at the end of the input */ const hasAddOn = (): boolean => { return !!addOn; } /** * Debounced (ie. temporised) version of the 'on change' callback. */ const debouncedOnChange = debounce ? useCallback(_debounce(onChange, debounce), [onChange, debounce]) : null; /** * Handle the change of content in the input field, and trigger the parent callback, if any */ const handleChange = (e: BaseSyntheticEvent): void => { const newValue = e.target.value; setInputValue(newValue); if (typeof onChange === 'function') { if (debounce) { debouncedOnChange(newValue); } else { onChange(newValue); } } } return (
{hasIcon() && {icon}} {hasAddOn() && {addOn}}
); } FabInput.defaultProps = { type: 'text', debounce: 0 };