2021-09-07 15:50:00 +08:00
|
|
|
// @flow
|
|
|
|
import type { Node } from 'react';
|
|
|
|
import React from 'react';
|
|
|
|
import parseDuration from 'parse-duration';
|
|
|
|
import { FormField } from 'component/common/form';
|
|
|
|
import Icon from 'component/common/icon';
|
|
|
|
import * as ICONS from 'constants/icons';
|
|
|
|
|
2022-02-26 20:23:42 +08:00
|
|
|
// prettier-ignore
|
|
|
|
const TOOLTIP = 'Units:\n • s: seconds\n • m: minutes \n • h: hours\n • d: days\n • b: months\n • month: months\n • y: years';
|
2021-09-07 15:50:00 +08:00
|
|
|
const ONE_HUNDRED_YEARS_IN_SECONDS = 3154000000;
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
name: string,
|
|
|
|
label?: string | Node,
|
|
|
|
placeholder?: string | number,
|
|
|
|
disabled?: boolean,
|
|
|
|
value: string | number,
|
|
|
|
onChange: (any) => void,
|
|
|
|
onResolve: (valueInSeconds: number) => void, // Returns parsed/resolved value in seconds; "-1" for invalid input.
|
|
|
|
maxDurationInSeconds?: number,
|
|
|
|
};
|
|
|
|
|
|
|
|
export default function FormFieldDuration(props: Props) {
|
|
|
|
const { name, label, placeholder, disabled, value, onChange, onResolve, maxDurationInSeconds } = props;
|
|
|
|
const [valueSec, setValueSec] = React.useState(-1);
|
|
|
|
const [valueErr, setValueErr] = React.useState('');
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
const handleInvalidInput = (errMsg: string) => {
|
|
|
|
if (valueSec !== -1) {
|
|
|
|
setValueSec(-1);
|
|
|
|
}
|
|
|
|
if (valueErr !== errMsg) {
|
|
|
|
setValueErr(errMsg);
|
|
|
|
}
|
|
|
|
onResolve(-1);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleValidInput = (seconds) => {
|
|
|
|
if (seconds !== valueSec) {
|
|
|
|
setValueSec(seconds);
|
|
|
|
onResolve(seconds);
|
|
|
|
}
|
|
|
|
if (valueErr) {
|
|
|
|
setValueErr('');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!value) {
|
|
|
|
handleValidInput(-1); // Reset
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const seconds = parseDuration(value, 's');
|
|
|
|
if (Number.isInteger(seconds) && seconds > 0) {
|
|
|
|
const max = maxDurationInSeconds || ONE_HUNDRED_YEARS_IN_SECONDS;
|
|
|
|
if (seconds > max) {
|
|
|
|
handleInvalidInput(__('Value exceeded maximum.'));
|
|
|
|
} else {
|
|
|
|
handleValidInput(seconds);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
handleInvalidInput(__('Invalid duration.'));
|
|
|
|
}
|
|
|
|
}, [value, valueSec, valueErr, maxDurationInSeconds, onResolve]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<FormField
|
|
|
|
name={name}
|
|
|
|
type="text"
|
|
|
|
disabled={disabled}
|
|
|
|
label={
|
|
|
|
<>
|
2022-02-26 20:35:04 +08:00
|
|
|
{label || __('Duration --[period e.g. ban duration]--')}
|
2022-02-26 20:23:42 +08:00
|
|
|
<Icon customTooltipText={__(TOOLTIP)} className="icon--help" icon={ICONS.HELP} tooltip size={16} />
|
2021-09-07 15:50:00 +08:00
|
|
|
</>
|
|
|
|
}
|
2022-02-26 20:23:42 +08:00
|
|
|
placeholder={placeholder || '30s, 10m, 1h, 2d, 3month, 1y'}
|
2021-09-07 15:50:00 +08:00
|
|
|
value={value}
|
|
|
|
onChange={onChange}
|
|
|
|
error={valueErr}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|