@tolle_/tolle-ui 18.2.15 → 18.2.18

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.
@@ -24,7 +24,10 @@ export class CountrySelectorComponent {
24
24
  defaultCountryCode = 'GH';
25
25
  returnValue = 'isoAlpha2';
26
26
  showName = true;
27
+ mergedPosition = 'none';
27
28
  onSelect = new EventEmitter();
29
+ onFocusChange = new EventEmitter();
30
+ onBlurChange = new EventEmitter();
28
31
  popover;
29
32
  searchInput;
30
33
  countryCodesService = inject(CountryCodesService);
@@ -48,19 +51,20 @@ export class CountrySelectorComponent {
48
51
  }
49
52
  }
50
53
  get computedTriggerClass() {
51
- return cn('flex w-full items-center justify-between rounded-md border transition-all duration-200', 'bg-background text-foreground', 'border-input shadow-sm', this.size === 'xs' && 'h-8 px-2 text-xs', this.size === 'sm' && 'h-9 px-3 text-sm', this.size === 'default' && 'h-10 px-3 text-sm', this.size === 'lg' && 'h-11 px-4 text-base', !(this.readonly || this.disabled) && [
52
- 'focus:outline-none',
53
- 'focus:ring-4',
54
- 'focus:ring-ring/30',
55
- 'focus:ring-offset-0',
56
- 'focus:border-primary/80',
57
- ], !(this.readonly || this.disabled) && 'hover:border-accent', this.error && [
58
- 'border-destructive',
59
- !(this.readonly || this.disabled) && [
60
- 'focus:border-destructive/80',
61
- 'focus:ring-destructive/30',
62
- ],
63
- ], this.disabled && 'cursor-not-allowed opacity-50 border-opacity-50', this.readonly && 'cursor-default border-dashed', this.class);
54
+ return cn('flex w-full items-center justify-between border transition-all duration-200', 'bg-background text-foreground', 'border-input shadow-sm', this.size === 'xs' && 'h-8 px-2 text-xs', this.size === 'sm' && 'h-9 px-3 text-sm', this.size === 'default' && 'h-10 px-3 text-sm', this.size === 'lg' && 'h-11 px-4 text-base',
55
+ // Merged position
56
+ this.mergedPosition === 'left' && 'rounded-l-md rounded-r-none border-r-0', this.mergedPosition === 'none' && 'rounded-md',
57
+ // Focus state
58
+ !(this.readonly || this.disabled) &&
59
+ this.isFocused && [
60
+ 'ring-4',
61
+ 'ring-ring/30',
62
+ 'ring-offset-0',
63
+ 'shadow-none',
64
+ this.error ? 'border-destructive/80' : 'border-primary/80',
65
+ ], !(this.readonly || this.disabled) && 'hover:border-accent',
66
+ // Error state
67
+ this.error && ['border-destructive', this.isFocused && 'ring-destructive/30'], this.disabled && 'cursor-not-allowed opacity-50 border-opacity-50', this.readonly && 'cursor-default border-dashed', this.class);
64
68
  }
65
69
  get computedLabelClass() {
66
70
  return cn('text-sm font-medium text-foreground leading-none transition-opacity duration-200', this.disabled && 'opacity-50');
@@ -104,10 +108,12 @@ export class CountrySelectorComponent {
104
108
  }
105
109
  onFocus() {
106
110
  this.isFocused = true;
111
+ this.onFocusChange.emit();
107
112
  }
108
113
  onBlur() {
109
114
  this.isFocused = false;
110
115
  this.onTouched();
116
+ this.onBlurChange.emit();
111
117
  }
112
118
  onPopoverOpen() {
113
119
  setTimeout(() => {
@@ -149,7 +155,7 @@ export class CountrySelectorComponent {
149
155
  this.disabled = isDisabled;
150
156
  }
151
157
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CountrySelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
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: [
158
+ 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", mergedPosition: "mergedPosition" }, outputs: { onSelect: "onSelect", onFocusChange: "onFocusChange", onBlurChange: "onBlurChange" }, providers: [
153
159
  {
154
160
  provide: NG_VALUE_ACCESSOR,
155
161
  useExisting: forwardRef(() => CountrySelectorComponent),
@@ -157,7 +163,7 @@ export class CountrySelectorComponent {
157
163
  },
158
164
  ], viewQueries: [{ propertyName: "popover", first: true, predicate: ["popover"], descendants: true }, { propertyName: "searchInput", first: true, predicate: ["searchInput"], descendants: true }], ngImport: i0, template: `
159
165
  <div class="flex w-full flex-col gap-1.5">
160
- <label *ngIf="label" [for]="id" [class]="computedLabelClass">
166
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
161
167
  {{ label }}
162
168
  </label>
163
169
 
@@ -234,7 +240,7 @@ export class CountrySelectorComponent {
234
240
  </div>
235
241
  </tolle-popover>
236
242
 
237
- <ng-container *ngIf="!disabled">
243
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
238
244
  <p
239
245
  *ngIf="hint && !error"
240
246
  class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
@@ -266,7 +272,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
266
272
  ],
267
273
  template: `
268
274
  <div class="flex w-full flex-col gap-1.5">
269
- <label *ngIf="label" [for]="id" [class]="computedLabelClass">
275
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
270
276
  {{ label }}
271
277
  </label>
272
278
 
@@ -343,7 +349,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
343
349
  </div>
344
350
  </tolle-popover>
345
351
 
346
- <ng-container *ngIf="!disabled">
352
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
347
353
  <p
348
354
  *ngIf="hint && !error"
349
355
  class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
@@ -388,8 +394,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
388
394
  type: Input
389
395
  }], showName: [{
390
396
  type: Input
397
+ }], mergedPosition: [{
398
+ type: Input
391
399
  }], onSelect: [{
392
400
  type: Output
401
+ }], onFocusChange: [{
402
+ type: Output
403
+ }], onBlurChange: [{
404
+ type: Output
393
405
  }], popover: [{
394
406
  type: ViewChild,
395
407
  args: ['popover']
@@ -397,4 +409,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
397
409
  type: ViewChild,
398
410
  args: ['searchInput']
399
411
  }] } });
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"]}
412
+ //# 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 && mergedPosition === 'none'\" [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 && mergedPosition === 'none'\">\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  @Input() mergedPosition: 'left' | 'none' = 'none';\n\n  @Output() onSelect = new EventEmitter<any>();\n  @Output() onFocusChange = new EventEmitter<void>();\n  @Output() onBlurChange = new EventEmitter<void>();\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 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      // Merged position\n      this.mergedPosition === 'left' && 'rounded-l-md rounded-r-none border-r-0',\n      this.mergedPosition === 'none' && 'rounded-md',\n      // Focus state\n      !(this.readonly || this.disabled) &&\n        this.isFocused && [\n          'ring-4',\n          'ring-ring/30',\n          'ring-offset-0',\n          'shadow-none',\n          this.error ? 'border-destructive/80' : 'border-primary/80',\n        ],\n      !(this.readonly || this.disabled) && 'hover:border-accent',\n      // Error state\n      this.error && ['border-destructive', this.isFocused && 'ring-destructive/30'],\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    this.onFocusChange.emit();\n  }\n\n  onBlur(): void {\n    this.isFocused = false;\n    this.onTouched();\n    this.onBlurChange.emit();\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"]}
@@ -234,7 +234,7 @@ export class DatePickerComponent {
234
234
  ></tolle-calendar>
235
235
  </div>
236
236
  </div>
237
- `, isInline: true, 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: MaskedInputComponent, selector: "tolle-masked-input", inputs: ["id", "label", "hint", "errorMessage", "mask", "placeholder", "type", "disabled", "readonly", "class", "containerClass", "error", "size", "returnRaw", "hideHintOnFocus"] }, { kind: "component", type: CalendarComponent, selector: "tolle-calendar", inputs: ["class", "mode", "disablePastDates", "showQuickActions", "minDate", "maxDate", "formatMonthFn", "formatYearFn", "formatDateFn"], outputs: ["dateSelect"] }] });
237
+ `, isInline: true, 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: MaskedInputComponent, selector: "tolle-masked-input", inputs: ["id", "label", "hint", "errorMessage", "mask", "placeholder", "type", "disabled", "readonly", "class", "containerClass", "error", "size", "returnRaw", "hideHintOnFocus", "externalFocused", "mergedPosition"] }, { kind: "component", type: CalendarComponent, selector: "tolle-calendar", inputs: ["class", "mode", "disablePastDates", "showQuickActions", "minDate", "maxDate", "formatMonthFn", "formatYearFn", "formatDateFn"], outputs: ["dateSelect"] }] });
238
238
  }
239
239
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DatePickerComponent, decorators: [{
240
240
  type: Component,
@@ -328,4 +328,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
328
328
  type: ViewChild,
329
329
  args: ['popover']
330
330
  }] } });
331
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/date-picker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, forwardRef, ElementRef, ViewChild, ChangeDetectorRef, OnDestroy\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { computePosition, flip, shift, offset, autoUpdate } from '@floating-ui/dom';\nimport { format, parse, isValid, startOfDay } from 'date-fns';\nimport { cn } from './utils/cn';\nimport { MaskedInputComponent } from './masked-input.component';\nimport { CalendarComponent, CalendarMode } from './calendar.component';\n\n@Component({\n  selector: 'tolle-date-picker',\n  standalone: true,\n  imports: [CommonModule, FormsModule, MaskedInputComponent, CalendarComponent],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DatePickerComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"relative w-full\" #triggerContainer>\n      <tolle-masked-input\n        #maskInput\n        [mask]=\"getMask()\"\n        [placeholder]=\"getPlaceholder()\"\n        [disabled]=\"disabled\"\n        [(ngModel)]=\"inputValue\"\n        (ngModelChange)=\"onInputChange($event)\"\n        [class]=\"cn(class)\">\n        <div suffix class=\"flex items-center gap-1.5 cursor-pointer\">\n          <i\n            *ngIf=\"value && !disabled && showClear\"\n            (click)=\"clear($event)\"\n            class=\"ri-close-line cursor-pointer text-muted-foreground hover:text-foreground transition-colors\"\n          ></i>\n\n          <i\n            (click)=\"togglePopover($event)\"\n            [class]=\"cn(\n              'cursor-pointer text-muted-foreground transition-colors',\n              'ri-calendar-line'\n            )\"\n          ></i>\n        </div>\n      </tolle-masked-input>\n\n      <div\n        #popover\n        *ngIf=\"isOpen\"\n        class=\"fixed z-[50]\"\n        style=\"visibility: hidden; top: 0; left: 0;\"\n      >\n        <tolle-calendar class=\"shadow-lg\"\n          [(ngModel)]=\"value\"\n          (ngModelChange)=\"onCalendarChange($event)\"\n          [mode]=\"mode\"\n          [disablePastDates]=\"disablePastDates\"\n          [minDate]=\"minDate\"\n          [maxDate]=\"maxDate\"\n          [showQuickActions]=\"showQuickActions\"\n          [formatMonthFn]=\"formatMonthFn\"\n          [formatYearFn]=\"formatYearFn\"\n        ></tolle-calendar>\n      </div>\n    </div>\n  `\n})\nexport class DatePickerComponent implements ControlValueAccessor, OnDestroy {\n  @Input() placeholder = 'MM/DD/YYYY';\n  @Input() disabled = false;\n  @Input() class = '';\n  @Input() disablePastDates = false;\n  @Input() showClear = true;\n  @Input() showQuickActions = true;\n  @Input() minDate?: Date;\n  @Input() maxDate?: Date;\n  @Input() mode: CalendarMode = 'date';\n  @Input() formatMonthFn?: (date: Date) => string;\n  @Input() formatYearFn?: (date: Date) => string;\n\n  // Format functions for display\n  @Input() displayFormat?: (date: Date, mode: CalendarMode) => string;\n\n  @ViewChild('triggerContainer') triggerContainer!: ElementRef;\n  @ViewChild('popover') popover!: ElementRef;\n\n  value: Date | null = null;\n  inputValue: string = '';\n  isOpen = false;\n  cleanupAutoUpdate?: () => void;\n\n  private _outsideClickHandler = (event: MouseEvent) => {\n    if (!this.triggerContainer.nativeElement.contains(event.target) && !this.popover?.nativeElement.contains(event.target)) {\n      this.close();\n    }\n  };\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  getMask(): string {\n    switch (this.mode) {\n      case 'date': return '00/00/0000';\n      case 'month': return '00/0000';\n      case 'year': return '0000';\n      default: return '00/00/0000';\n    }\n  }\n\n  getPlaceholder(): string {\n    switch (this.mode) {\n      case 'date': return 'MM/DD/YYYY';\n      case 'month': return 'MM/YYYY';\n      case 'year': return 'YYYY';\n      default: return 'MM/DD/YYYY';\n    }\n  }\n\n  getFormatString(): string {\n    switch (this.mode) {\n      case 'date': return 'MM/dd/yyyy';\n      case 'month': return 'MM/yyyy';\n      case 'year': return 'yyyy';\n      default: return 'MM/dd/yyyy';\n    }\n  }\n\n  formatDate(date: Date): string {\n    if (this.displayFormat) {\n      return this.displayFormat(date, this.mode);\n    }\n\n    switch (this.mode) {\n      case 'date': return format(date, 'MM/dd/yyyy');\n      case 'month': return format(date, 'MM/yyyy');\n      case 'year': return format(date, 'yyyy');\n      default: return format(date, 'MM/dd/yyyy');\n    }\n  }\n\n  parseDate(str: string): Date | null {\n    try {\n      const parsed = parse(str, this.getFormatString(), new Date());\n      return isValid(parsed) ? startOfDay(parsed) : null;\n    } catch {\n      return null;\n    }\n  }\n\n  onInputChange(str: string) {\n    const expectedLength = this.getFormatString().replace(/[^0]/g, '').length;\n\n    if (str?.length === expectedLength) {\n      const parsed = this.parseDate(str);\n      if (parsed) {\n        this.value = parsed;\n        this.onChange(this.value);\n      }\n    } else if (!str) {\n      this.value = null;\n      this.onChange(null);\n    }\n  }\n\n  onCalendarChange(date: Date | null) {\n    this.value = date;\n    if (date) {\n      this.inputValue = this.formatDate(date);\n    } else {\n      this.inputValue = '';\n    }\n    this.onChange(this.value);\n    this.close();\n  }\n\n  togglePopover(event: MouseEvent) {\n    event.stopPropagation();\n    if (this.disabled) return;\n    this.isOpen ? this.close() : this.open();\n  }\n\n  open() {\n    this.isOpen = true;\n    setTimeout(() => {\n      this.updatePosition();\n      document.addEventListener('mousedown', this._outsideClickHandler);\n    });\n  }\n\n  close() {\n    this.isOpen = false;\n    if (this.cleanupAutoUpdate) this.cleanupAutoUpdate();\n    document.removeEventListener('mousedown', this._outsideClickHandler);\n  }\n\n  clear(event: MouseEvent) {\n    event.stopPropagation();\n    this.value = null;\n    this.inputValue = '';\n    this.onChange(null);\n    this.cdr.markForCheck();\n  }\n\n  private updatePosition() {\n    if (!this.triggerContainer || !this.popover) return;\n\n    this.cleanupAutoUpdate = autoUpdate(\n      this.triggerContainer.nativeElement,\n      this.popover.nativeElement,\n      () => {\n        computePosition(this.triggerContainer.nativeElement, this.popover.nativeElement, {\n          strategy: 'fixed', // ADDED: Fixed strategy\n          placement: 'bottom-start', // Changed to bottom-start to align with input left edge\n          middleware: [\n            offset(4),\n            flip(),\n            shift({ padding: 8 })\n          ],\n        }).then(({ x, y, strategy }) => {\n          Object.assign(this.popover.nativeElement.style, {\n            position: strategy,\n            left: `${x}px`,\n            top: `${y}px`,\n            visibility: 'visible',\n          });\n        });\n      }\n    );\n  }\n\n  ngOnDestroy() {\n    if (this.cleanupAutoUpdate) this.cleanupAutoUpdate();\n    document.removeEventListener('mousedown', this._outsideClickHandler);\n  }\n\n  // CVA Implementation\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  writeValue(val: any): void {\n    if (val) {\n      const date = new Date(val);\n      if (isValid(date)) {\n        this.value = startOfDay(date);\n        this.inputValue = this.formatDate(this.value);\n      }\n    } else {\n      this.value = null;\n      this.inputValue = '';\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  protected cn = cn;\n}\n"]}
331
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/date-picker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, forwardRef, ElementRef, ViewChild, ChangeDetectorRef, OnDestroy\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { computePosition, flip, shift, offset, autoUpdate } from '@floating-ui/dom';\nimport { format, parse, isValid, startOfDay } from 'date-fns';\nimport { cn } from './utils/cn';\nimport { MaskedInputComponent } from './masked-input.component';\nimport { CalendarComponent, CalendarMode } from './calendar.component';\n\n@Component({\n  selector: 'tolle-date-picker',\n  standalone: true,\n  imports: [CommonModule, FormsModule, MaskedInputComponent, CalendarComponent],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DatePickerComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"relative w-full\" #triggerContainer>\n      <tolle-masked-input\n        #maskInput\n        [mask]=\"getMask()\"\n        [placeholder]=\"getPlaceholder()\"\n        [disabled]=\"disabled\"\n        [(ngModel)]=\"inputValue\"\n        (ngModelChange)=\"onInputChange($event)\"\n        [class]=\"cn(class)\">\n        <div suffix class=\"flex items-center gap-1.5 cursor-pointer\">\n          <i\n            *ngIf=\"value && !disabled && showClear\"\n            (click)=\"clear($event)\"\n            class=\"ri-close-line cursor-pointer text-muted-foreground hover:text-foreground transition-colors\"\n          ></i>\n\n          <i\n            (click)=\"togglePopover($event)\"\n            [class]=\"cn(\n              'cursor-pointer text-muted-foreground transition-colors',\n              'ri-calendar-line'\n            )\"\n          ></i>\n        </div>\n      </tolle-masked-input>\n\n      <div\n        #popover\n        *ngIf=\"isOpen\"\n        class=\"fixed z-[50]\"\n        style=\"visibility: hidden; top: 0; left: 0;\"\n      >\n        <tolle-calendar class=\"shadow-lg\"\n          [(ngModel)]=\"value\"\n          (ngModelChange)=\"onCalendarChange($event)\"\n          [mode]=\"mode\"\n          [disablePastDates]=\"disablePastDates\"\n          [minDate]=\"minDate\"\n          [maxDate]=\"maxDate\"\n          [showQuickActions]=\"showQuickActions\"\n          [formatMonthFn]=\"formatMonthFn\"\n          [formatYearFn]=\"formatYearFn\"\n        ></tolle-calendar>\n      </div>\n    </div>\n  `\n})\nexport class DatePickerComponent implements ControlValueAccessor, OnDestroy {\n  @Input() placeholder = 'MM/DD/YYYY';\n  @Input() disabled = false;\n  @Input() class = '';\n  @Input() disablePastDates = false;\n  @Input() showClear = true;\n  @Input() showQuickActions = true;\n  @Input() minDate?: Date;\n  @Input() maxDate?: Date;\n  @Input() mode: CalendarMode = 'date';\n  @Input() formatMonthFn?: (date: Date) => string;\n  @Input() formatYearFn?: (date: Date) => string;\n\n  // Format functions for display\n  @Input() displayFormat?: (date: Date, mode: CalendarMode) => string;\n\n  @ViewChild('triggerContainer') triggerContainer!: ElementRef;\n  @ViewChild('popover') popover!: ElementRef;\n\n  value: Date | null = null;\n  inputValue: string = '';\n  isOpen = false;\n  cleanupAutoUpdate?: () => void;\n\n  private _outsideClickHandler = (event: MouseEvent) => {\n    if (!this.triggerContainer.nativeElement.contains(event.target) && !this.popover?.nativeElement.contains(event.target)) {\n      this.close();\n    }\n  };\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  getMask(): string {\n    switch (this.mode) {\n      case 'date': return '00/00/0000';\n      case 'month': return '00/0000';\n      case 'year': return '0000';\n      default: return '00/00/0000';\n    }\n  }\n\n  getPlaceholder(): string {\n    switch (this.mode) {\n      case 'date': return 'MM/DD/YYYY';\n      case 'month': return 'MM/YYYY';\n      case 'year': return 'YYYY';\n      default: return 'MM/DD/YYYY';\n    }\n  }\n\n  getFormatString(): string {\n    switch (this.mode) {\n      case 'date': return 'MM/dd/yyyy';\n      case 'month': return 'MM/yyyy';\n      case 'year': return 'yyyy';\n      default: return 'MM/dd/yyyy';\n    }\n  }\n\n  formatDate(date: Date): string {\n    if (this.displayFormat) {\n      return this.displayFormat(date, this.mode);\n    }\n\n    switch (this.mode) {\n      case 'date': return format(date, 'MM/dd/yyyy');\n      case 'month': return format(date, 'MM/yyyy');\n      case 'year': return format(date, 'yyyy');\n      default: return format(date, 'MM/dd/yyyy');\n    }\n  }\n\n  parseDate(str: string): Date | null {\n    try {\n      const parsed = parse(str, this.getFormatString(), new Date());\n      return isValid(parsed) ? startOfDay(parsed) : null;\n    } catch {\n      return null;\n    }\n  }\n\n  onInputChange(str: string) {\n    const expectedLength = this.getFormatString().replace(/[^0]/g, '').length;\n\n    if (str?.length === expectedLength) {\n      const parsed = this.parseDate(str);\n      if (parsed) {\n        this.value = parsed;\n        this.onChange(this.value);\n      }\n    } else if (!str) {\n      this.value = null;\n      this.onChange(null);\n    }\n  }\n\n  onCalendarChange(date: Date | null) {\n    this.value = date;\n    if (date) {\n      this.inputValue = this.formatDate(date);\n    } else {\n      this.inputValue = '';\n    }\n    this.onChange(this.value);\n    this.close();\n  }\n\n  togglePopover(event: MouseEvent) {\n    event.stopPropagation();\n    if (this.disabled) return;\n    this.isOpen ? this.close() : this.open();\n  }\n\n  open() {\n    this.isOpen = true;\n    setTimeout(() => {\n      this.updatePosition();\n      document.addEventListener('mousedown', this._outsideClickHandler);\n    });\n  }\n\n  close() {\n    this.isOpen = false;\n    if (this.cleanupAutoUpdate) this.cleanupAutoUpdate();\n    document.removeEventListener('mousedown', this._outsideClickHandler);\n  }\n\n  clear(event: MouseEvent) {\n    event.stopPropagation();\n    this.value = null;\n    this.inputValue = '';\n    this.onChange(null);\n    this.cdr.markForCheck();\n  }\n\n  private updatePosition() {\n    if (!this.triggerContainer || !this.popover) return;\n\n    this.cleanupAutoUpdate = autoUpdate(\n      this.triggerContainer.nativeElement,\n      this.popover.nativeElement,\n      () => {\n        computePosition(this.triggerContainer.nativeElement, this.popover.nativeElement, {\n          strategy: 'fixed', // ADDED: Fixed strategy\n          placement: 'bottom-start', // Changed to bottom-start to align with input left edge\n          middleware: [\n            offset(4),\n            flip(),\n            shift({ padding: 8 })\n          ],\n        }).then(({ x, y, strategy }) => {\n          Object.assign(this.popover.nativeElement.style, {\n            position: strategy,\n            left: `${x}px`,\n            top: `${y}px`,\n            visibility: 'visible',\n          });\n        });\n      }\n    );\n  }\n\n  ngOnDestroy() {\n    if (this.cleanupAutoUpdate) this.cleanupAutoUpdate();\n    document.removeEventListener('mousedown', this._outsideClickHandler);\n  }\n\n  // CVA Implementation\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  writeValue(val: any): void {\n    if (val) {\n      const date = new Date(val);\n      if (isValid(date)) {\n        this.value = startOfDay(date);\n        this.inputValue = this.formatDate(this.value);\n      }\n    } else {\n      this.value = null;\n      this.inputValue = '';\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  protected cn = cn;\n}\n"]}