diff --git a/projects/demo-integrations/cypress/tests/addons/phone/phone-non-strict.cy.ts b/projects/demo-integrations/cypress/tests/addons/phone/phone-non-strict.cy.ts new file mode 100644 index 000000000..e1e3b8fc8 --- /dev/null +++ b/projects/demo-integrations/cypress/tests/addons/phone/phone-non-strict.cy.ts @@ -0,0 +1,201 @@ +import {DemoPath} from '@demo/constants'; + +import {BROWSER_SUPPORTS_REAL_EVENTS} from '../../../support/constants'; + +describe('Phone', () => { + describe('Non-strict', () => { + beforeEach(() => { + cy.visit(`/${DemoPath.PhonePackage}/API?strict=false`); + cy.get('#demo-content input') + .should('be.visible') + .first() + .focus() + .as('input'); + }); + + describe('basic typing (1 character per keydown)', () => { + const tests = [ + // [Typed value, Masked value, caretIndex] + ['7920', '+7 920', '+7 920'.length], + ['7920341', '+7 920 341', '+7 920 341'.length], + ['792034156', '+7 920 341-56', '+7 920 341-56'.length], + ['79203415627', '+7 920 341-56-27', '+7 920 341-56-27'.length], + ['792034156274234123', '+7 920 341-56-27', '+7 920 341-56-27'.length], + ['79 nd4 e', '+7 94', '+7 94'.length], + ] as const; + + tests.forEach(([typedValue, maskedValue, caretIndex]) => { + it(`Type "${typedValue}" => "${maskedValue}"`, () => { + cy.get('@input') + .type(typedValue) + .should('have.value', maskedValue) + .should('have.prop', 'selectionStart', caretIndex) + .should('have.prop', 'selectionEnd', caretIndex); + }); + }); + }); + + describe('basic erasing (value = "+7 920 424-11-32"', () => { + beforeEach(() => { + cy.get('@input').type('+79204241132'); + }); + + const tests = [ + // [How many times "Backspace"-key was pressed, caretPosition, Masked value] + [1, '+7 920 424-11-3'.length, '+7 920 424-11-3'], + [2, '+7 920 424-11'.length, '+7 920 424-11'], + [3, '+7 920 424-1'.length, '+7 920 424-1'], + [4, '+7 920 424'.length, '+7 920 424'], + [13, ''.length, ''], + ] as const; + + tests.forEach(([n, caretIndex, maskedValue]) => { + it(`Backspace x${n} => "${maskedValue}"`, () => { + cy.get('@input') + .type('{backspace}'.repeat(n)) + .should('have.value', maskedValue) + .should('have.prop', 'selectionStart', caretIndex) + .should('have.prop', 'selectionEnd', caretIndex); + }); + }); + }); + + describe('Editing somewhere in the middle of a value (NOT the last character)', () => { + beforeEach(() => { + cy.get('@input').type('+7 920 424-11-32'); + }); + + it('+7 920 424-1|1-32 => Backspace => +7 920 424-|13-2 => Type "2" => +7 920 424-2|1-32', () => { + cy.get('@input') + .type('{leftArrow}'.repeat('13-2'.length)) + .type('{backspace}') + .should('have.value', '+7 920 424-13-2') + .should('have.prop', 'selectionStart', '+7 920 424-'.length) + .should('have.prop', 'selectionEnd', '+7 920 424-'.length) + .type('2') + .should('have.value', '+7 920 424-21-32') + .should('have.prop', 'selectionStart', '+7 920 424-2'.length) + .should('have.prop', 'selectionEnd', '+7 920 424-2'.length); + }); + + it('+7 9|20 424-11-32 => Backspace => +7 2|04241132', () => { + cy.get('@input') + .type('{leftArrow}'.repeat('20 424-11-32'.length)) + .type('{backspace}') + .should('have.value', '+7 204241132') + .should('have.prop', 'selectionStart', '+7 '.length) + .should('have.prop', 'selectionEnd', '+7 '.length); + }); + }); + + describe('Text selection', () => { + beforeEach(() => { + cy.get('@input').type('+7 920 424-11-32'); + }); + + describe( + 'Select range and press Backspace', + BROWSER_SUPPORTS_REAL_EVENTS, + () => { + it('+7 920 424-11-32 => Select "+7 920 424-|11|-32" => Backspace => +7 920 424-|32', () => { + cy.get('@input') + .type('{leftArrow}'.repeat('-32'.length)) + .realPress([ + 'Shift', + ...Array('11'.length).fill('ArrowLeft'), + ]); + + cy.get('@input') + .type('{backspace}') + .should('have.value', '+7 920 424-32') + .should('have.prop', 'selectionStart', '+7 920 424-'.length) + .should('have.prop', 'selectionEnd', '+7 920 424-'.length); + }); + + it('+7 920 424-11-32 => Select "+7 920 424-1|1-3|2" => Backspace => +7 920 424-1|2', () => { + cy.get('@input') + .type('{leftArrow}') + .realPress([ + 'Shift', + ...Array('1-3'.length).fill('ArrowLeft'), + ]); + + cy.get('@input') + .type('{backspace}') + .should('have.value', '+7 920 424-12') + .should('have.prop', 'selectionStart', '+7 920 424-1'.length) + .should('have.prop', 'selectionEnd', '+7 920 424-1'.length); + }); + }, + ); + + describe( + 'Select range and type a digit', + BROWSER_SUPPORTS_REAL_EVENTS, + () => { + it('+7 920 424-11-32 => Select "+7 920 424-|11|-32" => Type "5" => +7 920 424-5|3-2', () => { + cy.get('@input') + .type('{leftArrow}'.repeat('-32'.length)) + .realPress([ + 'Shift', + ...Array('11'.length).fill('ArrowLeft'), + ]); + + cy.get('@input') + .type('5') + .should('have.value', '+7 920 424-53-2') + .should('have.prop', 'selectionStart', '+7 920 424-5'.length) + .should('have.prop', 'selectionEnd', '+7 920 424-5'.length); + }); + + it('+7 920 424-11-32 => Select "+7 920 424-1|1-3|2" => Type "5" => +7 920 424-15-|2', () => { + cy.get('@input') + .type('{leftArrow}') + .realPress([ + 'Shift', + ...Array('1-3'.length).fill('ArrowLeft'), + ]); + + cy.get('@input') + .type('5') + .should('have.value', '+7 920 424-15-2') + .should( + 'have.prop', + 'selectionStart', + '+7 920 424-15-'.length, + ) + .should('have.prop', 'selectionEnd', '+7 920 424-15-'.length); + }); + }, + ); + }); + }); + + describe('Some countries', () => { + it('US: +1 212 343-3355', () => { + openCounrty('US'); + + cy.get('@input').type('12123433355'); + cy.get('@input').should('have.value', '+1 212 343-3355'); + }); + + it('KZ: +7 771 931-1111', () => { + openCounrty('KZ'); + + cy.get('@input').type('77719311111'); + cy.get('@input').should('have.value', '+7 771 931-1111'); + }); + + it('BY: +375 44 748-82-69', () => { + openCounrty('BY'); + + cy.get('@input').type('375447488269'); + cy.get('@input').should('have.value', '+375 44 748-82-69'); + }); + }); +}); + +function openCounrty(code: string): void { + cy.visit(`/${DemoPath.PhonePackage}/API?strict=false&countryCode=${code}`); + cy.get('#demo-content input').should('be.visible').first().focus().as('input'); +} diff --git a/projects/demo/src/pages/phone/examples/3-non-strict/component.ts b/projects/demo/src/pages/phone/examples/3-non-strict/component.ts new file mode 100644 index 000000000..c9799fe84 --- /dev/null +++ b/projects/demo/src/pages/phone/examples/3-non-strict/component.ts @@ -0,0 +1,35 @@ +import {ChangeDetectionStrategy, Component} from '@angular/core'; +import {maskitoGetCountryFromNumber} from '@maskito/phone'; +import metadata from 'libphonenumber-js/min/metadata'; + +import mask from './mask'; + +@Component({ + selector: 'phone-doc-example-3', + template: ` + + Basic + + + +
Country code: {{ countryIsoCode }}
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class PhoneMaskDocExample3 { + value = ''; + readonly mask = mask; + + get countryIsoCode(): string { + return maskitoGetCountryFromNumber(this.value, metadata) || ''; + } +} diff --git a/projects/demo/src/pages/phone/examples/3-non-strict/mask.ts b/projects/demo/src/pages/phone/examples/3-non-strict/mask.ts new file mode 100644 index 000000000..fb8bf58ac --- /dev/null +++ b/projects/demo/src/pages/phone/examples/3-non-strict/mask.ts @@ -0,0 +1,8 @@ +import {maskitoPhoneOptionsGenerator} from '@maskito/phone'; +import metadata from 'libphonenumber-js/min/metadata'; + +export default maskitoPhoneOptionsGenerator({ + countryIsoCode: 'RU', + metadata, + strict: false, +}); diff --git a/projects/demo/src/pages/phone/examples/4-lazy-metadata/component.ts b/projects/demo/src/pages/phone/examples/4-lazy-metadata/component.ts new file mode 100644 index 000000000..a056fddb3 --- /dev/null +++ b/projects/demo/src/pages/phone/examples/4-lazy-metadata/component.ts @@ -0,0 +1,38 @@ +import {ChangeDetectionStrategy, Component, OnInit} from '@angular/core'; +import {MASKITO_DEFAULT_OPTIONS} from '@maskito/core'; +import {maskitoPhoneOptionsGenerator} from '@maskito/phone'; + +@Component({ + selector: 'phone-doc-example-4', + template: ` + + Lazy metadata + + + `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class PhoneMaskDocExample4 implements OnInit { + value = '+7 920 123-4567'; + mask = MASKITO_DEFAULT_OPTIONS; + + async ngOnInit(): Promise { + const metadata = await import('libphonenumber-js/min/metadata').then( + m => m.default, + ); + + this.mask = maskitoPhoneOptionsGenerator({ + countryIsoCode: 'RU', + metadata, + }); + } +} diff --git a/projects/demo/src/pages/phone/phone-doc.component.ts b/projects/demo/src/pages/phone/phone-doc.component.ts index 317cd1e1d..78661774c 100644 --- a/projects/demo/src/pages/phone/phone-doc.component.ts +++ b/projects/demo/src/pages/phone/phone-doc.component.ts @@ -30,8 +30,25 @@ export class PhoneDocComponent implements GeneratorOptions { ), }; + readonly nonStrict: TuiDocExample = { + [DocExamplePrimaryTab.MaskitoOptions]: import( + './examples/3-non-strict/mask.ts?raw' + ), + }; + + readonly lazyMetadata: TuiDocExample = { + [DocExamplePrimaryTab.MaskitoOptions]: import( + './examples/2-validation/mask.ts?raw' + ), + [DocExamplePrimaryTab.Angular]: import( + './examples/2-validation/component.ts?raw' + ), + }; + metadata = metadata; + strict = true; + countryCodeVariants = getCountries(this.metadata); countryIsoCode: CountryCode = 'RU'; diff --git a/projects/demo/src/pages/phone/phone-doc.module.ts b/projects/demo/src/pages/phone/phone-doc.module.ts index 0a2ac8adf..0735cd6b0 100644 --- a/projects/demo/src/pages/phone/phone-doc.module.ts +++ b/projects/demo/src/pages/phone/phone-doc.module.ts @@ -6,6 +6,7 @@ import {MaskitoModule} from '@maskito/angular'; import {TuiAddonDocModule, tuiGenerateRoutes} from '@taiga-ui/addon-doc'; import { TuiErrorModule, + TuiFlagPipeModule, TuiLinkModule, TuiNotificationModule, TuiTextfieldControllerModule, @@ -14,6 +15,8 @@ import {TuiFieldErrorPipeModule, TuiInputModule} from '@taiga-ui/kit'; import {PhoneMaskDocExample1} from './examples/1-basic/component'; import {PhoneMaskDocExample2} from './examples/2-validation/component'; +import {PhoneMaskDocExample3} from './examples/3-non-strict/component'; +import {PhoneMaskDocExample4} from './examples/4-lazy-metadata/component'; import {PhoneDocComponent} from './phone-doc.component'; @NgModule({ @@ -24,6 +27,7 @@ import {PhoneDocComponent} from './phone-doc.component'; MaskitoModule, TuiAddonDocModule, TuiInputModule, + TuiFlagPipeModule, TuiLinkModule, TuiErrorModule, TuiNotificationModule, @@ -31,7 +35,13 @@ import {PhoneDocComponent} from './phone-doc.component'; TuiTextfieldControllerModule, RouterModule.forChild(tuiGenerateRoutes(PhoneDocComponent)), ], - declarations: [PhoneDocComponent, PhoneMaskDocExample1, PhoneMaskDocExample2], + declarations: [ + PhoneDocComponent, + PhoneMaskDocExample1, + PhoneMaskDocExample2, + PhoneMaskDocExample3, + PhoneMaskDocExample4, + ], exports: [PhoneDocComponent], }) export class PhoneDocModule {} diff --git a/projects/demo/src/pages/phone/phone-doc.template.html b/projects/demo/src/pages/phone/phone-doc.template.html index cff431c91..292f3608d 100644 --- a/projects/demo/src/pages/phone/phone-doc.template.html +++ b/projects/demo/src/pages/phone/phone-doc.template.html @@ -61,6 +61,65 @@

Below is an example of a Hungarian phone mask with an angular validator.

+ + + + + +

+ Setting the + strict + option to + false + enables non-strict mask mode. There is no strict country + code in the mask in this mode. +

+

+ The + countryIsoCode + option is optional in that case, but if you specify it, the + mask will try to add that country's calling code when you + try to insert a phone number without a calling code. +

+
+
+ + + + + +

+ You can load metadata lazily, below is an example of how to + do it in Angular. +

+

+ You can also + + customize the metadata + + to reduce metadata size. See instructions + + here + +

+
+
@@ -92,6 +151,15 @@ > Country ISO-code + + Strict mode + diff --git a/projects/phone/src/index.ts b/projects/phone/src/index.ts index 6879aa146..7eec94a8c 100644 --- a/projects/phone/src/index.ts +++ b/projects/phone/src/index.ts @@ -1 +1 @@ -export {maskitoPhoneOptionsGenerator} from './lib/masks'; +export {maskitoGetCountryFromNumber, maskitoPhoneOptionsGenerator} from './lib/masks'; diff --git a/projects/phone/src/lib/masks/index.ts b/projects/phone/src/lib/masks/index.ts index 80d274f5e..8e101ca9c 100644 --- a/projects/phone/src/lib/masks/index.ts +++ b/projects/phone/src/lib/masks/index.ts @@ -1 +1 @@ -export {maskitoPhoneOptionsGenerator} from './phone'; +export {maskitoGetCountryFromNumber, maskitoPhoneOptionsGenerator} from './phone'; diff --git a/projects/phone/src/lib/masks/phone/index.ts b/projects/phone/src/lib/masks/phone/index.ts index 42ed7a2d9..a401b6298 100644 --- a/projects/phone/src/lib/masks/phone/index.ts +++ b/projects/phone/src/lib/masks/phone/index.ts @@ -5,3 +5,4 @@ export { phoneLengthPostprocessorGenerator, validatePhonePreprocessorGenerator, } from './processors'; +export {maskitoGetCountryFromNumber} from './utils'; diff --git a/projects/phone/src/lib/masks/phone/phone-mask-free.ts b/projects/phone/src/lib/masks/phone/phone-mask-free.ts new file mode 100644 index 000000000..e3746e1ab --- /dev/null +++ b/projects/phone/src/lib/masks/phone/phone-mask-free.ts @@ -0,0 +1,38 @@ +import {MASKITO_DEFAULT_OPTIONS, MaskitoOptions} from '@maskito/core'; +import {AsYouType, CountryCode, MetadataJson} from 'libphonenumber-js/core'; + +import { + phoneLengthPostprocessorGenerator, + validatePhonePreprocessorGenerator, +} from './processors'; +import {generatePhoneMask, getPhoneTemplate} from './utils'; + +export function maskitoPhoneFreeOptionsGenerator({ + defaultIsoCode, + metadata, +}: { + defaultIsoCode?: CountryCode; + metadata: MetadataJson; +}): Required { + const formatter = new AsYouType(defaultIsoCode, metadata); + const prefix = `+`; + + return { + ...MASKITO_DEFAULT_OPTIONS, + mask: ({value}) => { + const template = getPhoneTemplate(formatter, value); + + const mask = generatePhoneMask({value, template, prefix}); + + return template.length === 1 ? ['+', /\d/] : mask; + }, + postprocessors: [phoneLengthPostprocessorGenerator(metadata)], + preprocessors: [ + validatePhonePreprocessorGenerator({ + prefix, + countryIsoCode: defaultIsoCode, + metadata, + }), + ], + }; +} diff --git a/projects/phone/src/lib/masks/phone/phone-mask-strict.ts b/projects/phone/src/lib/masks/phone/phone-mask-strict.ts new file mode 100644 index 000000000..a3eb85030 --- /dev/null +++ b/projects/phone/src/lib/masks/phone/phone-mask-strict.ts @@ -0,0 +1,57 @@ +import {MASKITO_DEFAULT_OPTIONS, MaskitoOptions} from '@maskito/core'; +import { + maskitoAddOnFocusPlugin, + maskitoCaretGuard, + maskitoPrefixPostprocessorGenerator, + maskitoRemoveOnBlurPlugin, +} from '@maskito/kit'; +import { + AsYouType, + CountryCode, + getCountryCallingCode, + MetadataJson, +} from 'libphonenumber-js/core'; + +import { + cutInitCountryCodePreprocessor, + phoneLengthPostprocessorGenerator, + validatePhonePreprocessorGenerator, +} from './processors'; +import {generatePhoneMask, getPhoneTemplate} from './utils'; + +export function maskitoPhoneStrictOptionsGenerator({ + countryIsoCode, + metadata, +}: { + countryIsoCode: CountryCode; + metadata: MetadataJson; +}): Required { + const code = getCountryCallingCode(countryIsoCode, metadata); + const formatter = new AsYouType(countryIsoCode, metadata); + const prefix = `+${code} `; + + return { + ...MASKITO_DEFAULT_OPTIONS, + mask: ({value}) => { + const template = getPhoneTemplate(formatter, value); + + return generatePhoneMask({value, template, prefix}); + }, + plugins: [ + maskitoCaretGuard((value, [from, to]) => [ + from === to ? prefix.length : 0, + value.length, + ]), + maskitoRemoveOnBlurPlugin(prefix), + maskitoAddOnFocusPlugin(prefix), + ], + postprocessors: [ + maskitoPrefixPostprocessorGenerator(prefix), + phoneLengthPostprocessorGenerator(metadata), + ], + preprocessors: [ + cutInitCountryCodePreprocessor({countryIsoCode, metadata}), + validatePhonePreprocessorGenerator({prefix, countryIsoCode, metadata}), + ], + }; +} diff --git a/projects/phone/src/lib/masks/phone/phone-mask.ts b/projects/phone/src/lib/masks/phone/phone-mask.ts index 7b4826143..d7e88021b 100644 --- a/projects/phone/src/lib/masks/phone/phone-mask.ts +++ b/projects/phone/src/lib/masks/phone/phone-mask.ts @@ -1,57 +1,19 @@ -import {MASKITO_DEFAULT_OPTIONS, MaskitoOptions} from '@maskito/core'; -import { - maskitoAddOnFocusPlugin, - maskitoCaretGuard, - maskitoPrefixPostprocessorGenerator, - maskitoRemoveOnBlurPlugin, -} from '@maskito/kit'; -import { - AsYouType, - CountryCode, - getCountryCallingCode, - MetadataJson, -} from 'libphonenumber-js/core'; +import {MaskitoOptions} from '@maskito/core'; +import {CountryCode, MetadataJson} from 'libphonenumber-js/core'; -import { - cutInitCountryCodePreprocessor, - phoneLengthPostprocessorGenerator, - validatePhonePreprocessorGenerator, -} from './processors'; -import {generatePhoneMask, getPhoneTemplate} from './utils'; +import {maskitoPhoneFreeOptionsGenerator} from './phone-mask-free'; +import {maskitoPhoneStrictOptionsGenerator} from './phone-mask-strict'; export function maskitoPhoneOptionsGenerator({ countryIsoCode, metadata, + strict = true, }: { - countryIsoCode: CountryCode; + countryIsoCode?: CountryCode; metadata: MetadataJson; + strict?: boolean; }): Required { - const code = getCountryCallingCode(countryIsoCode, metadata); - const formatter = new AsYouType(countryIsoCode, metadata); - const prefix = `+${code} `; - - return { - ...MASKITO_DEFAULT_OPTIONS, - mask: ({value}) => { - const template = getPhoneTemplate(formatter, value); - - return generatePhoneMask({value, template, prefix}); - }, - plugins: [ - maskitoCaretGuard((value, [from, to]) => [ - from === to ? prefix.length : 0, - value.length, - ]), - maskitoRemoveOnBlurPlugin(prefix), - maskitoAddOnFocusPlugin(prefix), - ], - postprocessors: [ - maskitoPrefixPostprocessorGenerator(prefix), - phoneLengthPostprocessorGenerator(metadata, countryIsoCode), - ], - preprocessors: [ - cutInitCountryCodePreprocessor({countryIsoCode, metadata}), - validatePhonePreprocessorGenerator({prefix, countryIsoCode, metadata}), - ], - }; + return strict && countryIsoCode + ? maskitoPhoneStrictOptionsGenerator({countryIsoCode, metadata}) + : maskitoPhoneFreeOptionsGenerator({defaultIsoCode: countryIsoCode, metadata}); } diff --git a/projects/phone/src/lib/masks/phone/processors/phone-length-postprocessor.ts b/projects/phone/src/lib/masks/phone/processors/phone-length-postprocessor.ts index e92b1c3bb..c608f2d4b 100644 --- a/projects/phone/src/lib/masks/phone/processors/phone-length-postprocessor.ts +++ b/projects/phone/src/lib/masks/phone/processors/phone-length-postprocessor.ts @@ -1,17 +1,16 @@ import {MaskitoPostprocessor} from '@maskito/core'; -import {CountryCode, MetadataJson} from 'libphonenumber-js/core'; +import {MetadataJson} from 'libphonenumber-js/core'; import {cutPhoneByValidLength} from '../utils'; const MIN_LENGTH = 3; export function phoneLengthPostprocessorGenerator( metadata: MetadataJson, - countryIsoCode: CountryCode, ): MaskitoPostprocessor { return ({value, selection}) => ({ value: value.length > MIN_LENGTH - ? cutPhoneByValidLength({phone: value, countryIsoCode, metadata}) + ? cutPhoneByValidLength({phone: value, metadata}) : value, selection, }); diff --git a/projects/phone/src/lib/masks/phone/processors/validate-phone-preprocessor.ts b/projects/phone/src/lib/masks/phone/processors/validate-phone-preprocessor.ts index 6a77263a3..a91af76b2 100644 --- a/projects/phone/src/lib/masks/phone/processors/validate-phone-preprocessor.ts +++ b/projects/phone/src/lib/masks/phone/processors/validate-phone-preprocessor.ts @@ -12,7 +12,7 @@ export function validatePhonePreprocessorGenerator({ metadata, }: { prefix: string; - countryIsoCode: CountryCode; + countryIsoCode?: CountryCode; metadata: MetadataJson; }): MaskitoPreprocessor { return ({elementState, data}) => { @@ -29,14 +29,17 @@ export function validatePhonePreprocessorGenerator({ try { const validationError = validatePhoneNumberLength( data, - countryIsoCode, + {defaultCountry: countryIsoCode}, metadata, ); if (!validationError) { // handle past-event with different code, for example for 8 / +7 - const phone = parsePhoneNumber(data, countryIsoCode, metadata); - const nationalSignificantNumber = phone.nationalNumber; + const phone = countryIsoCode + ? parsePhoneNumber(data, countryIsoCode, metadata) + : parsePhoneNumber(data, metadata); + + const {nationalNumber, countryCallingCode} = phone; return { elementState: { @@ -44,8 +47,8 @@ export function validatePhonePreprocessorGenerator({ value: selectionIncludesPrefix ? '' : prefix, }, data: selectionIncludesPrefix - ? `${prefix}${nationalSignificantNumber}` - : nationalSignificantNumber, + ? `+${countryCallingCode} ${nationalNumber}` + : nationalNumber, }; } } catch { diff --git a/projects/phone/src/lib/masks/phone/utils/cut-phone-by-valid-length.ts b/projects/phone/src/lib/masks/phone/utils/cut-phone-by-valid-length.ts index 2bf765af2..8985d6546 100644 --- a/projects/phone/src/lib/masks/phone/utils/cut-phone-by-valid-length.ts +++ b/projects/phone/src/lib/masks/phone/utils/cut-phone-by-valid-length.ts @@ -1,16 +1,10 @@ -import { - CountryCode, - MetadataJson, - validatePhoneNumberLength, -} from 'libphonenumber-js/core'; +import {MetadataJson, validatePhoneNumberLength} from 'libphonenumber-js/core'; export function cutPhoneByValidLength({ phone, - countryIsoCode, metadata, }: { phone: string; - countryIsoCode: CountryCode; metadata: MetadataJson; }): string { const validationResult = validatePhoneNumberLength(phone, metadata); @@ -18,7 +12,6 @@ export function cutPhoneByValidLength({ if (validationResult === 'INVALID_LENGTH' || validationResult === 'TOO_LONG') { return cutPhoneByValidLength({ phone: phone.slice(0, phone.length - 1), - countryIsoCode, metadata, }); } diff --git a/projects/phone/src/lib/masks/phone/utils/get-country-from-number.ts b/projects/phone/src/lib/masks/phone/utils/get-country-from-number.ts new file mode 100644 index 000000000..8d00c0847 --- /dev/null +++ b/projects/phone/src/lib/masks/phone/utils/get-country-from-number.ts @@ -0,0 +1,12 @@ +import {AsYouType, CountryCode, MetadataJson} from 'libphonenumber-js/core'; + +export function maskitoGetCountryFromNumber( + number: string, + metadata: MetadataJson, +): CountryCode | undefined { + const formatter = new AsYouType({}, metadata); + + formatter.input(number); + + return formatter.getCountry(); +} diff --git a/projects/phone/src/lib/masks/phone/utils/index.ts b/projects/phone/src/lib/masks/phone/utils/index.ts index 84365115c..4bae960af 100644 --- a/projects/phone/src/lib/masks/phone/utils/index.ts +++ b/projects/phone/src/lib/masks/phone/utils/index.ts @@ -1,3 +1,4 @@ export * from './cut-phone-by-valid-length'; export * from './generate-phone-mask'; +export * from './get-country-from-number'; export * from './get-phone-template';