UNPKG

ngx-mat-intl-tel-input

Version:

An Angular Material package for entering and validating international telephone numbers. It adds a flag dropdown to any input, detects the user's country, displays a relevant placeholder and provides formatting/validation methods.

438 lines 195 kB
/* eslint-disable @typescript-eslint/member-ordering, no-underscore-dangle, id-blacklist, id-match, @typescript-eslint/naming-convention */ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostBinding, Input, Optional, Output, Self, ViewChild, } from '@angular/core'; import { MatFormFieldControl } from '@angular/material/form-field'; import { FormGroupDirective, FormsModule, NgControl, NgForm, NG_VALIDATORS, ReactiveFormsModule, } from '@angular/forms'; import { AsYouType, getExampleNumber, parsePhoneNumberFromString, } from 'libphonenumber-js'; import { CountryCode, Examples } from './data/country-code'; import { phoneNumberValidator } from './ngx-mat-intl-tel-input.validator'; import { FocusMonitor } from '@angular/cdk/a11y'; import { coerceBooleanProperty } from '@angular/cdk/coercion'; import { CommonModule } from '@angular/common'; import { MatButtonModule } from '@angular/material/button'; import { ErrorStateMatcher, mixinErrorState, } from '@angular/material/core'; import { MatDividerModule } from '@angular/material/divider'; import { MatInputModule } from '@angular/material/input'; import { MatMenu, MatMenuModule } from '@angular/material/menu'; import { Subject } from 'rxjs'; import { SearchPipe } from './search.pipe'; import * as i0 from "@angular/core"; import * as i1 from "./data/country-code"; import * as i2 from "@angular/cdk/a11y"; import * as i3 from "@angular/forms"; import * as i4 from "@angular/material/core"; import * as i5 from "@angular/common"; import * as i6 from "@angular/material/input"; import * as i7 from "@angular/material/menu"; import * as i8 from "@angular/material/button"; import * as i9 from "@angular/material/divider"; function NgxMatIntlTelInputComponent_span_3_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "span", 12); i0.ɵɵtext(1); i0.ɵɵelementEnd(); } if (rf & 2) { const ctx_r0 = i0.ɵɵnextContext(); i0.ɵɵadvance(1); i0.ɵɵtextInterpolate1("+", ctx_r0.selectedCountry.dialCode, ""); } } function NgxMatIntlTelInputComponent_input_6_Template(rf, ctx) { if (rf & 1) { const _r8 = i0.ɵɵgetCurrentView(); i0.ɵɵelementStart(0, "input", 13); i0.ɵɵlistener("ngModelChange", function NgxMatIntlTelInputComponent_input_6_Template_input_ngModelChange_0_listener($event) { i0.ɵɵrestoreView(_r8); const ctx_r7 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r7.searchCriteria = $event); })("click", function NgxMatIntlTelInputComponent_input_6_Template_input_click_0_listener($event) { return $event.stopPropagation(); }); i0.ɵɵelementEnd(); } if (rf & 2) { const ctx_r2 = i0.ɵɵnextContext(); i0.ɵɵproperty("ngModel", ctx_r2.searchCriteria)("placeholder", ctx_r2.searchPlaceholder); } } function NgxMatIntlTelInputComponent_button_7_span_5_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "span"); i0.ɵɵtext(1); i0.ɵɵelementEnd(); } if (rf & 2) { const country_r10 = i0.ɵɵnextContext().$implicit; i0.ɵɵadvance(1); i0.ɵɵtextInterpolate1("+", country_r10.dialCode, ""); } } function NgxMatIntlTelInputComponent_button_7_Template(rf, ctx) { if (rf & 1) { const _r14 = i0.ɵɵgetCurrentView(); i0.ɵɵelementStart(0, "button", 14); i0.ɵɵlistener("click", function NgxMatIntlTelInputComponent_button_7_Template_button_click_0_listener() { const restoredCtx = i0.ɵɵrestoreView(_r14); const country_r10 = restoredCtx.$implicit; const ctx_r13 = i0.ɵɵnextContext(); const _r6 = i0.ɵɵreference(11); return i0.ɵɵresetView(ctx_r13.onCountrySelect(country_r10, _r6)); }); i0.ɵɵelementStart(1, "div", 15); i0.ɵɵelement(2, "div", 16); i0.ɵɵelementEnd(); i0.ɵɵelementStart(3, "div", 17); i0.ɵɵtext(4); i0.ɵɵtemplate(5, NgxMatIntlTelInputComponent_button_7_span_5_Template, 2, 1, "span", 8); i0.ɵɵelementEnd()(); } if (rf & 2) { const country_r10 = ctx.$implicit; i0.ɵɵadvance(2); i0.ɵɵproperty("ngClass", country_r10.flagClass); i0.ɵɵadvance(2); i0.ɵɵtextInterpolate1(" ", country_r10.name, " "); i0.ɵɵadvance(1); i0.ɵɵproperty("ngIf", country_r10 == null ? null : country_r10.dialCode); } } function NgxMatIntlTelInputComponent_mat_divider_8_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelement(0, "mat-divider"); } } function NgxMatIntlTelInputComponent_ng_container_9_button_1_Template(rf, ctx) { if (rf & 1) { const _r19 = i0.ɵɵgetCurrentView(); i0.ɵɵelementStart(0, "button", 14); i0.ɵɵlistener("click", function NgxMatIntlTelInputComponent_ng_container_9_button_1_Template_button_click_0_listener() { i0.ɵɵrestoreView(_r19); const country_r15 = i0.ɵɵnextContext().$implicit; const ctx_r17 = i0.ɵɵnextContext(); const _r6 = i0.ɵɵreference(11); return i0.ɵɵresetView(ctx_r17.onCountrySelect(country_r15, _r6)); }); i0.ɵɵelementStart(1, "div", 15); i0.ɵɵelement(2, "div", 16); i0.ɵɵelementEnd(); i0.ɵɵelementStart(3, "div", 17); i0.ɵɵtext(4); i0.ɵɵelementEnd()(); } if (rf & 2) { const country_r15 = i0.ɵɵnextContext().$implicit; i0.ɵɵadvance(2); i0.ɵɵproperty("ngClass", country_r15.flagClass); i0.ɵɵadvance(2); i0.ɵɵtextInterpolate2(" ", country_r15.name, " +", country_r15.dialCode, " "); } } function NgxMatIntlTelInputComponent_ng_container_9_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementContainerStart(0); i0.ɵɵtemplate(1, NgxMatIntlTelInputComponent_ng_container_9_button_1_Template, 5, 3, "button", 18); i0.ɵɵpipe(2, "search"); i0.ɵɵelementContainerEnd(); } if (rf & 2) { const country_r15 = ctx.$implicit; const ctx_r5 = i0.ɵɵnextContext(); i0.ɵɵadvance(1); i0.ɵɵproperty("ngIf", i0.ɵɵpipeBind2(2, 1, country_r15, ctx_r5.searchCriteria)); } } class NgxMatIntlTelInputBase { constructor(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, /** @docs-private */ ngControl) { this._defaultErrorStateMatcher = _defaultErrorStateMatcher; this._parentForm = _parentForm; this._parentFormGroup = _parentFormGroup; this.ngControl = ngControl; this.stateChanges = new Subject(); } } const _NgxMatIntlTelInputMixinBase = mixinErrorState(NgxMatIntlTelInputBase); export class NgxMatIntlTelInputComponent extends _NgxMatIntlTelInputMixinBase { constructor(_changeDetectorRef, countryCodeData, fm, elRef, ngControl, _parentForm, _parentFormGroup, _defaultErrorStateMatcher) { super(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl); this._changeDetectorRef = _changeDetectorRef; this.countryCodeData = countryCodeData; this.fm = fm; this.elRef = elRef; this.ngControl = ngControl; this.preferredCountries = []; this.enablePlaceholder = true; this.onlyCountries = []; this.errorStateMatcher = new ErrorStateMatcher(); this.enableSearch = false; this.describedBy = ''; this._required = false; this._disabled = false; this.stateChanges = new Subject(); this.focused = false; this.id = `ngx-mat-intl-tel-input-${NgxMatIntlTelInputComponent.nextId++}`; this.allCountries = []; this.preferredCountriesInDropDown = []; this.countryChanged = new EventEmitter(); this._format = 'default'; this.onTouched = () => { }; this.propagateChange = (_) => { }; fm.monitor(elRef, true).subscribe((origin) => { if (this.focused && !origin) { this.onTouched(); } this.focused = !!origin; this.stateChanges.next(undefined); }); this.fetchCountryData(); if (this.ngControl != null) { this.ngControl.valueAccessor = this; } } get format() { return this._format; } set format(value) { this._format = value; this.phoneNumber = this.formattedPhoneNumber; this.stateChanges.next(undefined); } static getPhoneNumberPlaceHolder(countryISOCode) { const result = getExampleNumber(countryISOCode, Examples); return !!result ? result.number.toString() : undefined; } ngOnInit() { if (!this.searchPlaceholder) { this.searchPlaceholder = 'Search ...'; } if (this.preferredCountries.length) { this.preferredCountries.forEach((iso2) => { const preferredCountry = this.allCountries .filter((c) => c.iso2 === iso2) .shift(); if (preferredCountry) { this.preferredCountriesInDropDown.push(preferredCountry); } }); } if (this.onlyCountries.length) { this.allCountries = this.allCountries.filter((c) => this.onlyCountries.includes(c.iso2)); } if (this.numberInstance && this.numberInstance.country) { // If an existing number is present, we use it to determine selectedCountry this.selectedCountry = this.getCountry(this.numberInstance.country); } else { if (this.preferredCountriesInDropDown.length) { this.selectedCountry = this.preferredCountriesInDropDown[0]; } else { this.selectedCountry = this.allCountries[0]; } } this.countryChanged.emit(this.selectedCountry); this._changeDetectorRef.markForCheck(); this.stateChanges.next(undefined); } ngDoCheck() { if (this.ngControl) { this.updateErrorState(); } } onPhoneNumberChange() { try { this.numberInstance = parsePhoneNumberFromString(this.phoneNumber?.toString() || '', this.selectedCountry?.iso2.toUpperCase()); this.formatAsYouTypeIfEnabled(); this.value = this.numberInstance?.number; if (this.numberInstance && this.numberInstance.isValid()) { if (this.phoneNumber !== this.formattedPhoneNumber) { this.phoneNumber = this.formattedPhoneNumber; } if (this.selectedCountry?.iso2 !== this.numberInstance.country && !!this.numberInstance.country) { this.selectedCountry = this.getCountry(this.numberInstance.country); this.countryChanged.emit(this.selectedCountry); } } } catch (e) { // if no possible numbers are there, // then the full number is passed so that validator could be triggered and proper error could be shown this.value = this.phoneNumber?.toString(); } this.propagateChange(this.value); this._changeDetectorRef.markForCheck(); } onCountrySelect(country, el) { if (this.phoneNumber) { this.phoneNumber = this.numberInstance?.nationalNumber; } this.selectedCountry = country; this.countryChanged.emit(this.selectedCountry); this.onPhoneNumberChange(); el.focus(); } getCountry(code) { return (this.allCountries.find((c) => c.iso2 === code.toLowerCase()) || { name: 'UN', iso2: 'UN', dialCode: '', priority: 0, areaCodes: undefined, flagClass: 'UN', placeHolder: '', }); } onInputKeyPress(event) { const pattern = /[0-9+\- ]/; if (!pattern.test(event.key)) { event.preventDefault(); } } fetchCountryData() { this.countryCodeData.allCountries.forEach((c) => { const country = { name: c[0].toString(), iso2: c[1].toString(), dialCode: c[2].toString(), priority: +c[3] || 0, areaCodes: c[4] || undefined, flagClass: c[1].toString().toUpperCase(), placeHolder: '', }; if (this.enablePlaceholder) { country.placeHolder = NgxMatIntlTelInputComponent.getPhoneNumberPlaceHolder(country.iso2.toUpperCase()); } this.allCountries.push(country); }); } registerOnChange(fn) { this.propagateChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.disabled = isDisabled; this._changeDetectorRef.markForCheck(); this.stateChanges.next(undefined); } writeValue(value) { if (value) { this.numberInstance = parsePhoneNumberFromString(value); if (this.numberInstance) { const countryCode = this.numberInstance.country; this.phoneNumber = this.formattedPhoneNumber; if (!countryCode) { return; } setTimeout(() => { this.selectedCountry = this.getCountry(countryCode); if (this.selectedCountry.dialCode && !this.preferredCountries.includes(this.selectedCountry.iso2)) { this.preferredCountriesInDropDown.push(this.selectedCountry); } this.countryChanged.emit(this.selectedCountry); // Initial value is set this._changeDetectorRef.markForCheck(); this.stateChanges.next(undefined); }, 1); } else { this.phoneNumber = value; } } // Value is set from outside using setValue() this._changeDetectorRef.markForCheck(); this.stateChanges.next(undefined); } get empty() { return !this.phoneNumber; } get shouldLabelFloat() { return this.focused || !this.empty; } get placeholder() { return this._placeholder || ''; } set placeholder(value) { this._placeholder = value; this.stateChanges.next(undefined); } get required() { return this._required; } set required(value) { this._required = coerceBooleanProperty(value); this.stateChanges.next(undefined); } get disabled() { return this._disabled; } set disabled(value) { this._disabled = coerceBooleanProperty(value); this.stateChanges.next(undefined); } setDescribedByIds(ids) { this.describedBy = ids.join(' '); } onContainerClick(event) { if (event.target.tagName.toLowerCase() !== 'input') { // eslint-disable-next-line @typescript-eslint/no-non-null-assertion this.elRef.nativeElement.querySelector('input').focus(); } } reset() { this.phoneNumber = ''; this.propagateChange(null); this._changeDetectorRef.markForCheck(); this.stateChanges.next(undefined); } ngOnDestroy() { this.stateChanges.complete(); this.fm.stopMonitoring(this.elRef); } get formattedPhoneNumber() { if (!this.numberInstance) { return this.phoneNumber?.toString() || ''; } switch (this.format) { case 'national': return this.numberInstance.formatNational(); case 'international': return this.numberInstance.formatInternational(); default: return this.numberInstance.nationalNumber.toString(); } } formatAsYouTypeIfEnabled() { if (this.format === 'default') { return; } const asYouType = new AsYouType(this.selectedCountry?.iso2.toUpperCase()); // To avoid caret positioning we apply formatting only if the caret is at the end: if (this.phoneNumber ?.toString() .startsWith(this.previousFormattedNumber || '')) { this.phoneNumber = asYouType.input(this.phoneNumber.toString()); } this.previousFormattedNumber = this.phoneNumber?.toString(); } } NgxMatIntlTelInputComponent.nextId = 0; NgxMatIntlTelInputComponent.ɵfac = function NgxMatIntlTelInputComponent_Factory(t) { return new (t || NgxMatIntlTelInputComponent)(i0.ɵɵdirectiveInject(i0.ChangeDetectorRef), i0.ɵɵdirectiveInject(i1.CountryCode), i0.ɵɵdirectiveInject(i2.FocusMonitor), i0.ɵɵdirectiveInject(i0.ElementRef), i0.ɵɵdirectiveInject(i3.NgControl, 10), i0.ɵɵdirectiveInject(i3.NgForm, 8), i0.ɵɵdirectiveInject(i3.FormGroupDirective, 8), i0.ɵɵdirectiveInject(i4.ErrorStateMatcher)); }; NgxMatIntlTelInputComponent.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: NgxMatIntlTelInputComponent, selectors: [["ngx-mat-intl-tel-input"]], viewQuery: function NgxMatIntlTelInputComponent_Query(rf, ctx) { if (rf & 1) { i0.ɵɵviewQuery(MatMenu, 5); } if (rf & 2) { let _t; i0.ɵɵqueryRefresh(_t = i0.ɵɵloadQuery()) && (ctx.matMenu = _t.first); } }, hostVars: 3, hostBindings: function NgxMatIntlTelInputComponent_HostBindings(rf, ctx) { if (rf & 2) { i0.ɵɵhostProperty("id", ctx.id); i0.ɵɵclassProp("ngx-floating", ctx.shouldLabelFloat); } }, inputs: { preferredCountries: "preferredCountries", enablePlaceholder: "enablePlaceholder", inputPlaceholder: "inputPlaceholder", cssClass: "cssClass", name: "name", onlyCountries: "onlyCountries", errorStateMatcher: "errorStateMatcher", enableSearch: "enableSearch", searchPlaceholder: "searchPlaceholder", describedBy: "describedBy", format: "format", placeholder: "placeholder", required: "required", disabled: "disabled" }, outputs: { countryChanged: "countryChanged" }, standalone: true, features: [i0.ɵɵProvidersFeature([ CountryCode, { provide: MatFormFieldControl, useExisting: NgxMatIntlTelInputComponent }, { provide: NG_VALIDATORS, useValue: phoneNumberValidator, multi: true, }, ]), i0.ɵɵInheritDefinitionFeature, i0.ɵɵStandaloneFeature], decls: 12, vars: 14, consts: [[1, "ngx-mat-tel-input-container"], ["type", "button", "mat-button", "", 1, "country-selector", 3, "matMenuTriggerFor", "disabled"], [1, "country-selector-flag", "flag", 3, "ngClass"], ["class", "country-selector-code", 4, "ngIf"], ["backdropClass", "ngx-mat-tel-input-overlay-backdrop", "overlayPanelClass", "ngx-mat-tel-input-overlay-pane", 1, "ngx-mat-tel-input-mat-menu-panel"], ["menu", "matMenu"], ["class", "country-search", "type", "text", 3, "ngModel", "placeholder", "ngModelChange", "click", 4, "ngIf"], ["type", "button", "mat-menu-item", "", "class", "country-list-button", 3, "click", 4, "ngFor", "ngForOf"], [4, "ngIf"], [4, "ngFor", "ngForOf"], ["matInput", "", "type", "tel", "autocomplete", "off", 3, "ngClass", "ngModel", "errorStateMatcher", "placeholder", "disabled", "aria-describedby", "blur", "keypress", "ngModelChange"], ["focusable", ""], [1, "country-selector-code"], ["type", "text", 1, "country-search", 3, "ngModel", "placeholder", "ngModelChange", "click"], ["type", "button", "mat-menu-item", "", 1, "country-list-button", 3, "click"], [1, "icon-wrapper"], [1, "flag", 3, "ngClass"], [1, "label-wrapper"], ["type", "button", "mat-menu-item", "", "class", "country-list-button", 3, "click", 4, "ngIf"]], template: function NgxMatIntlTelInputComponent_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "div", 0)(1, "button", 1); i0.ɵɵelement(2, "span", 2); i0.ɵɵtemplate(3, NgxMatIntlTelInputComponent_span_3_Template, 2, 1, "span", 3); i0.ɵɵelementEnd(); i0.ɵɵelementStart(4, "mat-menu", 4, 5); i0.ɵɵtemplate(6, NgxMatIntlTelInputComponent_input_6_Template, 1, 2, "input", 6); i0.ɵɵtemplate(7, NgxMatIntlTelInputComponent_button_7_Template, 6, 3, "button", 7); i0.ɵɵtemplate(8, NgxMatIntlTelInputComponent_mat_divider_8_Template, 1, 0, "mat-divider", 8); i0.ɵɵtemplate(9, NgxMatIntlTelInputComponent_ng_container_9_Template, 3, 4, "ng-container", 9); i0.ɵɵelementEnd(); i0.ɵɵelementStart(10, "input", 10, 11); i0.ɵɵlistener("blur", function NgxMatIntlTelInputComponent_Template_input_blur_10_listener() { return ctx.onTouched(); })("keypress", function NgxMatIntlTelInputComponent_Template_input_keypress_10_listener($event) { return ctx.onInputKeyPress($event); })("ngModelChange", function NgxMatIntlTelInputComponent_Template_input_ngModelChange_10_listener($event) { return ctx.phoneNumber = $event; })("ngModelChange", function NgxMatIntlTelInputComponent_Template_input_ngModelChange_10_listener() { return ctx.onPhoneNumberChange(); }); i0.ɵɵelementEnd()(); } if (rf & 2) { const _r1 = i0.ɵɵreference(5); i0.ɵɵadvance(1); i0.ɵɵproperty("matMenuTriggerFor", _r1)("disabled", ctx.disabled); i0.ɵɵadvance(1); i0.ɵɵproperty("ngClass", ctx.selectedCountry == null ? null : ctx.selectedCountry.flagClass); i0.ɵɵadvance(1); i0.ɵɵproperty("ngIf", ctx.selectedCountry == null ? null : ctx.selectedCountry.dialCode); i0.ɵɵadvance(3); i0.ɵɵproperty("ngIf", ctx.enableSearch); i0.ɵɵadvance(1); i0.ɵɵproperty("ngForOf", ctx.preferredCountriesInDropDown); i0.ɵɵadvance(1); i0.ɵɵproperty("ngIf", ctx.preferredCountriesInDropDown == null ? null : ctx.preferredCountriesInDropDown.length); i0.ɵɵadvance(1); i0.ɵɵproperty("ngForOf", ctx.allCountries); i0.ɵɵadvance(1); i0.ɵɵproperty("ngClass", ctx.cssClass)("ngModel", ctx.phoneNumber)("errorStateMatcher", ctx.errorStateMatcher)("placeholder", ctx.inputPlaceholder)("disabled", ctx.disabled)("aria-describedby", ctx.describedBy); } }, dependencies: [CommonModule, i5.NgClass, i5.NgForOf, i5.NgIf, FormsModule, i3.DefaultValueAccessor, i3.NgControlStatus, i3.NgModel, MatInputModule, i6.MatInput, MatMenuModule, i7.MatMenu, i7.MatMenuItem, i7.MatMenuTrigger, MatButtonModule, i8.MatButton, MatDividerModule, i9.MatDivider, ReactiveFormsModule, SearchPipe], styles: ["input[_ngcontent-%COMP%]:not(.country-search){border:none;background:none;outline:none;font:inherit;width:100%;box-sizing:border-box;padding:0 6px 0 90px;position:relative;z-index:0;margin-top:0!important;margin-bottom:0!important;margin-right:0;margin-left:0}input.country-search[_ngcontent-%COMP%]{width:100%;height:34px;border:none;border-bottom:1px solid #ddd;font-size:14px;padding:20px 20px 24px}.icon-wrapper[_ngcontent-%COMP%]{padding-right:24px}.flag[_ngcontent-%COMP%]{background-image:url(