antd-phone-input
Advanced tools
Comparing version 0.2.4 to 0.3.0
211
index.cjs.js
"use strict"; | ||
var __rest = (this && this.__rest) || function (s, e) { | ||
var t = {}; | ||
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) | ||
t[p] = s[p]; | ||
if (s != null && typeof Object.getOwnPropertySymbols === "function") | ||
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { | ||
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) | ||
t[p[i]] = s[p[i]]; | ||
} | ||
return t; | ||
}; | ||
var __importDefault = (this && this.__importDefault) || function (mod) { | ||
@@ -8,29 +19,179 @@ return (mod && mod.__esModule) ? mod : { "default": mod }; | ||
const react_1 = require("react"); | ||
const style_1 = __importDefault(require("antd/es/input/style")); | ||
const config_provider_1 = require("antd/es/config-provider"); | ||
const useFormInstance_1 = __importDefault(require("antd/es/form/hooks/useFormInstance")); | ||
const context_1 = require("antd/es/form/context"); | ||
const statusUtils_1 = require("antd/es/_util/statusUtils"); | ||
const internal_1 = require("antd/es/theme/internal"); | ||
const legacy_1 = __importDefault(require("./legacy")); | ||
const style_2 = __importDefault(require("./style")); | ||
const PhoneInput = (inputLegacyProps) => { | ||
const { getPrefixCls } = (0, react_1.useContext)(config_provider_1.ConfigContext); | ||
const { status } = (0, react_1.useContext)(context_1.FormItemInputContext); | ||
const inputPrefixCls = getPrefixCls("input"); | ||
const dropdownPrefixCls = getPrefixCls("dropdown"); | ||
const [_1, inputCls] = (0, style_1.default)(inputPrefixCls); | ||
const [_2, dropdownCls] = (0, style_1.default)(dropdownPrefixCls); | ||
const [theme, token] = (0, internal_1.useToken)(); | ||
const inputClass = (0, react_1.useMemo)(() => { | ||
return `${inputCls} ` + (0, statusUtils_1.getStatusClassNames)(inputPrefixCls, status); | ||
}, [inputPrefixCls, inputCls, status]); | ||
const dropdownClass = (0, react_1.useMemo)(() => "ant-dropdown " + dropdownCls, [dropdownCls]); | ||
(0, internal_1.useStyleRegister)({ | ||
theme, | ||
token, | ||
hashId: "react-tel-input", | ||
path: ["antd-phone-input"], | ||
}, () => [(0, style_2.default)(token)]); | ||
return ((0, jsx_runtime_1.jsx)(legacy_1.default, Object.assign({}, inputLegacyProps, { inputClass: inputClass, dropdownClass: dropdownClass }))); | ||
const select_1 = __importDefault(require("antd/es/select")); | ||
const input_1 = __importDefault(require("antd/es/input")); | ||
const styles_1 = __importDefault(require("./styles")); | ||
const timezones_json_1 = __importDefault(require("./metadata/timezones.json")); | ||
const countries_json_1 = __importDefault(require("./metadata/countries.json")); | ||
const validations_json_1 = __importDefault(require("./metadata/validations.json")); | ||
(0, styles_1.default)(".ant-phone-input-select-item {display: flex; column-gap: 10px; align-items: center; } .ant-phone-input-search-wrapper .ant-input {margin: 0 3px 6px 3px; width: calc(100% - 6px); } .ant-phone-input-search-wrapper .ant-select-item-empty {margin: 0 6px 6px 6px; } .ant-phone-input-wrapper .ant-select-selector {border: none !important; } .ant-phone-input-wrapper .ant-select-selection-item {padding: 0 !important; } .ant-phone-input-wrapper .ant-input-group-addon * {display: flex; align-items: center; justify-content: center; } .flag {width: 16px; height: 11px; background-image: url(); } .flag.ad {background-position: -16px 0; } .flag.ae {background-position: -32px 0; } .flag.af {background-position: -48px 0; } .flag.ag {background-position: -64px 0; } .flag.ai {background-position: -80px 0; } .flag.al {background-position: -96px 0; } .flag.am {background-position: -112px 0; } .flag.ao {background-position: -128px 0; } .flag.ar {background-position: -144px 0; } .flag.as {background-position: -160px 0; } .flag.at {background-position: -176px 0; } .flag.au {background-position: -192px 0; } .flag.aw {background-position: -208px 0; } .flag.az {background-position: -224px 0; } .flag.ba {background-position: -240px 0; } .flag.bb {background-position: 0 -11px; } .flag.bd {background-position: -16px -11px; } .flag.be {background-position: -32px -11px; } .flag.bf {background-position: -48px -11px; } .flag.bg {background-position: -64px -11px; } .flag.bh {background-position: -80px -11px; } .flag.bi {background-position: -96px -11px; } .flag.bj {background-position: -112px -11px; } .flag.bm {background-position: -128px -11px; } .flag.bn {background-position: -144px -11px; } .flag.bo {background-position: -160px -11px; } .flag.br {background-position: -176px -11px; } .flag.bs {background-position: -192px -11px; } .flag.bt {background-position: -208px -11px; } .flag.bw {background-position: -224px -11px; } .flag.by {background-position: -240px -11px; } .flag.bz {background-position: 0 -22px; } .flag.ca {background-position: -16px -22px; } .flag.cd {background-position: -32px -22px; } .flag.cf {background-position: -48px -22px; } .flag.cg {background-position: -64px -22px; } .flag.ch {background-position: -80px -22px; } .flag.ci {background-position: -96px -22px; } .flag.ck {background-position: -112px -22px; } .flag.cl {background-position: -128px -22px; } .flag.cm {background-position: -144px -22px; } .flag.cn {background-position: -160px -22px; } .flag.co {background-position: -176px -22px; } .flag.cr {background-position: -192px -22px; } .flag.cu {background-position: -208px -22px; } .flag.cv {background-position: -224px -22px; } .flag.cw {background-position: -240px -22px; } .flag.cy {background-position: 0 -33px; } .flag.cz {background-position: -16px -33px; } .flag.de {background-position: -32px -33px; } .flag.dj {background-position: -48px -33px; } .flag.dk {background-position: -64px -33px; } .flag.dm {background-position: -80px -33px; } .flag.do {background-position: -96px -33px; } .flag.dz {background-position: -112px -33px; } .flag.ec {background-position: -128px -33px; } .flag.ee {background-position: -144px -33px; } .flag.eg {background-position: -160px -33px; } .flag.er {background-position: -176px -33px; } .flag.es {background-position: -192px -33px; } .flag.et {background-position: -208px -33px; } .flag.fi {background-position: -224px -33px; } .flag.fj {background-position: -240px -33px; } .flag.fk {background-position: 0 -44px; } .flag.fm {background-position: -16px -44px; } .flag.fo {background-position: -32px -44px; } .flag.fr {background-position: -48px -44px; } .flag.ga {background-position: -64px -44px; } .flag.gb {background-position: -80px -44px; } .flag.gd {background-position: -96px -44px; } .flag.ge {background-position: -112px -44px; } .flag.gf {background-position: -128px -44px; } .flag.gh {background-position: -144px -44px; } .flag.gi {background-position: -160px -44px; } .flag.gl {background-position: -176px -44px; } .flag.gm {background-position: -192px -44px; } .flag.gn {background-position: -208px -44px; } .flag.gp {background-position: -224px -44px; } .flag.gq {background-position: -240px -44px; } .flag.gr {background-position: 0 -55px; } .flag.gt {background-position: -16px -55px; } .flag.gu {background-position: -32px -55px; } .flag.gw {background-position: -48px -55px; } .flag.gy {background-position: -64px -55px; } .flag.hk {background-position: -80px -55px; } .flag.hn {background-position: -96px -55px; } .flag.hr {background-position: -112px -55px; } .flag.ht {background-position: -128px -55px; } .flag.hu {background-position: -144px -55px; } .flag.id {background-position: -160px -55px; } .flag.ie {background-position: -176px -55px; } .flag.il {background-position: -192px -55px; } .flag.in {background-position: -208px -55px; } .flag.io {background-position: -224px -55px; } .flag.iq {background-position: -240px -55px; } .flag.ir {background-position: 0 -66px; } .flag.is {background-position: -16px -66px; } .flag.it {background-position: -32px -66px; } .flag.je {background-position: -144px -154px; } .flag.jm {background-position: -48px -66px; } .flag.jo {background-position: -64px -66px; } .flag.jp {background-position: -80px -66px; } .flag.ke {background-position: -96px -66px; } .flag.kg {background-position: -112px -66px; } .flag.kh {background-position: -128px -66px; } .flag.ki {background-position: -144px -66px; } .flag.xk {background-position: -128px -154px; } .flag.km {background-position: -160px -66px; } .flag.kn {background-position: -176px -66px; } .flag.kp {background-position: -192px -66px; } .flag.kr {background-position: -208px -66px; } .flag.kw {background-position: -224px -66px; } .flag.ky {background-position: -240px -66px; } .flag.kz {background-position: 0 -77px; } .flag.la {background-position: -16px -77px; } .flag.lb {background-position: -32px -77px; } .flag.lc {background-position: -48px -77px; } .flag.li {background-position: -64px -77px; } .flag.lk {background-position: -80px -77px; } .flag.lr {background-position: -96px -77px; } .flag.ls {background-position: -112px -77px; } .flag.lt {background-position: -128px -77px; } .flag.lu {background-position: -144px -77px; } .flag.lv {background-position: -160px -77px; } .flag.ly {background-position: -176px -77px; } .flag.ma {background-position: -192px -77px; } .flag.mc {background-position: -208px -77px; } .flag.md {background-position: -224px -77px; } .flag.me {background-position: -112px -154px; height: 12px; } .flag.mg {background-position: 0 -88px; } .flag.mh {background-position: -16px -88px; } .flag.mk {background-position: -32px -88px; } .flag.ml {background-position: -48px -88px; } .flag.mm {background-position: -64px -88px; } .flag.mn {background-position: -80px -88px; } .flag.mo {background-position: -96px -88px; } .flag.mp {background-position: -112px -88px; } .flag.mq {background-position: -128px -88px; } .flag.mr {background-position: -144px -88px; } .flag.ms {background-position: -160px -88px; } .flag.mt {background-position: -176px -88px; } .flag.mu {background-position: -192px -88px; } .flag.mv {background-position: -208px -88px; } .flag.mw {background-position: -224px -88px; } .flag.mx {background-position: -240px -88px; } .flag.my {background-position: 0 -99px; } .flag.mz {background-position: -16px -99px; } .flag.na {background-position: -32px -99px; } .flag.nc {background-position: -48px -99px; } .flag.ne {background-position: -64px -99px; } .flag.nf {background-position: -80px -99px; } .flag.ng {background-position: -96px -99px; } .flag.ni {background-position: -112px -99px; } .flag.nl {background-position: -128px -99px; } .flag.bq {background-position: -128px -99px; } .flag.no {background-position: -144px -99px; } .flag.np {background-position: -160px -99px; } .flag.nr {background-position: -176px -99px; } .flag.nu {background-position: -192px -99px; } .flag.nz {background-position: -208px -99px; } .flag.om {background-position: -224px -99px; } .flag.pa {background-position: -240px -99px; } .flag.pe {background-position: 0 -110px; } .flag.pf {background-position: -16px -110px; } .flag.pg {background-position: -32px -110px; } .flag.ph {background-position: -48px -110px; } .flag.pk {background-position: -64px -110px; } .flag.pl {background-position: -80px -110px; } .flag.pm {background-position: -96px -110px; } .flag.pr {background-position: -112px -110px; } .flag.ps {background-position: -128px -110px; } .flag.pt {background-position: -144px -110px; } .flag.pw {background-position: -160px -110px; } .flag.py {background-position: -176px -110px; } .flag.qa {background-position: -192px -110px; } .flag.re {background-position: -208px -110px; } .flag.ro {background-position: -224px -110px; } .flag.rs {background-position: -240px -110px; } .flag.ru {background-position: 0 -121px; } .flag.rw {background-position: -16px -121px; } .flag.sa {background-position: -32px -121px; } .flag.sb {background-position: -48px -121px; } .flag.sc {background-position: -64px -121px; } .flag.sd {background-position: -80px -121px; } .flag.se {background-position: -96px -121px; } .flag.sg {background-position: -112px -121px; } .flag.sh {background-position: -128px -121px; } .flag.si {background-position: -144px -121px; } .flag.sk {background-position: -160px -121px; } .flag.sl {background-position: -176px -121px; } .flag.sm {background-position: -192px -121px; } .flag.sn {background-position: -208px -121px; } .flag.so {background-position: -224px -121px; } .flag.sr {background-position: -240px -121px; } .flag.ss {background-position: 0 -132px; } .flag.st {background-position: -16px -132px; } .flag.sv {background-position: -32px -132px; } .flag.sx {background-position: -48px -132px; } .flag.sy {background-position: -64px -132px; } .flag.sz {background-position: -80px -132px; } .flag.tc {background-position: -96px -132px; } .flag.td {background-position: -112px -132px; } .flag.tg {background-position: -128px -132px; } .flag.th {background-position: -144px -132px; } .flag.tj {background-position: -160px -132px; } .flag.tk {background-position: -176px -132px; } .flag.tl {background-position: -192px -132px; } .flag.tm {background-position: -208px -132px; } .flag.tn {background-position: -224px -132px; } .flag.to {background-position: -240px -132px; } .flag.tr {background-position: 0 -143px; } .flag.tt {background-position: -16px -143px; } .flag.tv {background-position: -32px -143px; } .flag.tw {background-position: -48px -143px; } .flag.tz {background-position: -64px -143px; } .flag.ua {background-position: -80px -143px; } .flag.ug {background-position: -96px -143px; } .flag.us {background-position: -112px -143px; } .flag.uy {background-position: -128px -143px; } .flag.uz {background-position: -144px -143px; } .flag.va {background-position: -160px -143px; } .flag.vc {background-position: -176px -143px; } .flag.ve {background-position: -192px -143px; } .flag.vg {background-position: -208px -143px; } .flag.vi {background-position: -224px -143px; } .flag.vn {background-position: -240px -143px; } .flag.vu {background-position: 0 -154px; } .flag.wf {background-position: -16px -154px; } .flag.ws {background-position: -32px -154px; } .flag.ye {background-position: -48px -154px; } .flag.za {background-position: -64px -154px; } .flag.zm {background-position: -80px -154px; } .flag.zw {background-position: -96px -154px; } "); | ||
const slots = new Set("."); | ||
const getMetadata = (rawValue, countriesList = countries_json_1.default, country = null) => { | ||
country = country == null && rawValue.startsWith("44") ? "gb" : country; | ||
if (country != null) { | ||
countriesList = countriesList.filter((c) => c[0] === country); | ||
countriesList = countriesList.sort((a, b) => b[2].length - a[2].length); | ||
} | ||
return countriesList.find((c) => rawValue.startsWith(c[2])); | ||
}; | ||
const getRawValue = (value) => { | ||
if (typeof value === "string") | ||
return value.replaceAll(/\D/g, ""); | ||
return [value === null || value === void 0 ? void 0 : value.countryCode, value === null || value === void 0 ? void 0 : value.areaCode, value === null || value === void 0 ? void 0 : value.phoneNumber].filter(Boolean).join(""); | ||
}; | ||
const displayFormat = (value) => { | ||
return value.replace(/[.\s\D]+$/, "").replace(/(\(\d+)$/, "$1)"); | ||
}; | ||
const cleanInput = (input, pattern) => { | ||
input = input.match(/\d/g) || []; | ||
return Array.from(pattern, c => input[0] === c || slots.has(c) ? input.shift() || c : c); | ||
}; | ||
const checkValidity = (metadata, strict = false) => { | ||
/** Checks if both the area code and phone number match the validation pattern */ | ||
const pattern = validations_json_1.default[metadata.isoCode][Number(strict)]; | ||
return new RegExp(pattern).test([metadata.areaCode, metadata.phoneNumber].filter(Boolean).join("")); | ||
}; | ||
const getDefaultISO2Code = () => { | ||
/** Returns the default ISO2 code, based on the user's timezone */ | ||
return (timezones_json_1.default[Intl.DateTimeFormat().resolvedOptions().timeZone] || "") || "us"; | ||
}; | ||
const parsePhoneNumber = (formattedNumber, countriesList = countries_json_1.default, country = null) => { | ||
var _a; | ||
const value = getRawValue(formattedNumber); | ||
const isoCode = ((_a = getMetadata(value, countriesList, country)) === null || _a === void 0 ? void 0 : _a[0]) || getDefaultISO2Code(); | ||
const countryCodePattern = /\+\d+/; | ||
const areaCodePattern = /\((\d+)\)/; | ||
/** Parses the matching partials of the phone number by predefined regex patterns */ | ||
const countryCodeMatch = formattedNumber ? (formattedNumber.match(countryCodePattern) || []) : []; | ||
const areaCodeMatch = formattedNumber ? (formattedNumber.match(areaCodePattern) || []) : []; | ||
/** Converts the parsed values of the country and area codes to integers if values present */ | ||
const countryCode = countryCodeMatch.length > 0 ? parseInt(countryCodeMatch[0]) : null; | ||
const areaCode = areaCodeMatch.length > 1 ? parseInt(areaCodeMatch[1]) : null; | ||
/** Parses the phone number by removing the country and area codes from the formatted value */ | ||
const phoneNumberPattern = new RegExp(`^${countryCode}${(areaCode || "")}(\\d+)`); | ||
const phoneNumberMatch = value ? (value.match(phoneNumberPattern) || []) : []; | ||
const phoneNumber = phoneNumberMatch.length > 1 ? phoneNumberMatch[1] : null; | ||
return { countryCode, areaCode, phoneNumber, isoCode }; | ||
}; | ||
const PhoneInput = (_a) => { | ||
var _b; | ||
var { value: initialValue = "", country = getDefaultISO2Code(), enableSearch = false, disableDropdown = false, onlyCountries = [], excludeCountries = [], preferredCountries = [], searchNotFound = "No country found", searchPlaceholder = "Search country", onMount: handleMount = () => null, onInput: handleInput = () => null, onChange: handleChange = () => null, onKeyDown: handleKeyDown = () => null } = _a, antInputProps = __rest(_a, ["value", "country", "enableSearch", "disableDropdown", "onlyCountries", "excludeCountries", "preferredCountries", "searchNotFound", "searchPlaceholder", "onMount", "onInput", "onChange", "onKeyDown"]); | ||
const defaultValue = getRawValue(initialValue); | ||
const defaultMetadata = getMetadata(defaultValue) || countries_json_1.default.find(([iso]) => iso === country); | ||
const defaultValueState = defaultValue || ((_b = countries_json_1.default.find(([iso]) => iso === (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[0]))) === null || _b === void 0 ? void 0 : _b[2]); | ||
const formInstance = (0, useFormInstance_1.default)(); | ||
const formContext = (0, react_1.useContext)(context_1.FormContext); | ||
const backRef = (0, react_1.useRef)(false); | ||
const initiatedRef = (0, react_1.useRef)(false); | ||
const [query, setQuery] = (0, react_1.useState)(""); | ||
const [value, setValue] = (0, react_1.useState)(defaultValueState); | ||
const [minWidth, setMinWidth] = (0, react_1.useState)(0); | ||
const [countryCode, setCountryCode] = (0, react_1.useState)(country); | ||
const countriesOnly = (0, react_1.useMemo)(() => { | ||
const allowList = onlyCountries.length > 0 ? onlyCountries : countries_json_1.default.map(([iso]) => iso); | ||
return countries_json_1.default.map(([iso]) => iso).filter((iso) => { | ||
return allowList.includes(iso) && !excludeCountries.includes(iso); | ||
}); | ||
}, [onlyCountries, excludeCountries]); | ||
const countriesList = (0, react_1.useMemo)(() => { | ||
const filteredCountries = countries_json_1.default.filter(([iso, name, _1, dial]) => { | ||
return countriesOnly.includes(iso) && (name.toLowerCase().startsWith(query.toLowerCase()) || dial.includes(query)); | ||
}); | ||
return [ | ||
...filteredCountries.filter(([iso]) => preferredCountries.includes(iso)), | ||
...filteredCountries.filter(([iso]) => !preferredCountries.includes(iso)), | ||
]; | ||
}, [countriesOnly, preferredCountries, query]); | ||
const metadata = (0, react_1.useMemo)(() => { | ||
const calculatedMetadata = getMetadata(getRawValue(value), countriesList, countryCode); | ||
if (countriesList.find(([iso]) => iso === (calculatedMetadata === null || calculatedMetadata === void 0 ? void 0 : calculatedMetadata[0]) || iso === (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[0]))) { | ||
return calculatedMetadata || defaultMetadata; | ||
} | ||
return countriesList[0]; | ||
}, [countriesList, countryCode, defaultMetadata, value]); | ||
const pattern = (0, react_1.useMemo)(() => { | ||
return (metadata === null || metadata === void 0 ? void 0 : metadata[3]) || (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[3]) || ""; | ||
}, [defaultMetadata, metadata]); | ||
const clean = (0, react_1.useCallback)((input) => { | ||
return cleanInput(input, pattern.replaceAll(/\d/g, ".")); | ||
}, [pattern]); | ||
const first = (0, react_1.useMemo)(() => { | ||
return [...pattern].findIndex(c => slots.has(c)); | ||
}, [pattern]); | ||
const prev = (0, react_1.useMemo)((j = 0) => { | ||
return Array.from(pattern.replaceAll(/\d/g, "."), (c, i) => { | ||
return slots.has(c) ? j = i + 1 : j; | ||
}); | ||
}, [pattern]); | ||
const selectValue = (0, react_1.useMemo)(() => { | ||
var _a, _b; | ||
const metadata = getMetadata(getRawValue(value), countriesList); | ||
return ((_a = (metadata || countriesList[0])) === null || _a === void 0 ? void 0 : _a[0]) + ((_b = (metadata || countriesList[0])) === null || _b === void 0 ? void 0 : _b[2]); | ||
}, [countriesList, value]); | ||
const setFieldValue = (0, react_1.useCallback)((value) => { | ||
if (formInstance) { | ||
let namePath = []; | ||
let formName = (formContext === null || formContext === void 0 ? void 0 : formContext.name) || ""; | ||
let fieldName = (antInputProps === null || antInputProps === void 0 ? void 0 : antInputProps.id) || ""; | ||
if (formName) { | ||
namePath.push(formName); | ||
fieldName = fieldName.slice(formName.length + 1); | ||
} | ||
formInstance.setFieldValue(namePath.concat(fieldName.split("_")), value); | ||
} | ||
}, [antInputProps, formContext, formInstance]); | ||
const format = (0, react_1.useCallback)(({ target }) => { | ||
const [i, j] = [target.selectionStart, target.selectionEnd].map((i) => { | ||
i = clean(target.value.slice(0, i)).findIndex(c => slots.has(c)); | ||
return i < 0 ? prev[prev.length - 1] : backRef.current ? prev[i - 1] || first : i; | ||
}); | ||
target.value = displayFormat(clean(target.value).join("")); | ||
target.setSelectionRange(i, j); | ||
backRef.current = false; | ||
setValue(target.value); | ||
}, [clean, first, prev]); | ||
const onKeyDown = (0, react_1.useCallback)((event) => { | ||
backRef.current = event.key === "Backspace"; | ||
handleKeyDown(event); | ||
}, [handleKeyDown]); | ||
const onChange = (0, react_1.useCallback)((event) => { | ||
const formattedNumber = displayFormat(clean(event.target.value).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList, countryCode); | ||
handleChange(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) }), event); | ||
}, [clean, countriesList, countryCode, handleChange]); | ||
const onInput = (0, react_1.useCallback)((event) => { | ||
handleInput(event); | ||
format(event); | ||
}, [format, handleInput]); | ||
const onMount = (0, react_1.useCallback)((value) => { | ||
setFieldValue(value); | ||
handleMount(value); | ||
}, [handleMount, setFieldValue]); | ||
(0, react_1.useEffect)(() => { | ||
if (initiatedRef.current) | ||
return; | ||
initiatedRef.current = true; | ||
let initialValue = getRawValue(value); | ||
if (!initialValue.startsWith(metadata === null || metadata === void 0 ? void 0 : metadata[2])) { | ||
initialValue = metadata === null || metadata === void 0 ? void 0 : metadata[2]; | ||
} | ||
const formattedNumber = displayFormat(clean(initialValue).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList); | ||
onMount(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) })); | ||
setCountryCode(phoneMetadata.isoCode); | ||
setValue(formattedNumber); | ||
}, [clean, countriesList, metadata, onMount, setFieldValue, value]); | ||
const countriesSelect = (0, react_1.useMemo)(() => ((0, jsx_runtime_1.jsx)(select_1.default, { suffixIcon: null, value: selectValue, open: disableDropdown ? false : undefined, onSelect: (selectedOption, { key: mask }) => { | ||
if (selectValue === selectedOption) | ||
return; | ||
const selectedCountryCode = selectedOption.slice(0, 2); | ||
const formattedNumber = displayFormat(cleanInput(mask, mask).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList, selectedCountryCode); | ||
setFieldValue(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) })); | ||
setCountryCode(selectedCountryCode); | ||
setValue(formattedNumber); | ||
}, optionLabelProp: "label", dropdownStyle: { minWidth }, notFoundContent: searchNotFound, dropdownRender: (menu) => ((0, jsx_runtime_1.jsxs)("div", { className: "ant-phone-input-search-wrapper", children: [enableSearch && ((0, jsx_runtime_1.jsx)(input_1.default, { placeholder: searchPlaceholder, onInput: ({ target }) => setQuery(target.value) })), menu] })), children: countriesList.map(([iso, name, dial, mask]) => ((0, jsx_runtime_1.jsx)(select_1.default.Option, { value: iso + dial, label: (0, jsx_runtime_1.jsx)("div", { className: `flag ${iso}` }), children: (0, jsx_runtime_1.jsxs)("div", { className: "ant-phone-input-select-item", children: [(0, jsx_runtime_1.jsx)("div", { className: `flag ${iso}` }), name, "\u00A0", displayFormat(mask)] }) }, mask))) })), [selectValue, disableDropdown, minWidth, searchNotFound, countriesList, setFieldValue, enableSearch, searchPlaceholder]); | ||
return ((0, jsx_runtime_1.jsx)("div", { className: "ant-phone-input-wrapper", ref: node => setMinWidth((node === null || node === void 0 ? void 0 : node.offsetWidth) || 0), children: (0, jsx_runtime_1.jsx)(input_1.default, Object.assign({ inputMode: "tel", value: value, onInput: onInput, onChange: onChange, onKeyDown: onKeyDown, addonBefore: countriesSelect }, antInputProps)) })); | ||
}; | ||
exports.default = PhoneInput; |
import { PhoneInputProps } from "./types"; | ||
declare const PhoneInput: (inputLegacyProps: PhoneInputProps) => import("react/jsx-runtime").JSX.Element; | ||
declare const PhoneInput: ({ value: initialValue, country, enableSearch, disableDropdown, onlyCountries, excludeCountries, preferredCountries, searchNotFound, searchPlaceholder, onMount: handleMount, onInput: handleInput, onChange: handleChange, onKeyDown: handleKeyDown, ...antInputProps }: PhoneInputProps) => import("react/jsx-runtime").JSX.Element; | ||
export default PhoneInput; |
217
index.js
@@ -1,30 +0,191 @@ | ||
import { jsx as _jsx } from "react/jsx-runtime"; | ||
import { useContext, useMemo } from "react"; | ||
import genComponentStyleHook from "antd/es/input/style"; | ||
import { ConfigContext } from "antd/es/config-provider"; | ||
import { FormItemInputContext } from "antd/es/form/context"; | ||
import { getStatusClassNames } from "antd/es/_util/statusUtils"; | ||
import { useStyleRegister, useToken } from "antd/es/theme/internal"; | ||
import InputLegacy from "./legacy"; | ||
import genPhoneInputStyle from "./style"; | ||
const PhoneInput = (inputLegacyProps) => { | ||
const { getPrefixCls } = useContext(ConfigContext); | ||
const { status } = useContext(FormItemInputContext); | ||
const inputPrefixCls = getPrefixCls("input"); | ||
const dropdownPrefixCls = getPrefixCls("dropdown"); | ||
const [_1, inputCls] = genComponentStyleHook(inputPrefixCls); | ||
const [_2, dropdownCls] = genComponentStyleHook(dropdownPrefixCls); | ||
const [theme, token] = useToken(); | ||
const inputClass = useMemo(() => { | ||
return `${inputCls} ` + getStatusClassNames(inputPrefixCls, status); | ||
}, [inputPrefixCls, inputCls, status]); | ||
const dropdownClass = useMemo(() => "ant-dropdown " + dropdownCls, [dropdownCls]); | ||
useStyleRegister({ | ||
theme, | ||
token, | ||
hashId: "react-tel-input", | ||
path: ["antd-phone-input"], | ||
}, () => [genPhoneInputStyle(token)]); | ||
return (_jsx(InputLegacy, Object.assign({}, inputLegacyProps, { inputClass: inputClass, dropdownClass: dropdownClass }))); | ||
var __rest = (this && this.__rest) || function (s, e) { | ||
var t = {}; | ||
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) | ||
t[p] = s[p]; | ||
if (s != null && typeof Object.getOwnPropertySymbols === "function") | ||
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { | ||
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) | ||
t[p[i]] = s[p[i]]; | ||
} | ||
return t; | ||
}; | ||
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; | ||
import { useCallback, useContext, useEffect, useMemo, useRef, useState } from "react"; | ||
import useFormInstance from "antd/es/form/hooks/useFormInstance"; | ||
import { FormContext } from "antd/es/form/context"; | ||
import Select from "antd/es/select"; | ||
import Input from "antd/es/input"; | ||
import styleInject from "./styles"; | ||
import timezones from "./metadata/timezones.json"; | ||
import countries from "./metadata/countries.json"; | ||
import validations from "./metadata/validations.json"; | ||
styleInject(".ant-phone-input-select-item {display: flex; column-gap: 10px; align-items: center; } .ant-phone-input-search-wrapper .ant-input {margin: 0 3px 6px 3px; width: calc(100% - 6px); } .ant-phone-input-search-wrapper .ant-select-item-empty {margin: 0 6px 6px 6px; } .ant-phone-input-wrapper .ant-select-selector {border: none !important; } .ant-phone-input-wrapper .ant-select-selection-item {padding: 0 !important; } .ant-phone-input-wrapper .ant-input-group-addon * {display: flex; align-items: center; justify-content: center; } .flag {width: 16px; height: 11px; background-image: url(); } .flag.ad {background-position: -16px 0; } .flag.ae {background-position: -32px 0; } .flag.af {background-position: -48px 0; } .flag.ag {background-position: -64px 0; } .flag.ai {background-position: -80px 0; } .flag.al {background-position: -96px 0; } .flag.am {background-position: -112px 0; } .flag.ao {background-position: -128px 0; } .flag.ar {background-position: -144px 0; } .flag.as {background-position: -160px 0; } .flag.at {background-position: -176px 0; } .flag.au {background-position: -192px 0; } .flag.aw {background-position: -208px 0; } .flag.az {background-position: -224px 0; } .flag.ba {background-position: -240px 0; } .flag.bb {background-position: 0 -11px; } .flag.bd {background-position: -16px -11px; } .flag.be {background-position: -32px -11px; } .flag.bf {background-position: -48px -11px; } .flag.bg {background-position: -64px -11px; } .flag.bh {background-position: -80px -11px; } .flag.bi {background-position: -96px -11px; } .flag.bj {background-position: -112px -11px; } .flag.bm {background-position: -128px -11px; } .flag.bn {background-position: -144px -11px; } .flag.bo {background-position: -160px -11px; } .flag.br {background-position: -176px -11px; } .flag.bs {background-position: -192px -11px; } .flag.bt {background-position: -208px -11px; } .flag.bw {background-position: -224px -11px; } .flag.by {background-position: -240px -11px; } .flag.bz {background-position: 0 -22px; } .flag.ca {background-position: -16px -22px; } .flag.cd {background-position: -32px -22px; } .flag.cf {background-position: -48px -22px; } .flag.cg {background-position: -64px -22px; } .flag.ch {background-position: -80px -22px; } .flag.ci {background-position: -96px -22px; } .flag.ck {background-position: -112px -22px; } .flag.cl {background-position: -128px -22px; } .flag.cm {background-position: -144px -22px; } .flag.cn {background-position: -160px -22px; } .flag.co {background-position: -176px -22px; } .flag.cr {background-position: -192px -22px; } .flag.cu {background-position: -208px -22px; } .flag.cv {background-position: -224px -22px; } .flag.cw {background-position: -240px -22px; } .flag.cy {background-position: 0 -33px; } .flag.cz {background-position: -16px -33px; } .flag.de {background-position: -32px -33px; } .flag.dj {background-position: -48px -33px; } .flag.dk {background-position: -64px -33px; } .flag.dm {background-position: -80px -33px; } .flag.do {background-position: -96px -33px; } .flag.dz {background-position: -112px -33px; } .flag.ec {background-position: -128px -33px; } .flag.ee {background-position: -144px -33px; } .flag.eg {background-position: -160px -33px; } .flag.er {background-position: -176px -33px; } .flag.es {background-position: -192px -33px; } .flag.et {background-position: -208px -33px; } .flag.fi {background-position: -224px -33px; } .flag.fj {background-position: -240px -33px; } .flag.fk {background-position: 0 -44px; } .flag.fm {background-position: -16px -44px; } .flag.fo {background-position: -32px -44px; } .flag.fr {background-position: -48px -44px; } .flag.ga {background-position: -64px -44px; } .flag.gb {background-position: -80px -44px; } .flag.gd {background-position: -96px -44px; } .flag.ge {background-position: -112px -44px; } .flag.gf {background-position: -128px -44px; } .flag.gh {background-position: -144px -44px; } .flag.gi {background-position: -160px -44px; } .flag.gl {background-position: -176px -44px; } .flag.gm {background-position: -192px -44px; } .flag.gn {background-position: -208px -44px; } .flag.gp {background-position: -224px -44px; } .flag.gq {background-position: -240px -44px; } .flag.gr {background-position: 0 -55px; } .flag.gt {background-position: -16px -55px; } .flag.gu {background-position: -32px -55px; } .flag.gw {background-position: -48px -55px; } .flag.gy {background-position: -64px -55px; } .flag.hk {background-position: -80px -55px; } .flag.hn {background-position: -96px -55px; } .flag.hr {background-position: -112px -55px; } .flag.ht {background-position: -128px -55px; } .flag.hu {background-position: -144px -55px; } .flag.id {background-position: -160px -55px; } .flag.ie {background-position: -176px -55px; } .flag.il {background-position: -192px -55px; } .flag.in {background-position: -208px -55px; } .flag.io {background-position: -224px -55px; } .flag.iq {background-position: -240px -55px; } .flag.ir {background-position: 0 -66px; } .flag.is {background-position: -16px -66px; } .flag.it {background-position: -32px -66px; } .flag.je {background-position: -144px -154px; } .flag.jm {background-position: -48px -66px; } .flag.jo {background-position: -64px -66px; } .flag.jp {background-position: -80px -66px; } .flag.ke {background-position: -96px -66px; } .flag.kg {background-position: -112px -66px; } .flag.kh {background-position: -128px -66px; } .flag.ki {background-position: -144px -66px; } .flag.xk {background-position: -128px -154px; } .flag.km {background-position: -160px -66px; } .flag.kn {background-position: -176px -66px; } .flag.kp {background-position: -192px -66px; } .flag.kr {background-position: -208px -66px; } .flag.kw {background-position: -224px -66px; } .flag.ky {background-position: -240px -66px; } .flag.kz {background-position: 0 -77px; } .flag.la {background-position: -16px -77px; } .flag.lb {background-position: -32px -77px; } .flag.lc {background-position: -48px -77px; } .flag.li {background-position: -64px -77px; } .flag.lk {background-position: -80px -77px; } .flag.lr {background-position: -96px -77px; } .flag.ls {background-position: -112px -77px; } .flag.lt {background-position: -128px -77px; } .flag.lu {background-position: -144px -77px; } .flag.lv {background-position: -160px -77px; } .flag.ly {background-position: -176px -77px; } .flag.ma {background-position: -192px -77px; } .flag.mc {background-position: -208px -77px; } .flag.md {background-position: -224px -77px; } .flag.me {background-position: -112px -154px; height: 12px; } .flag.mg {background-position: 0 -88px; } .flag.mh {background-position: -16px -88px; } .flag.mk {background-position: -32px -88px; } .flag.ml {background-position: -48px -88px; } .flag.mm {background-position: -64px -88px; } .flag.mn {background-position: -80px -88px; } .flag.mo {background-position: -96px -88px; } .flag.mp {background-position: -112px -88px; } .flag.mq {background-position: -128px -88px; } .flag.mr {background-position: -144px -88px; } .flag.ms {background-position: -160px -88px; } .flag.mt {background-position: -176px -88px; } .flag.mu {background-position: -192px -88px; } .flag.mv {background-position: -208px -88px; } .flag.mw {background-position: -224px -88px; } .flag.mx {background-position: -240px -88px; } .flag.my {background-position: 0 -99px; } .flag.mz {background-position: -16px -99px; } .flag.na {background-position: -32px -99px; } .flag.nc {background-position: -48px -99px; } .flag.ne {background-position: -64px -99px; } .flag.nf {background-position: -80px -99px; } .flag.ng {background-position: -96px -99px; } .flag.ni {background-position: -112px -99px; } .flag.nl {background-position: -128px -99px; } .flag.bq {background-position: -128px -99px; } .flag.no {background-position: -144px -99px; } .flag.np {background-position: -160px -99px; } .flag.nr {background-position: -176px -99px; } .flag.nu {background-position: -192px -99px; } .flag.nz {background-position: -208px -99px; } .flag.om {background-position: -224px -99px; } .flag.pa {background-position: -240px -99px; } .flag.pe {background-position: 0 -110px; } .flag.pf {background-position: -16px -110px; } .flag.pg {background-position: -32px -110px; } .flag.ph {background-position: -48px -110px; } .flag.pk {background-position: -64px -110px; } .flag.pl {background-position: -80px -110px; } .flag.pm {background-position: -96px -110px; } .flag.pr {background-position: -112px -110px; } .flag.ps {background-position: -128px -110px; } .flag.pt {background-position: -144px -110px; } .flag.pw {background-position: -160px -110px; } .flag.py {background-position: -176px -110px; } .flag.qa {background-position: -192px -110px; } .flag.re {background-position: -208px -110px; } .flag.ro {background-position: -224px -110px; } .flag.rs {background-position: -240px -110px; } .flag.ru {background-position: 0 -121px; } .flag.rw {background-position: -16px -121px; } .flag.sa {background-position: -32px -121px; } .flag.sb {background-position: -48px -121px; } .flag.sc {background-position: -64px -121px; } .flag.sd {background-position: -80px -121px; } .flag.se {background-position: -96px -121px; } .flag.sg {background-position: -112px -121px; } .flag.sh {background-position: -128px -121px; } .flag.si {background-position: -144px -121px; } .flag.sk {background-position: -160px -121px; } .flag.sl {background-position: -176px -121px; } .flag.sm {background-position: -192px -121px; } .flag.sn {background-position: -208px -121px; } .flag.so {background-position: -224px -121px; } .flag.sr {background-position: -240px -121px; } .flag.ss {background-position: 0 -132px; } .flag.st {background-position: -16px -132px; } .flag.sv {background-position: -32px -132px; } .flag.sx {background-position: -48px -132px; } .flag.sy {background-position: -64px -132px; } .flag.sz {background-position: -80px -132px; } .flag.tc {background-position: -96px -132px; } .flag.td {background-position: -112px -132px; } .flag.tg {background-position: -128px -132px; } .flag.th {background-position: -144px -132px; } .flag.tj {background-position: -160px -132px; } .flag.tk {background-position: -176px -132px; } .flag.tl {background-position: -192px -132px; } .flag.tm {background-position: -208px -132px; } .flag.tn {background-position: -224px -132px; } .flag.to {background-position: -240px -132px; } .flag.tr {background-position: 0 -143px; } .flag.tt {background-position: -16px -143px; } .flag.tv {background-position: -32px -143px; } .flag.tw {background-position: -48px -143px; } .flag.tz {background-position: -64px -143px; } .flag.ua {background-position: -80px -143px; } .flag.ug {background-position: -96px -143px; } .flag.us {background-position: -112px -143px; } .flag.uy {background-position: -128px -143px; } .flag.uz {background-position: -144px -143px; } .flag.va {background-position: -160px -143px; } .flag.vc {background-position: -176px -143px; } .flag.ve {background-position: -192px -143px; } .flag.vg {background-position: -208px -143px; } .flag.vi {background-position: -224px -143px; } .flag.vn {background-position: -240px -143px; } .flag.vu {background-position: 0 -154px; } .flag.wf {background-position: -16px -154px; } .flag.ws {background-position: -32px -154px; } .flag.ye {background-position: -48px -154px; } .flag.za {background-position: -64px -154px; } .flag.zm {background-position: -80px -154px; } .flag.zw {background-position: -96px -154px; } "); | ||
const slots = new Set("."); | ||
const getMetadata = (rawValue, countriesList = countries, country = null) => { | ||
country = country == null && rawValue.startsWith("44") ? "gb" : country; | ||
if (country != null) { | ||
countriesList = countriesList.filter((c) => c[0] === country); | ||
countriesList = countriesList.sort((a, b) => b[2].length - a[2].length); | ||
} | ||
return countriesList.find((c) => rawValue.startsWith(c[2])); | ||
}; | ||
const getRawValue = (value) => { | ||
if (typeof value === "string") | ||
return value.replaceAll(/\D/g, ""); | ||
return [value === null || value === void 0 ? void 0 : value.countryCode, value === null || value === void 0 ? void 0 : value.areaCode, value === null || value === void 0 ? void 0 : value.phoneNumber].filter(Boolean).join(""); | ||
}; | ||
const displayFormat = (value) => { | ||
return value.replace(/[.\s\D]+$/, "").replace(/(\(\d+)$/, "$1)"); | ||
}; | ||
const cleanInput = (input, pattern) => { | ||
input = input.match(/\d/g) || []; | ||
return Array.from(pattern, c => input[0] === c || slots.has(c) ? input.shift() || c : c); | ||
}; | ||
const checkValidity = (metadata, strict = false) => { | ||
/** Checks if both the area code and phone number match the validation pattern */ | ||
const pattern = validations[metadata.isoCode][Number(strict)]; | ||
return new RegExp(pattern).test([metadata.areaCode, metadata.phoneNumber].filter(Boolean).join("")); | ||
}; | ||
const getDefaultISO2Code = () => { | ||
/** Returns the default ISO2 code, based on the user's timezone */ | ||
return (timezones[Intl.DateTimeFormat().resolvedOptions().timeZone] || "") || "us"; | ||
}; | ||
const parsePhoneNumber = (formattedNumber, countriesList = countries, country = null) => { | ||
var _a; | ||
const value = getRawValue(formattedNumber); | ||
const isoCode = ((_a = getMetadata(value, countriesList, country)) === null || _a === void 0 ? void 0 : _a[0]) || getDefaultISO2Code(); | ||
const countryCodePattern = /\+\d+/; | ||
const areaCodePattern = /\((\d+)\)/; | ||
/** Parses the matching partials of the phone number by predefined regex patterns */ | ||
const countryCodeMatch = formattedNumber ? (formattedNumber.match(countryCodePattern) || []) : []; | ||
const areaCodeMatch = formattedNumber ? (formattedNumber.match(areaCodePattern) || []) : []; | ||
/** Converts the parsed values of the country and area codes to integers if values present */ | ||
const countryCode = countryCodeMatch.length > 0 ? parseInt(countryCodeMatch[0]) : null; | ||
const areaCode = areaCodeMatch.length > 1 ? parseInt(areaCodeMatch[1]) : null; | ||
/** Parses the phone number by removing the country and area codes from the formatted value */ | ||
const phoneNumberPattern = new RegExp(`^${countryCode}${(areaCode || "")}(\\d+)`); | ||
const phoneNumberMatch = value ? (value.match(phoneNumberPattern) || []) : []; | ||
const phoneNumber = phoneNumberMatch.length > 1 ? phoneNumberMatch[1] : null; | ||
return { countryCode, areaCode, phoneNumber, isoCode }; | ||
}; | ||
const PhoneInput = (_a) => { | ||
var _b; | ||
var { value: initialValue = "", country = getDefaultISO2Code(), enableSearch = false, disableDropdown = false, onlyCountries = [], excludeCountries = [], preferredCountries = [], searchNotFound = "No country found", searchPlaceholder = "Search country", onMount: handleMount = () => null, onInput: handleInput = () => null, onChange: handleChange = () => null, onKeyDown: handleKeyDown = () => null } = _a, antInputProps = __rest(_a, ["value", "country", "enableSearch", "disableDropdown", "onlyCountries", "excludeCountries", "preferredCountries", "searchNotFound", "searchPlaceholder", "onMount", "onInput", "onChange", "onKeyDown"]); | ||
const defaultValue = getRawValue(initialValue); | ||
const defaultMetadata = getMetadata(defaultValue) || countries.find(([iso]) => iso === country); | ||
const defaultValueState = defaultValue || ((_b = countries.find(([iso]) => iso === (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[0]))) === null || _b === void 0 ? void 0 : _b[2]); | ||
const formInstance = useFormInstance(); | ||
const formContext = useContext(FormContext); | ||
const backRef = useRef(false); | ||
const initiatedRef = useRef(false); | ||
const [query, setQuery] = useState(""); | ||
const [value, setValue] = useState(defaultValueState); | ||
const [minWidth, setMinWidth] = useState(0); | ||
const [countryCode, setCountryCode] = useState(country); | ||
const countriesOnly = useMemo(() => { | ||
const allowList = onlyCountries.length > 0 ? onlyCountries : countries.map(([iso]) => iso); | ||
return countries.map(([iso]) => iso).filter((iso) => { | ||
return allowList.includes(iso) && !excludeCountries.includes(iso); | ||
}); | ||
}, [onlyCountries, excludeCountries]); | ||
const countriesList = useMemo(() => { | ||
const filteredCountries = countries.filter(([iso, name, _1, dial]) => { | ||
return countriesOnly.includes(iso) && (name.toLowerCase().startsWith(query.toLowerCase()) || dial.includes(query)); | ||
}); | ||
return [ | ||
...filteredCountries.filter(([iso]) => preferredCountries.includes(iso)), | ||
...filteredCountries.filter(([iso]) => !preferredCountries.includes(iso)), | ||
]; | ||
}, [countriesOnly, preferredCountries, query]); | ||
const metadata = useMemo(() => { | ||
const calculatedMetadata = getMetadata(getRawValue(value), countriesList, countryCode); | ||
if (countriesList.find(([iso]) => iso === (calculatedMetadata === null || calculatedMetadata === void 0 ? void 0 : calculatedMetadata[0]) || iso === (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[0]))) { | ||
return calculatedMetadata || defaultMetadata; | ||
} | ||
return countriesList[0]; | ||
}, [countriesList, countryCode, defaultMetadata, value]); | ||
const pattern = useMemo(() => { | ||
return (metadata === null || metadata === void 0 ? void 0 : metadata[3]) || (defaultMetadata === null || defaultMetadata === void 0 ? void 0 : defaultMetadata[3]) || ""; | ||
}, [defaultMetadata, metadata]); | ||
const clean = useCallback((input) => { | ||
return cleanInput(input, pattern.replaceAll(/\d/g, ".")); | ||
}, [pattern]); | ||
const first = useMemo(() => { | ||
return [...pattern].findIndex(c => slots.has(c)); | ||
}, [pattern]); | ||
const prev = useMemo((j = 0) => { | ||
return Array.from(pattern.replaceAll(/\d/g, "."), (c, i) => { | ||
return slots.has(c) ? j = i + 1 : j; | ||
}); | ||
}, [pattern]); | ||
const selectValue = useMemo(() => { | ||
var _a, _b; | ||
const metadata = getMetadata(getRawValue(value), countriesList); | ||
return ((_a = (metadata || countriesList[0])) === null || _a === void 0 ? void 0 : _a[0]) + ((_b = (metadata || countriesList[0])) === null || _b === void 0 ? void 0 : _b[2]); | ||
}, [countriesList, value]); | ||
const setFieldValue = useCallback((value) => { | ||
if (formInstance) { | ||
let namePath = []; | ||
let formName = (formContext === null || formContext === void 0 ? void 0 : formContext.name) || ""; | ||
let fieldName = (antInputProps === null || antInputProps === void 0 ? void 0 : antInputProps.id) || ""; | ||
if (formName) { | ||
namePath.push(formName); | ||
fieldName = fieldName.slice(formName.length + 1); | ||
} | ||
formInstance.setFieldValue(namePath.concat(fieldName.split("_")), value); | ||
} | ||
}, [antInputProps, formContext, formInstance]); | ||
const format = useCallback(({ target }) => { | ||
const [i, j] = [target.selectionStart, target.selectionEnd].map((i) => { | ||
i = clean(target.value.slice(0, i)).findIndex(c => slots.has(c)); | ||
return i < 0 ? prev[prev.length - 1] : backRef.current ? prev[i - 1] || first : i; | ||
}); | ||
target.value = displayFormat(clean(target.value).join("")); | ||
target.setSelectionRange(i, j); | ||
backRef.current = false; | ||
setValue(target.value); | ||
}, [clean, first, prev]); | ||
const onKeyDown = useCallback((event) => { | ||
backRef.current = event.key === "Backspace"; | ||
handleKeyDown(event); | ||
}, [handleKeyDown]); | ||
const onChange = useCallback((event) => { | ||
const formattedNumber = displayFormat(clean(event.target.value).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList, countryCode); | ||
handleChange(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) }), event); | ||
}, [clean, countriesList, countryCode, handleChange]); | ||
const onInput = useCallback((event) => { | ||
handleInput(event); | ||
format(event); | ||
}, [format, handleInput]); | ||
const onMount = useCallback((value) => { | ||
setFieldValue(value); | ||
handleMount(value); | ||
}, [handleMount, setFieldValue]); | ||
useEffect(() => { | ||
if (initiatedRef.current) | ||
return; | ||
initiatedRef.current = true; | ||
let initialValue = getRawValue(value); | ||
if (!initialValue.startsWith(metadata === null || metadata === void 0 ? void 0 : metadata[2])) { | ||
initialValue = metadata === null || metadata === void 0 ? void 0 : metadata[2]; | ||
} | ||
const formattedNumber = displayFormat(clean(initialValue).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList); | ||
onMount(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) })); | ||
setCountryCode(phoneMetadata.isoCode); | ||
setValue(formattedNumber); | ||
}, [clean, countriesList, metadata, onMount, setFieldValue, value]); | ||
const countriesSelect = useMemo(() => (_jsx(Select, { suffixIcon: null, value: selectValue, open: disableDropdown ? false : undefined, onSelect: (selectedOption, { key: mask }) => { | ||
if (selectValue === selectedOption) | ||
return; | ||
const selectedCountryCode = selectedOption.slice(0, 2); | ||
const formattedNumber = displayFormat(cleanInput(mask, mask).join("")); | ||
const phoneMetadata = parsePhoneNumber(formattedNumber, countriesList, selectedCountryCode); | ||
setFieldValue(Object.assign(Object.assign({}, phoneMetadata), { valid: (strict) => checkValidity(phoneMetadata, strict) })); | ||
setCountryCode(selectedCountryCode); | ||
setValue(formattedNumber); | ||
}, optionLabelProp: "label", dropdownStyle: { minWidth }, notFoundContent: searchNotFound, dropdownRender: (menu) => (_jsxs("div", { className: "ant-phone-input-search-wrapper", children: [enableSearch && (_jsx(Input, { placeholder: searchPlaceholder, onInput: ({ target }) => setQuery(target.value) })), menu] })), children: countriesList.map(([iso, name, dial, mask]) => (_jsx(Select.Option, { value: iso + dial, label: _jsx("div", { className: `flag ${iso}` }), children: _jsxs("div", { className: "ant-phone-input-select-item", children: [_jsx("div", { className: `flag ${iso}` }), name, "\u00A0", displayFormat(mask)] }) }, mask))) })), [selectValue, disableDropdown, minWidth, searchNotFound, countriesList, setFieldValue, enableSearch, searchPlaceholder]); | ||
return (_jsx("div", { className: "ant-phone-input-wrapper", ref: node => setMinWidth((node === null || node === void 0 ? void 0 : node.offsetWidth) || 0), children: _jsx(Input, Object.assign({ inputMode: "tel", value: value, onInput: onInput, onChange: onChange, onKeyDown: onKeyDown, addonBefore: countriesSelect }, antInputProps)) })); | ||
}; | ||
export default PhoneInput; |
{ | ||
"version": "0.2.4", | ||
"version": "0.3.0", | ||
"name": "antd-phone-input", | ||
@@ -35,9 +35,2 @@ "description": "Advanced, highly customizable phone input component for Ant Design.", | ||
}, | ||
"./legacy": { | ||
"import": "./legacy/index.js", | ||
"require": "./legacy/index.cjs.js", | ||
"types": { | ||
"default": "./legacy/index.d.ts" | ||
} | ||
}, | ||
"./types": { | ||
@@ -50,4 +43,8 @@ "import": "./types.js", | ||
}, | ||
"./legacy/style": { | ||
"default": "./legacy/style.less" | ||
"./styles": { | ||
"import": "./styles.js", | ||
"require": "./styles.cjs.js", | ||
"types": { | ||
"default": "./styles.d.ts" | ||
} | ||
}, | ||
@@ -58,13 +55,13 @@ "./package.json": "./package.json" | ||
"index*", | ||
"style*", | ||
"types*", | ||
"legacy", | ||
"styles*", | ||
"LICENSE", | ||
"metadata", | ||
"README.md" | ||
], | ||
"scripts": { | ||
"rename": "bash -c 'for file in $1*.js; do if [[ \"${file%.js}\" =~ ^[^.]*$ ]]; then mv \"$file\" \"${file%.js}.$0.js\"; fi; done'", | ||
"compile": "tsc --module commonjs && npm run rename -- cjs && npm run rename -- cjs legacy/ && tsc --declaration", | ||
"build": "npm run compile && tsx scripts/prepare-styles.ts", | ||
"prebuild": "rm -r legacy index* style* types* || true", | ||
"rename": "bash -c 'for file in *.js; do mv $file \"${file%.js}.$0.js\"; done'", | ||
"build": "tsc --module commonjs && npm run rename -- cjs && tsc --declaration", | ||
"prebuild": "rm -r metadata index* types* styles* || true", | ||
"postbuild": "tsx scripts/prepare-styles.ts", | ||
"postpack": "tsx scripts/prepare-package.ts", | ||
@@ -81,6 +78,5 @@ "test": "jest --config jestconfig.json" | ||
"@testing-library/user-event": "^14.5.1", | ||
"@types/jest": "^29.5.5", | ||
"@types/react": "^18.2.21", | ||
"antd": "npm:antd@5.2.0", | ||
"antd4": "npm:antd@^4.24.8", | ||
"@types/jest": "^29.5.7", | ||
"@types/react": "^18.2.34", | ||
"antd": "*", | ||
"identity-obj-proxy": "^3.0.0", | ||
@@ -93,6 +89,3 @@ "jest": "^29.7.0", | ||
"typescript": "^5.2.2" | ||
}, | ||
"dependencies": { | ||
"react-phone-input-2": "^2.15.1" | ||
} | ||
} |
@@ -12,4 +12,4 @@ # Antd Phone Input | ||
countries and is compatible with [`antd`](https://github.com/ant-design/ant-design) 4 and 5 versions. It has built-in | ||
support for area codes and provides validation to ensure that the entered numbers are valid. This open-source project | ||
is designed to simplify the process of collecting phone numbers from users. | ||
support for area codes and provides [strict validation](#validation) to ensure the entered numbers are valid. This | ||
open-source project is designed to simplify the process of collecting phone numbers from users. | ||
@@ -28,7 +28,6 @@ ## Installation | ||
The latest version does not require any additional actions for loading the styles as it uses | ||
the [`cssinjs`](https://github.com/ant-design/cssinjs) ecosystem. | ||
The library is designed to work with the `4.x` and `5.x` series of versions in the same way. It can be used as a regular | ||
Ant [Input](https://ant.design/components/input) (see the sample below). More usage examples can be found in | ||
the [examples](examples) directory. | ||
### Antd 5.x | ||
```javascript | ||
@@ -48,27 +47,6 @@ import React from "react"; | ||
![latest](https://user-images.githubusercontent.com/44609997/227775101-72b03e76-52bc-421d-8e75-a03c9d0d6d08.png) | ||
### Antd 4.x | ||
For `4.x` versions, you should use the `legacy` endpoint. | ||
```javascript | ||
import PhoneInput from "antd-phone-input/legacy"; | ||
``` | ||
For including the styles, you should import them in the main `less` file after importing either | ||
the `antd/dist/antd.less` or `antd/dist/antd.dark.less` styles. | ||
```diff | ||
@import "~antd/dist/antd"; | ||
+ @import "~antd-phone-input/legacy/style"; | ||
``` | ||
![legacy](https://user-images.githubusercontent.com/44609997/227775155-9e22bc63-2148-4714-ba8a-9bb4e44c0128.png) | ||
## Value | ||
The value of the component is an object containing the parts of a phone number. This format of value gives a wide range | ||
of opportunities for handling the data in your custom way. For example, you can easily merge the parts of the phone | ||
number into a single string. | ||
The value of the component is an object containing the parts of the phone number. This format of value gives a wide | ||
range of opportunities for handling the data in your desired way. | ||
@@ -81,3 +59,3 @@ ```javascript | ||
isoCode: "us", | ||
valid: function valid() | ||
valid: function valid(strict) | ||
} | ||
@@ -88,10 +66,9 @@ ``` | ||
The `valid` function of the value object returns the validity of the phone number depending on the selected country. So | ||
this can be used in a `validator` like this: | ||
The `valid` function of the value object returns the current validity of the entered phone number based on the selected | ||
country. So this can be used in a `validator` like this: | ||
```javascript | ||
const validator = (_, {valid}) => { | ||
if (valid()) { | ||
return Promise.resolve(); | ||
} | ||
// if (valid(true)) return Promise.resolve(); // strict validation | ||
if (valid()) return Promise.resolve(); // non-strict validation | ||
return Promise.reject("Invalid phone number"); | ||
@@ -107,29 +84,24 @@ } | ||
By default, the `valid` function validates the phone number based on the possible supported lengths of the selected | ||
country. But it also supports a strict validation that apart from the length also checks if the area code is valid for | ||
the selected country. To enable strict validation, pass `true` as the first argument of the `valid` function. | ||
## Props | ||
Apart from the below-described phone-specific properties, all [Input](https://ant.design/components/input#input) | ||
properties that are supported by the used `antd` version, can be applied to the phone input component. | ||
| Property | Description | Type | | ||
|--------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------| | ||
| size | Either `large`, `middle` or `small`. Default value is `middle`. See at ant [docs][antInputProps] for more. | string | | ||
| value | An object containing a parsed phone number or the raw number. This also applies to the `initialValue` property of [Form.Item](https://ant.design/components/form#formitem). | [object](#value) / string | | ||
| style | Applies CSS styles to the container element. | CSSProperties | | ||
| className | The value will be assigned to the container element. | string | | ||
| disabled | Disables the whole input component. | boolean | | ||
| country | Country code to be selected by default. By default, it will show the flag of the user's country. | string | | ||
| enableSearch | Enables search in the country selection dropdown menu. Default value is `false`. | boolean | | ||
| searchNotFound | The value is shown if `enableSearch` is `true` and the query does not match any country. Default value is `No country found`. | string | | ||
| searchPlaceholder | The value is shown if `enableSearch` is `true`. Default value is `Search country`. | string | | ||
| disableDropdown | Disables the manual country selection through the dropdown menu. | boolean | | ||
| inputProps | [HTML properties of input][htmlInputProps] to pass into the input. E.g. `inputProps={{autoFocus: true}}`. | InputHTMLAttributes | | ||
| searchPlaceholder | The value is shown if `enableSearch` is `true`. Default value is `search`. | string | | ||
| searchNotFound | The value is shown if `enableSearch` is `true` and the query does not match any country. Default value is `No entries to show`. | string | | ||
| placeholder | Custom placeholder. Default placeholder is `1 (702) 123-4567`. | string | | ||
| country | Country code to be selected by default. By default, it will show the flag of the user's country. | string | | ||
| regions | Show only the countries of the specified regions. See the list of [available regions][reactPhoneRegions]. | string[] | | ||
| onlyCountries | Country codes to be included in the list. E.g. `onlyCountries={['us', 'ca', 'uk']}`. | string[] | | ||
| excludeCountries | Country codes to be excluded from the list of countries. E.g. `excludeCountries={['us', 'ca', 'uk']}`. | string[] | | ||
| preferredCountries | Country codes to be at the top of the list. E.g. `preferredCountries={['us', 'ca', 'uk']}`. | string[] | | ||
| onChange | Callback when the user is inputting. See at ant [docs][antInputProps] for more. | function(value, e) | | ||
| onPressEnter | The callback function that is triggered when <kbd>Enter</kbd> key is pressed. | function(e) | | ||
| onFocus | The callback is triggered when the input element is focused. | function(e, value) | | ||
| onClick | The callback is triggered when the user clicks on the input element. | function(e, value) | | ||
| onBlur | The callback is triggered when the input element gets blurred or unfocused. | function(e, value) | | ||
| onKeyDown | The callback is triggered when any key is pressed down. | function(e) | | ||
| onMount | The callback is triggered once the component gets mounted. | function(e) | | ||
| onChange | The only difference from the original `onChange` is that value comes first. | function(value, event) | | ||
| onMount | The callback is triggered once the component gets mounted. | function(value) | | ||
@@ -144,7 +116,1 @@ ## Contribute | ||
Copyright (C) 2023 Artyom Vancyan. [MIT](LICENSE) | ||
[antInputProps]:https://ant.design/components/input#input | ||
[reactPhoneRegions]:https://github.com/bl00mber/react-phone-input-2#regions | ||
[htmlInputProps]:https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#attributes |
@@ -1,6 +0,3 @@ | ||
import { ChangeEvent, CSSProperties, FocusEvent, InputHTMLAttributes, KeyboardEvent, MouseEvent } from "react"; | ||
export interface CountryData { | ||
countryCode: string; | ||
dialCode?: string; | ||
} | ||
import { ChangeEvent, KeyboardEvent } from "react"; | ||
import { InputProps } from "antd/es/input"; | ||
export interface PhoneNumber { | ||
@@ -11,41 +8,19 @@ countryCode?: number | null; | ||
isoCode?: string; | ||
dialChanged?: boolean; | ||
valid?(): boolean; | ||
valid?(strict?: boolean): boolean; | ||
} | ||
export interface AntInputProps { | ||
size?: "small" | "middle" | "large"; | ||
export interface PhoneInputProps extends Omit<InputProps, "value" | "onChange"> { | ||
value?: PhoneNumber | string; | ||
style?: CSSProperties; | ||
className?: string; | ||
disabled?: boolean; | ||
onChange?(value: PhoneNumber, event: ChangeEvent<HTMLInputElement>): void; | ||
onPressEnter?(event: KeyboardEvent<HTMLInputElement>): void; | ||
} | ||
export interface ReactPhoneInputProps { | ||
inputProps?: InputHTMLAttributes<HTMLInputElement>; | ||
country?: string; | ||
enableSearch?: boolean; | ||
searchNotFound?: string; | ||
searchPlaceholder?: string; | ||
searchNotFound?: string; | ||
dropdownClass?: string; | ||
inputClass?: string; | ||
placeholder?: string; | ||
enableSearch?: boolean; | ||
disableDropdown?: boolean; | ||
country?: string; | ||
regions?: string[]; | ||
onlyCountries?: string[]; | ||
excludeCountries?: string[]; | ||
preferredCountries?: string[]; | ||
onFocus?(event: FocusEvent<HTMLInputElement>, value: PhoneNumber): void; | ||
onClick?(event: MouseEvent<HTMLInputElement>, value: PhoneNumber): void; | ||
onBlur?(event: FocusEvent<HTMLInputElement>, value: PhoneNumber): void; | ||
onMount?(value: PhoneNumber): void; | ||
onInput?(event: ChangeEvent<HTMLInputElement>): void; | ||
onKeyDown?(event: KeyboardEvent<HTMLInputElement>): void; | ||
onMount?(value: PhoneNumber): void; | ||
/** NOTE: This differs from the antd Input onChange interface */ | ||
onChange?(value: PhoneNumber, event: ChangeEvent<HTMLInputElement>): void; | ||
} | ||
export interface ReactPhoneOnChange { | ||
(value: string, data: CountryData, event: ChangeEvent<HTMLInputElement>, formattedNumber: string): void; | ||
} | ||
export interface ReactPhoneOnMount { | ||
(value: string, event: ChangeEvent<HTMLInputElement> & CountryData, formattedNumber: string): void; | ||
} | ||
export interface PhoneInputProps extends AntInputProps, ReactPhoneInputProps { | ||
} |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
206898
2
12
5608
15
110
1
- Removedreact-phone-input-2@^2.15.1
- Removedjs-tokens@4.0.0(transitive)
- Removedlodash.debounce@4.0.8(transitive)
- Removedlodash.memoize@4.1.2(transitive)
- Removedlodash.reduce@4.6.0(transitive)
- Removedlodash.startswith@4.2.1(transitive)
- Removedloose-envify@1.4.0(transitive)
- Removedobject-assign@4.1.1(transitive)
- Removedprop-types@15.8.1(transitive)
- Removedreact-is@16.13.1(transitive)
- Removedreact-phone-input-2@2.15.1(transitive)