@tolle_/tolle-ui 18.2.13 → 18.2.15

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, Input, forwardRef, inject, ViewChild, ChangeDetectorRef, Output, EventEmitter } from '@angular/core';
1
+ import { Component, Input, forwardRef, inject, ViewChild, ChangeDetectorRef, Output, EventEmitter, } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
4
4
  import { DomSanitizer } from '@angular/platform-browser';
@@ -26,6 +26,7 @@ export class CountrySelectorComponent {
26
26
  showName = true;
27
27
  onSelect = new EventEmitter();
28
28
  popover;
29
+ searchInput;
29
30
  countryCodesService = inject(CountryCodesService);
30
31
  sanitizer = inject(DomSanitizer);
31
32
  cdr = inject(ChangeDetectorRef);
@@ -52,20 +53,20 @@ export class CountrySelectorComponent {
52
53
  'focus:ring-4',
53
54
  'focus:ring-ring/30',
54
55
  'focus:ring-offset-0',
55
- 'focus:border-primary/80'
56
+ 'focus:border-primary/80',
56
57
  ], !(this.readonly || this.disabled) && 'hover:border-accent', this.error && [
57
58
  'border-destructive',
58
59
  !(this.readonly || this.disabled) && [
59
60
  'focus:border-destructive/80',
60
- 'focus:ring-destructive/30'
61
- ]
61
+ 'focus:ring-destructive/30',
62
+ ],
62
63
  ], this.disabled && 'cursor-not-allowed opacity-50 border-opacity-50', this.readonly && 'cursor-default border-dashed', this.class);
63
64
  }
64
65
  get computedLabelClass() {
65
- return cn("text-sm font-medium text-foreground leading-none transition-opacity duration-200", this.disabled && "opacity-50");
66
+ return cn('text-sm font-medium text-foreground leading-none transition-opacity duration-200', this.disabled && 'opacity-50');
66
67
  }
67
68
  get iconClass() {
68
- return cn('ri-arrow-down-s-line text-muted-foreground ml-2 transition-transform duration-200', this.popover?.isOpen ? 'rotate-180' : '', (this.size === 'xs' || this.size === 'sm') ? 'text-[14px]' : 'text-[18px]', (this.disabled || this.readonly) && 'opacity-30');
69
+ return cn('ri-arrow-down-s-line text-muted-foreground ml-2 transition-transform duration-200', this.popover?.isOpen ? 'rotate-180' : '', this.size === 'xs' || this.size === 'sm' ? 'text-[14px]' : 'text-[18px]', (this.disabled || this.readonly) && 'opacity-30');
69
70
  }
70
71
  getItemClass(country) {
71
72
  const isSelected = this.selectedCountry?.isoAlpha2 === country.isoAlpha2;
@@ -89,11 +90,16 @@ export class CountrySelectorComponent {
89
90
  }
90
91
  getReturnValue(country) {
91
92
  switch (this.returnValue) {
92
- case 'object': return country;
93
- case 'isoAlpha2': return country.isoAlpha2;
94
- case 'dialCode': return country.dialCode;
95
- case 'name': return country.name;
96
- default: return country.isoAlpha2;
93
+ case 'object':
94
+ return country;
95
+ case 'isoAlpha2':
96
+ return country.isoAlpha2;
97
+ case 'dialCode':
98
+ return country.dialCode;
99
+ case 'name':
100
+ return country.name;
101
+ default:
102
+ return country.isoAlpha2;
97
103
  }
98
104
  }
99
105
  onFocus() {
@@ -103,6 +109,11 @@ export class CountrySelectorComponent {
103
109
  this.isFocused = false;
104
110
  this.onTouched();
105
111
  }
112
+ onPopoverOpen() {
113
+ setTimeout(() => {
114
+ this.searchInput?.nativeElement?.focus();
115
+ }, 0);
116
+ }
106
117
  onPopoverClose() {
107
118
  this.searchQuery = '';
108
119
  this.filterCountries('');
@@ -128,26 +139,33 @@ export class CountrySelectorComponent {
128
139
  }
129
140
  this.cdr.markForCheck();
130
141
  }
131
- registerOnChange(fn) { this.onChange = fn; }
132
- registerOnTouched(fn) { this.onTouched = fn; }
133
- setDisabledState(isDisabled) { this.disabled = isDisabled; }
142
+ registerOnChange(fn) {
143
+ this.onChange = fn;
144
+ }
145
+ registerOnTouched(fn) {
146
+ this.onTouched = fn;
147
+ }
148
+ setDisabledState(isDisabled) {
149
+ this.disabled = isDisabled;
150
+ }
134
151
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CountrySelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
135
152
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CountrySelectorComponent, isStandalone: true, selector: "tolle-country-selector", inputs: { id: "id", label: "label", hint: "hint", errorMessage: "errorMessage", error: "error", hideHintOnFocus: "hideHintOnFocus", placeholder: "placeholder", class: "class", disabled: "disabled", readonly: "readonly", size: "size", defaultCountryCode: "defaultCountryCode", returnValue: "returnValue", showName: "showName" }, outputs: { onSelect: "onSelect" }, providers: [
136
153
  {
137
154
  provide: NG_VALUE_ACCESSOR,
138
155
  useExisting: forwardRef(() => CountrySelectorComponent),
139
- multi: true
140
- }
141
- ], viewQueries: [{ propertyName: "popover", first: true, predicate: ["popover"], descendants: true }], ngImport: i0, template: `
142
- <div class="flex flex-col gap-1.5 w-full">
143
- <label
144
- *ngIf="label"
145
- [for]="id"
146
- [class]="computedLabelClass">
156
+ multi: true,
157
+ },
158
+ ], viewQueries: [{ propertyName: "popover", first: true, predicate: ["popover"], descendants: true }, { propertyName: "searchInput", first: true, predicate: ["searchInput"], descendants: true }], ngImport: i0, template: `
159
+ <div class="flex w-full flex-col gap-1.5">
160
+ <label *ngIf="label" [for]="id" [class]="computedLabelClass">
147
161
  {{ label }}
148
162
  </label>
149
163
 
150
- <tolle-popover #popover [placement]="'bottom-start'" (onClose)="onPopoverClose()">
164
+ <tolle-popover
165
+ #popover
166
+ [placement]="'bottom-start'"
167
+ (onOpen)="onPopoverOpen()"
168
+ (onClose)="onPopoverClose()">
151
169
  <div trigger class="w-full">
152
170
  <button
153
171
  type="button"
@@ -157,19 +175,17 @@ export class CountrySelectorComponent {
157
175
  (blur)="onBlur()"
158
176
  (focus)="onFocus()"
159
177
  [attr.aria-invalid]="error"
160
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
161
- >
178
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null">
162
179
  <div class="flex items-center gap-2 truncate">
163
180
  <img
164
181
  *ngIf="selectedCountry"
165
182
  [src]="getFlagUrl(selectedCountry.flag)"
166
- class="h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0"
167
- [alt]="selectedCountry.name"
168
- />
183
+ class="h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover"
184
+ [alt]="selectedCountry.name" />
169
185
  <span *ngIf="selectedCountry && showName" class="truncate font-medium">
170
186
  {{ selectedCountry.name }}
171
187
  </span>
172
- <span *ngIf="!selectedCountry" class="text-muted-foreground truncate">
188
+ <span *ngIf="!selectedCountry" class="truncate text-muted-foreground">
173
189
  {{ placeholder }}
174
190
  </span>
175
191
  </div>
@@ -177,16 +193,17 @@ export class CountrySelectorComponent {
177
193
  </button>
178
194
  </div>
179
195
 
180
- <div class="flex flex-col bg-popover rounded-md border border-border shadow-md min-w-[300px] max-w-[400px] overflow-hidden">
181
- <div class="p-2 border-b border-border bg-popover shadow-sm sticky top-0 z-10">
196
+ <div
197
+ class="flex min-w-[300px] max-w-[400px] flex-col overflow-hidden rounded-md border border-border bg-popover shadow-md"
198
+ (mousedown)="$event.stopPropagation()">
199
+ <div class="sticky top-0 z-10 border-b border-border bg-popover p-2 shadow-sm">
182
200
  <tolle-input
183
201
  size="sm"
184
202
  placeholder="Search country..."
185
203
  [(ngModel)]="searchQuery"
186
204
  (ngModelChange)="filterCountries($event)"
187
205
  class="w-full"
188
- #searchInput
189
- >
206
+ #searchInput>
190
207
  <i prefix class="ri-search-line"></i>
191
208
  </tolle-input>
192
209
  </div>
@@ -195,20 +212,22 @@ export class CountrySelectorComponent {
195
212
  <div
196
213
  *ngFor="let country of shadowCountries"
197
214
  (click)="selectCountry(country)"
198
- [class]="getItemClass(country)"
199
- >
200
- <div class="flex items-center gap-3 w-full">
215
+ [class]="getItemClass(country)">
216
+ <div class="flex w-full items-center gap-3">
201
217
  <img
202
218
  [src]="getFlagUrl(country.flag)"
203
- class="h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0"
204
- [alt]="country.name"
205
- />
206
- <span class="text-sm flex-1 truncate">{{ country.name }}</span>
219
+ class="h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover"
220
+ [alt]="country.name" />
221
+ <span class="flex-1 truncate text-sm">{{ country.name }}</span>
207
222
  <span class="text-xs text-muted-foreground">{{ country.dialCode }}</span>
208
- <i *ngIf="selectedCountry?.isoAlpha2 === country.isoAlpha2" class="ri-check-line text-primary"></i>
223
+ <i
224
+ *ngIf="selectedCountry?.isoAlpha2 === country.isoAlpha2"
225
+ class="ri-check-line text-primary"></i>
209
226
  </div>
210
227
  </div>
211
- <div *ngIf="shadowCountries.length === 0" class="py-6 text-center text-sm text-muted-foreground">
228
+ <div
229
+ *ngIf="shadowCountries.length === 0"
230
+ class="py-6 text-center text-sm text-muted-foreground">
212
231
  No countries found.
213
232
  </div>
214
233
  </div>
@@ -218,16 +237,14 @@ export class CountrySelectorComponent {
218
237
  <ng-container *ngIf="!disabled">
219
238
  <p
220
239
  *ngIf="hint && !error"
221
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
222
- [class.opacity-0]="isFocused && hideHintOnFocus"
223
- >
240
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
241
+ [class.opacity-0]="isFocused && hideHintOnFocus">
224
242
  {{ hint }}
225
243
  </p>
226
244
  <p
227
245
  *ngIf="error && errorMessage"
228
246
  [id]="id + '-error'"
229
- class="text-xs text-destructive px-1 font-medium"
230
- >
247
+ class="px-1 text-xs font-medium text-destructive">
231
248
  {{ errorMessage }}
232
249
  </p>
233
250
  </ng-container>
@@ -244,19 +261,20 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
244
261
  {
245
262
  provide: NG_VALUE_ACCESSOR,
246
263
  useExisting: forwardRef(() => CountrySelectorComponent),
247
- multi: true
248
- }
264
+ multi: true,
265
+ },
249
266
  ],
250
267
  template: `
251
- <div class="flex flex-col gap-1.5 w-full">
252
- <label
253
- *ngIf="label"
254
- [for]="id"
255
- [class]="computedLabelClass">
268
+ <div class="flex w-full flex-col gap-1.5">
269
+ <label *ngIf="label" [for]="id" [class]="computedLabelClass">
256
270
  {{ label }}
257
271
  </label>
258
272
 
259
- <tolle-popover #popover [placement]="'bottom-start'" (onClose)="onPopoverClose()">
273
+ <tolle-popover
274
+ #popover
275
+ [placement]="'bottom-start'"
276
+ (onOpen)="onPopoverOpen()"
277
+ (onClose)="onPopoverClose()">
260
278
  <div trigger class="w-full">
261
279
  <button
262
280
  type="button"
@@ -266,19 +284,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
266
284
  (blur)="onBlur()"
267
285
  (focus)="onFocus()"
268
286
  [attr.aria-invalid]="error"
269
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
270
- >
287
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null">
271
288
  <div class="flex items-center gap-2 truncate">
272
289
  <img
273
290
  *ngIf="selectedCountry"
274
291
  [src]="getFlagUrl(selectedCountry.flag)"
275
- class="h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0"
276
- [alt]="selectedCountry.name"
277
- />
292
+ class="h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover"
293
+ [alt]="selectedCountry.name" />
278
294
  <span *ngIf="selectedCountry && showName" class="truncate font-medium">
279
295
  {{ selectedCountry.name }}
280
296
  </span>
281
- <span *ngIf="!selectedCountry" class="text-muted-foreground truncate">
297
+ <span *ngIf="!selectedCountry" class="truncate text-muted-foreground">
282
298
  {{ placeholder }}
283
299
  </span>
284
300
  </div>
@@ -286,16 +302,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
286
302
  </button>
287
303
  </div>
288
304
 
289
- <div class="flex flex-col bg-popover rounded-md border border-border shadow-md min-w-[300px] max-w-[400px] overflow-hidden">
290
- <div class="p-2 border-b border-border bg-popover shadow-sm sticky top-0 z-10">
305
+ <div
306
+ class="flex min-w-[300px] max-w-[400px] flex-col overflow-hidden rounded-md border border-border bg-popover shadow-md"
307
+ (mousedown)="$event.stopPropagation()">
308
+ <div class="sticky top-0 z-10 border-b border-border bg-popover p-2 shadow-sm">
291
309
  <tolle-input
292
310
  size="sm"
293
311
  placeholder="Search country..."
294
312
  [(ngModel)]="searchQuery"
295
313
  (ngModelChange)="filterCountries($event)"
296
314
  class="w-full"
297
- #searchInput
298
- >
315
+ #searchInput>
299
316
  <i prefix class="ri-search-line"></i>
300
317
  </tolle-input>
301
318
  </div>
@@ -304,20 +321,22 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
304
321
  <div
305
322
  *ngFor="let country of shadowCountries"
306
323
  (click)="selectCountry(country)"
307
- [class]="getItemClass(country)"
308
- >
309
- <div class="flex items-center gap-3 w-full">
324
+ [class]="getItemClass(country)">
325
+ <div class="flex w-full items-center gap-3">
310
326
  <img
311
327
  [src]="getFlagUrl(country.flag)"
312
- class="h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0"
313
- [alt]="country.name"
314
- />
315
- <span class="text-sm flex-1 truncate">{{ country.name }}</span>
328
+ class="h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover"
329
+ [alt]="country.name" />
330
+ <span class="flex-1 truncate text-sm">{{ country.name }}</span>
316
331
  <span class="text-xs text-muted-foreground">{{ country.dialCode }}</span>
317
- <i *ngIf="selectedCountry?.isoAlpha2 === country.isoAlpha2" class="ri-check-line text-primary"></i>
332
+ <i
333
+ *ngIf="selectedCountry?.isoAlpha2 === country.isoAlpha2"
334
+ class="ri-check-line text-primary"></i>
318
335
  </div>
319
336
  </div>
320
- <div *ngIf="shadowCountries.length === 0" class="py-6 text-center text-sm text-muted-foreground">
337
+ <div
338
+ *ngIf="shadowCountries.length === 0"
339
+ class="py-6 text-center text-sm text-muted-foreground">
321
340
  No countries found.
322
341
  </div>
323
342
  </div>
@@ -327,21 +346,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
327
346
  <ng-container *ngIf="!disabled">
328
347
  <p
329
348
  *ngIf="hint && !error"
330
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
331
- [class.opacity-0]="isFocused && hideHintOnFocus"
332
- >
349
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
350
+ [class.opacity-0]="isFocused && hideHintOnFocus">
333
351
  {{ hint }}
334
352
  </p>
335
353
  <p
336
354
  *ngIf="error && errorMessage"
337
355
  [id]="id + '-error'"
338
- class="text-xs text-destructive px-1 font-medium"
339
- >
356
+ class="px-1 text-xs font-medium text-destructive">
340
357
  {{ errorMessage }}
341
358
  </p>
342
359
  </ng-container>
343
360
  </div>
344
- `
361
+ `,
345
362
  }]
346
363
  }], propDecorators: { id: [{
347
364
  type: Input
@@ -376,5 +393,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
376
393
  }], popover: [{
377
394
  type: ViewChild,
378
395
  args: ['popover']
396
+ }], searchInput: [{
397
+ type: ViewChild,
398
+ args: ['searchInput']
379
399
  }] } });
380
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"country-selector.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/country-selector.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EACL,UAAU,EACV,MAAM,EAEN,SAAS,EACT,iBAAiB,EACjB,MAAM,EACN,YAAY,EACb,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,YAAY,EAAmB,MAAM,2BAA2B,CAAC;AAC1E,OAAO,EAAE,mBAAmB,EAAE,MAAM,yBAAyB,CAAC;AAC9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AACvD,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AACnD,OAAO,EAAE,EAAE,EAAE,MAAM,YAAY,CAAC;;;;AA6GhC,MAAM,OAAO,wBAAwB;IAC1B,EAAE,GAAG,oBAAoB,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;IACnE,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,gBAAgB,CAAC;IAC/B,KAAK,GAAG,EAAE,CAAC;IACX,QAAQ,GAAG,KAAK,CAAC;IACjB,QAAQ,GAAG,KAAK,CAAC;IACjB,IAAI,GAAmC,SAAS,CAAC;IACjD,kBAAkB,GAAG,IAAI,CAAC;IAC1B,WAAW,GAAiD,WAAW,CAAC;IACxE,QAAQ,GAAG,IAAI,CAAC;IAEf,QAAQ,GAAG,IAAI,YAAY,EAAO,CAAC;IAEvB,OAAO,CAAoB;IAEzC,mBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;IAClD,SAAS,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;IACjC,GAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;IAExC,KAAK,GAAQ,IAAI,CAAC;IAClB,eAAe,GAAQ,IAAI,CAAC;IAC5B,WAAW,GAAG,EAAE,CAAC;IACjB,eAAe,GAAU,EAAE,CAAC;IAC5B,SAAS,GAAG,KAAK,CAAC;IAElB,QAAQ,GAAQ,GAAG,EAAE,GAAG,CAAC,CAAC;IAC1B,SAAS,GAAQ,GAAG,EAAE,GAAG,CAAC,CAAC;IAEjB,EAAE,GAAG,EAAE,CAAC;IAElB,QAAQ;QACN,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC;QAC1D,IAAI,IAAI,CAAC,kBAAkB,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3C,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,IAAI,CAC5D,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC,kBAAkB,CAC7C,CAAC;YACF,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;gBACzB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YACzD,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,oBAAoB;QACtB,OAAO,EAAE,CACP,wFAAwF,EACxF,+BAA+B,EAC/B,wBAAwB,EACxB,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,kBAAkB,EACxC,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,kBAAkB,EACxC,IAAI,CAAC,IAAI,KAAK,SAAS,IAAI,mBAAmB,EAC9C,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,qBAAqB,EAC3C,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI;YACnC,oBAAoB;YACpB,cAAc;YACd,oBAAoB;YACpB,qBAAqB;YACrB,yBAAyB;SAC1B,EACD,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,qBAAqB,EAC1D,IAAI,CAAC,KAAK,IAAI;YACZ,oBAAoB;YACpB,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI;gBACnC,6BAA6B;gBAC7B,2BAA2B;aAC5B;SACF,EACD,IAAI,CAAC,QAAQ,IAAI,iDAAiD,EAClE,IAAI,CAAC,QAAQ,IAAI,8BAA8B,EAC/C,IAAI,CAAC,KAAK,CACX,CAAC;IACJ,CAAC;IAED,IAAI,kBAAkB;QACpB,OAAO,EAAE,CACP,kFAAkF,EAClF,IAAI,CAAC,QAAQ,IAAI,YAAY,CAC9B,CAAC;IACJ,CAAC;IAED,IAAI,SAAS;QACX,OAAO,EAAE,CACP,mFAAmF,EACnF,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,EACxC,CAAC,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,EAC1E,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,YAAY,CACjD,CAAC;IACJ,CAAC;IAED,YAAY,CAAC,OAAY;QACvB,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,EAAE,SAAS,KAAK,OAAO,CAAC,SAAS,CAAC;QACzE,OAAO,EAAE,CACP,6GAA6G,EAC7G,UAAU,CAAC,CAAC,CAAC,kCAAkC,CAAC,CAAC,CAAC,oCAAoC,CACvF,CAAC;IACJ,CAAC;IAED,UAAU,CAAC,UAAkB;QAC3B,OAAO,IAAI,CAAC,SAAS,CAAC,8BAA8B,CAClD,sBAAsB,GAAG,UAAU,CACpC,CAAC;IACJ,CAAC;IAED,eAAe,CAAC,KAAa;QAC3B,MAAM,MAAM,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;QAClD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CACnE,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC;YACrC,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC3B,CAAC,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,CAC3C,CAAC;IACJ,CAAC;IAED,aAAa,CAAC,OAAY;QACxB,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;QAC/B,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC5B,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;IACvB,CAAC;IAEO,cAAc,CAAC,OAAY;QACjC,QAAQ,IAAI,CAAC,WAAW,EAAE,CAAC;YACzB,KAAK,QAAQ,CAAC,CAAC,OAAO,OAAO,CAAC;YAC9B,KAAK,WAAW,CAAC,CAAC,OAAO,OAAO,CAAC,SAAS,CAAC;YAC3C,KAAK,UAAU,CAAC,CAAC,OAAO,OAAO,CAAC,QAAQ,CAAC;YACzC,KAAK,MAAM,CAAC,CAAC,OAAO,OAAO,CAAC,IAAI,CAAC;YACjC,OAAO,CAAC,CAAC,OAAO,OAAO,CAAC,SAAS,CAAC;QACpC,CAAC;IACH,CAAC;IAED,OAAO;QACL,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;IACxB,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;QACzB,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAED,UAAU,CAAC,KAAU;QACnB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE;gBACjE,IAAI,IAAI,CAAC,WAAW,KAAK,QAAQ;oBAAE,OAAO,CAAC,CAAC,SAAS,KAAK,KAAK,CAAC,SAAS,CAAC;gBAC1E,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW;oBAAE,OAAO,CAAC,CAAC,SAAS,KAAK,KAAK,CAAC;gBACnE,IAAI,IAAI,CAAC,WAAW,KAAK,UAAU;oBAAE,OAAO,CAAC,CAAC,QAAQ,KAAK,KAAK,CAAC;gBACjE,IAAI,IAAI,CAAC,WAAW,KAAK,MAAM;oBAAE,OAAO,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC;gBACzD,OAAO,KAAK,CAAC;YACf,CAAC,CAAC,CAAC;QACL,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;QAC9B,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED,gBAAgB,CAAC,EAAO,IAAU,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC,CAAC,CAAC;IACvD,iBAAiB,CAAC,EAAO,IAAU,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,CAAC;IACzD,gBAAgB,CAAC,UAAmB,IAAU,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC,CAAC,CAAC;wGAvKhE,wBAAwB;4FAAxB,wBAAwB,gbAvGxB;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,wBAAwB,CAAC;gBACvD,KAAK,EAAE,IAAI;aACZ;SACF,8HACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8FT,2DAtGS,YAAY,+PAAE,WAAW,+VAAE,gBAAgB,iHAAE,cAAc;;4FAwG1D,wBAAwB;kBA3GpC,SAAS;mBAAC;oBACT,QAAQ,EAAE,wBAAwB;oBAClC,UAAU,EAAE,IAAI;oBAChB,OAAO,EAAE,CAAC,YAAY,EAAE,WAAW,EAAE,gBAAgB,EAAE,cAAc,CAAC;oBACtE,SAAS,EAAE;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,yBAAyB,CAAC;4BACvD,KAAK,EAAE,IAAI;yBACZ;qBACF;oBACD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8FT;iBACF;8BAEU,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,KAAK;sBAAb,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBAEI,QAAQ;sBAAjB,MAAM;gBAEe,OAAO;sBAA5B,SAAS;uBAAC,SAAS","sourcesContent":["import {\n  Component,\n  Input,\n  forwardRef,\n  inject,\n  OnInit,\n  ViewChild,\n  ChangeDetectorRef,\n  Output,\n  EventEmitter\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { CountryCodesService } from './country-codes.service';\nimport { PopoverComponent } from './popover.component';\nimport { InputComponent } from './input.component';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-country-selector',\n  standalone: true,\n  imports: [CommonModule, FormsModule, PopoverComponent, InputComponent],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CountrySelectorComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"flex flex-col gap-1.5 w-full\">\n      <label\n        *ngIf=\"label\"\n        [for]=\"id\"\n        [class]=\"computedLabelClass\">\n        {{ label }}\n      </label>\n\n      <tolle-popover #popover [placement]=\"'bottom-start'\" (onClose)=\"onPopoverClose()\">\n        <div trigger class=\"w-full\">\n          <button\n            type=\"button\"\n            [id]=\"id\"\n            [disabled]=\"disabled\"\n            [class]=\"computedTriggerClass\"\n            (blur)=\"onBlur()\"\n            (focus)=\"onFocus()\"\n            [attr.aria-invalid]=\"error\"\n            [attr.aria-describedby]=\"error && errorMessage ? id + '-error' : null\"\n          >\n            <div class=\"flex items-center gap-2 truncate\">\n              <img\n                *ngIf=\"selectedCountry\"\n                [src]=\"getFlagUrl(selectedCountry.flag)\"\n                class=\"h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0\"\n                [alt]=\"selectedCountry.name\"\n              />\n              <span *ngIf=\"selectedCountry && showName\" class=\"truncate font-medium\">\n                {{ selectedCountry.name }}\n              </span>\n              <span *ngIf=\"!selectedCountry\" class=\"text-muted-foreground truncate\">\n                {{ placeholder }}\n              </span>\n            </div>\n            <i [class]=\"iconClass\"></i>\n          </button>\n        </div>\n\n        <div class=\"flex flex-col bg-popover rounded-md border border-border shadow-md min-w-[300px] max-w-[400px] overflow-hidden\">\n          <div class=\"p-2 border-b border-border bg-popover shadow-sm sticky top-0 z-10\">\n            <tolle-input\n              size=\"sm\"\n              placeholder=\"Search country...\"\n              [(ngModel)]=\"searchQuery\"\n              (ngModelChange)=\"filterCountries($event)\"\n              class=\"w-full\"\n              #searchInput\n            >\n              <i prefix class=\"ri-search-line\"></i>\n            </tolle-input>\n          </div>\n\n          <div class=\"max-h-[300px] overflow-y-auto p-1\">\n            <div\n              *ngFor=\"let country of shadowCountries\"\n              (click)=\"selectCountry(country)\"\n              [class]=\"getItemClass(country)\"\n            >\n              <div class=\"flex items-center gap-3 w-full\">\n                <img\n                  [src]=\"getFlagUrl(country.flag)\"\n                  class=\"h-4 w-6 rounded-sm border border-border object-cover flex-shrink-0\"\n                  [alt]=\"country.name\"\n                />\n                <span class=\"text-sm flex-1 truncate\">{{ country.name }}</span>\n                <span class=\"text-xs text-muted-foreground\">{{ country.dialCode }}</span>\n                <i *ngIf=\"selectedCountry?.isoAlpha2 === country.isoAlpha2\" class=\"ri-check-line text-primary\"></i>\n              </div>\n            </div>\n            <div *ngIf=\"shadowCountries.length === 0\" class=\"py-6 text-center text-sm text-muted-foreground\">\n              No countries found.\n            </div>\n          </div>\n        </div>\n      </tolle-popover>\n\n      <ng-container *ngIf=\"!disabled\">\n        <p\n          *ngIf=\"hint && !error\"\n          class=\"text-xs text-muted-foreground px-1 transition-opacity duration-200\"\n          [class.opacity-0]=\"isFocused && hideHintOnFocus\"\n        >\n          {{ hint }}\n        </p>\n        <p\n          *ngIf=\"error && errorMessage\"\n          [id]=\"id + '-error'\"\n          class=\"text-xs text-destructive px-1 font-medium\"\n        >\n          {{ errorMessage }}\n        </p>\n      </ng-container>\n    </div>\n  `\n})\nexport class CountrySelectorComponent implements OnInit, ControlValueAccessor {\n  @Input() id = `country-selector-${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 = 'Select country';\n  @Input() class = '';\n  @Input() disabled = false;\n  @Input() readonly = false;\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  @Input() defaultCountryCode = 'GH';\n  @Input() returnValue: 'object' | 'isoAlpha2' | 'dialCode' | 'name' = 'isoAlpha2';\n  @Input() showName = true;\n\n  @Output() onSelect = new EventEmitter<any>();\n\n  @ViewChild('popover') popover!: PopoverComponent;\n\n  private countryCodesService = inject(CountryCodesService);\n  private sanitizer = inject(DomSanitizer);\n  private cdr = inject(ChangeDetectorRef);\n\n  value: any = null;\n  selectedCountry: any = null;\n  searchQuery = '';\n  shadowCountries: any[] = [];\n  isFocused = false;\n\n  onChange: any = () => { };\n  onTouched: any = () => { };\n\n  protected cn = cn;\n\n  ngOnInit() {\n    this.shadowCountries = this.countryCodesService.countries;\n    if (this.defaultCountryCode && !this.value) {\n      this.selectedCountry = this.countryCodesService.countries.find(\n        c => c.isoAlpha2 === this.defaultCountryCode\n      );\n      if (this.selectedCountry) {\n        this.value = this.getReturnValue(this.selectedCountry);\n      }\n    }\n  }\n\n  get computedTriggerClass() {\n    return cn(\n      'flex w-full items-center justify-between rounded-md border transition-all duration-200',\n      'bg-background text-foreground',\n      'border-input shadow-sm',\n      this.size === 'xs' && 'h-8 px-2 text-xs',\n      this.size === 'sm' && 'h-9 px-3 text-sm',\n      this.size === 'default' && 'h-10 px-3 text-sm',\n      this.size === 'lg' && 'h-11 px-4 text-base',\n      !(this.readonly || this.disabled) && [\n        'focus:outline-none',\n        'focus:ring-4',\n        'focus:ring-ring/30',\n        'focus:ring-offset-0',\n        'focus:border-primary/80'\n      ],\n      !(this.readonly || this.disabled) && 'hover:border-accent',\n      this.error && [\n        'border-destructive',\n        !(this.readonly || this.disabled) && [\n          'focus:border-destructive/80',\n          'focus:ring-destructive/30'\n        ]\n      ],\n      this.disabled && 'cursor-not-allowed opacity-50 border-opacity-50',\n      this.readonly && 'cursor-default border-dashed',\n      this.class\n    );\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 iconClass() {\n    return cn(\n      'ri-arrow-down-s-line text-muted-foreground ml-2 transition-transform duration-200',\n      this.popover?.isOpen ? 'rotate-180' : '',\n      (this.size === 'xs' || this.size === 'sm') ? 'text-[14px]' : 'text-[18px]',\n      (this.disabled || this.readonly) && 'opacity-30'\n    );\n  }\n\n  getItemClass(country: any) {\n    const isSelected = this.selectedCountry?.isoAlpha2 === country.isoAlpha2;\n    return cn(\n      'flex items-center justify-between px-3 py-2 cursor-pointer transition-colors duration-150 rounded-sm w-full',\n      isSelected ? 'bg-accent text-accent-foreground' : 'hover:bg-accent/50 text-foreground'\n    );\n  }\n\n  getFlagUrl(flagBase64: string): SafeResourceUrl {\n    return this.sanitizer.bypassSecurityTrustResourceUrl(\n      'data:image/*;base64,' + flagBase64\n    );\n  }\n\n  filterCountries(query: string) {\n    const filter = (query || '').toLowerCase().trim();\n    this.shadowCountries = this.countryCodesService.countries.filter(c =>\n      c.name.toLowerCase().includes(filter) ||\n      c.dialCode.includes(filter) ||\n      c.isoAlpha2.toLowerCase().includes(filter)\n    );\n  }\n\n  selectCountry(country: any) {\n    this.selectedCountry = country;\n    this.value = this.getReturnValue(country);\n    this.onChange(this.value);\n    this.onSelect.emit(country);\n    this.popover.close();\n  }\n\n  private getReturnValue(country: any) {\n    switch (this.returnValue) {\n      case 'object': return country;\n      case 'isoAlpha2': return country.isoAlpha2;\n      case 'dialCode': return country.dialCode;\n      case 'name': return country.name;\n      default: return country.isoAlpha2;\n    }\n  }\n\n  onFocus(): void {\n    this.isFocused = true;\n  }\n\n  onBlur(): void {\n    this.isFocused = false;\n    this.onTouched();\n  }\n\n  onPopoverClose() {\n    this.searchQuery = '';\n    this.filterCountries('');\n    this.onBlur();\n  }\n\n  writeValue(value: any): void {\n    this.value = value;\n    if (value) {\n      this.selectedCountry = this.countryCodesService.countries.find(c => {\n        if (this.returnValue === 'object') return c.isoAlpha2 === value.isoAlpha2;\n        if (this.returnValue === 'isoAlpha2') return c.isoAlpha2 === value;\n        if (this.returnValue === 'dialCode') return c.dialCode === value;\n        if (this.returnValue === 'name') return c.name === value;\n        return false;\n      });\n    } else {\n      this.selectedCountry = null;\n    }\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: any): void { this.onChange = fn; }\n  registerOnTouched(fn: any): void { this.onTouched = fn; }\n  setDisabledState(isDisabled: boolean): void { this.disabled = isDisabled; }\n}\n"]}
400
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"country-selector.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/country-selector.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  forwardRef,\n  inject,\n  OnInit,\n  ViewChild,\n  ElementRef,\n  ChangeDetectorRef,\n  Output,\n  EventEmitter,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { CountryCodesService } from './country-codes.service';\nimport { PopoverComponent } from './popover.component';\nimport { InputComponent } from './input.component';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-country-selector',\n  standalone: true,\n  imports: [CommonModule, FormsModule, PopoverComponent, InputComponent],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CountrySelectorComponent),\n      multi: true,\n    },\n  ],\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      <tolle-popover\n        #popover\n        [placement]=\"'bottom-start'\"\n        (onOpen)=\"onPopoverOpen()\"\n        (onClose)=\"onPopoverClose()\">\n        <div trigger class=\"w-full\">\n          <button\n            type=\"button\"\n            [id]=\"id\"\n            [disabled]=\"disabled\"\n            [class]=\"computedTriggerClass\"\n            (blur)=\"onBlur()\"\n            (focus)=\"onFocus()\"\n            [attr.aria-invalid]=\"error\"\n            [attr.aria-describedby]=\"error && errorMessage ? id + '-error' : null\">\n            <div class=\"flex items-center gap-2 truncate\">\n              <img\n                *ngIf=\"selectedCountry\"\n                [src]=\"getFlagUrl(selectedCountry.flag)\"\n                class=\"h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover\"\n                [alt]=\"selectedCountry.name\" />\n              <span *ngIf=\"selectedCountry && showName\" class=\"truncate font-medium\">\n                {{ selectedCountry.name }}\n              </span>\n              <span *ngIf=\"!selectedCountry\" class=\"truncate text-muted-foreground\">\n                {{ placeholder }}\n              </span>\n            </div>\n            <i [class]=\"iconClass\"></i>\n          </button>\n        </div>\n\n        <div\n          class=\"flex min-w-[300px] max-w-[400px] flex-col overflow-hidden rounded-md border border-border bg-popover shadow-md\"\n          (mousedown)=\"$event.stopPropagation()\">\n          <div class=\"sticky top-0 z-10 border-b border-border bg-popover p-2 shadow-sm\">\n            <tolle-input\n              size=\"sm\"\n              placeholder=\"Search country...\"\n              [(ngModel)]=\"searchQuery\"\n              (ngModelChange)=\"filterCountries($event)\"\n              class=\"w-full\"\n              #searchInput>\n              <i prefix class=\"ri-search-line\"></i>\n            </tolle-input>\n          </div>\n\n          <div class=\"max-h-[300px] overflow-y-auto p-1\">\n            <div\n              *ngFor=\"let country of shadowCountries\"\n              (click)=\"selectCountry(country)\"\n              [class]=\"getItemClass(country)\">\n              <div class=\"flex w-full items-center gap-3\">\n                <img\n                  [src]=\"getFlagUrl(country.flag)\"\n                  class=\"h-4 w-6 flex-shrink-0 rounded-sm border border-border object-cover\"\n                  [alt]=\"country.name\" />\n                <span class=\"flex-1 truncate text-sm\">{{ country.name }}</span>\n                <span class=\"text-xs text-muted-foreground\">{{ country.dialCode }}</span>\n                <i\n                  *ngIf=\"selectedCountry?.isoAlpha2 === country.isoAlpha2\"\n                  class=\"ri-check-line text-primary\"></i>\n              </div>\n            </div>\n            <div\n              *ngIf=\"shadowCountries.length === 0\"\n              class=\"py-6 text-center text-sm text-muted-foreground\">\n              No countries found.\n            </div>\n          </div>\n        </div>\n      </tolle-popover>\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          [class.opacity-0]=\"isFocused && hideHintOnFocus\">\n          {{ hint }}\n        </p>\n        <p\n          *ngIf=\"error && errorMessage\"\n          [id]=\"id + '-error'\"\n          class=\"px-1 text-xs font-medium text-destructive\">\n          {{ errorMessage }}\n        </p>\n      </ng-container>\n    </div>\n  `,\n})\nexport class CountrySelectorComponent implements OnInit, ControlValueAccessor {\n  @Input() id = `country-selector-${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 = 'Select country';\n  @Input() class = '';\n  @Input() disabled = false;\n  @Input() readonly = false;\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  @Input() defaultCountryCode = 'GH';\n  @Input() returnValue: 'object' | 'isoAlpha2' | 'dialCode' | 'name' = 'isoAlpha2';\n  @Input() showName = true;\n\n  @Output() onSelect = new EventEmitter<any>();\n\n  @ViewChild('popover') popover!: PopoverComponent;\n  @ViewChild('searchInput') searchInput!: ElementRef<HTMLInputElement>;\n\n  private countryCodesService = inject(CountryCodesService);\n  private sanitizer = inject(DomSanitizer);\n  private cdr = inject(ChangeDetectorRef);\n\n  value: any = null;\n  selectedCountry: any = null;\n  searchQuery = '';\n  shadowCountries: any[] = [];\n  isFocused = false;\n\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  protected cn = cn;\n\n  ngOnInit() {\n    this.shadowCountries = this.countryCodesService.countries;\n    if (this.defaultCountryCode && !this.value) {\n      this.selectedCountry = this.countryCodesService.countries.find(\n        c => c.isoAlpha2 === this.defaultCountryCode\n      );\n      if (this.selectedCountry) {\n        this.value = this.getReturnValue(this.selectedCountry);\n      }\n    }\n  }\n\n  get computedTriggerClass() {\n    return cn(\n      'flex w-full items-center justify-between rounded-md border transition-all duration-200',\n      'bg-background text-foreground',\n      'border-input shadow-sm',\n      this.size === 'xs' && 'h-8 px-2 text-xs',\n      this.size === 'sm' && 'h-9 px-3 text-sm',\n      this.size === 'default' && 'h-10 px-3 text-sm',\n      this.size === 'lg' && 'h-11 px-4 text-base',\n      !(this.readonly || this.disabled) && [\n        'focus:outline-none',\n        'focus:ring-4',\n        'focus:ring-ring/30',\n        'focus:ring-offset-0',\n        'focus:border-primary/80',\n      ],\n      !(this.readonly || this.disabled) && 'hover:border-accent',\n      this.error && [\n        'border-destructive',\n        !(this.readonly || this.disabled) && [\n          'focus:border-destructive/80',\n          'focus:ring-destructive/30',\n        ],\n      ],\n      this.disabled && 'cursor-not-allowed opacity-50 border-opacity-50',\n      this.readonly && 'cursor-default border-dashed',\n      this.class\n    );\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 iconClass() {\n    return cn(\n      'ri-arrow-down-s-line text-muted-foreground ml-2 transition-transform duration-200',\n      this.popover?.isOpen ? 'rotate-180' : '',\n      this.size === 'xs' || this.size === 'sm' ? 'text-[14px]' : 'text-[18px]',\n      (this.disabled || this.readonly) && 'opacity-30'\n    );\n  }\n\n  getItemClass(country: any) {\n    const isSelected = this.selectedCountry?.isoAlpha2 === country.isoAlpha2;\n    return cn(\n      'flex items-center justify-between px-3 py-2 cursor-pointer transition-colors duration-150 rounded-sm w-full',\n      isSelected ? 'bg-accent text-accent-foreground' : 'hover:bg-accent/50 text-foreground'\n    );\n  }\n\n  getFlagUrl(flagBase64: string): SafeResourceUrl {\n    return this.sanitizer.bypassSecurityTrustResourceUrl('data:image/*;base64,' + flagBase64);\n  }\n\n  filterCountries(query: string) {\n    const filter = (query || '').toLowerCase().trim();\n    this.shadowCountries = this.countryCodesService.countries.filter(\n      c =>\n        c.name.toLowerCase().includes(filter) ||\n        c.dialCode.includes(filter) ||\n        c.isoAlpha2.toLowerCase().includes(filter)\n    );\n  }\n\n  selectCountry(country: any) {\n    this.selectedCountry = country;\n    this.value = this.getReturnValue(country);\n    this.onChange(this.value);\n    this.onSelect.emit(country);\n    this.popover.close();\n  }\n\n  private getReturnValue(country: any) {\n    switch (this.returnValue) {\n      case 'object':\n        return country;\n      case 'isoAlpha2':\n        return country.isoAlpha2;\n      case 'dialCode':\n        return country.dialCode;\n      case 'name':\n        return country.name;\n      default:\n        return country.isoAlpha2;\n    }\n  }\n\n  onFocus(): void {\n    this.isFocused = true;\n  }\n\n  onBlur(): void {\n    this.isFocused = false;\n    this.onTouched();\n  }\n\n  onPopoverOpen() {\n    setTimeout(() => {\n      this.searchInput?.nativeElement?.focus();\n    }, 0);\n  }\n\n  onPopoverClose() {\n    this.searchQuery = '';\n    this.filterCountries('');\n    this.onBlur();\n  }\n\n  writeValue(value: any): void {\n    this.value = value;\n    if (value) {\n      this.selectedCountry = this.countryCodesService.countries.find(c => {\n        if (this.returnValue === 'object') return c.isoAlpha2 === value.isoAlpha2;\n        if (this.returnValue === 'isoAlpha2') return c.isoAlpha2 === value;\n        if (this.returnValue === 'dialCode') return c.dialCode === value;\n        if (this.returnValue === 'name') return c.name === value;\n        return false;\n      });\n    } else {\n      this.selectedCountry = null;\n    }\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n}\n"]}