@tolle_/tolle-ui 18.2.16 → 18.2.19

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,4 @@
1
- import { Component, EventEmitter, forwardRef, inject, Input, Output, ChangeDetectorRef } from '@angular/core';
1
+ import { Component, EventEmitter, forwardRef, inject, Input, Output, ChangeDetectorRef, } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
4
4
  import { CountrySelectorComponent } from './country-selector.component';
@@ -6,7 +6,8 @@ import { MaskedInputComponent } from './masked-input.component';
6
6
  import parsePhoneNumber from 'libphonenumber-js';
7
7
  import { cn } from './utils/cn';
8
8
  import * as i0 from "@angular/core";
9
- import * as i1 from "@angular/forms";
9
+ import * as i1 from "@angular/common";
10
+ import * as i2 from "@angular/forms";
10
11
  export class PhoneNumberInputComponent {
11
12
  cdr = inject(ChangeDetectorRef);
12
13
  id = `phone-input-${Math.random().toString(36).substr(2, 9)}`;
@@ -34,6 +35,27 @@ export class PhoneNumberInputComponent {
34
35
  constructor() {
35
36
  this.selectedIso = this.defaultCountryCode;
36
37
  }
38
+ get computedLabelClass() {
39
+ return cn('text-sm font-medium text-foreground leading-none transition-opacity duration-200', this.disabled && 'opacity-50');
40
+ }
41
+ get computedMergedClass() {
42
+ return cn('group relative flex items-center w-full rounded-md border transition-all duration-200', 'bg-background border-input shadow-sm', this.size === 'xs' && 'h-8', this.size === 'sm' && 'h-9', this.size === 'default' && 'h-10', this.size === 'lg' && 'h-11',
43
+ // Focus state
44
+ !this.readonly &&
45
+ !this.disabled && [
46
+ 'focus-within:ring-4',
47
+ 'focus-within:ring-ring/30',
48
+ 'focus-within:ring-offset-0',
49
+ 'focus-within:shadow-none',
50
+ this.error ? 'focus-within:border-destructive/80' : 'focus-within:border-primary/80',
51
+ ],
52
+ // Error state
53
+ this.error && 'border-destructive', this.error && !this.readonly && !this.disabled && 'focus-within:ring-destructive/30',
54
+ // Disabled state
55
+ this.disabled && ['cursor-not-allowed opacity-50', 'border-opacity-50'],
56
+ // Readonly state
57
+ this.readonly && 'cursor-default border-dashed', this.class);
58
+ }
37
59
  writeValue(value) {
38
60
  if (value) {
39
61
  if (typeof value === 'object' && value.number) {
@@ -107,75 +129,99 @@ export class PhoneNumberInputComponent {
107
129
  {
108
130
  provide: NG_VALUE_ACCESSOR,
109
131
  useExisting: forwardRef(() => PhoneNumberInputComponent),
110
- multi: true
111
- }
132
+ multi: true,
133
+ },
112
134
  ], ngImport: i0, template: `
113
- <tolle-masked-input
114
- [id]="id"
115
- [label]="label"
116
- [hint]="hint"
117
- [error]="error"
118
- [errorMessage]="errorMessage"
119
- [hideHintOnFocus]="hideHintOnFocus"
120
- [mask]="mask"
121
- [size]="size"
122
- [disabled]="disabled"
123
- [readonly]="readonly"
124
- [placeholder]="placeholder"
125
- [(ngModel)]="displayValue"
126
- (ngModelChange)="onMaskInputChange($event)"
127
- [containerClass]="cn('pl-0', class)"
128
- >
129
- <tolle-country-selector
130
- prefix
131
- class="country-selector-override"
132
- [showName]="false"
133
- [size]="size"
134
- [disabled]="disabled || !enableCountrySelector"
135
- [readonly]="readonly"
136
- [(ngModel)]="selectedIso"
137
- (ngModelChange)="onCountryChange($event)"
138
- ></tolle-country-selector>
139
- </tolle-masked-input>
140
- `, isInline: true, styles: [":host{display:block;width:100%}::ng-deep .country-selector-override{display:flex;align-items:center;border-right:1px solid var(--border, #e5e7eb)}::ng-deep .country-selector-override button{border:none!important;border-radius:0!important;border-top-left-radius:calc(var(--radius, .5rem) - 1px)!important;border-bottom-left-radius:calc(var(--radius, .5rem) - 1px)!important;background:transparent!important;box-shadow:none!important;padding-left:.75rem!important;padding-right:.5rem!important;display:flex!important;align-items:center!important;justify-content:center!important;gap:.25rem!important}::ng-deep .country-selector-override button .flex{gap:.25rem!important}::ng-deep .country-selector-override button i{margin-left:0!important;line-height:1!important;display:flex!important;align-items:center!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: CountrySelectorComponent, selector: "tolle-country-selector", inputs: ["id", "label", "hint", "errorMessage", "error", "hideHintOnFocus", "placeholder", "class", "disabled", "readonly", "size", "defaultCountryCode", "returnValue", "showName"], outputs: ["onSelect"] }, { kind: "component", type: MaskedInputComponent, selector: "tolle-masked-input", inputs: ["id", "label", "hint", "errorMessage", "mask", "placeholder", "type", "disabled", "readonly", "class", "containerClass", "error", "size", "returnRaw", "hideHintOnFocus"] }] });
135
+ <div class="flex w-full flex-col gap-1.5">
136
+ <label *ngIf="label" [for]="id" [class]="computedLabelClass">
137
+ {{ label }}
138
+ </label>
139
+
140
+ <div [class]="computedMergedClass">
141
+ <tolle-country-selector
142
+ [showName]="false"
143
+ [size]="size"
144
+ [disabled]="disabled || !enableCountrySelector"
145
+ [readonly]="readonly"
146
+ [mergedPosition]="'left'"
147
+ [(ngModel)]="selectedIso"
148
+ (ngModelChange)="onCountryChange($event)"></tolle-country-selector>
149
+
150
+ <tolle-masked-input
151
+ [id]="id"
152
+ [mask]="mask"
153
+ [size]="size"
154
+ [disabled]="disabled"
155
+ [readonly]="readonly"
156
+ [placeholder]="placeholder"
157
+ [error]="error"
158
+ [mergedPosition]="'right'"
159
+ [(ngModel)]="displayValue"
160
+ (ngModelChange)="onMaskInputChange($event)"></tolle-masked-input>
161
+ </div>
162
+
163
+ <ng-container *ngIf="!disabled">
164
+ <p
165
+ *ngIf="hint && !error"
166
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200">
167
+ {{ hint }}
168
+ </p>
169
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
170
+ {{ errorMessage }}
171
+ </p>
172
+ </ng-container>
173
+ </div>
174
+ `, isInline: true, styles: [":host{display:block;width:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: CountrySelectorComponent, selector: "tolle-country-selector", inputs: ["id", "label", "hint", "errorMessage", "error", "hideHintOnFocus", "placeholder", "class", "disabled", "readonly", "size", "defaultCountryCode", "returnValue", "showName", "mergedPosition"], outputs: ["onSelect", "onFocusChange", "onBlurChange"] }, { kind: "component", type: MaskedInputComponent, selector: "tolle-masked-input", inputs: ["id", "label", "hint", "errorMessage", "mask", "placeholder", "type", "disabled", "readonly", "class", "containerClass", "error", "size", "returnRaw", "hideHintOnFocus", "mergedPosition"] }] });
141
175
  }
142
176
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PhoneNumberInputComponent, decorators: [{
143
177
  type: Component,
144
178
  args: [{ selector: 'tolle-phone-number-input', standalone: true, imports: [CommonModule, FormsModule, CountrySelectorComponent, MaskedInputComponent], template: `
145
- <tolle-masked-input
146
- [id]="id"
147
- [label]="label"
148
- [hint]="hint"
149
- [error]="error"
150
- [errorMessage]="errorMessage"
151
- [hideHintOnFocus]="hideHintOnFocus"
152
- [mask]="mask"
153
- [size]="size"
154
- [disabled]="disabled"
155
- [readonly]="readonly"
156
- [placeholder]="placeholder"
157
- [(ngModel)]="displayValue"
158
- (ngModelChange)="onMaskInputChange($event)"
159
- [containerClass]="cn('pl-0', class)"
160
- >
161
- <tolle-country-selector
162
- prefix
163
- class="country-selector-override"
164
- [showName]="false"
165
- [size]="size"
166
- [disabled]="disabled || !enableCountrySelector"
167
- [readonly]="readonly"
168
- [(ngModel)]="selectedIso"
169
- (ngModelChange)="onCountryChange($event)"
170
- ></tolle-country-selector>
171
- </tolle-masked-input>
179
+ <div class="flex w-full flex-col gap-1.5">
180
+ <label *ngIf="label" [for]="id" [class]="computedLabelClass">
181
+ {{ label }}
182
+ </label>
183
+
184
+ <div [class]="computedMergedClass">
185
+ <tolle-country-selector
186
+ [showName]="false"
187
+ [size]="size"
188
+ [disabled]="disabled || !enableCountrySelector"
189
+ [readonly]="readonly"
190
+ [mergedPosition]="'left'"
191
+ [(ngModel)]="selectedIso"
192
+ (ngModelChange)="onCountryChange($event)"></tolle-country-selector>
193
+
194
+ <tolle-masked-input
195
+ [id]="id"
196
+ [mask]="mask"
197
+ [size]="size"
198
+ [disabled]="disabled"
199
+ [readonly]="readonly"
200
+ [placeholder]="placeholder"
201
+ [error]="error"
202
+ [mergedPosition]="'right'"
203
+ [(ngModel)]="displayValue"
204
+ (ngModelChange)="onMaskInputChange($event)"></tolle-masked-input>
205
+ </div>
206
+
207
+ <ng-container *ngIf="!disabled">
208
+ <p
209
+ *ngIf="hint && !error"
210
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200">
211
+ {{ hint }}
212
+ </p>
213
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
214
+ {{ errorMessage }}
215
+ </p>
216
+ </ng-container>
217
+ </div>
172
218
  `, providers: [
173
219
  {
174
220
  provide: NG_VALUE_ACCESSOR,
175
221
  useExisting: forwardRef(() => PhoneNumberInputComponent),
176
- multi: true
177
- }
178
- ], styles: [":host{display:block;width:100%}::ng-deep .country-selector-override{display:flex;align-items:center;border-right:1px solid var(--border, #e5e7eb)}::ng-deep .country-selector-override button{border:none!important;border-radius:0!important;border-top-left-radius:calc(var(--radius, .5rem) - 1px)!important;border-bottom-left-radius:calc(var(--radius, .5rem) - 1px)!important;background:transparent!important;box-shadow:none!important;padding-left:.75rem!important;padding-right:.5rem!important;display:flex!important;align-items:center!important;justify-content:center!important;gap:.25rem!important}::ng-deep .country-selector-override button .flex{gap:.25rem!important}::ng-deep .country-selector-override button i{margin-left:0!important;line-height:1!important;display:flex!important;align-items:center!important}\n"] }]
222
+ multi: true,
223
+ },
224
+ ], styles: [":host{display:block;width:100%}\n"] }]
179
225
  }], ctorParameters: () => [], propDecorators: { id: [{
180
226
  type: Input
181
227
  }], label: [{
@@ -209,4 +255,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
209
255
  }], onSelect: [{
210
256
  type: Output
211
257
  }] } });
212
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"phone-number-input.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/phone-number-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    inject,\n    Input,\n    Output,\n    ViewChild,\n    ChangeDetectorRef\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { CountrySelectorComponent } from './country-selector.component';\nimport { MaskedInputComponent } from './masked-input.component';\nimport parsePhoneNumber from 'libphonenumber-js';\nimport { cn } from './utils/cn';\n\n@Component({\n    selector: 'tolle-phone-number-input',\n    standalone: true,\n    imports: [CommonModule, FormsModule, CountrySelectorComponent, MaskedInputComponent],\n    template: `\n    <tolle-masked-input\n      [id]=\"id\"\n      [label]=\"label\"\n      [hint]=\"hint\"\n      [error]=\"error\"\n      [errorMessage]=\"errorMessage\"\n      [hideHintOnFocus]=\"hideHintOnFocus\"\n      [mask]=\"mask\"\n      [size]=\"size\"\n      [disabled]=\"disabled\"\n      [readonly]=\"readonly\"\n      [placeholder]=\"placeholder\"\n      [(ngModel)]=\"displayValue\"\n      (ngModelChange)=\"onMaskInputChange($event)\"\n      [containerClass]=\"cn('pl-0', class)\"\n    >\n      <tolle-country-selector\n        prefix\n        class=\"country-selector-override\"\n        [showName]=\"false\"\n        [size]=\"size\"\n        [disabled]=\"disabled || !enableCountrySelector\"\n        [readonly]=\"readonly\"\n        [(ngModel)]=\"selectedIso\"\n        (ngModelChange)=\"onCountryChange($event)\"\n      ></tolle-country-selector>\n    </tolle-masked-input>\n  `,\n    styles: [`\n    :host {\n      display: block;\n      width: 100%;\n    }\n    ::ng-deep .country-selector-override {\n       display: flex;\n       align-items: center;\n       border-right: 1px solid var(--border, #e5e7eb);\n    }\n    ::ng-deep .country-selector-override button {\n      border: none !important;\n      border-radius: 0 !important;\n      border-top-left-radius: calc(var(--radius, 0.5rem) - 1px) !important;\n      border-bottom-left-radius: calc(var(--radius, 0.5rem) - 1px) !important;\n      background: transparent !important;\n      box-shadow: none !important;\n      padding-left: 0.75rem !important;\n      padding-right: 0.5rem !important;\n      display: flex !important;\n      align-items: center !important;\n      justify-content: center !important;\n      gap: 0.25rem !important;\n    }\n    ::ng-deep .country-selector-override button .flex {\n      gap: 0.25rem !important;\n    }\n    ::ng-deep .country-selector-override button i {\n      margin-left: 0 !important;\n      line-height: 1 !important;\n      display: flex !important;\n      align-items: center !important;\n    }\n  `],\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => PhoneNumberInputComponent),\n            multi: true\n        }\n    ]\n})\nexport class PhoneNumberInputComponent implements ControlValueAccessor {\n    private cdr = inject(ChangeDetectorRef);\n\n    @Input() id = `phone-input-${Math.random().toString(36).substr(2, 9)}`;\n    @Input() label = '';\n    @Input() hint = '';\n    @Input() errorMessage = '';\n    @Input() error = false;\n    @Input() hideHintOnFocus = true;\n    @Input() placeholder = 'Phone number';\n    @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n    @Input() disabled = false;\n    @Input() readonly = false;\n    @Input() defaultCountryCode = 'GH';\n    @Input() dataType: 'NumberOnly' | 'FullObject' = 'FullObject';\n    @Input() mask = '(000) 000-0000';\n    @Input() class = '';\n    @Input() enableCountrySelector = true;\n\n    @Output() onSelect = new EventEmitter<any>();\n\n    displayValue = '';\n    selectedIso = '';\n    private rawValue = '';\n\n    protected cn = cn;\n\n    onChange: any = () => { };\n    onTouched: any = () => { };\n\n    constructor() {\n        this.selectedIso = this.defaultCountryCode;\n    }\n\n    writeValue(value: any): void {\n        if (value) {\n            if (typeof value === 'object' && value.number) {\n                this.selectedIso = value.country || this.defaultCountryCode;\n                this.displayValue = value.number;\n                this.rawValue = value.number.replace(/\\D/g, '');\n            } else {\n                this.displayValue = value;\n                this.rawValue = value.toString().replace(/\\D/g, '');\n                try {\n                    const parsed = parsePhoneNumber(value);\n                    if (parsed) {\n                        this.selectedIso = parsed.country || this.defaultCountryCode;\n                    }\n                } catch (e) { }\n            }\n        } else {\n            this.displayValue = '';\n            this.rawValue = '';\n        }\n        this.cdr.markForCheck();\n    }\n\n    registerOnChange(fn: any): void {\n        this.onChange = fn;\n    }\n\n    registerOnTouched(fn: any): void {\n        this.onTouched = fn;\n    }\n\n    setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n        this.cdr.markForCheck();\n    }\n\n    onCountryChange(iso: string) {\n        this.selectedIso = iso;\n        this.updateModel();\n    }\n\n    onMaskInputChange(value: string) {\n        this.displayValue = value;\n        this.rawValue = value.replace(/\\D/g, '');\n        this.updateModel();\n    }\n\n    private updateModel() {\n        if (!this.rawValue) {\n            this.onChange(null);\n            return;\n        }\n\n        try {\n            const parsed = parsePhoneNumber(this.rawValue, this.selectedIso as any);\n            if (parsed) {\n                if (this.dataType === 'FullObject') {\n                    this.onChange(parsed);\n                } else {\n                    this.onChange(parsed.number);\n                }\n                this.onSelect.emit(parsed);\n            } else {\n                this.onChange(this.rawValue);\n            }\n        } catch (e) {\n            this.onChange(this.rawValue);\n        }\n    }\n}\n"]}
258
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"phone-number-input.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/phone-number-input.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,YAAY,EACZ,UAAU,EACV,MAAM,EACN,KAAK,EACL,MAAM,EACN,iBAAiB,GAClB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAwB,iBAAiB,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AACtF,OAAO,EAAE,wBAAwB,EAAE,MAAM,8BAA8B,CAAC;AACxE,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,gBAAgB,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,EAAE,EAAE,MAAM,YAAY,CAAC;;;;AA+DhC,MAAM,OAAO,yBAAyB;IAC5B,GAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;IAE/B,EAAE,GAAG,eAAe,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;IAC9D,KAAK,GAAG,EAAE,CAAC;IACX,IAAI,GAAG,EAAE,CAAC;IACV,YAAY,GAAG,EAAE,CAAC;IAClB,KAAK,GAAG,KAAK,CAAC;IACd,eAAe,GAAG,IAAI,CAAC;IACvB,WAAW,GAAG,cAAc,CAAC;IAC7B,IAAI,GAAmC,SAAS,CAAC;IACjD,QAAQ,GAAG,KAAK,CAAC;IACjB,QAAQ,GAAG,KAAK,CAAC;IACjB,kBAAkB,GAAG,IAAI,CAAC;IAC1B,QAAQ,GAAgC,YAAY,CAAC;IACrD,IAAI,GAAG,gBAAgB,CAAC;IACxB,KAAK,GAAG,EAAE,CAAC;IACX,qBAAqB,GAAG,IAAI,CAAC;IAE5B,QAAQ,GAAG,IAAI,YAAY,EAAO,CAAC;IAE7C,YAAY,GAAG,EAAE,CAAC;IAClB,WAAW,GAAG,EAAE,CAAC;IACT,QAAQ,GAAG,EAAE,CAAC;IAEZ,EAAE,GAAG,EAAE,CAAC;IAElB,QAAQ,GAAQ,GAAG,EAAE,GAAE,CAAC,CAAC;IACzB,SAAS,GAAQ,GAAG,EAAE,GAAE,CAAC,CAAC;IAE1B;QACE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC;IAC7C,CAAC;IAED,IAAI,kBAAkB;QACpB,OAAO,EAAE,CACP,kFAAkF,EAClF,IAAI,CAAC,QAAQ,IAAI,YAAY,CAC9B,CAAC;IACJ,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,EAAE,CACP,uFAAuF,EACvF,sCAAsC,EACtC,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,KAAK,EAC3B,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,KAAK,EAC3B,IAAI,CAAC,IAAI,KAAK,SAAS,IAAI,MAAM,EACjC,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,MAAM;QAC5B,cAAc;QACd,CAAC,IAAI,CAAC,QAAQ;YACZ,CAAC,IAAI,CAAC,QAAQ,IAAI;YAChB,qBAAqB;YACrB,2BAA2B;YAC3B,4BAA4B;YAC5B,0BAA0B;YAC1B,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,oCAAoC,CAAC,CAAC,CAAC,gCAAgC;SACrF;QACH,cAAc;QACd,IAAI,CAAC,KAAK,IAAI,oBAAoB,EAClC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,kCAAkC;QACpF,iBAAiB;QACjB,IAAI,CAAC,QAAQ,IAAI,CAAC,+BAA+B,EAAE,mBAAmB,CAAC;QACvE,iBAAiB;QACjB,IAAI,CAAC,QAAQ,IAAI,8BAA8B,EAC/C,IAAI,CAAC,KAAK,CACX,CAAC;IACJ,CAAC;IAED,UAAU,CAAC,KAAU;QACnB,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;gBAC9C,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,OAAO,IAAI,IAAI,CAAC,kBAAkB,CAAC;gBAC5D,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,MAAM,CAAC;gBACjC,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;YAClD,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;gBAC1B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBACpD,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAC;oBACvC,IAAI,MAAM,EAAE,CAAC;wBACX,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC,kBAAkB,CAAC;oBAC/D,CAAC;gBACH,CAAC;gBAAC,OAAO,CAAC,EAAE,CAAC,CAAA,CAAC;YAChB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;YACvB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACrB,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,gBAAgB,CAAC,UAAmB;QAClC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;QAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,eAAe,CAAC,GAAW;QACzB,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC;QACvB,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,KAAa;QAC7B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACzC,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAEO,WAAW;QACjB,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YACpB,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,gBAAgB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,WAAkB,CAAC,CAAC;YACxE,IAAI,MAAM,EAAE,CAAC;gBACX,IAAI,IAAI,CAAC,QAAQ,KAAK,YAAY,EAAE,CAAC;oBACnC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBACxB,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;gBACD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC7B,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC/B,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;wGAzIU,yBAAyB;4FAAzB,yBAAyB,odARzB;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,yBAAyB,CAAC;gBACxD,KAAK,EAAE,IAAI;aACZ;SACF,0BAvDS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwCT,0GAzCS,YAAY,kIAAE,WAAW,+VAAE,wBAAwB,mUAAE,oBAAoB;;4FA0DxE,yBAAyB;kBA7DrC,SAAS;+BACE,0BAA0B,cACxB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,wBAAwB,EAAE,oBAAoB,CAAC,YAC1E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwCT,aASU;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,0BAA0B,CAAC;4BACxD,KAAK,EAAE,IAAI;yBACZ;qBACF;wDAKQ,EAAE;sBAAV,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,qBAAqB;sBAA7B,KAAK;gBAEI,QAAQ;sBAAjB,MAAM","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  forwardRef,\n  inject,\n  Input,\n  Output,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { CountrySelectorComponent } from './country-selector.component';\nimport { MaskedInputComponent } from './masked-input.component';\nimport parsePhoneNumber from 'libphonenumber-js';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-phone-number-input',\n  standalone: true,\n  imports: [CommonModule, FormsModule, CountrySelectorComponent, MaskedInputComponent],\n  template: `\n    <div class=\"flex w-full flex-col gap-1.5\">\n      <label *ngIf=\"label\" [for]=\"id\" [class]=\"computedLabelClass\">\n        {{ label }}\n      </label>\n\n      <div [class]=\"computedMergedClass\">\n        <tolle-country-selector\n          [showName]=\"false\"\n          [size]=\"size\"\n          [disabled]=\"disabled || !enableCountrySelector\"\n          [readonly]=\"readonly\"\n          [mergedPosition]=\"'left'\"\n          [(ngModel)]=\"selectedIso\"\n          (ngModelChange)=\"onCountryChange($event)\"></tolle-country-selector>\n\n        <tolle-masked-input\n          [id]=\"id\"\n          [mask]=\"mask\"\n          [size]=\"size\"\n          [disabled]=\"disabled\"\n          [readonly]=\"readonly\"\n          [placeholder]=\"placeholder\"\n          [error]=\"error\"\n          [mergedPosition]=\"'right'\"\n          [(ngModel)]=\"displayValue\"\n          (ngModelChange)=\"onMaskInputChange($event)\"></tolle-masked-input>\n      </div>\n\n      <ng-container *ngIf=\"!disabled\">\n        <p\n          *ngIf=\"hint && !error\"\n          class=\"px-1 text-xs text-muted-foreground transition-opacity duration-200\">\n          {{ hint }}\n        </p>\n        <p *ngIf=\"error && errorMessage\" [id]=\"id + '-error'\" class=\"px-1 text-xs text-destructive\">\n          {{ errorMessage }}\n        </p>\n      </ng-container>\n    </div>\n  `,\n  styles: [\n    `\n      :host {\n        display: block;\n        width: 100%;\n      }\n    `,\n  ],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => PhoneNumberInputComponent),\n      multi: true,\n    },\n  ],\n})\nexport class PhoneNumberInputComponent implements ControlValueAccessor {\n  private cdr = inject(ChangeDetectorRef);\n\n  @Input() id = `phone-input-${Math.random().toString(36).substr(2, 9)}`;\n  @Input() label = '';\n  @Input() hint = '';\n  @Input() errorMessage = '';\n  @Input() error = false;\n  @Input() hideHintOnFocus = true;\n  @Input() placeholder = 'Phone number';\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  @Input() disabled = false;\n  @Input() readonly = false;\n  @Input() defaultCountryCode = 'GH';\n  @Input() dataType: 'NumberOnly' | 'FullObject' = 'FullObject';\n  @Input() mask = '(000) 000-0000';\n  @Input() class = '';\n  @Input() enableCountrySelector = true;\n\n  @Output() onSelect = new EventEmitter<any>();\n\n  displayValue = '';\n  selectedIso = '';\n  private rawValue = '';\n\n  protected cn = cn;\n\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  constructor() {\n    this.selectedIso = this.defaultCountryCode;\n  }\n\n  get computedLabelClass() {\n    return cn(\n      'text-sm font-medium text-foreground leading-none transition-opacity duration-200',\n      this.disabled && 'opacity-50'\n    );\n  }\n\n  get computedMergedClass() {\n    return cn(\n      'group relative flex items-center w-full rounded-md border transition-all duration-200',\n      'bg-background border-input shadow-sm',\n      this.size === 'xs' && 'h-8',\n      this.size === 'sm' && 'h-9',\n      this.size === 'default' && 'h-10',\n      this.size === 'lg' && 'h-11',\n      // Focus state\n      !this.readonly &&\n        !this.disabled && [\n          'focus-within:ring-4',\n          'focus-within:ring-ring/30',\n          'focus-within:ring-offset-0',\n          'focus-within:shadow-none',\n          this.error ? 'focus-within:border-destructive/80' : 'focus-within:border-primary/80',\n        ],\n      // Error state\n      this.error && 'border-destructive',\n      this.error && !this.readonly && !this.disabled && 'focus-within:ring-destructive/30',\n      // Disabled state\n      this.disabled && ['cursor-not-allowed opacity-50', 'border-opacity-50'],\n      // Readonly state\n      this.readonly && 'cursor-default border-dashed',\n      this.class\n    );\n  }\n\n  writeValue(value: any): void {\n    if (value) {\n      if (typeof value === 'object' && value.number) {\n        this.selectedIso = value.country || this.defaultCountryCode;\n        this.displayValue = value.number;\n        this.rawValue = value.number.replace(/\\D/g, '');\n      } else {\n        this.displayValue = value;\n        this.rawValue = value.toString().replace(/\\D/g, '');\n        try {\n          const parsed = parsePhoneNumber(value);\n          if (parsed) {\n            this.selectedIso = parsed.country || this.defaultCountryCode;\n          }\n        } catch (e) {}\n      }\n    } else {\n      this.displayValue = '';\n      this.rawValue = '';\n    }\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this.cdr.markForCheck();\n  }\n\n  onCountryChange(iso: string) {\n    this.selectedIso = iso;\n    this.updateModel();\n  }\n\n  onMaskInputChange(value: string) {\n    this.displayValue = value;\n    this.rawValue = value.replace(/\\D/g, '');\n    this.updateModel();\n  }\n\n  private updateModel() {\n    if (!this.rawValue) {\n      this.onChange(null);\n      return;\n    }\n\n    try {\n      const parsed = parsePhoneNumber(this.rawValue, this.selectedIso as any);\n      if (parsed) {\n        if (this.dataType === 'FullObject') {\n          this.onChange(parsed);\n        } else {\n          this.onChange(parsed.number);\n        }\n        this.onSelect.emit(parsed);\n      } else {\n        this.onChange(this.rawValue);\n      }\n    } catch (e) {\n      this.onChange(this.rawValue);\n    }\n  }\n}\n"]}