@tolle_/tolle-ui 18.2.16 → 18.2.19

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,4 @@
1
- import { Component, Input, forwardRef, ViewChild } from '@angular/core';
1
+ import { Component, Input, forwardRef, ViewChild, } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
4
4
  import { cn } from './utils/cn';
@@ -22,13 +22,18 @@ export class MaskedInputComponent {
22
22
  size = 'default';
23
23
  returnRaw = false;
24
24
  hideHintOnFocus = true;
25
+ mergedPosition = 'none';
25
26
  inputEl;
26
27
  hasPrefix = false;
27
28
  hasSuffix = false;
28
29
  displayValue = '';
29
30
  isFocused = false;
30
31
  tokens = {
31
- '0': /\d/, '9': /\d/, 'a': /[a-z]/i, 'A': /[a-z]/i, '*': /[a-z0-9]/i
32
+ '0': /\d/,
33
+ '9': /\d/,
34
+ a: /[a-z]/i,
35
+ A: /[a-z]/i,
36
+ '*': /[a-z0-9]/i,
32
37
  };
33
38
  onChange = () => { };
34
39
  onTouched = () => { };
@@ -46,58 +51,48 @@ export class MaskedInputComponent {
46
51
  }
47
52
  }
48
53
  get computedLabelClass() {
49
- return cn("text-sm font-medium text-foreground leading-none transition-opacity duration-200", this.disabled && "opacity-50");
54
+ return cn('text-sm font-medium text-foreground leading-none transition-opacity duration-200', this.disabled && 'opacity-50');
50
55
  }
51
56
  get computedContainerClass() {
52
57
  return cn(
53
58
  // Base styles
54
- "group relative flex items-center w-full rounded-md border transition-all duration-200", "bg-background",
55
- // Border and shadow
56
- "border-input shadow-sm",
59
+ 'group relative flex items-center w-full border transition-all duration-200', 'bg-background border-input shadow-sm',
57
60
  // Sizing
58
- this.size === 'xs' && "h-8 px-2 gap-1.5 text-xs", this.size === 'sm' && "h-9 px-3 gap-2 text-sm", this.size === 'default' && "h-10 px-3 gap-2 text-sm", this.size === 'lg' && "h-11 px-4 gap-3 text-base",
59
- // Focus state - SIMPLE LIKE ZARDUI
61
+ this.size === 'xs' && 'h-8 px-2 gap-1.5 text-xs', this.size === 'sm' && 'h-9 px-3 gap-2 text-sm', this.size === 'default' && 'h-10 px-3 gap-2 text-sm', this.size === 'lg' && 'h-11 px-4 gap-3 text-base',
62
+ // Merged position - handle border radius
63
+ this.mergedPosition === 'left' && 'rounded-l-md rounded-r-none border-r-0 pr-0', this.mergedPosition === 'right' && 'rounded-r-md rounded-l-none border-l-0 pl-0 flex-1', this.mergedPosition === 'none' && 'rounded-md',
64
+ // Focus state
60
65
  !(this.readonly || this.disabled) && [
61
- "focus-within:border-primary/80",
62
- "focus-within:ring-4",
63
- "focus-within:ring-ring/30",
64
- "focus-within:ring-offset-0",
65
- "focus-within:shadow-none",
66
+ 'focus-within:ring-4',
67
+ 'focus-within:ring-ring/30',
68
+ 'focus-within:ring-offset-0',
69
+ 'focus-within:shadow-none',
70
+ this.error ? 'focus-within:border-destructive/80' : 'focus-within:border-primary/80',
66
71
  ],
67
72
  // Error state
68
73
  this.error && [
69
- "border-destructive",
70
- !(this.readonly || this.disabled) && [
71
- "focus-within:border-destructive/80",
72
- "focus-within:ring-destructive/30"
73
- ]
74
+ 'border-destructive',
75
+ !(this.readonly || this.disabled) && 'focus-within:ring-destructive/30',
74
76
  ],
75
77
  // Disabled state
76
- this.disabled && [
77
- "cursor-not-allowed opacity-50",
78
- "border-opacity-50"
79
- ],
78
+ this.disabled && ['cursor-not-allowed opacity-50', 'border-opacity-50'],
80
79
  // Readonly state
81
- this.readonly && [
82
- "cursor-default",
83
- "border-dashed",
84
- !this.disabled && "focus-within:ring-0 focus-within:border-opacity-100"
85
- ], this.containerClass);
80
+ this.readonly && ['cursor-default', 'border-dashed'], this.containerClass);
86
81
  }
87
82
  get computedInputClass() {
88
83
  return cn(
89
84
  // Base styles
90
- "flex-1 bg-transparent border-none p-0", "placeholder:text-muted-foreground",
85
+ 'flex-1 bg-transparent border-none p-0', 'placeholder:text-muted-foreground',
91
86
  // Remove all default focus styles
92
- "focus:outline-none focus:ring-0 focus:shadow-none",
87
+ 'focus:outline-none focus:ring-0 focus:shadow-none',
93
88
  // Text sizing
94
- this.size === 'xs' && "text-xs", this.size === 'sm' && "text-sm", this.size === 'default' && "text-sm", this.size === 'lg' && "text-base",
89
+ this.size === 'xs' && 'text-xs', this.size === 'sm' && 'text-sm', this.size === 'default' && 'text-sm', this.size === 'lg' && 'text-base',
95
90
  // Cursor states
96
- this.disabled && "cursor-not-allowed", this.readonly && "cursor-default",
91
+ this.disabled && 'cursor-not-allowed', this.readonly && 'cursor-default',
97
92
  // Text color
98
- "text-foreground",
93
+ 'text-foreground',
99
94
  // Selection color
100
- "selection:bg-primary/20 selection:text-foreground", this.class);
95
+ 'selection:bg-primary/20 selection:text-foreground', this.class);
101
96
  }
102
97
  focusInput() {
103
98
  if (!this.disabled && this.inputEl) {
@@ -167,28 +162,22 @@ export class MaskedInputComponent {
167
162
  }
168
163
  cn = cn;
169
164
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MaskedInputComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
170
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MaskedInputComponent, isStandalone: true, selector: "tolle-masked-input", inputs: { id: "id", label: "label", hint: "hint", errorMessage: "errorMessage", mask: "mask", placeholder: "placeholder", type: "type", disabled: "disabled", readonly: "readonly", class: "class", containerClass: "containerClass", error: "error", size: "size", returnRaw: "returnRaw", hideHintOnFocus: "hideHintOnFocus" }, providers: [
165
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MaskedInputComponent, isStandalone: true, selector: "tolle-masked-input", inputs: { id: "id", label: "label", hint: "hint", errorMessage: "errorMessage", mask: "mask", placeholder: "placeholder", type: "type", disabled: "disabled", readonly: "readonly", class: "class", containerClass: "containerClass", error: "error", size: "size", returnRaw: "returnRaw", hideHintOnFocus: "hideHintOnFocus", mergedPosition: "mergedPosition" }, providers: [
171
166
  {
172
167
  provide: NG_VALUE_ACCESSOR,
173
168
  useExisting: forwardRef(() => MaskedInputComponent),
174
- multi: true
175
- }
169
+ multi: true,
170
+ },
176
171
  ], viewQueries: [{ propertyName: "inputEl", first: true, predicate: ["inputEl"], descendants: true, static: true }], ngImport: i0, template: `
177
- <div class="flex flex-col gap-1.5 w-full">
178
- <label
179
- *ngIf="label"
180
- [for]="id"
181
- [class]="computedLabelClass"
182
- >
172
+ <div class="flex w-full flex-col gap-1.5">
173
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
183
174
  {{ label }}
184
175
  </label>
185
176
 
186
- <div
187
- [class]="computedContainerClass"
188
- (click)="focusInput()"
189
- >
177
+ <div [class]="computedContainerClass" (click)="focusInput()">
190
178
  <!-- Prefix Icon -->
191
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
179
+ <div
180
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
192
181
  <ng-content select="[prefix]"></ng-content>
193
182
  </div>
194
183
 
@@ -205,28 +194,23 @@ export class MaskedInputComponent {
205
194
  (focus)="onFocus()"
206
195
  [class]="computedInputClass"
207
196
  [attr.aria-invalid]="error"
208
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
209
- />
197
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null" />
210
198
 
211
199
  <!-- Suffix Icon -->
212
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
200
+ <div
201
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
213
202
  <ng-content select="[suffix]"></ng-content>
214
203
  </div>
215
204
  </div>
216
205
 
217
- <ng-container *ngIf="!disabled">
206
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
218
207
  <p
219
208
  *ngIf="hint && !error"
220
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
221
- [class.opacity-0]="isFocused && hideHintOnFocus"
222
- >
209
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
210
+ [class.opacity-0]="isFocused && hideHintOnFocus">
223
211
  {{ hint }}
224
212
  </p>
225
- <p
226
- *ngIf="error && errorMessage"
227
- [id]="id + '-error'"
228
- class="text-xs text-destructive px-1"
229
- >
213
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
230
214
  {{ errorMessage }}
231
215
  </p>
232
216
  </ng-container>
@@ -243,25 +227,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
243
227
  {
244
228
  provide: NG_VALUE_ACCESSOR,
245
229
  useExisting: forwardRef(() => MaskedInputComponent),
246
- multi: true
247
- }
230
+ multi: true,
231
+ },
248
232
  ],
249
233
  template: `
250
- <div class="flex flex-col gap-1.5 w-full">
251
- <label
252
- *ngIf="label"
253
- [for]="id"
254
- [class]="computedLabelClass"
255
- >
234
+ <div class="flex w-full flex-col gap-1.5">
235
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
256
236
  {{ label }}
257
237
  </label>
258
238
 
259
- <div
260
- [class]="computedContainerClass"
261
- (click)="focusInput()"
262
- >
239
+ <div [class]="computedContainerClass" (click)="focusInput()">
263
240
  <!-- Prefix Icon -->
264
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
241
+ <div
242
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
265
243
  <ng-content select="[prefix]"></ng-content>
266
244
  </div>
267
245
 
@@ -278,33 +256,28 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
278
256
  (focus)="onFocus()"
279
257
  [class]="computedInputClass"
280
258
  [attr.aria-invalid]="error"
281
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
282
- />
259
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null" />
283
260
 
284
261
  <!-- Suffix Icon -->
285
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
262
+ <div
263
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
286
264
  <ng-content select="[suffix]"></ng-content>
287
265
  </div>
288
266
  </div>
289
267
 
290
- <ng-container *ngIf="!disabled">
268
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
291
269
  <p
292
270
  *ngIf="hint && !error"
293
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
294
- [class.opacity-0]="isFocused && hideHintOnFocus"
295
- >
271
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
272
+ [class.opacity-0]="isFocused && hideHintOnFocus">
296
273
  {{ hint }}
297
274
  </p>
298
- <p
299
- *ngIf="error && errorMessage"
300
- [id]="id + '-error'"
301
- class="text-xs text-destructive px-1"
302
- >
275
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
303
276
  {{ errorMessage }}
304
277
  </p>
305
278
  </ng-container>
306
279
  </div>
307
- `
280
+ `,
308
281
  }]
309
282
  }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }], propDecorators: { id: [{
310
283
  type: Input
@@ -336,8 +309,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
336
309
  type: Input
337
310
  }], hideHintOnFocus: [{
338
311
  type: Input
312
+ }], mergedPosition: [{
313
+ type: Input
339
314
  }], inputEl: [{
340
315
  type: ViewChild,
341
316
  args: ['inputEl', { static: true }]
342
317
  }] } });
343
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"masked-input.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/masked-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, forwardRef, ElementRef, ViewChild,\n  AfterContentChecked, ChangeDetectorRef\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-masked-input',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MaskedInputComponent),\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      >\n        {{ label }}\n      </label>\n\n      <div\n        [class]=\"computedContainerClass\"\n        (click)=\"focusInput()\"\n      >\n        <!-- Prefix Icon -->\n        <div class=\"h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200\">\n          <ng-content select=\"[prefix]\"></ng-content>\n        </div>\n\n        <input\n          #inputEl\n          [id]=\"id\"\n          [type]=\"type\"\n          [placeholder]=\"placeholder\"\n          [disabled]=\"disabled\"\n          [readOnly]=\"readonly\"\n          [value]=\"displayValue\"\n          (input)=\"onInput($event)\"\n          (blur)=\"onBlur()\"\n          (focus)=\"onFocus()\"\n          [class]=\"computedInputClass\"\n          [attr.aria-invalid]=\"error\"\n          [attr.aria-describedby]=\"error && errorMessage ? id + '-error' : null\"\n        />\n\n        <!-- Suffix Icon -->\n        <div class=\"h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200\">\n          <ng-content select=\"[suffix]\"></ng-content>\n        </div>\n      </div>\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\"\n        >\n          {{ errorMessage }}\n        </p>\n      </ng-container>\n    </div>\n  `\n})\nexport class MaskedInputComponent implements ControlValueAccessor, AfterContentChecked {\n  @Input() id: string = `masked-input-${Math.random().toString(36).substr(2, 9)}`;\n  @Input() label: string = '';\n  @Input() hint: string = '';\n  @Input() errorMessage: string = '';\n  @Input() mask: string = '';\n  @Input() placeholder = '';\n  @Input() type = 'text';\n  @Input() disabled = false;\n  @Input() readonly = false;\n  @Input() class = '';\n  @Input() containerClass: string = '';\n  @Input() error: boolean = false;\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  @Input() returnRaw = false;\n  @Input() hideHintOnFocus: boolean = true;\n\n  @ViewChild('inputEl', { static: true }) inputEl!: ElementRef<HTMLInputElement>;\n\n  hasPrefix = false;\n  hasSuffix = false;\n  displayValue = '';\n  isFocused: boolean = false;\n\n  private tokens: { [key: string]: RegExp } = {\n    '0': /\\d/, '9': /\\d/, 'a': /[a-z]/i, 'A': /[a-z]/i, '*': /[a-z0-9]/i\n  };\n\n  onChange: any = () => { };\n  onTouched: any = () => { };\n\n  constructor(private el: ElementRef, private cdr: ChangeDetectorRef) { }\n\n  ngAfterContentChecked() {\n    const prefix = this.el.nativeElement.querySelector('[prefix]');\n    const suffix = this.el.nativeElement.querySelector('[suffix]');\n\n    if (this.hasPrefix !== !!prefix || this.hasSuffix !== !!suffix) {\n      this.hasPrefix = !!prefix;\n      this.hasSuffix = !!suffix;\n      this.cdr.detectChanges();\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 computedContainerClass() {\n    return cn(\n      // Base styles\n      \"group relative flex items-center w-full rounded-md border transition-all duration-200\",\n      \"bg-background\",\n\n      // Border and shadow\n      \"border-input shadow-sm\",\n\n      // Sizing\n      this.size === 'xs' && \"h-8 px-2 gap-1.5 text-xs\",\n      this.size === 'sm' && \"h-9 px-3 gap-2 text-sm\",\n      this.size === 'default' && \"h-10 px-3 gap-2 text-sm\",\n      this.size === 'lg' && \"h-11 px-4 gap-3 text-base\",\n\n      // Focus state - SIMPLE LIKE ZARDUI\n      !(this.readonly || this.disabled) && [\n        \"focus-within:border-primary/80\",\n        \"focus-within:ring-4\",\n        \"focus-within:ring-ring/30\",\n        \"focus-within:ring-offset-0\",\n        \"focus-within:shadow-none\",\n      ],\n\n      // Error state\n      this.error && [\n        \"border-destructive\",\n        !(this.readonly || this.disabled) && [\n          \"focus-within:border-destructive/80\",\n          \"focus-within:ring-destructive/30\"\n        ]\n      ],\n\n      // Disabled state\n      this.disabled && [\n        \"cursor-not-allowed opacity-50\",\n        \"border-opacity-50\"\n      ],\n\n      // Readonly state\n      this.readonly && [\n        \"cursor-default\",\n        \"border-dashed\",\n        !this.disabled && \"focus-within:ring-0 focus-within:border-opacity-100\"\n      ],\n\n      this.containerClass\n    );\n  }\n\n  get computedInputClass() {\n    return cn(\n      // Base styles\n      \"flex-1 bg-transparent border-none p-0\",\n      \"placeholder:text-muted-foreground\",\n\n      // Remove all default focus styles\n      \"focus:outline-none focus:ring-0 focus:shadow-none\",\n\n      // Text sizing\n      this.size === 'xs' && \"text-xs\",\n      this.size === 'sm' && \"text-sm\",\n      this.size === 'default' && \"text-sm\",\n      this.size === 'lg' && \"text-base\",\n\n      // Cursor states\n      this.disabled && \"cursor-not-allowed\",\n      this.readonly && \"cursor-default\",\n\n      // Text color\n      \"text-foreground\",\n\n      // Selection color\n      \"selection:bg-primary/20 selection:text-foreground\",\n\n      this.class\n    );\n  }\n\n  focusInput(): void {\n    if (!this.disabled && this.inputEl) {\n      this.inputEl.nativeElement.focus();\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  // --- Masking Logic ---\n  onInput(event: Event) {\n    if (this.readonly || this.disabled) return;\n    const input = event.target as HTMLInputElement;\n    const raw = this.unmask(input.value);\n    const masked = this.applyMask(raw);\n    this.displayValue = masked;\n    input.value = masked;\n    this.returnRaw ? this.onChange(raw) : this.onChange(masked);\n  }\n\n  private applyMask(rawValue: string): string {\n    let rawIndex = 0;\n    let formatted = '';\n    for (let i = 0; i < this.mask.length; i++) {\n      if (rawIndex >= rawValue.length) break;\n      const maskChar = this.mask[i];\n      const rawChar = rawValue[rawIndex];\n      if (this.tokens[maskChar]) {\n        if (this.tokens[maskChar].test(rawChar)) {\n          formatted += rawChar;\n          rawIndex++;\n        } else {\n          rawIndex++;\n          i--;\n        }\n      } else {\n        formatted += maskChar;\n        if (rawChar === maskChar) rawIndex++;\n      }\n    }\n    return formatted;\n  }\n\n  private unmask(val: string): string {\n    return val.replace(/[^a-zA-Z0-9]/g, '');\n  }\n\n  writeValue(value: any): void {\n    this.displayValue = value ? this.applyMask(this.unmask(value.toString())) : '';\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  protected cn = cn;\n}\n"]}
318
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"masked-input.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/masked-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  forwardRef,\n  ElementRef,\n  ViewChild,\n  AfterContentChecked,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-masked-input',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MaskedInputComponent),\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      <div [class]=\"computedContainerClass\" (click)=\"focusInput()\">\n        <!-- Prefix Icon -->\n        <div\n          class=\"flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary\">\n          <ng-content select=\"[prefix]\"></ng-content>\n        </div>\n\n        <input\n          #inputEl\n          [id]=\"id\"\n          [type]=\"type\"\n          [placeholder]=\"placeholder\"\n          [disabled]=\"disabled\"\n          [readOnly]=\"readonly\"\n          [value]=\"displayValue\"\n          (input)=\"onInput($event)\"\n          (blur)=\"onBlur()\"\n          (focus)=\"onFocus()\"\n          [class]=\"computedInputClass\"\n          [attr.aria-invalid]=\"error\"\n          [attr.aria-describedby]=\"error && errorMessage ? id + '-error' : null\" />\n\n        <!-- Suffix Icon -->\n        <div\n          class=\"flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary\">\n          <ng-content select=\"[suffix]\"></ng-content>\n        </div>\n      </div>\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 *ngIf=\"error && errorMessage\" [id]=\"id + '-error'\" class=\"px-1 text-xs text-destructive\">\n          {{ errorMessage }}\n        </p>\n      </ng-container>\n    </div>\n  `,\n})\nexport class MaskedInputComponent implements ControlValueAccessor, AfterContentChecked {\n  @Input() id: string = `masked-input-${Math.random().toString(36).substr(2, 9)}`;\n  @Input() label: string = '';\n  @Input() hint: string = '';\n  @Input() errorMessage: string = '';\n  @Input() mask: string = '';\n  @Input() placeholder = '';\n  @Input() type = 'text';\n  @Input() disabled = false;\n  @Input() readonly = false;\n  @Input() class = '';\n  @Input() containerClass: string = '';\n  @Input() error: boolean = false;\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  @Input() returnRaw = false;\n  @Input() hideHintOnFocus: boolean = true;\n  @Input() mergedPosition: 'left' | 'right' | 'none' = 'none';\n\n  @ViewChild('inputEl', { static: true }) inputEl!: ElementRef<HTMLInputElement>;\n\n  hasPrefix = false;\n  hasSuffix = false;\n  displayValue = '';\n  isFocused: boolean = false;\n\n  private tokens: { [key: string]: RegExp } = {\n    '0': /\\d/,\n    '9': /\\d/,\n    a: /[a-z]/i,\n    A: /[a-z]/i,\n    '*': /[a-z0-9]/i,\n  };\n\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  constructor(\n    private el: ElementRef,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngAfterContentChecked() {\n    const prefix = this.el.nativeElement.querySelector('[prefix]');\n    const suffix = this.el.nativeElement.querySelector('[suffix]');\n\n    if (this.hasPrefix !== !!prefix || this.hasSuffix !== !!suffix) {\n      this.hasPrefix = !!prefix;\n      this.hasSuffix = !!suffix;\n      this.cdr.detectChanges();\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 computedContainerClass() {\n    return cn(\n      // Base styles\n      'group relative flex items-center w-full border transition-all duration-200',\n      'bg-background border-input shadow-sm',\n\n      // Sizing\n      this.size === 'xs' && 'h-8 px-2 gap-1.5 text-xs',\n      this.size === 'sm' && 'h-9 px-3 gap-2 text-sm',\n      this.size === 'default' && 'h-10 px-3 gap-2 text-sm',\n      this.size === 'lg' && 'h-11 px-4 gap-3 text-base',\n\n      // Merged position - handle border radius\n      this.mergedPosition === 'left' && 'rounded-l-md rounded-r-none border-r-0 pr-0',\n      this.mergedPosition === 'right' && 'rounded-r-md rounded-l-none border-l-0 pl-0 flex-1',\n      this.mergedPosition === 'none' && 'rounded-md',\n\n      // Focus state\n      !(this.readonly || 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\n      // Error state\n      this.error && [\n        'border-destructive',\n        !(this.readonly || this.disabled) && 'focus-within:ring-destructive/30',\n      ],\n\n      // Disabled state\n      this.disabled && ['cursor-not-allowed opacity-50', 'border-opacity-50'],\n\n      // Readonly state\n      this.readonly && ['cursor-default', 'border-dashed'],\n\n      this.containerClass\n    );\n  }\n\n  get computedInputClass() {\n    return cn(\n      // Base styles\n      'flex-1 bg-transparent border-none p-0',\n      'placeholder:text-muted-foreground',\n\n      // Remove all default focus styles\n      'focus:outline-none focus:ring-0 focus:shadow-none',\n\n      // Text sizing\n      this.size === 'xs' && 'text-xs',\n      this.size === 'sm' && 'text-sm',\n      this.size === 'default' && 'text-sm',\n      this.size === 'lg' && 'text-base',\n\n      // Cursor states\n      this.disabled && 'cursor-not-allowed',\n      this.readonly && 'cursor-default',\n\n      // Text color\n      'text-foreground',\n\n      // Selection color\n      'selection:bg-primary/20 selection:text-foreground',\n\n      this.class\n    );\n  }\n\n  focusInput(): void {\n    if (!this.disabled && this.inputEl) {\n      this.inputEl.nativeElement.focus();\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  // --- Masking Logic ---\n  onInput(event: Event) {\n    if (this.readonly || this.disabled) return;\n    const input = event.target as HTMLInputElement;\n    const raw = this.unmask(input.value);\n    const masked = this.applyMask(raw);\n    this.displayValue = masked;\n    input.value = masked;\n    this.returnRaw ? this.onChange(raw) : this.onChange(masked);\n  }\n\n  private applyMask(rawValue: string): string {\n    let rawIndex = 0;\n    let formatted = '';\n    for (let i = 0; i < this.mask.length; i++) {\n      if (rawIndex >= rawValue.length) break;\n      const maskChar = this.mask[i];\n      const rawChar = rawValue[rawIndex];\n      if (this.tokens[maskChar]) {\n        if (this.tokens[maskChar].test(rawChar)) {\n          formatted += rawChar;\n          rawIndex++;\n        } else {\n          rawIndex++;\n          i--;\n        }\n      } else {\n        formatted += maskChar;\n        if (rawChar === maskChar) rawIndex++;\n      }\n    }\n    return formatted;\n  }\n\n  private unmask(val: string): string {\n    return val.replace(/[^a-zA-Z0-9]/g, '');\n  }\n\n  writeValue(value: any): void {\n    this.displayValue = value ? this.applyMask(this.unmask(value.toString())) : '';\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  protected cn = cn;\n}\n"]}