2022-07-13 16:08:59 -07:00
|
|
|
import { parsePhoneNumber, AsYouType } from 'libphonenumber-js';
|
2022-07-13 15:37:40 -07:00
|
|
|
import React, { useState, useEffect } from 'react';
|
2022-07-13 07:42:58 -07:00
|
|
|
|
2022-07-13 15:37:40 -07:00
|
|
|
import { CountryCode } from 'soapbox/utils/phone';
|
2022-07-13 07:42:58 -07:00
|
|
|
|
|
|
|
import Input from '../input/input';
|
|
|
|
|
2022-07-13 09:40:02 -07:00
|
|
|
import CountryCodeDropdown from './country-code-dropdown';
|
|
|
|
|
|
|
|
interface IPhoneInput extends Pick<React.InputHTMLAttributes<HTMLInputElement>, 'required' | 'autoFocus'> {
|
2022-07-13 15:37:40 -07:00
|
|
|
/** E164 phone number. */
|
2023-02-15 13:26:27 -08:00
|
|
|
value?: string
|
2022-07-13 15:37:40 -07:00
|
|
|
/** Change handler which receives the E164 phone string. */
|
2023-02-15 13:26:27 -08:00
|
|
|
onChange?: (phone: string | undefined) => void
|
2022-07-13 15:37:40 -07:00
|
|
|
/** Country code that's selected on mount. */
|
2023-02-15 13:26:27 -08:00
|
|
|
defaultCountryCode?: CountryCode
|
2022-07-13 07:42:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Internationalized phone input with country code picker. */
|
|
|
|
const PhoneInput: React.FC<IPhoneInput> = (props) => {
|
2022-07-13 15:37:40 -07:00
|
|
|
const { value, onChange, defaultCountryCode = '1', ...rest } = props;
|
|
|
|
|
|
|
|
const [countryCode, setCountryCode] = useState<CountryCode>(defaultCountryCode);
|
|
|
|
const [nationalNumber, setNationalNumber] = useState<string>('');
|
2022-07-13 09:40:02 -07:00
|
|
|
|
2022-07-13 07:42:58 -07:00
|
|
|
const handleChange: React.ChangeEventHandler<HTMLInputElement> = ({ target }) => {
|
2022-07-13 16:08:59 -07:00
|
|
|
// HACK: AsYouType is not meant to be used this way. But it works!
|
|
|
|
const asYouType = new AsYouType({ defaultCallingCode: countryCode });
|
|
|
|
const formatted = asYouType.input(target.value);
|
|
|
|
|
2022-07-13 17:50:07 -07:00
|
|
|
// If the new value is the same as before, we might be backspacing,
|
|
|
|
// so use the actual event value instead of the formatted value.
|
2022-07-13 16:08:59 -07:00
|
|
|
if (formatted === nationalNumber && target.value !== nationalNumber) {
|
|
|
|
setNationalNumber(target.value);
|
|
|
|
} else {
|
|
|
|
setNationalNumber(formatted);
|
|
|
|
}
|
2022-07-13 15:37:40 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
// When the internal state changes, update the external state.
|
|
|
|
useEffect(() => {
|
2022-07-13 07:42:58 -07:00
|
|
|
if (onChange) {
|
2022-07-13 15:37:40 -07:00
|
|
|
try {
|
|
|
|
const opts = { defaultCallingCode: countryCode, extract: false } as any;
|
|
|
|
const result = parsePhoneNumber(nationalNumber, opts);
|
|
|
|
|
2022-07-13 17:50:07 -07:00
|
|
|
// Throw if the number is invalid, but catch it below.
|
|
|
|
// We'll only ever call `onChange` with a valid E164 string or `undefined`.
|
2022-07-13 15:37:40 -07:00
|
|
|
if (!result.isPossible()) {
|
|
|
|
throw result;
|
|
|
|
}
|
|
|
|
|
|
|
|
onChange(result.format('E.164'));
|
|
|
|
} catch (e) {
|
|
|
|
// The value returned is always a valid E164 string.
|
|
|
|
// If it's not valid, it'll return undefined.
|
|
|
|
onChange(undefined);
|
|
|
|
}
|
2022-07-13 07:42:58 -07:00
|
|
|
}
|
2022-07-13 15:37:40 -07:00
|
|
|
}, [countryCode, nationalNumber]);
|
2022-07-13 07:42:58 -07:00
|
|
|
|
2022-07-14 11:08:59 -07:00
|
|
|
useEffect(() => {
|
|
|
|
handleChange({ target: { value: nationalNumber } } as any);
|
|
|
|
}, [countryCode, nationalNumber]);
|
|
|
|
|
2022-07-13 07:42:58 -07:00
|
|
|
return (
|
2022-07-14 08:55:00 -07:00
|
|
|
<Input
|
|
|
|
onChange={handleChange}
|
|
|
|
value={nationalNumber}
|
2022-09-07 05:24:44 -07:00
|
|
|
prepend={
|
2022-07-13 16:45:42 -07:00
|
|
|
<CountryCodeDropdown
|
|
|
|
countryCode={countryCode}
|
|
|
|
onChange={setCountryCode}
|
|
|
|
/>
|
2022-07-14 08:55:00 -07:00
|
|
|
}
|
|
|
|
{...rest}
|
|
|
|
/>
|
2022-07-13 07:42:58 -07:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default PhoneInput;
|