@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.
- package/esm2022/lib/country-selector.component.mjs +35 -23
- package/esm2022/lib/date-picker.component.mjs +2 -2
- package/esm2022/lib/masked-input.component.mjs +61 -86
- package/esm2022/lib/phone-number-input.component.mjs +109 -63
- package/fesm2022/tolle-ui.mjs +199 -167
- package/fesm2022/tolle-ui.mjs.map +1 -1
- package/lib/button.component.d.ts +1 -1
- package/lib/country-selector.component.d.ts +4 -1
- package/lib/masked-input.component.d.ts +2 -1
- package/lib/phone-number-input.component.d.ts +2 -0
- package/lib/toggle.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -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/
|
|
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
|
-
<
|
|
114
|
-
[
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
[
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
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
|
-
<
|
|
146
|
-
[
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
[
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
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%}
|
|
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":"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","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"]}
|