@tolle_/tolle-ui 18.2.16 → 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.
@@ -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,19 @@ export class MaskedInputComponent {
22
22
  size = 'default';
23
23
  returnRaw = false;
24
24
  hideHintOnFocus = true;
25
+ externalFocused;
26
+ mergedPosition = 'none';
25
27
  inputEl;
26
28
  hasPrefix = false;
27
29
  hasSuffix = false;
28
30
  displayValue = '';
29
31
  isFocused = false;
30
32
  tokens = {
31
- '0': /\d/, '9': /\d/, 'a': /[a-z]/i, 'A': /[a-z]/i, '*': /[a-z0-9]/i
33
+ '0': /\d/,
34
+ '9': /\d/,
35
+ a: /[a-z]/i,
36
+ A: /[a-z]/i,
37
+ '*': /[a-z0-9]/i,
32
38
  };
33
39
  onChange = () => { };
34
40
  onTouched = () => { };
@@ -46,58 +52,52 @@ export class MaskedInputComponent {
46
52
  }
47
53
  }
48
54
  get computedLabelClass() {
49
- return cn("text-sm font-medium text-foreground leading-none transition-opacity duration-200", this.disabled && "opacity-50");
55
+ return cn('text-sm font-medium text-foreground leading-none transition-opacity duration-200', this.disabled && 'opacity-50');
50
56
  }
51
57
  get computedContainerClass() {
58
+ const focused = this.externalFocused !== undefined ? this.externalFocused : this.isFocused;
52
59
  return cn(
53
60
  // Base styles
54
- "group relative flex items-center w-full rounded-md border transition-all duration-200", "bg-background",
61
+ 'group relative flex items-center w-full border transition-all duration-200', 'bg-background',
55
62
  // Border and shadow
56
- "border-input shadow-sm",
63
+ 'border-input shadow-sm',
57
64
  // 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
60
- !(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",
65
+ 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',
66
+ // Merged position - handle border radius
67
+ 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', this.mergedPosition === 'none' && 'rounded-md',
68
+ // Focus state
69
+ !(this.readonly || this.disabled) &&
70
+ focused && [
71
+ 'ring-4',
72
+ 'ring-ring/30',
73
+ 'ring-offset-0',
74
+ 'shadow-none',
75
+ this.error ? 'border-destructive/80' : 'border-primary/80',
66
76
  ],
67
77
  // Error state
68
78
  this.error && [
69
- "border-destructive",
70
- !(this.readonly || this.disabled) && [
71
- "focus-within:border-destructive/80",
72
- "focus-within:ring-destructive/30"
73
- ]
79
+ 'border-destructive',
80
+ !(this.readonly || this.disabled) && focused && 'ring-destructive/30',
74
81
  ],
75
82
  // Disabled state
76
- this.disabled && [
77
- "cursor-not-allowed opacity-50",
78
- "border-opacity-50"
79
- ],
83
+ this.disabled && ['cursor-not-allowed opacity-50', 'border-opacity-50'],
80
84
  // 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);
85
+ this.readonly && ['cursor-default', 'border-dashed', !this.disabled && !focused && 'ring-0'], this.containerClass);
86
86
  }
87
87
  get computedInputClass() {
88
88
  return cn(
89
89
  // Base styles
90
- "flex-1 bg-transparent border-none p-0", "placeholder:text-muted-foreground",
90
+ 'flex-1 bg-transparent border-none p-0', 'placeholder:text-muted-foreground',
91
91
  // Remove all default focus styles
92
- "focus:outline-none focus:ring-0 focus:shadow-none",
92
+ 'focus:outline-none focus:ring-0 focus:shadow-none',
93
93
  // Text sizing
94
- this.size === 'xs' && "text-xs", this.size === 'sm' && "text-sm", this.size === 'default' && "text-sm", this.size === 'lg' && "text-base",
94
+ this.size === 'xs' && 'text-xs', this.size === 'sm' && 'text-sm', this.size === 'default' && 'text-sm', this.size === 'lg' && 'text-base',
95
95
  // Cursor states
96
- this.disabled && "cursor-not-allowed", this.readonly && "cursor-default",
96
+ this.disabled && 'cursor-not-allowed', this.readonly && 'cursor-default',
97
97
  // Text color
98
- "text-foreground",
98
+ 'text-foreground',
99
99
  // Selection color
100
- "selection:bg-primary/20 selection:text-foreground", this.class);
100
+ 'selection:bg-primary/20 selection:text-foreground', this.class);
101
101
  }
102
102
  focusInput() {
103
103
  if (!this.disabled && this.inputEl) {
@@ -167,28 +167,22 @@ export class MaskedInputComponent {
167
167
  }
168
168
  cn = cn;
169
169
  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: [
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", externalFocused: "externalFocused", mergedPosition: "mergedPosition" }, providers: [
171
171
  {
172
172
  provide: NG_VALUE_ACCESSOR,
173
173
  useExisting: forwardRef(() => MaskedInputComponent),
174
- multi: true
175
- }
174
+ multi: true,
175
+ },
176
176
  ], 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
- >
177
+ <div class="flex w-full flex-col gap-1.5">
178
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
183
179
  {{ label }}
184
180
  </label>
185
181
 
186
- <div
187
- [class]="computedContainerClass"
188
- (click)="focusInput()"
189
- >
182
+ <div [class]="computedContainerClass" (click)="focusInput()">
190
183
  <!-- Prefix Icon -->
191
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
184
+ <div
185
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
192
186
  <ng-content select="[prefix]"></ng-content>
193
187
  </div>
194
188
 
@@ -205,28 +199,23 @@ export class MaskedInputComponent {
205
199
  (focus)="onFocus()"
206
200
  [class]="computedInputClass"
207
201
  [attr.aria-invalid]="error"
208
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
209
- />
202
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null" />
210
203
 
211
204
  <!-- Suffix Icon -->
212
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
205
+ <div
206
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
213
207
  <ng-content select="[suffix]"></ng-content>
214
208
  </div>
215
209
  </div>
216
210
 
217
- <ng-container *ngIf="!disabled">
211
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
218
212
  <p
219
213
  *ngIf="hint && !error"
220
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
221
- [class.opacity-0]="isFocused && hideHintOnFocus"
222
- >
214
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
215
+ [class.opacity-0]="isFocused && hideHintOnFocus">
223
216
  {{ hint }}
224
217
  </p>
225
- <p
226
- *ngIf="error && errorMessage"
227
- [id]="id + '-error'"
228
- class="text-xs text-destructive px-1"
229
- >
218
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
230
219
  {{ errorMessage }}
231
220
  </p>
232
221
  </ng-container>
@@ -243,25 +232,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
243
232
  {
244
233
  provide: NG_VALUE_ACCESSOR,
245
234
  useExisting: forwardRef(() => MaskedInputComponent),
246
- multi: true
247
- }
235
+ multi: true,
236
+ },
248
237
  ],
249
238
  template: `
250
- <div class="flex flex-col gap-1.5 w-full">
251
- <label
252
- *ngIf="label"
253
- [for]="id"
254
- [class]="computedLabelClass"
255
- >
239
+ <div class="flex w-full flex-col gap-1.5">
240
+ <label *ngIf="label && mergedPosition === 'none'" [for]="id" [class]="computedLabelClass">
256
241
  {{ label }}
257
242
  </label>
258
243
 
259
- <div
260
- [class]="computedContainerClass"
261
- (click)="focusInput()"
262
- >
244
+ <div [class]="computedContainerClass" (click)="focusInput()">
263
245
  <!-- Prefix Icon -->
264
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
246
+ <div
247
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
265
248
  <ng-content select="[prefix]"></ng-content>
266
249
  </div>
267
250
 
@@ -278,33 +261,28 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
278
261
  (focus)="onFocus()"
279
262
  [class]="computedInputClass"
280
263
  [attr.aria-invalid]="error"
281
- [attr.aria-describedby]="error && errorMessage ? id + '-error' : null"
282
- />
264
+ [attr.aria-describedby]="error && errorMessage ? id + '-error' : null" />
283
265
 
284
266
  <!-- Suffix Icon -->
285
- <div class="h-full flex items-center text-muted-foreground group-focus-within:text-primary transition-colors duration-200">
267
+ <div
268
+ class="flex h-full items-center text-muted-foreground transition-colors duration-200 group-focus-within:text-primary">
286
269
  <ng-content select="[suffix]"></ng-content>
287
270
  </div>
288
271
  </div>
289
272
 
290
- <ng-container *ngIf="!disabled">
273
+ <ng-container *ngIf="!disabled && mergedPosition === 'none'">
291
274
  <p
292
275
  *ngIf="hint && !error"
293
- class="text-xs text-muted-foreground px-1 transition-opacity duration-200"
294
- [class.opacity-0]="isFocused && hideHintOnFocus"
295
- >
276
+ class="px-1 text-xs text-muted-foreground transition-opacity duration-200"
277
+ [class.opacity-0]="isFocused && hideHintOnFocus">
296
278
  {{ hint }}
297
279
  </p>
298
- <p
299
- *ngIf="error && errorMessage"
300
- [id]="id + '-error'"
301
- class="text-xs text-destructive px-1"
302
- >
280
+ <p *ngIf="error && errorMessage" [id]="id + '-error'" class="px-1 text-xs text-destructive">
303
281
  {{ errorMessage }}
304
282
  </p>
305
283
  </ng-container>
306
284
  </div>
307
- `
285
+ `,
308
286
  }]
309
287
  }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }], propDecorators: { id: [{
310
288
  type: Input
@@ -336,8 +314,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
336
314
  type: Input
337
315
  }], hideHintOnFocus: [{
338
316
  type: Input
317
+ }], externalFocused: [{
318
+ type: Input
319
+ }], mergedPosition: [{
320
+ type: Input
339
321
  }], inputEl: [{
340
322
  type: ViewChild,
341
323
  args: ['inputEl', { static: true }]
342
324
  }] } });
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"]}
325
+ //# 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() externalFocused: boolean | undefined;\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    const focused = this.externalFocused !== undefined ? this.externalFocused : this.isFocused;\n    return cn(\n      // Base styles\n      'group relative flex items-center w-full 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      // 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',\n      this.mergedPosition === 'none' && 'rounded-md',\n\n      // Focus state\n      !(this.readonly || this.disabled) &&\n        focused && [\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\n      // Error state\n      this.error && [\n        'border-destructive',\n        !(this.readonly || this.disabled) && focused && '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', !this.disabled && !focused && 'ring-0'],\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"]}