@lbgm/phone-input
Version:
An Angular phone input module
275 lines • 53.4 kB
JavaScript
import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
import parsePhoneNumber from "libphonenumber-js";
import allCountries from './all-countries';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "@angular/forms";
import * as i3 from "./typing.directive";
export var FormControlEvent;
(function (FormControlEvent) {
FormControlEvent["INVALID"] = "INVALID";
FormControlEvent["VALID"] = "VALID";
})(FormControlEvent || (FormControlEvent = {}));
export class PhoneInputComponent {
constructor(el, cd) {
this.el = el;
this.cd = cd;
this.value = "";
this.label = "";
this.hasError = false;
this.hasSuccess = false;
this.placeholder = "";
this.name = "lbgm-phone-input";
this.required = false;
this.defaultCountry = 'BJ';
this.arrow = true;
this.listHeight = 150;
this.allowed = ([]);
this.phoneEvent = new EventEmitter(true);
this.phoneData = new EventEmitter(true);
this.country = new EventEmitter(true);
this.countries = allCountries;
this.openSelect = false;
this.phone = "";
this.popupPos = "bottom";
this.focus = false;
}
/**
* TrackBy for *ngFor
* @param index
* @param item
* @returns
*/
trackByCountry(index, country) {
return index;
// you can also return item.X;
}
ngOnInit() {
// initialize default country selected
this.defaultSelected = this.formatPhoneInput(this.value);
}
ngOnChanges(changes) {
//
// watch changes on @Input() here
}
ngDoCheck() {
// this.cd.markForCheck();
}
ngAfterViewInit() {
this.emitAll();
// outside
document.addEventListener("click", (event) => {
var _a, _b;
if (((_a = this.basePhoneArrow) === null || _a === void 0 ? void 0 : _a.nativeElement) &&
!((_b = this.basePhoneArrow) === null || _b === void 0 ? void 0 : _b.nativeElement).contains(event.target)) {
this.openSelect = false;
}
});
}
/**
* used to send custom Event: usable in case of scroll turning off when popup is under
*/
cev_dash_select() {
var _a;
const event = new CustomEvent("CEV_SELECT_POPUP", {
detail: { opened: this.openSelect, target: (_a = this.selectPhone) === null || _a === void 0 ? void 0 : _a.nativeElement },
});
document.body.dispatchEvent(event);
}
/**
* filt allowedCountries from props
*/
get allowedCountries() {
const tbl = this.allowed.length !== 0
? this.countries.filter((o) => this.allowed.includes(o.iso2))
: this.countries;
return tbl;
}
get getGroup() {
return this.group;
}
get getName() {
return this.name;
}
get fieldError() {
var _a;
if (!this.controls)
return (_a = this.hasError) !== null && _a !== void 0 ? _a : false;
const f = this.controls[this.name];
return f.status === FormControlEvent.INVALID && f.touched && this.required;
}
get fieldSuccess() {
var _a;
if (!this.controls)
return (_a = this.hasSuccess) !== null && _a !== void 0 ? _a : false;
const f = this.controls[this.name];
return f.status === FormControlEvent.VALID && f.touched && this.required;
}
/**
* ToggleSelect
* to open countries list
*/
toggleSelect() {
var _a;
this.openSelect = !this.openSelect;
// calculate popup position: top or bottom
const selectRect = (_a = this.selectPhone) === null || _a === void 0 ? void 0 : _a.nativeElement.getBoundingClientRect();
// y
this.popupPos = selectRect.bottom < this.listHeight ? "top" : "bottom";
//
this.cev_dash_select();
}
;
/**
* formatPhoneInput
* used to format Phone Input
* @param val
*/
formatPhoneInput(val) {
var _a;
const phoneNumber = parsePhoneNumber(`+${val}`);
if (phoneNumber) {
this.phone = phoneNumber.nationalNumber;
return {
iso2: phoneNumber === null || phoneNumber === void 0 ? void 0 : phoneNumber.country,
dialCode: phoneNumber === null || phoneNumber === void 0 ? void 0 : phoneNumber.countryCallingCode,
name: (_a = this.countries.find((o) => o.iso2 === (phoneNumber === null || phoneNumber === void 0 ? void 0 : phoneNumber.country))) === null || _a === void 0 ? void 0 : _a.name,
};
}
// else
return Object.assign({}, this.countries.find((o) => o.iso2 === this.defaultCountry));
}
;
/**
* emitPhone
* used to emit phone in internationalFormat
*/
emitPhone() {
if (this.phone)
this.phoneEvent.emit(`${this.defaultSelected.dialCode}${this.phone}`);
else
this.phoneEvent.emit("");
}
;
/**
* emitPhoneData
* Used to emit phoneData as an object
* @returns {}
*/
emitPhoneData(event) {
void event;
const ph = parsePhoneNumber(`+${this.defaultSelected.dialCode}${this.phone}`);
this.phoneData.emit({
country: ph === null || ph === void 0 ? void 0 : ph.country,
dialCode: ph === null || ph === void 0 ? void 0 : ph.countryCallingCode,
nationalNumber: ph === null || ph === void 0 ? void 0 : ph.nationalNumber,
number: ph === null || ph === void 0 ? void 0 : ph.number,
isValid: ph === null || ph === void 0 ? void 0 : ph.isValid(),
});
}
;
/**
* emitAll
* used to emit all event
*/
emitAll() {
this.country.emit(this.defaultSelected.iso2);
this.emitPhone();
this.emitPhoneData();
}
/**
* to select any country
* @param country
*/
choose(country) {
this.defaultSelected = country;
this.openSelect = false;
this.emitAll();
}
;
/**
* bind on input
* @param event
*/
onPhoneInput(event) {
event.target.value = this.phone = String(event.target.value).replace(/\D/g, "");
this.emitPhone();
}
;
focusOn(event) {
switch (event.type) {
case "mouseenter":
this.focus = true;
break;
case "mouseleave":
this.focus = false;
break;
default:
break;
}
}
}
PhoneInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: PhoneInputComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
PhoneInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: PhoneInputComponent, selector: "lbgm-phone-input", inputs: { value: "value", label: "label", hasError: "hasError", hasSuccess: "hasSuccess", placeholder: "placeholder", name: "name", required: "required", defaultCountry: "defaultCountry", arrow: "arrow", listHeight: "listHeight", allowed: "allowed", group: "group", controls: "controls" }, outputs: { phoneEvent: "phoneEvent", phoneData: "phoneData", country: "country" }, host: { listeners: { "mouseleave": "focusOn($event)", "mouseenter": "focusOn($event)" } }, viewQueries: [{ propertyName: "basePhoneArrow", first: true, predicate: ["basePhoneArrow"], descendants: true }, { propertyName: "inputBase", first: true, predicate: ["inputBase"], descendants: true }, { propertyName: "arrowIcon", first: true, predicate: ["arrowIcon"], descendants: true, static: true }, { propertyName: "selectPhone", first: true, predicate: ["selectPhone"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #selectPhone\n data-widget-item=\"baseinput\"\n class=\"flex flex-col relative\"\n data-lbgm-phonenumberinput=\"component\"\n>\n <!--phone-number-input core-->\n <div\n data-children=\"core\"\n class=\"w-full flex flex-col relative\"\n >\n <!---->\n <label\n *ngIf=\"label\"\n class=\"cursor-pointer baseinput-label p-3px text-xs text-left leading-3 tracking-mz1px text-3dark font-medium select-none bg-white absolute left-2 -translate-y-2\"\n [ngClass]=\"{ 'text-red': fieldError, 'text-blue': focus || fieldSuccess }\"\n [for]=\"name\"\n data-children=\"label\"\n >\n <span data-children=\"labelText\">{{ label }}</span>\n <!---->\n <span\n *ngIf=\"required\"\n data-children=\"requiredStar\"\n class=\"text-left text-DA1414 font-semibold opacity-80 text-xs\"\n > *</span\n >\n </label>\n <!--input-->\n <div\n data-children=\"inputcore\"\n #selectPhoneButton\n class=\"bg-white baseinput-core border w-full border-3dark rounded-md py-2 px-4 flex flex-shrink flex-nowrap items-center space-x-2\"\n [ngClass]=\"{ 'border-red': fieldError, 'border-blue': fieldSuccess, 'border-blue-50': focus }\"\n >\n <span\n (click)=\"toggleSelect()\"\n class=\"inline-flex flex-nowrap items-center space-x-2 cursor-pointer\"\n #basePhoneArrow\n data-children=\"arrowGroup\"\n >\n <span *ngIf=\"arrow; then iconTemplate\"></span>\n <ng-template #iconTemplate>\n <span #arrowIcon [attr.data-arrow-icon]=\"arrowIcon.children.length\" class=\"inline-flex flex-shrink-0 select-none\">\n <ng-content select=\"[arrow]\"></ng-content>\n </span>\n <span *ngIf=\"!arrowIcon.children.length\" [attr.data-arrow-icon]=\"arrowIcon.children.length\" class=\"inline-flex flex-shrink-0\">\n <svg\n width=\"12\"\n height=\"6\"\n viewBox=\"0 0 12 6\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n fill-rule=\"evenodd\"\n clip-rule=\"evenodd\"\n d=\"M0.96967 0.21967C1.26256 -0.0732233 1.73744 -0.0732233 2.03033 0.21967L6 4.18934L9.96967 0.21967C10.2626 -0.0732233 10.7374 -0.0732233 11.0303 0.21967C11.3232 0.512563 11.3232 0.987437 11.0303 1.28033L6.53033 5.78033C6.23744 6.07322 5.76256 6.07322 5.46967 5.78033L0.96967 1.28033C0.676777 0.987437 0.676777 0.512563 0.96967 0.21967Z\"\n [attr.fill]=\"focus ? 'rgb(29 144 237)' : 'rgba(51, 51, 51, 0.5)'\"\n />\n </svg>\n </span>\n </ng-template>\n <span\n class=\"opacity-80 select-none inline-flex flex-whrink-0 font-medium text-3dark text-left text-xs leading-4\"\n [ngClass]=\"{ 'text-blue': focus }\"\n >\n {{ '+' + defaultSelected.dialCode }}\n </span>\n </span>\n <!-- INPUT ELEMENT -->\n <span *ngIf=\"group; then withGroup; else withoutGroup;\"></span>\n <ng-template #withGroup>\n <div class=\"w-full\" [formGroup]=\"getGroup\">\n <input\n #inputBase\n inputTyping\n data-children=\"htmlInput\"\n [placeholder]=\"placeholder\"\n [formControlName]=\"getName\"\n [name]=\"name\"\n [id]=\"name\"\n [value]=\"phone\"\n [autocomplete]=\"'off'\"\n [inputInterval]=\"300\"\n (input)=\"onPhoneInput($event)\"\n (finish)=\"emitPhoneData()\"\n spellcheck=\"false\"\n class=\"border-0 outline-none appearance-none flex-shrink w-full bg-transparent text-3dark text-base\"\n [ngClass]=\"{ 'text-blue': focus }\"\n type=\"text\"\n />\n </div>\n </ng-template>\n\n <ng-template #withoutGroup>\n <input\n #inputBase\n inputTyping\n data-children=\"htmlInput\"\n [placeholder]=\"placeholder\"\n [name]=\"name\"\n [id]=\"name\"\n [value]=\"phone\"\n [autocomplete]=\"'off'\"\n [inputInterval]=\"300\"\n (input)=\"onPhoneInput($event)\"\n (finish)=\"emitPhoneData()\"\n spellcheck=\"false\"\n class=\"border-0 outline-none appearance-none flex-shrink w-full bg-transparent text-3dark text-base\"\n [ngClass]=\"{ 'text-blue': focus }\"\n type=\"text\"\n />\n </ng-template>\n </div>\n <!--select option-->\n <div\n #selectOptions\n class=\"w-full rounded border border-3dark-2 bg-white absolute z-1 lbgm-phone-scrll\"\n *ngIf=\"openSelect\"\n data-children=\"countriesList\"\n [ngClass]=\"{ 'bottom-0': popupPos === 'top', 'mt-z281rem top-full': popupPos === 'bottom' }\"\n [style.maxHeight]=\"listHeight + 'px'\"\n >\n <div\n class=\"w-full py-2 px-4 cursor-pointer text-left hover:bg-3dark-1\"\n *ngFor=\"let country of allowedCountries; let lastitr = last; trackBy: trackByCountry\"\n (click)=\"choose(country)\"\n [attr.data-country]=\"country.iso2\"\n >\n <span class=\"font-semibold text-xs text-3dark\">\n {{ country.name }}\n </span>\n </div>\n </div>\n <!-- end core-->\n </div>\n\n <!-- any slot -->\n <ng-content select=\"[slot]\"></ng-content>\n</div>", styles: ["*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:currentColor}:before,:after{--tw-content: \"\"}html{line-height:1.5;-webkit-text-size-adjust:100%;tab-size:4;font-family:ui-sans-serif,system-ui,-apple-system,BlinkMacSystemFont,\"Segoe UI\",Roboto,\"Helvetica Neue\",Arial,\"Noto Sans\",sans-serif,\"Apple Color Emoji\",\"Segoe UI Emoji\",\"Segoe UI Symbol\",\"Noto Color Emoji\";font-feature-settings:normal;font-variation-settings:normal}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,\"Liberation Mono\",\"Courier New\",monospace;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-size:100%;font-weight:inherit;line-height:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,[type=button],[type=reset],[type=submit]{-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dl,dd,h1,h2,h3,h4,h5,h6,hr,figure,p,pre{margin:0}fieldset{margin:0;padding:0}legend{padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}textarea{resize:vertical}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]{display:none}*,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: }::-webkit-backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: }.static{position:static}.absolute{position:absolute}.relative{position:relative}.bottom-0{bottom:0px}.left-2{left:.5rem}.top-full{top:100%}.z-1{z-index:1}.mt-z281rem{margin-top:.281rem}.flex{display:flex}.inline-flex{display:inline-flex}.w-full{width:100%}.flex-shrink{flex-shrink:1}.flex-shrink-0{flex-shrink:0}.-translate-y-2{--tw-translate-y: -.5rem;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.cursor-pointer{cursor:pointer}.select-none{-webkit-user-select:none;user-select:none}.appearance-none{-webkit-appearance:none;appearance:none}.flex-col{flex-direction:column}.flex-nowrap{flex-wrap:nowrap}.items-center{align-items:center}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(.5rem * var(--tw-space-x-reverse));margin-left:calc(.5rem * calc(1 - var(--tw-space-x-reverse)))}.rounded{border-radius:.25rem}.rounded-md{border-radius:.375rem}.border{border-width:1px}.border-0{border-width:0px}.border-3dark{border-color:#33333380}.border-3dark-2{border-color:#3333}.border-blue{--tw-border-opacity: 1;border-color:rgb(29 144 237/var(--tw-border-opacity))}.border-blue-50{border-color:#1d90ed80}.border-red{--tw-border-opacity: 1;border-color:rgb(255 0 0/var(--tw-border-opacity))}.bg-transparent{background-color:transparent}.bg-white{--tw-bg-opacity: 1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}.p-3px{padding:3px}.px-4{padding-left:1rem;padding-right:1rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.text-left{text-align:left}.text-base{font-size:1rem;line-height:1.5rem}.text-xs{font-size:.75rem;line-height:1rem}.font-medium{font-weight:500}.font-semibold{font-weight:600}.leading-3{line-height:.75rem}.leading-4{line-height:1rem}.tracking-mz1px{letter-spacing:-.1px}.text-3dark{color:#33333380}.text-DA1414{--tw-text-opacity: 1;color:rgb(218 20 20/var(--tw-text-opacity))}.text-blue{--tw-text-opacity: 1;color:rgb(29 144 237/var(--tw-text-opacity))}.text-red{--tw-text-opacity: 1;color:rgb(255 0 0/var(--tw-text-opacity))}.opacity-80{opacity:.8}.outline-none{outline:2px solid transparent;outline-offset:2px}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.hover\\:bg-3dark-1:hover{background-color:#3333331a}\n", ":host{display:block;font-family:-apple-system,BlinkMacSystemFont,\"Segoe UI\",Roboto,Helvetica,Arial,sans-serif,\"Apple Color Emoji\",\"Segoe UI Emoji\",\"Segoe UI Symbol\";color:#333;box-sizing:border-box;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;position:relative}@media screen and (-webkit-min-device-pixel-ratio: 0){.lbgm-phone-scrll::-webkit-scrollbar{width:6px;height:6px}.lbgm-phone-scrll::-webkit-scrollbar-track{background:transparent}.lbgm-phone-scrll::-webkit-scrollbar-thumb{border-radius:6px;background:rgba(0,106,82,.8);border:0;display:none}.lbgm-phone-scrll:hover.lbgm-phone-scrll::-webkit-scrollbar-thumb{display:initial}.lbgm-phone-scrll::-webkit-scrollbar-corner{display:none}}@-moz-document url-prefix(){.lbgm-phone-scrll{scrollbar-color:rgba(0,106,82,.8) transparent;scrollbar-width:none}.lbgm-phone-scrll:hover{scrollbar-width:thin}}[data-widget-item=baseinput]{transition:all .2s ease-in-out}[data-widget-item=baseinput] *{box-sizing:border-box}[data-widget-item=baseinput] label.error{color:red}[data-widget-item=baseinput] label.success{color:#1d90ed}[data-widget-item=baseinput] [data-children=countriesList]{overflow-y:auto;filter:drop-shadow(0 5px 15px rgba(0,0,0,.15))}[data-widget-item=baseinput] [data-children=inputcore] input::placeholder{font-weight:normal;color:#333}[data-widget-item=baseinput] [data-children=inputcore] input:-webkit-autofill,[data-widget-item=baseinput] [data-children=inputcore] input:-webkit-autofill:hover,[data-widget-item=baseinput] [data-children=inputcore] input:-webkit-autofill:focus,[data-widget-item=baseinput] [data-children=inputcore] input:-webkit-autofill:active{-webkit-transition:background-color 5000s ease-in-out 0s;transition:background-color 5000s ease-in-out 0s}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i3.TypingDirective, selector: "[inputTyping]", inputs: ["inputInterval"], outputs: ["run", "finish"] }, { type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i2.FormControlName, selector: "[formControlName]", inputs: ["disabled", "formControlName", "ngModel"], outputs: ["ngModelChange"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: PhoneInputComponent, decorators: [{
type: Component,
args: [{
selector: 'lbgm-phone-input',
templateUrl: './phone-input.component.html',
styleUrls: [
'./tailwind.scss',
'./phone-input.component.scss',
]
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { value: [{
type: Input
}], label: [{
type: Input
}], hasError: [{
type: Input
}], hasSuccess: [{
type: Input
}], placeholder: [{
type: Input
}], name: [{
type: Input
}], required: [{
type: Input
}], defaultCountry: [{
type: Input
}], arrow: [{
type: Input
}], listHeight: [{
type: Input
}], allowed: [{
type: Input
}], group: [{
type: Input
}], controls: [{
type: Input
}], basePhoneArrow: [{
type: ViewChild,
args: ['basePhoneArrow']
}], inputBase: [{
type: ViewChild,
args: ['inputBase']
}], arrowIcon: [{
type: ViewChild,
args: ['arrowIcon', { static: true }]
}], selectPhone: [{
type: ViewChild,
args: ['selectPhone']
}], phoneEvent: [{
type: Output
}], phoneData: [{
type: Output
}], country: [{
type: Output
}], focusOn: [{
type: HostListener,
args: ['mouseleave', ['$event']]
}, {
type: HostListener,
args: ['mouseenter', ['$event']]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"phone-input.component.js","sourceRoot":"","sources":["../../../../projects/phone-input/src/lib/phone-input.component.ts","../../../../projects/phone-input/src/lib/phone-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, OnInit, Output, SimpleChanges, ViewChild } from '@angular/core';\nimport parsePhoneNumber, { PhoneNumber } from \"libphonenumber-js\";\n\nimport allCountries, { T_Country } from './all-countries';\nimport { AbstractControl, FormGroup } from '@angular/forms';\n\nexport { T_Country };\n\nexport type T_FormFieldControl = { [key: string]: AbstractControl; };\n\nexport interface PhoneDATA {\n  country?: string;\n  dialCode?: string | number;\n  nationalNumber?: string | number;\n  number?: string | number;\n  isValid?: boolean;\n}\n\nexport enum FormControlEvent {\n  INVALID = 'INVALID',\n  VALID = 'VALID'\n}\n\n@Component({\n  selector: 'lbgm-phone-input',\n  templateUrl: './phone-input.component.html',\n  styleUrls: [\n    './tailwind.scss',\n    './phone-input.component.scss',\n  ]\n})\nexport class PhoneInputComponent implements OnInit, AfterViewInit, OnChanges {\n\n  @Input() value?: string = \"\";\n  @Input() label?: string = \"\";\n  @Input() hasError?: boolean = false;\n  @Input() hasSuccess?: boolean = false;\n  @Input() placeholder?: string = \"\"\n  @Input() name?: string = \"lbgm-phone-input\"\n  @Input() required?: boolean = false;\n  @Input() defaultCountry?: string = 'BJ';\n  @Input() arrow?: boolean = true;\n  @Input() listHeight?: number = 150;\n  @Input() allowed?: string[] = ([]);\n\n  @Input() group?: FormGroup;\n  @Input() controls?: T_FormFieldControl;\n\n  @ViewChild('basePhoneArrow') basePhoneArrow?: ElementRef;\n  @ViewChild('inputBase') inputBase?: ElementRef;\n  @ViewChild('arrowIcon', { static: true }) arrowIcon?: ElementRef;\n  @ViewChild('selectPhone') selectPhone?: ElementRef;\n\n  @Output() phoneEvent = new EventEmitter<string>(true);\n  @Output() phoneData = new EventEmitter<PhoneDATA>(true);\n  @Output() country = new EventEmitter<string>(true);\n\n  countries: T_Country[] = allCountries;\n  openSelect: boolean = false;\n  defaultSelected!: T_Country;\n  phone: string = \"\";\n  popupPos: string = \"bottom\";\n  focus: boolean = false;\n\n  constructor(public el: ElementRef, private cd: ChangeDetectorRef) { }\n\n\n  /**\n   * TrackBy for *ngFor\n   * @param index\n   * @param item\n   * @returns\n   */\n   trackByCountry(index: number, country: T_Country): number {\n    return index;\n    // you can also return item.X;\n  }\n\n  ngOnInit(): void {\n    // initialize default country selected\n    this.defaultSelected = this.formatPhoneInput(this.value as string);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    //\n    // watch changes on @Input() here\n  }\n\n  ngDoCheck() {\n    // this.cd.markForCheck();\n  }\n\n  ngAfterViewInit(): void {\n    this.emitAll();\n    // outside\n    document.addEventListener(\"click\", (event) => {\n      if (\n        this.basePhoneArrow?.nativeElement &&\n        !(this.basePhoneArrow?.nativeElement as HTMLElement).contains(event.target as Node)\n      ) {\n        this.openSelect = false;\n      }\n    });\n  }\n\n\n  /**\n  * used to send custom Event: usable in case of scroll turning off when popup is under\n  */\n  cev_dash_select(): void {\n\n    const event = new CustomEvent(\"CEV_SELECT_POPUP\", {\n      detail: { opened: this.openSelect, target: this.selectPhone?.nativeElement },\n    });\n\n    document.body.dispatchEvent(event);\n  }\n\n  /**\n  * filt allowedCountries from props\n  */\n  get allowedCountries(): T_Country[] {\n    const tbl: T_Country[] =\n      (this.allowed as string[]).length !== 0\n        ? this.countries.filter((o: T_Country) => (this.allowed as string[]).includes(o.iso2))\n        : this.countries;\n    return tbl;\n  }\n\n  get getGroup (): FormGroup {\n    return this.group as FormGroup;\n  }\n\n  get getName (): string {\n    return this.name as string;\n  }\n\n  get fieldError(): boolean {\n    if(!this.controls) return this.hasError ?? false;\n    const f = (this.controls as T_FormFieldControl)[this.name as string];\n    return f.status === FormControlEvent.INVALID && f.touched && this.required as boolean;\n  }\n\n  get fieldSuccess(): boolean {\n    if(!this.controls) return this.hasSuccess ?? false;\n    const f = (this.controls as T_FormFieldControl)[this.name as string];\n    return f.status === FormControlEvent.VALID && f.touched && this.required as boolean;\n  }\n\n\n  /**\n  * ToggleSelect\n  * to open countries list\n  */\n  toggleSelect(): void {\n    this.openSelect = !this.openSelect;\n\n    // calculate popup position: top or bottom\n    const selectRect = this.selectPhone?.nativeElement.getBoundingClientRect();\n    // y\n    this.popupPos = selectRect.bottom < (this.listHeight as number) ? \"top\" : \"bottom\";\n    //\n    this.cev_dash_select();\n  };\n\n\n  /**\n  * formatPhoneInput\n  * used to format Phone Input\n  * @param val\n  */\n  formatPhoneInput (val: string): T_Country {\n    const phoneNumber: PhoneNumber | undefined = parsePhoneNumber(`+${val}`);\n    if (phoneNumber) {\n      this.phone = phoneNumber.nationalNumber;\n\n      return {\n        iso2: phoneNumber?.country as string,\n        dialCode: phoneNumber?.countryCallingCode as string,\n        name: this.countries.find((o: T_Country) => o.iso2 === phoneNumber?.country as string)?.name as string,\n      };\n    }\n    // else\n    return {\n      ...this.countries.find((o: T_Country) => o.iso2 === this.defaultCountry) as T_Country,\n    };\n  };\n\n  /**\n  * emitPhone\n  * used to emit phone in internationalFormat\n  */\n  emitPhone(): void {\n    if (this.phone)\n      this.phoneEvent.emit(`${this.defaultSelected.dialCode}${this.phone}`);\n    else this.phoneEvent.emit(\"\");\n  };\n\n\n  /**\n  * emitPhoneData\n  * Used to emit phoneData as an object\n  * @returns {}\n  */\n  emitPhoneData(event?: Event): void {\n    void event;\n    const ph = parsePhoneNumber(\n      `+${this.defaultSelected.dialCode}${this.phone}`\n    );\n    this.phoneData.emit({\n      country: ph?.country,\n      dialCode: ph?.countryCallingCode,\n      nationalNumber: ph?.nationalNumber,\n      number: ph?.number,\n      isValid: ph?.isValid(),\n    });\n  };\n\n  /**\n  * emitAll\n  * used to emit all event\n  */\n  emitAll(): void {\n    this.country.emit(this.defaultSelected.iso2);\n    this.emitPhone();\n    this.emitPhoneData();\n  }\n\n  /**\n  * to select any country\n  * @param country\n  */\n  choose(country: T_Country) {\n    this.defaultSelected = country;\n    this.openSelect = false;\n    this.emitAll();\n  };\n\n\n  /**\n  * bind on input\n  * @param event\n  */\n  onPhoneInput(event: Event): void {\n    (event.target as HTMLInputElement).value = this.phone = String((event.target as HTMLInputElement).value).replace(\n      /\\D/g,\n      \"\"\n    );\n    this.emitPhone();\n  };\n\n\n  @HostListener('mouseleave', ['$event'])\n  @HostListener('mouseenter', ['$event'])\n  focusOn(event: Event): void {\n   switch (event.type) {\n    case \"mouseenter\":\n      this.focus = true;\n    break;\n    case \"mouseleave\":\n      this.focus = false;\n    break;\n    default:\n    break;\n   }\n  }\n\n}\n","<div\n    #selectPhone\n    data-widget-item=\"baseinput\"\n    class=\"flex flex-col relative\"\n    data-lbgm-phonenumberinput=\"component\"\n>\n        <!--phone-number-input core-->\n        <div\n          data-children=\"core\"\n          class=\"w-full flex flex-col relative\"\n        >\n          <!---->\n          <label\n            *ngIf=\"label\"\n            class=\"cursor-pointer baseinput-label p-3px text-xs text-left leading-3 tracking-mz1px text-3dark font-medium select-none bg-white absolute left-2 -translate-y-2\"\n            [ngClass]=\"{ 'text-red': fieldError, 'text-blue': focus || fieldSuccess }\"\n            [for]=\"name\"\n            data-children=\"label\"\n          >\n            <span data-children=\"labelText\">{{ label }}</span>\n            <!---->\n            <span\n              *ngIf=\"required\"\n              data-children=\"requiredStar\"\n              class=\"text-left text-DA1414 font-semibold opacity-80 text-xs\"\n              >&thinsp;*</span\n            >\n          </label>\n          <!--input-->\n          <div\n            data-children=\"inputcore\"\n            #selectPhoneButton\n            class=\"bg-white baseinput-core border w-full border-3dark rounded-md py-2 px-4 flex flex-shrink flex-nowrap items-center space-x-2\"\n            [ngClass]=\"{ 'border-red': fieldError, 'border-blue': fieldSuccess, 'border-blue-50': focus }\"\n          >\n            <span\n              (click)=\"toggleSelect()\"\n              class=\"inline-flex flex-nowrap items-center space-x-2 cursor-pointer\"\n              #basePhoneArrow\n              data-children=\"arrowGroup\"\n            >\n              <span *ngIf=\"arrow; then iconTemplate\"></span>\n              <ng-template #iconTemplate>\n                <span #arrowIcon [attr.data-arrow-icon]=\"arrowIcon.children.length\" class=\"inline-flex flex-shrink-0 select-none\">\n                  <ng-content select=\"[arrow]\"></ng-content>\n                </span>\n                <span *ngIf=\"!arrowIcon.children.length\" [attr.data-arrow-icon]=\"arrowIcon.children.length\" class=\"inline-flex flex-shrink-0\">\n                  <svg\n                    width=\"12\"\n                    height=\"6\"\n                    viewBox=\"0 0 12 6\"\n                    fill=\"none\"\n                    xmlns=\"http://www.w3.org/2000/svg\"\n                  >\n                    <path\n                      fill-rule=\"evenodd\"\n                      clip-rule=\"evenodd\"\n                      d=\"M0.96967 0.21967C1.26256 -0.0732233 1.73744 -0.0732233 2.03033 0.21967L6 4.18934L9.96967 0.21967C10.2626 -0.0732233 10.7374 -0.0732233 11.0303 0.21967C11.3232 0.512563 11.3232 0.987437 11.0303 1.28033L6.53033 5.78033C6.23744 6.07322 5.76256 6.07322 5.46967 5.78033L0.96967 1.28033C0.676777 0.987437 0.676777 0.512563 0.96967 0.21967Z\"\n                      [attr.fill]=\"focus ? 'rgb(29 144 237)' : 'rgba(51, 51, 51, 0.5)'\"\n                    />\n                  </svg>\n                </span>\n              </ng-template>\n              <span\n                class=\"opacity-80 select-none inline-flex flex-whrink-0 font-medium text-3dark text-left text-xs leading-4\"\n                [ngClass]=\"{ 'text-blue': focus }\"\n              >\n              {{ '+' + defaultSelected.dialCode }}\n              </span>\n            </span>\n            <!-- INPUT ELEMENT -->\n            <span *ngIf=\"group; then withGroup; else withoutGroup;\"></span>\n            <ng-template #withGroup>\n              <div class=\"w-full\" [formGroup]=\"getGroup\">\n                <input\n                    #inputBase\n                    inputTyping\n                    data-children=\"htmlInput\"\n                    [placeholder]=\"placeholder\"\n                    [formControlName]=\"getName\"\n             