Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

antd-phone-input

Package Overview
Dependencies
Maintainers
1
Versions
27
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

antd-phone-input - npm Package Compare versions

Comparing version 0.2.4 to 0.3.0

metadata/countries.json

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;

2

index.d.ts
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;

@@ -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 {
}
SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc