ngx-mat-intl-tel-input-v16
Version:
319 lines • 186 kB
JavaScript
/* 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";
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 {
static { this.nextId = 0; }
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;
}
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.label = '';
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;
}
}
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();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.8", ngImport: i0, type: NgxMatIntlTelInputComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.CountryCode }, { token: i2.FocusMonitor }, { token: i0.ElementRef }, { token: i3.NgControl, optional: true, self: true }, { token: i3.NgForm, optional: true }, { token: i3.FormGroupDirective, optional: true }, { token: i4.ErrorStateMatcher }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.8", type: NgxMatIntlTelInputComponent, isStandalone: true, selector: "ngx-mat-intl-tel-input", inputs: { preferredCountries: "preferredCountries", enablePlaceholder: "enablePlaceholder", inputPlaceholder: "inputPlaceholder", cssClass: "cssClass", name: "name", onlyCountries: "onlyCountries", errorStateMatcher: "errorStateMatcher", enableSearch: "enableSearch", searchPlaceholder: "searchPlaceholder", describedBy: "describedBy", label: "label", format: "format", placeholder: "placeholder", required: "required", disabled: "disabled" }, outputs: { countryChanged: "countryChanged" }, host: { properties: { "id": "this.id", "class.ngx-floating": "this.shouldLabelFloat" } }, providers: [
CountryCode,
{ provide: MatFormFieldControl, useExisting: NgxMatIntlTelInputComponent },
{
provide: NG_VALIDATORS,
useValue: phoneNumberValidator,
multi: true,
},
], viewQueries: [{ propertyName: "matMenu", first: true, predicate: MatMenu, descendants: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"ngx-mat-tel-input-container\">\r\n <button\r\n type=\"button\"\r\n mat-button\r\n [matMenuTriggerFor]=\"menu\"\r\n class=\"country-selector\"\r\n [disabled]=\"disabled\"\r\n >\r\n <span\r\n class=\"country-selector-flag flag\"\r\n [ngClass]=\"selectedCountry?.flagClass\"\r\n ></span>\r\n <span class=\"country-selector-code\" *ngIf=\"selectedCountry?.dialCode\"\r\n >+{{ selectedCountry.dialCode }}</span\r\n >\r\n </button>\r\n <mat-menu #menu=\"matMenu\"\r\n class=\"ngx-mat-tel-input-mat-menu-panel\"\r\n backdropClass=\"ngx-mat-tel-input-overlay-backdrop\"\r\n overlayPanelClass=\"ngx-mat-tel-input-overlay-pane\">\r\n <input\r\n *ngIf=\"enableSearch\"\r\n class=\"country-search\"\r\n [(ngModel)]=\"searchCriteria\"\r\n type=\"text\"\r\n [placeholder]=\"searchPlaceholder\"\r\n (click)=\"$event.stopPropagation()\"\r\n />\r\n <button\r\n type=\"button\"\r\n mat-menu-item\r\n class=\"country-list-button\"\r\n *ngFor=\"let country of preferredCountriesInDropDown\"\r\n (click)=\"onCountrySelect(country, focusable)\"\r\n >\r\n <div class=\"icon-wrapper\">\r\n <div class=\"flag\" [ngClass]=\"country.flagClass\"></div>\r\n </div>\r\n <div class=\"label-wrapper\">\r\n {{ country.name }}\r\n <span *ngIf=\"country?.dialCode\">+{{ country.dialCode }}</span>\r\n </div>\r\n </button>\r\n <mat-divider *ngIf=\"preferredCountriesInDropDown?.length\"></mat-divider>\r\n <ng-container *ngFor=\"let country of allCountries\">\r\n <button\r\n type=\"button\"\r\n mat-menu-item\r\n class=\"country-list-button\"\r\n *ngIf=\"country | search: searchCriteria\"\r\n (click)=\"onCountrySelect(country, focusable)\"\r\n >\r\n <div class=\"icon-wrapper\">\r\n <div class=\"flag\" [ngClass]=\"country.flagClass\"></div>\r\n </div>\r\n <div class=\"label-wrapper\">\r\n {{ country.name }} +{{ country.dialCode }}\r\n </div>\r\n </button>\r\n </ng-container>\r\n </mat-menu>\r\n <label [for]=\"focusable.id\" class=\"sr-only\">{{label||focusable.id}}</label>\r\n <input\r\n matInput\r\n type=\"tel\"\r\n autocomplete=\"off\"\r\n [ngClass]=\"cssClass\"\r\n (blur)=\"onTouched()\"\r\n (keypress)=\"onInputKeyPress($event)\"\r\n [(ngModel)]=\"phoneNumber\"\r\n (ngModelChange)=\"onPhoneNumberChange()\"\r\n [errorStateMatcher]=\"errorStateMatcher\"\r\n [placeholder]=\"inputPlaceholder\"\r\n [disabled]=\"disabled\"\r\n [aria-describedby]=\"describedBy\"\r\n #focusable\r\n />\r\n</div>\r\n", styles: ["input: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{width:100%;height:34px;border:none;border-bottom:1px solid #ddd;font-size:14px;padding:10px}.icon-wrapper{padding-right:24px}.flag{background-image:url(