diff --git a/src/area-select.tsx b/src/area-select.antd.tsx similarity index 100% rename from src/area-select.tsx rename to src/area-select.antd.tsx diff --git a/src/index.tsx b/src/index.tsx index 0582d23..af71860 100644 --- a/src/index.tsx +++ b/src/index.tsx @@ -1,134 +1,6 @@ -import React, { useCallback, useState, useEffect, useRef } from 'react'; -import { Input } from 'antd'; -import { GroupProps, InputProps } from 'antd/es/input'; -import AreaSelect, { AreaSelectProps } from './area-select'; -import compactAreas, { Area } from './sources'; -import './styles.less'; - -export * from './area-select'; +import { CountryPhoneInput } from './phone-input.antd'; +export default CountryPhoneInput; +export * from './phone-input.antd'; +export * from './area-select.antd'; export * from './config'; export * from './sources'; - -export interface CountryPhoneInputProps - extends Omit { - onChange?: (value: CountryPhoneInputValue) => void; - value?: CountryPhoneInputValue; - defaultValue?: CountryPhoneInputValue; - selectProps?: AreaSelectProps; - inputGroupProps?: GroupProps; - inline?: boolean; - className?: string; -} - -export type CountryPhoneInputValue = { - code?: number; - phone?: string; - short?: string; -}; - -function CountryPhoneInput({ - defaultValue, - onChange, - selectProps = {}, - inputGroupProps, - inline, - className, - ...inputProps -}: CountryPhoneInputProps) { - if (defaultValue) { - defaultValue.short = defaultValue.short?.toUpperCase(); - } - if (inputProps.value) { - inputProps.value.short = inputProps.value.short?.toUpperCase(); - } - - const defaultArea: Area | undefined = compactAreas.find((area) => { - if (defaultValue) { - return area.short === defaultValue.short; - } - return area.short === 'CN'; - }); - const [area, setArea] = useState(defaultArea); - const [phone, setPhone] = useState(defaultValue?.phone); - - const latestInputPropsRef = useRef(inputProps); - useEffect(() => { - latestInputPropsRef.current = inputProps; - }, [inputProps]); - useEffect(() => { - if (!('value' in latestInputPropsRef.current)) { - return; - } - - const value = inputProps.value; - if (value === undefined) { - setArea(undefined); - setPhone(undefined); - return; - } - if (value.short) { - setArea(compactAreas.find((area) => area.short === value.short)); - } else { - setArea(compactAreas.find((area) => area.phoneCode === value.code)); - } - setPhone(value.phone); - }, [inputProps.value]); - - const triggerChange = useCallback( - (phone?: string, area?: Area) => { - const result: CountryPhoneInputValue = { - phone, - code: area?.phoneCode, - short: area?.short, - }; - onChange?.(result); - }, - [onChange] - ); - - const handleAreaChange = useCallback( - (value: string) => { - const area = compactAreas.find((area) => area.short === value); - if (!area) { - return; - } - setArea(area); - triggerChange(phone, area); - }, - [setArea, triggerChange, phone] - ); - - const handlePhoneChange = useCallback( - (e: React.ChangeEvent) => { - const currentValue = e.target.value; - setPhone(currentValue); - triggerChange(currentValue, area); - }, - [setPhone, area, triggerChange] - ); - - const areaSelect = ( - - ); - - if (inline) { - inputProps.addonBefore = areaSelect; - } else { - inputProps.prefix = areaSelect; - } - - return ( - - ); -} - -export default CountryPhoneInput; diff --git a/src/phone-input.antd.tsx b/src/phone-input.antd.tsx new file mode 100644 index 0000000..cf39853 --- /dev/null +++ b/src/phone-input.antd.tsx @@ -0,0 +1,130 @@ +import React, { useCallback, useState, useEffect, useRef } from 'react'; +import { Input } from 'antd'; +import { GroupProps, InputProps } from 'antd/es/input'; +import AreaSelect, { AreaSelectProps } from './area-select.antd'; +import compactAreas, { Area } from './sources'; +import './styles.less'; + +export interface CountryPhoneInputProps + extends Omit { + onChange?: (value: CountryPhoneInputValue) => void; + value?: CountryPhoneInputValue; + defaultValue?: CountryPhoneInputValue; + selectProps?: AreaSelectProps; + inputGroupProps?: GroupProps; + inline?: boolean; + className?: string; +} + +export type CountryPhoneInputValue = { + code?: number; + phone?: string; + short?: string; +}; + +export const CountryPhoneInput = ({ + defaultValue, + onChange, + selectProps = {}, + inputGroupProps, + inline, + className, + ...inputProps +}: CountryPhoneInputProps) => { + if (defaultValue) { + defaultValue.short = defaultValue.short?.toUpperCase(); + } + if (inputProps.value) { + inputProps.value.short = inputProps.value.short?.toUpperCase(); + } + + const defaultArea: Area | undefined = compactAreas.find((area) => { + if (defaultValue) { + return area.short === defaultValue.short; + } + return area.short === 'CN'; + }); + const [area, setArea] = useState(defaultArea); + const [phone, setPhone] = useState(defaultValue?.phone); + + const latestInputPropsRef = useRef(inputProps); + useEffect(() => { + latestInputPropsRef.current = inputProps; + }, [inputProps]); + useEffect(() => { + if (!('value' in latestInputPropsRef.current)) { + return; + } + + const value = inputProps.value; + if (value === undefined) { + setArea(undefined); + setPhone(undefined); + return; + } + if (value.short) { + setArea(compactAreas.find((area) => area.short === value.short)); + } else { + setArea(compactAreas.find((area) => area.phoneCode === value.code)); + } + setPhone(value.phone); + }, [inputProps.value]); + + const triggerChange = useCallback( + (phone?: string, area?: Area) => { + const result: CountryPhoneInputValue = { + phone, + code: area?.phoneCode, + short: area?.short, + }; + onChange?.(result); + }, + [onChange] + ); + + const handleAreaChange = useCallback( + (value: string) => { + const area = compactAreas.find((area) => area.short === value); + if (!area) { + return; + } + setArea(area); + triggerChange(phone, area); + }, + [setArea, triggerChange, phone] + ); + + const handlePhoneChange = useCallback( + (e: React.ChangeEvent) => { + const currentValue = e.target.value; + setPhone(currentValue); + triggerChange(currentValue, area); + }, + [setPhone, area, triggerChange] + ); + + const areaSelect = ( + + ); + + if (inline) { + inputProps.addonBefore = areaSelect; + } else { + inputProps.prefix = areaSelect; + } + + return ( + + ); +}; + +export default CountryPhoneInput;