2021-03-30 15:56:36 +02:00
|
|
|
import React, { BaseSyntheticEvent, ReactNode, useCallback, useEffect, useState } from 'react';
|
2021-03-24 17:31:50 +01:00
|
|
|
import { debounce as _debounce } from 'lodash';
|
|
|
|
|
|
|
|
interface FabInputProps {
|
|
|
|
id: string,
|
2021-06-23 17:00:15 +02:00
|
|
|
onChange?: (value: string, validity?: ValidityState) => void,
|
2021-04-02 17:16:27 +02:00
|
|
|
defaultValue: any,
|
2021-03-24 17:31:50 +01:00
|
|
|
icon?: ReactNode,
|
|
|
|
addOn?: ReactNode,
|
|
|
|
addOnClassName?: string,
|
|
|
|
className?: string,
|
|
|
|
disabled?: boolean,
|
|
|
|
required?: boolean,
|
|
|
|
debounce?: number,
|
2021-04-06 17:47:47 +02:00
|
|
|
readOnly?: boolean,
|
2021-04-07 16:21:12 +02:00
|
|
|
maxLength?: number,
|
|
|
|
pattern?: string,
|
|
|
|
placeholder?: string,
|
2021-04-08 10:00:19 +02:00
|
|
|
error?: string,
|
2021-03-24 17:31:50 +01:00
|
|
|
type?: 'text' | 'date' | 'password' | 'url' | 'time' | 'tel' | 'search' | 'number' | 'month' | 'email' | 'datetime-local' | 'week',
|
2021-06-22 17:56:13 +02:00
|
|
|
step?: number | 'any',
|
2021-06-23 17:00:15 +02:00
|
|
|
min?: number,
|
|
|
|
max?: number,
|
2021-03-24 17:31:50 +01:00
|
|
|
}
|
|
|
|
|
2021-04-07 16:21:12 +02:00
|
|
|
/**
|
|
|
|
* This component is a template for an input component that wraps the application style
|
|
|
|
*/
|
2021-06-23 17:00:15 +02:00
|
|
|
export const FabInput: React.FC<FabInputProps> = ({ id, onChange, defaultValue, icon, className, disabled, type, required, debounce, addOn, addOnClassName, readOnly, maxLength, pattern, placeholder, error, step, min, max }) => {
|
2021-04-02 17:16:27 +02:00
|
|
|
const [inputValue, setInputValue] = useState<any>(defaultValue);
|
2021-03-30 11:26:47 +02:00
|
|
|
|
2021-04-07 16:21:12 +02:00
|
|
|
/**
|
|
|
|
* When the component is mounted, initialize the default value for the input.
|
|
|
|
* If the default value changes, update the value of the input until there's no content in it.
|
|
|
|
*/
|
2021-03-30 15:56:36 +02:00
|
|
|
useEffect(() => {
|
2021-04-02 17:16:27 +02:00
|
|
|
if (!inputValue) {
|
|
|
|
setInputValue(defaultValue);
|
2021-04-06 17:47:47 +02:00
|
|
|
if (typeof onChange === 'function') {
|
|
|
|
onChange(defaultValue);
|
|
|
|
}
|
2021-03-31 16:03:51 +02:00
|
|
|
}
|
2021-04-02 17:16:27 +02:00
|
|
|
}, [defaultValue]);
|
2021-03-30 15:56:36 +02:00
|
|
|
|
2021-03-24 17:31:50 +01:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
|
2021-04-08 10:00:19 +02:00
|
|
|
/**
|
|
|
|
* Check if the current component was provided an error string to display, on the input
|
|
|
|
*/
|
|
|
|
const hasError = (): boolean => {
|
|
|
|
return !!error;
|
|
|
|
}
|
|
|
|
|
2021-03-24 17:31:50 +01:00
|
|
|
/**
|
|
|
|
* Debounced (ie. temporised) version of the 'on change' callback.
|
|
|
|
*/
|
2021-04-06 17:47:47 +02:00
|
|
|
const debouncedOnChange = debounce ? useCallback(_debounce(onChange, debounce), [onChange, debounce]) : null;
|
2021-03-24 17:31:50 +01:00
|
|
|
|
|
|
|
/**
|
2021-04-02 16:02:50 +02:00
|
|
|
* Handle the change of content in the input field, and trigger the parent callback, if any
|
2021-03-24 17:31:50 +01:00
|
|
|
*/
|
2021-03-30 11:26:47 +02:00
|
|
|
const handleChange = (e: BaseSyntheticEvent): void => {
|
2021-04-07 16:21:12 +02:00
|
|
|
const { value, validity } = e.target;
|
|
|
|
setInputValue(value);
|
2021-03-24 17:31:50 +01:00
|
|
|
if (typeof onChange === 'function') {
|
2021-03-30 11:26:47 +02:00
|
|
|
if (debounce) {
|
2021-04-07 16:21:12 +02:00
|
|
|
debouncedOnChange(value, validity);
|
2021-03-30 11:26:47 +02:00
|
|
|
} else {
|
2021-04-07 16:21:12 +02:00
|
|
|
onChange(value, validity);
|
2021-03-30 11:26:47 +02:00
|
|
|
}
|
2021-03-24 17:31:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className={`fab-input ${className ? className : ''}`}>
|
2021-04-08 10:00:19 +02:00
|
|
|
<div className={`input-wrapper ${hasError() ? 'input-error' : ''}`}>
|
|
|
|
{hasIcon() && <span className="fab-input--icon">{icon}</span>}
|
|
|
|
<input id={id}
|
|
|
|
type={type}
|
2021-06-22 17:56:13 +02:00
|
|
|
step={step}
|
2021-06-23 17:00:15 +02:00
|
|
|
min={min}
|
|
|
|
max={max}
|
2021-04-08 10:00:19 +02:00
|
|
|
className="fab-input--input"
|
|
|
|
value={inputValue}
|
|
|
|
onChange={handleChange}
|
|
|
|
disabled={disabled}
|
|
|
|
required={required}
|
|
|
|
readOnly={readOnly}
|
|
|
|
maxLength={maxLength}
|
|
|
|
pattern={pattern}
|
|
|
|
placeholder={placeholder} />
|
|
|
|
{hasAddOn() && <span className={`fab-input--addon ${addOnClassName ? addOnClassName : ''}`}>{addOn}</span>}
|
|
|
|
</div>
|
|
|
|
{hasError() && <span className="fab-input--error">{error}</span> }
|
2021-03-24 17:31:50 +01:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
FabInput.defaultProps = { type: 'text', debounce: 0 };
|