fn-input 0.0.11 → 0.0.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/fesm2022/fn-input.mjs +14 -158
- package/fesm2022/fn-input.mjs.map +1 -1
- package/lib/fn-input.component.d.ts +2 -8
- package/lib/fn-input.component.d.ts.map +1 -1
- package/package.json +2 -4
- package/esm2022/fn-input.mjs +0 -5
- package/esm2022/lib/fn-input.component.mjs +0 -1192
- package/esm2022/lib/fn-input.types.mjs +0 -12
- package/esm2022/public-api.mjs +0 -3
|
@@ -1,1192 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, EventEmitter, inject, Input, Output, signal, ViewChild, ViewChildren, } from '@angular/core';
|
|
2
|
-
import { CommonModule } from '@angular/common';
|
|
3
|
-
// Removed HttpClient import
|
|
4
|
-
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
5
|
-
import { TranslateModule, TranslateService } from '@ngx-translate/core';
|
|
6
|
-
import { Subscription, merge } from 'rxjs';
|
|
7
|
-
import { startWith } from 'rxjs/operators';
|
|
8
|
-
import { FNLabel } from 'fn-label';
|
|
9
|
-
import { DEFAULT_CURRENCY_META, FN_TOAST_SERVICE, } from './fn-input.types';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "@angular/forms";
|
|
12
|
-
import * as i2 from "@angular/common";
|
|
13
|
-
import * as i3 from "@ngx-translate/core";
|
|
14
|
-
const CORE_ICONS = {
|
|
15
|
-
'eye-open': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M2 12s4-8 10-8 10 8 10 8-4 8-10 8-10-8-10-8z" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><circle cx="12" cy="12" r="3" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
16
|
-
'eye-close': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M2 10a11 11 0 0 0 20 0" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><path d="M5 13l-1 2M12 14v2M19 13l1 2" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
17
|
-
'two-square': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><rect x="8" y="4" width="12" height="12" rx="2" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><path d="M4 8v10a2 2 0 0 0 2 2h10" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
18
|
-
'round-arrow-top-left': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M23 4v6h-6M1 20v-6h6M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
19
|
-
'check-circle': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2"/><path d="M8 12l3 3 5-5" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
20
|
-
'info-circle': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="16" x2="12" y2="12" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="8" x2="12.01" y2="8" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
21
|
-
'alert-circle': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="8" x2="12" y2="12" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="16" x2="12.01" y2="16" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
22
|
-
'exclamation-circle': `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="8" x2="12" y2="12" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><line x1="12" y1="16" x2="12.01" y2="16" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
23
|
-
cross: `<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M18 6L6 18M6 6l12 12" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>`,
|
|
24
|
-
};
|
|
25
|
-
export class FNInput {
|
|
26
|
-
cdr;
|
|
27
|
-
field = {};
|
|
28
|
-
helperHandle;
|
|
29
|
-
toastService;
|
|
30
|
-
injectedToastService = inject(FN_TOAST_SERVICE, { optional: true });
|
|
31
|
-
get effectiveToastService() {
|
|
32
|
-
return this.toastService || this.injectedToastService || undefined;
|
|
33
|
-
}
|
|
34
|
-
currencyMeta = DEFAULT_CURRENCY_META;
|
|
35
|
-
defaultLocale = 'en-US';
|
|
36
|
-
form;
|
|
37
|
-
valueChange = new EventEmitter();
|
|
38
|
-
fieldBlur = new EventEmitter();
|
|
39
|
-
translateService = inject(TranslateService);
|
|
40
|
-
hasFocus = signal(false);
|
|
41
|
-
textareaElement;
|
|
42
|
-
iconContainers;
|
|
43
|
-
// Internal Toast State
|
|
44
|
-
toasts = signal([]);
|
|
45
|
-
// Removed HttpClient injection
|
|
46
|
-
subs = new Subscription();
|
|
47
|
-
sizeMap = {
|
|
48
|
-
extrasmall: 10,
|
|
49
|
-
small: 16,
|
|
50
|
-
medium: 20,
|
|
51
|
-
large: 24,
|
|
52
|
-
'x-large': 32,
|
|
53
|
-
xxlarge: 48,
|
|
54
|
-
};
|
|
55
|
-
getIconSizeName(size) {
|
|
56
|
-
switch (size) {
|
|
57
|
-
case 16:
|
|
58
|
-
return 'small';
|
|
59
|
-
case 20:
|
|
60
|
-
return 'medium';
|
|
61
|
-
case 24:
|
|
62
|
-
return 'large';
|
|
63
|
-
case 32:
|
|
64
|
-
return 'x-large';
|
|
65
|
-
default:
|
|
66
|
-
return 'medium';
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
get isAlphanumeric() {
|
|
70
|
-
return this.field.isAlphanumeric !== false;
|
|
71
|
-
}
|
|
72
|
-
get isEmailField() {
|
|
73
|
-
return this.field.type === 'email' || this.field.name?.toLowerCase().includes('email');
|
|
74
|
-
}
|
|
75
|
-
get alphanumericPattern() {
|
|
76
|
-
if (this.field.isAddressLine) {
|
|
77
|
-
return /[^A-Za-z0-9 \-_&(),/]/g;
|
|
78
|
-
}
|
|
79
|
-
return /[^A-Za-z0-9 \-_&()]/g;
|
|
80
|
-
}
|
|
81
|
-
isDisabled = false;
|
|
82
|
-
control;
|
|
83
|
-
isVisible = signal(true); // Track visibility state
|
|
84
|
-
helperText = '';
|
|
85
|
-
// Use a counter-based approach for unique IDs (safer than Math.random())
|
|
86
|
-
static idCounter = 0;
|
|
87
|
-
uniqueId = `fn-input-${++FNInput.idCounter}`;
|
|
88
|
-
constructor(cdr) {
|
|
89
|
-
this.cdr = cdr;
|
|
90
|
-
}
|
|
91
|
-
ngOnInit() {
|
|
92
|
-
if (this.field.hidden)
|
|
93
|
-
return;
|
|
94
|
-
this.helperText = this.field.helperText ?? '';
|
|
95
|
-
this.initFormControl();
|
|
96
|
-
if (this.field.value)
|
|
97
|
-
this.control.setValue(this.field.value);
|
|
98
|
-
this.setupVisibilityCondition();
|
|
99
|
-
this.setupFieldMessageListener();
|
|
100
|
-
this.setupToastSubscription();
|
|
101
|
-
}
|
|
102
|
-
ngOnDestroy() {
|
|
103
|
-
this.subs.unsubscribe();
|
|
104
|
-
}
|
|
105
|
-
ngOnChanges(changes) {
|
|
106
|
-
if (changes['field'] || changes['form']) {
|
|
107
|
-
this.updateAllIcons();
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
setupFieldMessageListener() {
|
|
111
|
-
if (!this.control)
|
|
112
|
-
return;
|
|
113
|
-
const val$ = this.control.valueChanges.pipe(startWith(this.control.value));
|
|
114
|
-
const status$ = this.control.statusChanges.pipe(startWith(this.control.status));
|
|
115
|
-
this.subs.add(merge(val$, status$).subscribe(() => {
|
|
116
|
-
try {
|
|
117
|
-
this.cdr.detectChanges();
|
|
118
|
-
}
|
|
119
|
-
catch (e) {
|
|
120
|
-
console.warn('CDR detectChanges failed during field message update:', e);
|
|
121
|
-
this.cdr.markForCheck();
|
|
122
|
-
}
|
|
123
|
-
}));
|
|
124
|
-
}
|
|
125
|
-
ngAfterViewInit() {
|
|
126
|
-
// Access the textarea element by ViewChild or native element reference
|
|
127
|
-
this.autoResizeInitial();
|
|
128
|
-
if (this.field.type === 'textarea' && this.control?.value) {
|
|
129
|
-
setTimeout(() => this.autoResizeInitial(), 30);
|
|
130
|
-
}
|
|
131
|
-
// Apply initial formatting after view is initialized
|
|
132
|
-
if (this.field.type === 'number' && this.field.isCurrency && this.control?.value) {
|
|
133
|
-
this.formatInitialCurrencyValue();
|
|
134
|
-
}
|
|
135
|
-
this.setupIconContainersListener();
|
|
136
|
-
}
|
|
137
|
-
setupIconContainersListener() {
|
|
138
|
-
this.subs.add(this.iconContainers.changes.subscribe(() => {
|
|
139
|
-
this.updateAllIcons();
|
|
140
|
-
}));
|
|
141
|
-
}
|
|
142
|
-
enforceLowercase(event) {
|
|
143
|
-
const input = event.target;
|
|
144
|
-
const start = input.selectionStart;
|
|
145
|
-
const end = input.selectionEnd;
|
|
146
|
-
input.value = input.value.toLowerCase();
|
|
147
|
-
if (this.control) {
|
|
148
|
-
this.control.setValue(input.value, { emitEvent: false });
|
|
149
|
-
}
|
|
150
|
-
this.valueChange.emit({ name: this.field.name, value: input.value });
|
|
151
|
-
// restore cursor position so typing is smooth
|
|
152
|
-
input.setSelectionRange(start, end);
|
|
153
|
-
}
|
|
154
|
-
/**
|
|
155
|
-
* Handle email input - allows only valid email characters and enforces lowercase.
|
|
156
|
-
* Allowed: a-z, 0-9, . _ + - @
|
|
157
|
-
* Rules:
|
|
158
|
-
* - Only one @ allowed
|
|
159
|
-
* - No consecutive dots (..)
|
|
160
|
-
* - Automatic lowercase
|
|
161
|
-
*/
|
|
162
|
-
handleEmailInput(event) {
|
|
163
|
-
const input = event.target;
|
|
164
|
-
const cursorPosition = input.selectionStart || 0;
|
|
165
|
-
const originalValue = input.value;
|
|
166
|
-
// 1. Convert to lowercase
|
|
167
|
-
const lowerValue = originalValue.toLowerCase();
|
|
168
|
-
// 2. Split at the first @ to handle local and domain parts separately
|
|
169
|
-
const parts = lowerValue.split('@');
|
|
170
|
-
let localPart = parts[0];
|
|
171
|
-
let domainPart = parts.length > 1 ? parts.slice(1).join('') : null;
|
|
172
|
-
// 3. Filter local part: a-z, 0-9, ., _, +, -
|
|
173
|
-
// Rule: Cannot start with a dot (.)
|
|
174
|
-
localPart = localPart.replaceAll(/[^a-z0-9._+-]/g, '');
|
|
175
|
-
while (localPart.startsWith('.')) {
|
|
176
|
-
localPart = localPart.substring(1);
|
|
177
|
-
}
|
|
178
|
-
while (localPart.includes('..')) {
|
|
179
|
-
localPart = localPart.replaceAll('..', '.');
|
|
180
|
-
}
|
|
181
|
-
// 4. Filter domain part: a-z, 0-9, ., -
|
|
182
|
-
// Rule: No underscores (_), no plus signs (+), cannot start with a hyphen (-)
|
|
183
|
-
let filteredValue = localPart;
|
|
184
|
-
if (domainPart !== null) {
|
|
185
|
-
domainPart = domainPart.replaceAll(/[^a-z0-9.-]/g, '');
|
|
186
|
-
while (domainPart.startsWith('-')) {
|
|
187
|
-
domainPart = domainPart.substring(1);
|
|
188
|
-
}
|
|
189
|
-
while (domainPart.includes('..')) {
|
|
190
|
-
domainPart = domainPart.replaceAll('..', '.');
|
|
191
|
-
}
|
|
192
|
-
filteredValue += '@' + domainPart;
|
|
193
|
-
}
|
|
194
|
-
if (originalValue !== filteredValue) {
|
|
195
|
-
const charsRemovedBeforeCursor = this.countRemovedCharsBeforeCursor(originalValue.toLowerCase(), filteredValue, cursorPosition);
|
|
196
|
-
input.value = filteredValue;
|
|
197
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
198
|
-
const newCursorPos = Math.max(0, cursorPosition - charsRemovedBeforeCursor);
|
|
199
|
-
input.setSelectionRange(newCursorPos, newCursorPos);
|
|
200
|
-
}
|
|
201
|
-
else if (originalValue !== lowerValue) {
|
|
202
|
-
// If only case changed (e.g., typed Uppercase)
|
|
203
|
-
input.value = filteredValue;
|
|
204
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
205
|
-
input.setSelectionRange(cursorPosition, cursorPosition);
|
|
206
|
-
}
|
|
207
|
-
if (this.field.type === 'password' && this.field.feedback) {
|
|
208
|
-
this.checkPasswordStrength(filteredValue);
|
|
209
|
-
}
|
|
210
|
-
this.valueChange.emit({ name: this.field.name, value: filteredValue });
|
|
211
|
-
this.control.markAsTouched();
|
|
212
|
-
}
|
|
213
|
-
onFocus() {
|
|
214
|
-
this.hasFocus.set(true);
|
|
215
|
-
}
|
|
216
|
-
formatInitialCurrencyValue() {
|
|
217
|
-
const numericValue = Number(this.control.value);
|
|
218
|
-
if (!Number.isNaN(numericValue)) {
|
|
219
|
-
setTimeout(() => {
|
|
220
|
-
const inputElement = document.getElementById(this.uniqueId);
|
|
221
|
-
if (inputElement) {
|
|
222
|
-
inputElement.value = this.formatCurrencyWithCommas(numericValue);
|
|
223
|
-
}
|
|
224
|
-
}, 50);
|
|
225
|
-
}
|
|
226
|
-
}
|
|
227
|
-
initFormControl() {
|
|
228
|
-
const controller = this.form.controls?.[this.field.name];
|
|
229
|
-
this.control = controller;
|
|
230
|
-
if (this.field.value)
|
|
231
|
-
this.control.setValue(this.field.value);
|
|
232
|
-
if (this.control) {
|
|
233
|
-
// Handle disabled state
|
|
234
|
-
if (this.field.disabled ?? false) {
|
|
235
|
-
this.control?.disable();
|
|
236
|
-
}
|
|
237
|
-
else {
|
|
238
|
-
this.control?.enable();
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
}
|
|
242
|
-
autoResizeInitial() {
|
|
243
|
-
const textarea = this.textareaElement?.nativeElement;
|
|
244
|
-
if (textarea) {
|
|
245
|
-
// Get the number of rows (default to 2 if not specified)
|
|
246
|
-
const rows = this.field.rows || 1;
|
|
247
|
-
// Calculate line height from computed styles
|
|
248
|
-
const computedStyle = globalThis.getComputedStyle(textarea);
|
|
249
|
-
const lineHeight = Number.parseFloat(computedStyle.lineHeight) || 32; // Default to 24px if not set
|
|
250
|
-
// Calculate minimum height based on rows
|
|
251
|
-
const minHeight = rows * lineHeight;
|
|
252
|
-
// Reset height to auto to get accurate scrollHeight
|
|
253
|
-
textarea.style.height = 'auto';
|
|
254
|
-
// Use the larger of scrollHeight or minHeight
|
|
255
|
-
const newHeight = Math.max(textarea.scrollHeight, minHeight);
|
|
256
|
-
textarea.style.height = newHeight + 'px';
|
|
257
|
-
textarea.style.overflow = 'hidden';
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
handleTextArea(event) {
|
|
261
|
-
const element = event.target;
|
|
262
|
-
const cursorPosition = element.selectionStart;
|
|
263
|
-
const originalValue = element.value;
|
|
264
|
-
// Apply alphanumeric filtering if enabled
|
|
265
|
-
if (this.field.type === 'textarea' && this.isAlphanumeric) {
|
|
266
|
-
// Only allow characters based on the selected pattern
|
|
267
|
-
const alphanumericPattern = this.alphanumericPattern;
|
|
268
|
-
let filteredValue = originalValue.replaceAll(alphanumericPattern, '');
|
|
269
|
-
// Prevent leading spaces
|
|
270
|
-
if (filteredValue.startsWith(' ')) {
|
|
271
|
-
filteredValue = filteredValue.trimStart();
|
|
272
|
-
}
|
|
273
|
-
// Replace multiple consecutive spaces with single space
|
|
274
|
-
filteredValue = filteredValue.replaceAll(/\s{2,}/g, ' ');
|
|
275
|
-
if (originalValue !== filteredValue) {
|
|
276
|
-
// Calculate the number of characters removed before the cursor position
|
|
277
|
-
const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;
|
|
278
|
-
element.value = filteredValue;
|
|
279
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
280
|
-
// Restore cursor position - account for removed characters
|
|
281
|
-
const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);
|
|
282
|
-
element.setSelectionRange(newCursorPos, newCursorPos);
|
|
283
|
-
}
|
|
284
|
-
// increase height
|
|
285
|
-
this.autoResizeInitial();
|
|
286
|
-
this.valueChange.emit({ name: this.field.name, value: filteredValue });
|
|
287
|
-
}
|
|
288
|
-
else {
|
|
289
|
-
let filteredValue = originalValue;
|
|
290
|
-
// Strict filtering if !isAlphanumeric
|
|
291
|
-
if (!this.isAlphanumeric) {
|
|
292
|
-
console.log('isAlphanumeric', this.isAlphanumeric);
|
|
293
|
-
const allowedPattern = /[^a-zA-Z0-9\s(){}[\];,."='+\-*/<>!&|%_@$?:]/g;
|
|
294
|
-
filteredValue = filteredValue.replaceAll(allowedPattern, '');
|
|
295
|
-
}
|
|
296
|
-
// Prevent leading spaces
|
|
297
|
-
if (filteredValue.startsWith(' ')) {
|
|
298
|
-
filteredValue = filteredValue.trimStart();
|
|
299
|
-
}
|
|
300
|
-
// Replace multiple consecutive spaces with single space
|
|
301
|
-
filteredValue = filteredValue.replaceAll(/\s{2,}/g, ' ');
|
|
302
|
-
if (originalValue !== filteredValue) {
|
|
303
|
-
// Calculate the number of characters removed before the cursor position
|
|
304
|
-
const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;
|
|
305
|
-
element.value = filteredValue;
|
|
306
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
307
|
-
// Restore cursor position - account for removed characters
|
|
308
|
-
const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);
|
|
309
|
-
element.setSelectionRange(newCursorPos, newCursorPos);
|
|
310
|
-
}
|
|
311
|
-
// increase height
|
|
312
|
-
this.autoResizeInitial();
|
|
313
|
-
this.valueChange.emit({ name: this.field.name, value: filteredValue });
|
|
314
|
-
}
|
|
315
|
-
this.control.markAsTouched();
|
|
316
|
-
}
|
|
317
|
-
togglePasswordVisibility() {
|
|
318
|
-
this.showPassword = !this.showPassword;
|
|
319
|
-
this.updateAllIcons();
|
|
320
|
-
}
|
|
321
|
-
handleInput(event) {
|
|
322
|
-
const target = event.target;
|
|
323
|
-
const cursorPosition = target.selectionStart;
|
|
324
|
-
const originalValue = target.value;
|
|
325
|
-
let filteredValue = originalValue;
|
|
326
|
-
// IF NOT alphanumeric, user wants strict filtering:
|
|
327
|
-
// "only accept tihs.) {} [] ; , . " ' = + - * / < > ! & | % &_( )-/.;@!, $% : dont allow emoji"
|
|
328
|
-
// Interpretation: Allow alphanumeric + specified special chars. Ban emojis.
|
|
329
|
-
// Actually, the user said: "if isAlphanumeric false only accept..."
|
|
330
|
-
// So if isAlphanumeric is TRUE, we use the existing handleAlphanumericInput logic (or similar).
|
|
331
|
-
// The current logic in handleInput is generic.
|
|
332
|
-
// The existing code has a separate `handleAlphanumericInput` method but it's not called here?
|
|
333
|
-
// Wait, the template likely calls `handleAlphanumericInput` if `isAlphanumeric` is true, or `handleInput` otherwise?
|
|
334
|
-
// Let's check the template.
|
|
335
|
-
// BUT assuming `handleInput` is the generic handler:
|
|
336
|
-
if (!this.isAlphanumeric) {
|
|
337
|
-
// Regex to allow: a-z, A-Z, 0-9, and: ) {} [] ; , . " ' = + - * / < > ! & | % _ ( - @ $ ? :
|
|
338
|
-
// Note: User list: ) {} [] ; , . " ' = + - * / < > ! & | % &_( )-/.;@!, $%
|
|
339
|
-
// Combined unique special chars: ) { } [ ] ; , . " ' = + - * / < > ! & | % _ ( @ $ ? :
|
|
340
|
-
// Also space is implied? usually yes.
|
|
341
|
-
// And NO EMOJIS.
|
|
342
|
-
// easier to just replace anything NOT in the allowed set.
|
|
343
|
-
// Allowed chars regex pattern:
|
|
344
|
-
// A-Za-z0-9
|
|
345
|
-
// Space: \s
|
|
346
|
-
// Special: (){}[\];,."='+\-*/<>!&|%_@$?:
|
|
347
|
-
// Note: - inside [] needs to be escaped or at end. ] needs escape. \ needs escape? (not in list).
|
|
348
|
-
// User list has: `.` `"` `'` `?` (wait, ? not in list?)
|
|
349
|
-
// User list: `) {} [] ; , . " ' = + - * / < > ! & | % &_( )-/.;@!, $%`
|
|
350
|
-
// Uniques: ) { } [ ] ; , . " ' = + - * / < > ! & | % _ ( @ $
|
|
351
|
-
// (The `&` is repeated, `.` repeated, `,` repeated etc.)
|
|
352
|
-
// I will create a regex that matches anything NOT in this list and replace with empty string.
|
|
353
|
-
// [^a-zA-Z0-9\s(){}[\];,."='+\-*/<>!&|%_@$]
|
|
354
|
-
// Note: `*`, `+`, `?` etc need escaping in regex if outside [], but inside [] they are literals mostly.
|
|
355
|
-
// `-` needs to be last or escaped. `]` needs escape. `^` needs escape if first? No, invalid here.
|
|
356
|
-
// Let's verify valid chars:
|
|
357
|
-
// a-z A-Z 0-9
|
|
358
|
-
// space
|
|
359
|
-
// ) { } [ ] ; , . " ' = + - * / < > ! & | % _ ( @ $ :
|
|
360
|
-
const allowedPattern = /[^a-zA-Z0-9\s(){}[\];,."='+\-*/<>!&|%_@$?:]/g;
|
|
361
|
-
filteredValue = filteredValue.replaceAll(allowedPattern, '');
|
|
362
|
-
}
|
|
363
|
-
// Prevent leading spaces
|
|
364
|
-
if (filteredValue.startsWith(' ')) {
|
|
365
|
-
filteredValue = filteredValue.trimStart();
|
|
366
|
-
}
|
|
367
|
-
// replaceAll multiple consecutive spaces with single space
|
|
368
|
-
filteredValue = filteredValue.replaceAll(/\s{2,}/g, ' ');
|
|
369
|
-
if (originalValue !== filteredValue) {
|
|
370
|
-
// Calculate the number of characters removed before the cursor position
|
|
371
|
-
const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;
|
|
372
|
-
target.value = filteredValue;
|
|
373
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
374
|
-
// Restore cursor position - account for removed characters
|
|
375
|
-
const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);
|
|
376
|
-
target.setSelectionRange(newCursorPos, newCursorPos);
|
|
377
|
-
}
|
|
378
|
-
// Trailing spaces will be trimmed on blur, not during input
|
|
379
|
-
if (this.field.type === 'password' && this.field.feedback) {
|
|
380
|
-
this.checkPasswordStrength(filteredValue);
|
|
381
|
-
}
|
|
382
|
-
this.valueChange.emit({ name: this.field.name, value: filteredValue });
|
|
383
|
-
this.control.markAsTouched();
|
|
384
|
-
}
|
|
385
|
-
/**
|
|
386
|
-
* Handle alphanumeric input - allows only letters, numbers, and specific special characters
|
|
387
|
-
* Allowed: A-Z, a-z, 0-9, space, hyphen, underscore, ampersand, and parentheses
|
|
388
|
-
* Also prevents leading spaces and multiple consecutive spaces
|
|
389
|
-
*/
|
|
390
|
-
handleAlphanumericInput(event) {
|
|
391
|
-
const target = event.target;
|
|
392
|
-
const cursorPosition = target.selectionStart;
|
|
393
|
-
const originalValue = target.value;
|
|
394
|
-
// Only allow characters based on the selected pattern
|
|
395
|
-
const alphanumericPattern = this.alphanumericPattern;
|
|
396
|
-
let filteredValue = originalValue.replaceAll(alphanumericPattern, '');
|
|
397
|
-
// Prevent leading spaces
|
|
398
|
-
if (filteredValue.startsWith(' ')) {
|
|
399
|
-
filteredValue = filteredValue.trimStart();
|
|
400
|
-
}
|
|
401
|
-
// Replace multiple consecutive spaces with single space
|
|
402
|
-
filteredValue = filteredValue.replaceAll(/\s{2,}/g, ' ');
|
|
403
|
-
if (originalValue !== filteredValue) {
|
|
404
|
-
// Calculate the number of characters removed before the cursor position
|
|
405
|
-
const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;
|
|
406
|
-
target.value = filteredValue;
|
|
407
|
-
this.control.setValue(filteredValue, { emitEvent: false });
|
|
408
|
-
// Restore cursor position - account for removed characters
|
|
409
|
-
const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);
|
|
410
|
-
target.setSelectionRange(newCursorPos, newCursorPos);
|
|
411
|
-
}
|
|
412
|
-
if (this.field.type === 'password' && this.field.feedback) {
|
|
413
|
-
this.checkPasswordStrength(filteredValue);
|
|
414
|
-
}
|
|
415
|
-
this.valueChange.emit({ name: this.field.name, value: filteredValue });
|
|
416
|
-
this.control.markAsTouched();
|
|
417
|
-
}
|
|
418
|
-
handleBlur(e, maxDecimals = 2) {
|
|
419
|
-
this.hasFocus.set(false);
|
|
420
|
-
const target = e.target;
|
|
421
|
-
const cleanValueString = target.value;
|
|
422
|
-
if (!cleanValueString) {
|
|
423
|
-
this.fieldBlur.emit({ name: this.field.name, value: '' });
|
|
424
|
-
this.cdr.detectChanges();
|
|
425
|
-
return;
|
|
426
|
-
}
|
|
427
|
-
// Delegate to type-specific handlers
|
|
428
|
-
if (this.field.type === 'text' || this.field.type === 'textarea') {
|
|
429
|
-
this.handleTextFieldBlur(target, cleanValueString);
|
|
430
|
-
}
|
|
431
|
-
else if (this.field.type === 'number') {
|
|
432
|
-
this.handleNumberFieldBlur(target, cleanValueString, maxDecimals);
|
|
433
|
-
}
|
|
434
|
-
// Emit final value
|
|
435
|
-
this.fieldBlur.emit({
|
|
436
|
-
name: this.field.name,
|
|
437
|
-
value: target.value,
|
|
438
|
-
});
|
|
439
|
-
this.cdr.detectChanges();
|
|
440
|
-
}
|
|
441
|
-
/**
|
|
442
|
-
* Handle blur for text and textarea fields - trim trailing spaces
|
|
443
|
-
*/
|
|
444
|
-
handleTextFieldBlur(target, cleanValueString) {
|
|
445
|
-
const trimmed = cleanValueString.trim();
|
|
446
|
-
if (cleanValueString !== trimmed) {
|
|
447
|
-
target.value = trimmed;
|
|
448
|
-
this.control.setValue(trimmed, { emitEvent: false });
|
|
449
|
-
}
|
|
450
|
-
}
|
|
451
|
-
/**
|
|
452
|
-
* Handle blur for number fields - validate and format
|
|
453
|
-
*/
|
|
454
|
-
handleNumberFieldBlur(target, cleanValueString, maxDecimals) {
|
|
455
|
-
// Remove commas for parsing/validation
|
|
456
|
-
const numString = cleanValueString.replaceAll(',', '');
|
|
457
|
-
// Check if the string is a valid number format
|
|
458
|
-
const isValidNumberFormat = /^-?\d+(\.\d+)?$/.test(numString);
|
|
459
|
-
// If not a valid number format, clear the field
|
|
460
|
-
if (!isValidNumberFormat) {
|
|
461
|
-
target.value = '';
|
|
462
|
-
this.control.setValue('');
|
|
463
|
-
return;
|
|
464
|
-
}
|
|
465
|
-
// Format currency fields or set raw value for non-currency
|
|
466
|
-
if (this.field.type === 'number' && this.field.isCurrency) {
|
|
467
|
-
this.processCurrencyValue(target, numString, maxDecimals);
|
|
468
|
-
}
|
|
469
|
-
else {
|
|
470
|
-
// Non-currency: preserve the raw string value
|
|
471
|
-
target.value = numString;
|
|
472
|
-
}
|
|
473
|
-
}
|
|
474
|
-
/**
|
|
475
|
-
* Process currency value - handle formatting and decimal places
|
|
476
|
-
*/
|
|
477
|
-
processCurrencyValue(target, numString, maxDecimals) {
|
|
478
|
-
// Handle negative values - convert to positive
|
|
479
|
-
let processedValue = numString.startsWith('-') ? numString.substring(1) : numString;
|
|
480
|
-
// Handle edge case of "-0" or just "-"
|
|
481
|
-
if (processedValue === '' || processedValue === '0') {
|
|
482
|
-
processedValue = '0';
|
|
483
|
-
}
|
|
484
|
-
// Format the value with proper decimal places
|
|
485
|
-
const cleanValueString = this.formatDecimalPlaces(processedValue, maxDecimals);
|
|
486
|
-
const formattedValue = this.formatCurrencyWithCommas(cleanValueString, maxDecimals);
|
|
487
|
-
// Update the control value with the numeric string (not formatted)
|
|
488
|
-
this.control.setValue(cleanValueString, { emitEvent: false });
|
|
489
|
-
// Trigger validators and mark as touched
|
|
490
|
-
this.control.updateValueAndValidity();
|
|
491
|
-
this.control.markAsTouched();
|
|
492
|
-
// Set display value with commas after Angular's form control update
|
|
493
|
-
setTimeout(() => {
|
|
494
|
-
target.value = formattedValue;
|
|
495
|
-
}, 0);
|
|
496
|
-
}
|
|
497
|
-
/**
|
|
498
|
-
* Format decimal places - pad or truncate to maxDecimals
|
|
499
|
-
*/
|
|
500
|
-
formatDecimalPlaces(value, maxDecimals) {
|
|
501
|
-
let [intPart, decPart] = value.includes('.') ? value.split('.') : [value, ''];
|
|
502
|
-
// Pad or truncate decimal part to maxDecimals
|
|
503
|
-
if (decPart.length < maxDecimals) {
|
|
504
|
-
decPart = decPart.padEnd(maxDecimals, '0');
|
|
505
|
-
}
|
|
506
|
-
else if (decPart.length > maxDecimals) {
|
|
507
|
-
decPart = decPart.slice(0, maxDecimals);
|
|
508
|
-
}
|
|
509
|
-
// Build the clean value string (without commas)
|
|
510
|
-
return maxDecimals > 0 ? `${intPart}.${decPart}` : intPart;
|
|
511
|
-
}
|
|
512
|
-
handleNumberInput(event, maxDecimals = 2) {
|
|
513
|
-
const target = event.target;
|
|
514
|
-
const originalValue = target.value;
|
|
515
|
-
const cursorPosition = target.selectionStart ?? 0;
|
|
516
|
-
// Step 1: Filter and format the input value
|
|
517
|
-
let value = this.filterNumericInput(target.value, maxDecimals);
|
|
518
|
-
// Step 2: Update cursor position if value changed
|
|
519
|
-
this.updateCursorPositionAfterFilter(target, originalValue, value, cursorPosition);
|
|
520
|
-
// Step 3: Update form control and validate
|
|
521
|
-
this.updateFormControlValue(value, maxDecimals);
|
|
522
|
-
// Step 4: Emit changes
|
|
523
|
-
this.valueChange.emit({ name: this.field.name, value: target.value });
|
|
524
|
-
this.control.markAsTouched();
|
|
525
|
-
}
|
|
526
|
-
/**
|
|
527
|
-
* Filter numeric input to only allow digits and decimal point
|
|
528
|
-
* Ensures only one decimal point and enforces max decimal places
|
|
529
|
-
*/
|
|
530
|
-
filterNumericInput(value, maxDecimals) {
|
|
531
|
-
// Remove ALL non-numeric characters except decimal point
|
|
532
|
-
let filtered = value.replaceAll(/[^0-9.]/g, '');
|
|
533
|
-
// Ensure only one decimal point
|
|
534
|
-
filtered = this.filterDecimalPoints(filtered);
|
|
535
|
-
// Restrict to maxDecimals decimal places if decimal exists
|
|
536
|
-
filtered = this.enforceDecimalLimit(filtered, maxDecimals);
|
|
537
|
-
return filtered;
|
|
538
|
-
}
|
|
539
|
-
/**
|
|
540
|
-
* Ensure only one decimal point exists in the value
|
|
541
|
-
*/
|
|
542
|
-
filterDecimalPoints(value) {
|
|
543
|
-
const parts = value.split('.');
|
|
544
|
-
if (parts.length > 2) {
|
|
545
|
-
return parts[0] + '.' + parts.slice(1).join('');
|
|
546
|
-
}
|
|
547
|
-
return value;
|
|
548
|
-
}
|
|
549
|
-
/**
|
|
550
|
-
* Enforce maximum decimal places limit
|
|
551
|
-
*/
|
|
552
|
-
enforceDecimalLimit(value, maxDecimals) {
|
|
553
|
-
if (!value.includes('.')) {
|
|
554
|
-
return value;
|
|
555
|
-
}
|
|
556
|
-
const [integerPart, decimalPart] = value.split('.');
|
|
557
|
-
if (decimalPart && decimalPart.length > maxDecimals) {
|
|
558
|
-
return integerPart + '.' + decimalPart.slice(0, maxDecimals);
|
|
559
|
-
}
|
|
560
|
-
return value;
|
|
561
|
-
}
|
|
562
|
-
/**
|
|
563
|
-
* Update cursor position after filtering input
|
|
564
|
-
*/
|
|
565
|
-
updateCursorPositionAfterFilter(target, originalValue, newValue, cursorPosition) {
|
|
566
|
-
if (target.value === newValue) {
|
|
567
|
-
return;
|
|
568
|
-
}
|
|
569
|
-
const charsRemovedBeforeCursor = this.countRemovedCharsBeforeCursor(originalValue, newValue, cursorPosition);
|
|
570
|
-
target.value = newValue;
|
|
571
|
-
const newPosition = Math.max(0, cursorPosition - charsRemovedBeforeCursor);
|
|
572
|
-
target.setSelectionRange(newPosition, newPosition);
|
|
573
|
-
}
|
|
574
|
-
/**
|
|
575
|
-
* Update form control value and perform validation
|
|
576
|
-
*/
|
|
577
|
-
updateFormControlValue(value, maxDecimals) {
|
|
578
|
-
this.control.setValue(value, { emitEvent: false });
|
|
579
|
-
this.validateIntegerDigits(value, maxDecimals);
|
|
580
|
-
this.clearNumericError();
|
|
581
|
-
}
|
|
582
|
-
/**
|
|
583
|
-
* Count how many characters were removed before the cursor position
|
|
584
|
-
* This helps maintain correct cursor position after filtering input
|
|
585
|
-
*/
|
|
586
|
-
countRemovedCharsBeforeCursor(original, filtered, cursorPos) {
|
|
587
|
-
let removedCount = 0;
|
|
588
|
-
let filteredIndex = 0;
|
|
589
|
-
for (let i = 0; i < cursorPos && i < original.length; i++) {
|
|
590
|
-
if (filteredIndex < filtered.length && original[i] === filtered[filteredIndex]) {
|
|
591
|
-
filteredIndex++;
|
|
592
|
-
}
|
|
593
|
-
else {
|
|
594
|
-
removedCount++;
|
|
595
|
-
}
|
|
596
|
-
}
|
|
597
|
-
return removedCount;
|
|
598
|
-
}
|
|
599
|
-
/**
|
|
600
|
-
* Handle paste event for number inputs
|
|
601
|
-
* Prevents pasting of non-numeric characters and enforces digit limits
|
|
602
|
-
*/
|
|
603
|
-
handleNumberPaste(event, maxDecimals = 2) {
|
|
604
|
-
event.preventDefault();
|
|
605
|
-
const target = event.target;
|
|
606
|
-
const pastedText = event.clipboardData?.getData('text') || '';
|
|
607
|
-
const isCurrency = this.field.type === 'number' && !!this.field.isCurrency;
|
|
608
|
-
const filteredText = isCurrency
|
|
609
|
-
? this.getFilteredCurrencyPastedText(pastedText)
|
|
610
|
-
: this.getFilteredIntegerPastedText(pastedText);
|
|
611
|
-
const finalValue = this.buildFinalPasteValue(target, filteredText, isCurrency, maxDecimals);
|
|
612
|
-
// Update the input
|
|
613
|
-
target.value = finalValue;
|
|
614
|
-
this.control.setValue(finalValue, { emitEvent: false });
|
|
615
|
-
this.validateIntegerDigits(finalValue, maxDecimals);
|
|
616
|
-
this.clearNumericError();
|
|
617
|
-
this.valueChange.emit({ name: this.field.name, value: finalValue });
|
|
618
|
-
this.control.markAsTouched();
|
|
619
|
-
this.setCursorAfterPaste(target, filteredText, finalValue);
|
|
620
|
-
}
|
|
621
|
-
/**
|
|
622
|
-
* Filter pasted text for currency input (allows digits and decimal point)
|
|
623
|
-
*/
|
|
624
|
-
getFilteredCurrencyPastedText(pastedText) {
|
|
625
|
-
return pastedText.replaceAll(/[^0-9.]/g, '');
|
|
626
|
-
}
|
|
627
|
-
/**
|
|
628
|
-
* Filter pasted text for integer-only input (allows digits only)
|
|
629
|
-
*/
|
|
630
|
-
getFilteredIntegerPastedText(pastedText) {
|
|
631
|
-
return pastedText.replaceAll(/\D/g, '');
|
|
632
|
-
}
|
|
633
|
-
/**
|
|
634
|
-
* Build final value after paste operation
|
|
635
|
-
*/
|
|
636
|
-
buildFinalPasteValue(target, filteredText, isCurrency, maxDecimals) {
|
|
637
|
-
const currentValue = target.value;
|
|
638
|
-
const start = target.selectionStart;
|
|
639
|
-
const end = target.selectionEnd;
|
|
640
|
-
let finalValue;
|
|
641
|
-
// If selection API is not supported or no selection, replace entire content
|
|
642
|
-
if (start === null || end === null || (start === 0 && end === 0)) {
|
|
643
|
-
finalValue = filteredText;
|
|
644
|
-
}
|
|
645
|
-
else {
|
|
646
|
-
finalValue = currentValue.substring(0, start) + filteredText + currentValue.substring(end);
|
|
647
|
-
}
|
|
648
|
-
return isCurrency ? this.applyDecimalRestrictions(finalValue, maxDecimals) : finalValue;
|
|
649
|
-
}
|
|
650
|
-
/**
|
|
651
|
-
* Apply decimal restrictions for currency fields
|
|
652
|
-
*/
|
|
653
|
-
applyDecimalRestrictions(value, maxDecimals) {
|
|
654
|
-
let result = value;
|
|
655
|
-
// Ensure only one decimal point
|
|
656
|
-
const parts = result.split('.');
|
|
657
|
-
if (parts.length > 2) {
|
|
658
|
-
result = parts[0] + '.' + parts.slice(1).join('');
|
|
659
|
-
}
|
|
660
|
-
// Restrict to maxDecimals decimal places
|
|
661
|
-
if (result.includes('.')) {
|
|
662
|
-
const [intPart, decPart] = result.split('.');
|
|
663
|
-
if (decPart.length > maxDecimals) {
|
|
664
|
-
result = intPart + '.' + decPart.slice(0, maxDecimals);
|
|
665
|
-
}
|
|
666
|
-
}
|
|
667
|
-
return result;
|
|
668
|
-
}
|
|
669
|
-
/**
|
|
670
|
-
* Validate integer digit limit and set/clear errors
|
|
671
|
-
*/
|
|
672
|
-
validateIntegerDigits(value, maxDecimals) {
|
|
673
|
-
if (this.field.type !== 'number' || !this.field.maxIntegerDigits) {
|
|
674
|
-
return;
|
|
675
|
-
}
|
|
676
|
-
const maxIntDigits = this.field.maxIntegerDigits ?? 15;
|
|
677
|
-
const [intPart] = value.includes('.') ? value.split('.') : [value];
|
|
678
|
-
if (intPart.length > maxIntDigits) {
|
|
679
|
-
this.setMaxIntegerDigitsError(intPart.length, maxIntDigits, maxDecimals);
|
|
680
|
-
}
|
|
681
|
-
else {
|
|
682
|
-
this.clearMaxIntegerDigitsError();
|
|
683
|
-
}
|
|
684
|
-
}
|
|
685
|
-
/**
|
|
686
|
-
* Set max integer digits error
|
|
687
|
-
*/
|
|
688
|
-
setMaxIntegerDigitsError(actual, max, maxDecimals) {
|
|
689
|
-
const currentErrors = this.control.errors || {};
|
|
690
|
-
this.control.setErrors({
|
|
691
|
-
...currentErrors,
|
|
692
|
-
maxIntegerDigits: {
|
|
693
|
-
actual,
|
|
694
|
-
max,
|
|
695
|
-
maxDecimals,
|
|
696
|
-
message: `Please shorten your input to ${max} whole digits and ${maxDecimals} decimal or fewer.`,
|
|
697
|
-
},
|
|
698
|
-
});
|
|
699
|
-
}
|
|
700
|
-
/**
|
|
701
|
-
* Clear max integer digits error if present
|
|
702
|
-
*/
|
|
703
|
-
clearMaxIntegerDigitsError() {
|
|
704
|
-
if (!this.control.hasError('maxIntegerDigits')) {
|
|
705
|
-
return;
|
|
706
|
-
}
|
|
707
|
-
const errors = { ...this.control.errors };
|
|
708
|
-
delete errors['maxIntegerDigits'];
|
|
709
|
-
this.control.setErrors(Object.keys(errors).length ? errors : null);
|
|
710
|
-
}
|
|
711
|
-
/**
|
|
712
|
-
* Clear numeric error if present
|
|
713
|
-
*/
|
|
714
|
-
clearNumericError() {
|
|
715
|
-
if (!this.control.hasError('numeric')) {
|
|
716
|
-
return;
|
|
717
|
-
}
|
|
718
|
-
const errors = { ...this.control.errors };
|
|
719
|
-
delete errors['numeric'];
|
|
720
|
-
this.control.setErrors(Object.keys(errors).length ? errors : null);
|
|
721
|
-
}
|
|
722
|
-
/**
|
|
723
|
-
* Set cursor position after paste operation
|
|
724
|
-
*/
|
|
725
|
-
setCursorAfterPaste(target, filteredText, finalValue) {
|
|
726
|
-
// Input type="number" does not support selection APIs
|
|
727
|
-
if (target.type === 'number') {
|
|
728
|
-
return;
|
|
729
|
-
}
|
|
730
|
-
const start = target.selectionStart || 0;
|
|
731
|
-
const newCursorPos = start + filteredText.length;
|
|
732
|
-
const actualPos = Math.min(newCursorPos, finalValue.length);
|
|
733
|
-
target.setSelectionRange(actualPos, actualPos);
|
|
734
|
-
}
|
|
735
|
-
// Format number with comma separators
|
|
736
|
-
// Uses string-based formatting for large numbers to preserve precision
|
|
737
|
-
formatCurrencyWithCommas(value, maxDecimals = 2) {
|
|
738
|
-
if (this.field.type !== 'number' || !this.field.isCurrency)
|
|
739
|
-
return value.toString();
|
|
740
|
-
const strValue = value.toString();
|
|
741
|
-
// Return original value for empty or invalid inputs
|
|
742
|
-
if (!strValue || strValue === 'NaN')
|
|
743
|
-
return '';
|
|
744
|
-
// Use string-based formatting to preserve precision for large numbers
|
|
745
|
-
const [integerPart, decimalPart] = strValue.includes('.')
|
|
746
|
-
? strValue.split('.')
|
|
747
|
-
: [strValue, ''];
|
|
748
|
-
// Format integer part with comma separators (every 3 digits from right)
|
|
749
|
-
// Using iterative approach instead of regex to avoid ReDoS concerns
|
|
750
|
-
let formattedInteger = '';
|
|
751
|
-
const len = integerPart.length;
|
|
752
|
-
for (let i = 0; i < len; i++) {
|
|
753
|
-
if (i > 0 && (len - i) % 3 === 0) {
|
|
754
|
-
formattedInteger += ',';
|
|
755
|
-
}
|
|
756
|
-
formattedInteger += integerPart[i];
|
|
757
|
-
}
|
|
758
|
-
// Format decimal part (pad or truncate to maxDecimals)
|
|
759
|
-
let formattedDecimal = decimalPart || '';
|
|
760
|
-
if (formattedDecimal.length < maxDecimals) {
|
|
761
|
-
formattedDecimal = formattedDecimal.padEnd(maxDecimals, '0');
|
|
762
|
-
}
|
|
763
|
-
else if (formattedDecimal.length > maxDecimals) {
|
|
764
|
-
formattedDecimal = formattedDecimal.slice(0, maxDecimals);
|
|
765
|
-
}
|
|
766
|
-
return maxDecimals > 0 ? `${formattedInteger}.${formattedDecimal}` : formattedInteger;
|
|
767
|
-
}
|
|
768
|
-
handleNumberKeydown(event, isCurrency) {
|
|
769
|
-
if (isCurrency)
|
|
770
|
-
return;
|
|
771
|
-
// Allow: Backspace, Delete, Tab, Arrow keys
|
|
772
|
-
if (['Backspace', 'Delete', 'Tab', 'ArrowLeft', 'ArrowRight'].includes(event.key)) {
|
|
773
|
-
return;
|
|
774
|
-
}
|
|
775
|
-
// Allow clipboard operations (Ctrl+V, Ctrl+C, Ctrl+X, Cmd+V, Cmd+C, Cmd+X)
|
|
776
|
-
// and selection (Ctrl+A, Cmd+A)
|
|
777
|
-
if (event.ctrlKey || event.metaKey) {
|
|
778
|
-
return;
|
|
779
|
-
}
|
|
780
|
-
// Block everything that is not 0-9
|
|
781
|
-
if (!/^\d$/.test(event.key)) {
|
|
782
|
-
event.preventDefault();
|
|
783
|
-
}
|
|
784
|
-
}
|
|
785
|
-
// Get locale from currency code using currencyMeta
|
|
786
|
-
getLocaleFromCurrency(currencyCode) {
|
|
787
|
-
const currencyObj = this.currencyMeta.find((c) => c.code === currencyCode);
|
|
788
|
-
return currencyObj?.locale || this.defaultLocale;
|
|
789
|
-
}
|
|
790
|
-
showPassword = false;
|
|
791
|
-
passwordStrengthLabel = '';
|
|
792
|
-
strengthClass = '';
|
|
793
|
-
passwordStrengthPercent = 0;
|
|
794
|
-
isPasswordFocused = false;
|
|
795
|
-
onPasswordFocus() {
|
|
796
|
-
this.hasFocus.set(true);
|
|
797
|
-
this.isPasswordFocused = true;
|
|
798
|
-
this.cdr.detectChanges();
|
|
799
|
-
}
|
|
800
|
-
onPasswordBlur() {
|
|
801
|
-
this.isPasswordFocused = false;
|
|
802
|
-
this.cdr.detectChanges();
|
|
803
|
-
}
|
|
804
|
-
handlePasswordBlur(e) {
|
|
805
|
-
this.handleBlur(e); // Original blur functionality (sets hasFocus to false)
|
|
806
|
-
this.onPasswordBlur(); // Hide password feedback
|
|
807
|
-
}
|
|
808
|
-
checkPasswordStrength(value) {
|
|
809
|
-
if (!value || this.field.type !== 'password') {
|
|
810
|
-
this.passwordStrengthLabel = '';
|
|
811
|
-
this.strengthClass = '';
|
|
812
|
-
this.passwordStrengthPercent = 0;
|
|
813
|
-
return;
|
|
814
|
-
}
|
|
815
|
-
let score = 0;
|
|
816
|
-
if (value.length >= 8)
|
|
817
|
-
score++;
|
|
818
|
-
if (/[A-Z]/.test(value))
|
|
819
|
-
score++;
|
|
820
|
-
if (/\d/.test(value))
|
|
821
|
-
score++;
|
|
822
|
-
if (/[\W_]/.test(value))
|
|
823
|
-
score++;
|
|
824
|
-
if (score <= 1) {
|
|
825
|
-
this.passwordStrengthLabel = this.field.weakLabel || 'Weak';
|
|
826
|
-
this.strengthClass = 'text-[var(--RHB-Red-100)]';
|
|
827
|
-
this.passwordStrengthPercent = 25;
|
|
828
|
-
}
|
|
829
|
-
else if (score === 2 || score === 3) {
|
|
830
|
-
this.passwordStrengthLabel = this.field.mediumLabel || 'Medium';
|
|
831
|
-
this.strengthClass = 'text-[var(--Orange-100)]';
|
|
832
|
-
this.passwordStrengthPercent = score === 2 ? 50 : 75;
|
|
833
|
-
}
|
|
834
|
-
else {
|
|
835
|
-
this.passwordStrengthLabel = this.field.strongLabel || 'Strong';
|
|
836
|
-
this.strengthClass = 'text-[var(--Green-100)]';
|
|
837
|
-
this.passwordStrengthPercent = 100;
|
|
838
|
-
}
|
|
839
|
-
}
|
|
840
|
-
increment() {
|
|
841
|
-
if (this.field.type !== 'number')
|
|
842
|
-
return;
|
|
843
|
-
const step = this.field.step || 1;
|
|
844
|
-
const current = Number(this.control.value) || 0;
|
|
845
|
-
let nextValue = current + step;
|
|
846
|
-
if (typeof this.field.max === 'number') {
|
|
847
|
-
nextValue = Math.min(nextValue, this.field.max);
|
|
848
|
-
}
|
|
849
|
-
this.control.setValue(nextValue);
|
|
850
|
-
this.control.markAsTouched();
|
|
851
|
-
this.valueChange.emit({ name: this.field.name, value: nextValue });
|
|
852
|
-
// Update display value for currency inputs
|
|
853
|
-
if (this.field.isCurrency) {
|
|
854
|
-
setTimeout(() => {
|
|
855
|
-
const inputElement = document.getElementById(this.uniqueId);
|
|
856
|
-
if (inputElement) {
|
|
857
|
-
inputElement.value = this.formatCurrencyWithCommas(nextValue);
|
|
858
|
-
}
|
|
859
|
-
});
|
|
860
|
-
}
|
|
861
|
-
}
|
|
862
|
-
decrement() {
|
|
863
|
-
if (this.field.type !== 'number')
|
|
864
|
-
return;
|
|
865
|
-
const step = this.field.step || 1;
|
|
866
|
-
const current = Number(this.control.value) || 0;
|
|
867
|
-
let nextValue = current - step;
|
|
868
|
-
// Ensure minimum value is 0 (no negative values)
|
|
869
|
-
const minValue = Math.max(this.field.min || 0, 0);
|
|
870
|
-
nextValue = Math.max(nextValue, minValue);
|
|
871
|
-
this.control.setValue(nextValue);
|
|
872
|
-
this.control.markAsTouched();
|
|
873
|
-
this.valueChange.emit({ name: this.field.name, value: nextValue });
|
|
874
|
-
// Update display value for currency inputs
|
|
875
|
-
if (this.field.isCurrency) {
|
|
876
|
-
setTimeout(() => {
|
|
877
|
-
const inputElement = document.getElementById(this.uniqueId);
|
|
878
|
-
if (inputElement) {
|
|
879
|
-
inputElement.value = this.formatCurrencyWithCommas(nextValue);
|
|
880
|
-
}
|
|
881
|
-
});
|
|
882
|
-
}
|
|
883
|
-
}
|
|
884
|
-
copyText(e) {
|
|
885
|
-
e.stopPropagation();
|
|
886
|
-
if (!this.control.value)
|
|
887
|
-
return;
|
|
888
|
-
navigator.clipboard.writeText(this.control.value).then(() => {
|
|
889
|
-
const service = this.effectiveToastService;
|
|
890
|
-
if (service) {
|
|
891
|
-
// Position set to top as requested
|
|
892
|
-
service.success('Copied to clipboard', 'Success', 2000);
|
|
893
|
-
}
|
|
894
|
-
});
|
|
895
|
-
}
|
|
896
|
-
prefixClick(e) {
|
|
897
|
-
if (this.field.prefix?.onClick) {
|
|
898
|
-
this.field.prefix.onClick(e);
|
|
899
|
-
}
|
|
900
|
-
}
|
|
901
|
-
suffixClick(e) {
|
|
902
|
-
if (this.field.suffix?.onClick) {
|
|
903
|
-
this.field.suffix.onClick(e);
|
|
904
|
-
}
|
|
905
|
-
}
|
|
906
|
-
/**
|
|
907
|
-
* Set up visibility condition listener
|
|
908
|
-
* Watches the dependent field and shows/hides this field based on conditions
|
|
909
|
-
*/
|
|
910
|
-
setupVisibilityCondition() {
|
|
911
|
-
if (!this.field.visibilityCondition || !this.form)
|
|
912
|
-
return;
|
|
913
|
-
const { dependsOn } = this.field.visibilityCondition;
|
|
914
|
-
const dependentControl = this.form.get(dependsOn);
|
|
915
|
-
if (!dependentControl) {
|
|
916
|
-
return;
|
|
917
|
-
}
|
|
918
|
-
// Initial visibility check
|
|
919
|
-
this.updateVisibility(dependentControl.value);
|
|
920
|
-
// Subscribe to value changes
|
|
921
|
-
dependentControl.valueChanges.subscribe((value) => {
|
|
922
|
-
this.updateVisibility(value);
|
|
923
|
-
});
|
|
924
|
-
}
|
|
925
|
-
/**
|
|
926
|
-
* Update field visibility based on dependent field value
|
|
927
|
-
*/
|
|
928
|
-
updateVisibility(dependentValue) {
|
|
929
|
-
if (!this.field.visibilityCondition)
|
|
930
|
-
return;
|
|
931
|
-
const { showWhen, hideWhen } = this.field.visibilityCondition;
|
|
932
|
-
let shouldBeVisible = false;
|
|
933
|
-
// Check showWhen condition
|
|
934
|
-
if (showWhen !== undefined) {
|
|
935
|
-
if (Array.isArray(showWhen)) {
|
|
936
|
-
shouldBeVisible = showWhen.includes(dependentValue);
|
|
937
|
-
}
|
|
938
|
-
else {
|
|
939
|
-
shouldBeVisible = dependentValue === showWhen;
|
|
940
|
-
}
|
|
941
|
-
}
|
|
942
|
-
// Check hideWhen condition (overrides showWhen)
|
|
943
|
-
if (hideWhen !== undefined) {
|
|
944
|
-
const shouldHide = Array.isArray(hideWhen)
|
|
945
|
-
? hideWhen.includes(dependentValue)
|
|
946
|
-
: dependentValue === hideWhen;
|
|
947
|
-
if (shouldHide) {
|
|
948
|
-
shouldBeVisible = false;
|
|
949
|
-
}
|
|
950
|
-
}
|
|
951
|
-
// Update visibility
|
|
952
|
-
this.isVisible.set(shouldBeVisible);
|
|
953
|
-
// Only clear value when hiding if there's no initial value
|
|
954
|
-
// This preserves data when editing existing records
|
|
955
|
-
if (!shouldBeVisible && this.control && !this.control.value) {
|
|
956
|
-
this.control.setValue(null);
|
|
957
|
-
this.control.markAsUntouched();
|
|
958
|
-
this.control.updateValueAndValidity();
|
|
959
|
-
}
|
|
960
|
-
this.cdr.detectChanges();
|
|
961
|
-
}
|
|
962
|
-
setupToastSubscription() {
|
|
963
|
-
const service = this.effectiveToastService;
|
|
964
|
-
if (service) {
|
|
965
|
-
this.subs.add(service.toasts$.subscribe((toasts) => {
|
|
966
|
-
this.toasts.set(toasts);
|
|
967
|
-
// Wait for DOM to update then render toast icons
|
|
968
|
-
this.updateAllIcons();
|
|
969
|
-
}));
|
|
970
|
-
}
|
|
971
|
-
}
|
|
972
|
-
removeToast(id) {
|
|
973
|
-
this.effectiveToastService?.remove(id);
|
|
974
|
-
}
|
|
975
|
-
getTranslatedMessage(message) {
|
|
976
|
-
if (this.helperHandle && typeof this.helperHandle.translate === 'function') {
|
|
977
|
-
return this.helperHandle.translate(message);
|
|
978
|
-
}
|
|
979
|
-
return message;
|
|
980
|
-
}
|
|
981
|
-
getToastIconName(toast) {
|
|
982
|
-
switch (toast.type) {
|
|
983
|
-
case 'success':
|
|
984
|
-
return 'check-circle';
|
|
985
|
-
case 'warn':
|
|
986
|
-
case 'info':
|
|
987
|
-
return 'info-circle';
|
|
988
|
-
case 'error':
|
|
989
|
-
return 'exclamation-circle';
|
|
990
|
-
default:
|
|
991
|
-
return 'info-circle';
|
|
992
|
-
}
|
|
993
|
-
}
|
|
994
|
-
showFormFieldMessage(control, helperText) {
|
|
995
|
-
if (this.helperHandle) {
|
|
996
|
-
return this.helperHandle.showFormFieldMessage(control, helperText);
|
|
997
|
-
}
|
|
998
|
-
const isError = control?.touched && !!control?.errors;
|
|
999
|
-
const isHelper = !isError && Boolean(helperText ?? '');
|
|
1000
|
-
return isError || isHelper;
|
|
1001
|
-
}
|
|
1002
|
-
getTranslatedLabel(label) {
|
|
1003
|
-
if (!label)
|
|
1004
|
-
return '';
|
|
1005
|
-
try {
|
|
1006
|
-
const labelStr = label.toString();
|
|
1007
|
-
return (labelStr
|
|
1008
|
-
.split('::')
|
|
1009
|
-
.map((part) => part.trim())
|
|
1010
|
-
.filter((part) => !!part)
|
|
1011
|
-
.map((part) => this.translateService.instant(part))
|
|
1012
|
-
.join(' ') + this.getEndSymbol(labelStr));
|
|
1013
|
-
}
|
|
1014
|
-
catch (e) {
|
|
1015
|
-
console.warn('Translation failed in getTranslatedLabel:', label, e);
|
|
1016
|
-
return label;
|
|
1017
|
-
}
|
|
1018
|
-
}
|
|
1019
|
-
getEndSymbol(label) {
|
|
1020
|
-
return label.toString().includes('required') && label.toString().includes('FNFieldMessage')
|
|
1021
|
-
? '.'
|
|
1022
|
-
: '';
|
|
1023
|
-
}
|
|
1024
|
-
// --- ICON LOGIC ---
|
|
1025
|
-
updateAllIcons() {
|
|
1026
|
-
// We'll call this after ViewInit or when inputs change
|
|
1027
|
-
setTimeout(() => {
|
|
1028
|
-
this.iconContainers?.forEach((container) => {
|
|
1029
|
-
const ds = container.nativeElement.dataset;
|
|
1030
|
-
const name = ds['iconName'];
|
|
1031
|
-
const variant = ds['iconVariant'] || 'Line';
|
|
1032
|
-
const size = (ds['iconSize'] || 'medium');
|
|
1033
|
-
const color = ds['iconColor'];
|
|
1034
|
-
const disabled = ds['iconDisabled'] === 'true';
|
|
1035
|
-
if (name) {
|
|
1036
|
-
this.loadIconToContainer(container.nativeElement, name, variant, size, color, disabled);
|
|
1037
|
-
}
|
|
1038
|
-
});
|
|
1039
|
-
}, 0);
|
|
1040
|
-
}
|
|
1041
|
-
loadIconToContainer(container, name, variant, sizeName, color, disabled) {
|
|
1042
|
-
const size = this.sizeMap[sizeName];
|
|
1043
|
-
// 1. Check if raw SVG string is provided
|
|
1044
|
-
const trimmedName = name.trim();
|
|
1045
|
-
if (trimmedName.startsWith('<svg')) {
|
|
1046
|
-
this.renderSvgToContainer(container, trimmedName, color, size, 'custom', disabled);
|
|
1047
|
-
return;
|
|
1048
|
-
}
|
|
1049
|
-
// 1b. Check if it's just raw SVG inner content (like `<path ...>`)
|
|
1050
|
-
if (trimmedName.startsWith('<path') ||
|
|
1051
|
-
trimmedName.startsWith('<g') ||
|
|
1052
|
-
trimmedName.startsWith('<circle') ||
|
|
1053
|
-
trimmedName.startsWith('<rect') ||
|
|
1054
|
-
trimmedName.startsWith('<polygon')) {
|
|
1055
|
-
const wrappedSvg = `<svg viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">${trimmedName}</svg>`;
|
|
1056
|
-
this.renderSvgToContainer(container, wrappedSvg, color, size, 'custom', disabled);
|
|
1057
|
-
return;
|
|
1058
|
-
}
|
|
1059
|
-
// 2. Check built-in icons registry
|
|
1060
|
-
// Strip possible --size suffix if it was passed by mistake from old logic
|
|
1061
|
-
const baseName = trimmedName.split('--')[0].trim();
|
|
1062
|
-
if (CORE_ICONS[baseName]) {
|
|
1063
|
-
this.renderSvgToContainer(container, CORE_ICONS[baseName], color, size, baseName, disabled);
|
|
1064
|
-
return;
|
|
1065
|
-
}
|
|
1066
|
-
// No fallback to HTTP
|
|
1067
|
-
console.warn(`[fn-input] Icon "${name}" not found in core registry and no raw SVG provided.`);
|
|
1068
|
-
container.innerHTML = '';
|
|
1069
|
-
this.cdr.detectChanges();
|
|
1070
|
-
}
|
|
1071
|
-
renderSvgToContainer(container, raw, color, size, name, disabled) {
|
|
1072
|
-
try {
|
|
1073
|
-
const parser = new DOMParser();
|
|
1074
|
-
const doc = parser.parseFromString(raw, 'image/svg+xml');
|
|
1075
|
-
const svg = doc.querySelector('svg');
|
|
1076
|
-
if (!svg) {
|
|
1077
|
-
container.innerHTML = '';
|
|
1078
|
-
return;
|
|
1079
|
-
}
|
|
1080
|
-
// SVG Normalization logic
|
|
1081
|
-
for (const s of svg.querySelectorAll('style'))
|
|
1082
|
-
s.remove();
|
|
1083
|
-
if (color) {
|
|
1084
|
-
for (const el of svg.querySelectorAll('*')) {
|
|
1085
|
-
const style = el.getAttribute('style');
|
|
1086
|
-
if (style) {
|
|
1087
|
-
const cleaned = style
|
|
1088
|
-
.replaceAll(/fill\s*:\s*[^;]+;?/gi, '')
|
|
1089
|
-
.replaceAll(/stroke\s*:\s*[^;]+;?/gi, '');
|
|
1090
|
-
if (cleaned)
|
|
1091
|
-
el.setAttribute('style', cleaned);
|
|
1092
|
-
else
|
|
1093
|
-
el.removeAttribute('style');
|
|
1094
|
-
}
|
|
1095
|
-
const fill = el.getAttribute('fill');
|
|
1096
|
-
if (fill && fill !== 'none' && !fill.startsWith('url(')) {
|
|
1097
|
-
el.setAttribute('fill', 'currentColor');
|
|
1098
|
-
}
|
|
1099
|
-
const stroke = el.getAttribute('stroke');
|
|
1100
|
-
if (stroke && stroke !== 'none' && !stroke.startsWith('url(')) {
|
|
1101
|
-
el.setAttribute('stroke', 'currentColor');
|
|
1102
|
-
}
|
|
1103
|
-
}
|
|
1104
|
-
}
|
|
1105
|
-
svg.removeAttribute('width');
|
|
1106
|
-
svg.removeAttribute('height');
|
|
1107
|
-
if (!svg.getAttribute('viewBox')) {
|
|
1108
|
-
svg.setAttribute('viewBox', `0 0 ${size} ${size}`);
|
|
1109
|
-
}
|
|
1110
|
-
svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
|
|
1111
|
-
svg.setAttribute('focusable', 'false');
|
|
1112
|
-
svg.setAttribute('aria-hidden', 'true');
|
|
1113
|
-
svg.setAttribute('width', '100%');
|
|
1114
|
-
svg.setAttribute('height', '100%');
|
|
1115
|
-
const style = `;display:block;max-width:100%;max-height:100%;`;
|
|
1116
|
-
svg.setAttribute('style', (svg.getAttribute('style') || '') + style);
|
|
1117
|
-
container.innerHTML = '';
|
|
1118
|
-
if (color) {
|
|
1119
|
-
container.style.color = color;
|
|
1120
|
-
}
|
|
1121
|
-
if (disabled) {
|
|
1122
|
-
container.style.opacity = '0.5';
|
|
1123
|
-
container.style.cursor = 'not-allowed';
|
|
1124
|
-
}
|
|
1125
|
-
else {
|
|
1126
|
-
container.style.opacity = '1';
|
|
1127
|
-
container.style.cursor = '';
|
|
1128
|
-
}
|
|
1129
|
-
container.appendChild(svg);
|
|
1130
|
-
this.cdr.detectChanges();
|
|
1131
|
-
}
|
|
1132
|
-
catch (error) {
|
|
1133
|
-
console.error(`[fn-input] Error parsing SVG for icon "${name}":`, error);
|
|
1134
|
-
container.innerHTML = '';
|
|
1135
|
-
}
|
|
1136
|
-
}
|
|
1137
|
-
// Removed getIconPath as HTTP loading is disabled
|
|
1138
|
-
// --- MESSAGE LOGIC ---
|
|
1139
|
-
getFieldMessage() {
|
|
1140
|
-
if (!this.control)
|
|
1141
|
-
return this.field?.helperText || '';
|
|
1142
|
-
if (this.control.touched && this.control.errors) {
|
|
1143
|
-
for (const key of Object.keys(this.control.errors)) {
|
|
1144
|
-
const errorValue = this.control.errors[key];
|
|
1145
|
-
if (errorValue && typeof errorValue === 'object' && errorValue.message) {
|
|
1146
|
-
return errorValue.message;
|
|
1147
|
-
}
|
|
1148
|
-
if (this.field?.errors?.[key]) {
|
|
1149
|
-
return this.field.errors[key];
|
|
1150
|
-
}
|
|
1151
|
-
}
|
|
1152
|
-
return (this.field?.errors?.['default'] ||
|
|
1153
|
-
'Please enter ' + this.translateService.instant(this.field?.label));
|
|
1154
|
-
}
|
|
1155
|
-
return this.field?.helperText || '';
|
|
1156
|
-
}
|
|
1157
|
-
get isError() {
|
|
1158
|
-
return !!(this.control?.touched && this.control?.errors);
|
|
1159
|
-
}
|
|
1160
|
-
get isSuccess() {
|
|
1161
|
-
return !!(this.control?.valid && this.control?.touched);
|
|
1162
|
-
}
|
|
1163
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FNInput, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
1164
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: FNInput, isStandalone: true, selector: "fn-input", inputs: { field: "field", helperHandle: "helperHandle", toastService: "toastService", currencyMeta: "currencyMeta", defaultLocale: "defaultLocale", form: "form" }, outputs: { valueChange: "valueChange", fieldBlur: "fieldBlur" }, viewQueries: [{ propertyName: "textareaElement", first: true, predicate: ["fnTextarea"], descendants: true }, { propertyName: "iconContainers", predicate: ["iconContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "@if (field.name && isVisible() && !field.hidden) {\r\n <div class=\"fn-input-container\">\r\n @if (field.type! !== 'hidden') {\r\n <fn-label\r\n [for]=\"uniqueId\"\r\n [label]=\"field.label\"\r\n [required]=\"field.required || false\"\r\n [hideOptional]=\"field.hideOptional || false\"\r\n [color]=\"field.color || '#03182b'\"\r\n [variant]=\"field.labelVariant || 'p1'\"\r\n [statusLabel]=\"field.statusLabel\"\r\n ></fn-label>\r\n }\r\n\r\n <ng-container>\r\n @switch (field.type) {\r\n <!-- Textarea Field -->\r\n @case ('textarea') {\r\n <textarea\r\n #fnTextarea\r\n style=\"resize: none\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"field.disabled || false\"\r\n [rows]=\"field.rows || 1\"\r\n [readonly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n (input)=\"handleTextArea($event)\"\r\n (blur)=\"handleBlur($event)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n ></textarea>\r\n }\r\n\r\n <!-- Password Field -->\r\n @case ('password') {\r\n <div class=\"relative w-full\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"showPassword ? 'text' : 'password'\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [disabled]=\"field.disabled || false\"\r\n [formControl]=\"control\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [readonly]=\"field.readOnly || false\"\r\n (input)=\"isAlphanumeric ? handleAlphanumericInput($event) : handleInput($event)\"\r\n (focus)=\"onPasswordFocus()\"\r\n (blur)=\"handlePasswordBlur($event)\"\r\n class=\"fn-input-field overflow-hidden text-ellipsis\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-12':\r\n (field.toggleMask && !field.isCopyText) ||\r\n (!field.toggleMask && field.isCopyText),\r\n 'pr-24': field.toggleMask && field.isCopyText && !field.hasGenerateKey,\r\n 'pr-32': field.toggleMask && field.isCopyText && field.hasGenerateKey,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n />\r\n\r\n <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n @if (field.toggleMask && !field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"togglePasswordVisibility()\"\r\n (keydown)=\"togglePasswordVisibility()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n } @else if (field.isCopyText && !field.toggleMask) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n } @else if (field.toggleMask && field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"togglePasswordVisibility()\"\r\n (keydown)=\"togglePasswordVisibility()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n [ngClass]=\"{ 'opacity-50 cursor-not-allowed': !control.value }\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n }\r\n </div>\r\n </div>\r\n\r\n @if (field.feedback && isPasswordFocused) {\r\n <div class=\"mt-2 space-y-1\">\r\n <div class=\"flex items-center justify-between text-xs\">\r\n <span class=\"font-medium\" [ngClass]=\"strengthClass\">\r\n {{ passwordStrengthLabel | translate }}\r\n </span>\r\n <div class=\"flex items-center gap-1\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"\r\n passwordStrengthPercent > 50 ? 'check-circle' : 'info-circle'\r\n \"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"extrasmall\"\r\n [attr.data-icon-color]=\"\r\n passwordStrengthPercent <= 25\r\n ? '#ef3e42'\r\n : passwordStrengthPercent <= 75\r\n ? '#ff9f00'\r\n : '#6cc24a'\r\n \"\r\n ></div>\r\n </div>\r\n </div>\r\n <div class=\"h-1.5 w-full bg-[#eef0f2] rounded-full overflow-hidden\">\r\n <div\r\n class=\"h-full transition-all duration-300 rounded-full\"\r\n [ngClass]=\"{\r\n 'bg-[#ef3e42]': passwordStrengthPercent <= 25,\r\n 'bg-[#ff9f00]': passwordStrengthPercent > 25 && passwordStrengthPercent <= 75,\r\n 'bg-[#6cc24a]': passwordStrengthPercent > 75,\r\n }\"\r\n [style.width.%]=\"passwordStrengthPercent\"\r\n ></div>\r\n </div>\r\n </div>\r\n }\r\n }\r\n\r\n <!-- Number Field -->\r\n @case ('number') {\r\n <div class=\"relative w-full\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"field.type === 'number' && field.isCurrency ? 'text' : 'number'\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"isDisabled || false\"\r\n [readOnly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n [min]=\"field.type === 'number' && !field.isCurrency ? 0 : null\"\r\n [max]=\"field.type === 'number' && !field.isCurrency ? field.max : null\"\r\n [step]=\"field.type === 'number' && !field.isCurrency ? field.step || 1 : null\"\r\n (input)=\"handleNumberInput($event, field.minFractionDigits || 2)\"\r\n (paste)=\"handleNumberPaste($event, field.minFractionDigits || 2)\"\r\n (blur)=\"handleBlur($event, field.minFractionDigits || 2)\"\r\n (keydown)=\"handleNumberKeydown($event, field.isCurrency || false)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-10': field.suffix?.icon && !field.suffix?.text,\r\n 'has-suffix-text': !!field.suffix?.text,\r\n 'text-[24px] font-bold': field.isCurrency,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n />\r\n </div>\r\n }\r\n\r\n <!-- Default Input Field (text, email, etc.) -->\r\n @default {\r\n <div class=\"relative z-0\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"field.type\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"field.disabled || false\"\r\n [readOnly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n (input)=\"\r\n isEmailField\r\n ? handleEmailInput($event)\r\n : field.type === 'text' && isAlphanumeric\r\n ? handleAlphanumericInput($event)\r\n : handleInput($event)\r\n \"\r\n (blur)=\"handleBlur($event)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-10':\r\n (field.icon || field.suffix?.icon) && !field.isCopyText && !field.suffix?.text,\r\n 'has-suffix-text': !!field.suffix?.text,\r\n 'pr-20':\r\n field.isCopyText && !field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n 'pr-30':\r\n field.isCopyText && field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n 'pr-32':\r\n field.isCopyText && field.hasGenerateKey && (field.icon || field.suffix?.icon),\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n />\r\n\r\n <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n @if (field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n }\r\n\r\n @if (field.icon || field.suffix?.icon || field.suffix?.text) {\r\n @if (field.suffix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"suffixClick($event)\"\r\n (keydown.enter)=\"suffixClick($event)\"\r\n [attr.tabindex]=\"field.suffix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.suffix?.onClick\"\r\n >{{ field.suffix!.text }}</span\r\n >\r\n }\r\n @if (field.icon || field.suffix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.suffix?.icon || field.icon!.name\"\r\n [attr.data-icon-variant]=\"field.icon?.variant || 'Line'\"\r\n [attr.data-icon-size]=\"field.icon ? getIconSizeName(field.icon.size) : 'medium'\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n }\r\n </ng-container>\r\n\r\n @if (showFormFieldMessage(control, helperText)) {\r\n <div class=\"fn-field-message-container\">\r\n <span\r\n class=\"fn-field-message-text\"\r\n [ngClass]=\"{\r\n error: isError,\r\n success: isSuccess,\r\n }\"\r\n >\r\n {{ getFieldMessage() | translate }}\r\n </span>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Integrated Toast Notifications -->\r\n <div class=\"toast-container\">\r\n @for (toast of toasts(); track toast.id) {\r\n <div class=\"toast toast-{{ toast.type }}\" [class.toast-enter]=\"toast.isVisible\">\r\n <span class=\"toast-icon\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"getToastIconName(toast)\"\r\n data-icon-variant=\"Colour\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"\r\n toast.type === 'success'\r\n ? '#6cc24a'\r\n : toast.type === 'warn'\r\n ? '#ff9f00'\r\n : toast.type === 'error'\r\n ? '#ef3e42'\r\n : '#03182b'\r\n \"\r\n ></div>\r\n </span>\r\n <div class=\"toast-content\">\r\n <span class=\"toast-message\">\r\n {{ getTranslatedMessage(toast.message) | translate }}\r\n </span>\r\n </div>\r\n <button class=\"toast-close\" (click)=\"removeToast(toast.id)\" aria-label=\"Close\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"cross\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n data-icon-color=\"#ffffff\"\r\n ></div>\r\n </button>\r\n </div>\r\n }\r\n </div>\r\n}\r\n", styles: [".fn-input-container{display:flex;flex-direction:column;width:100%}.fn-label{font-family:Outfit,sans-serif;font-weight:500;margin-bottom:8px;display:flex;align-items:center}.fn-label span{color:#c4cdd2;font-size:14px;font-weight:400}.fn-input-field{width:100%;height:48px;padding:0 16px;background-color:#f7f8f9;border-bottom:1.5px solid #c4cdd2;font-family:Outfit,sans-serif;font-size:16px;color:#03182b;transition:all .2s ease;border-radius:4px 4px 0 0}.fn-input-field::placeholder{color:#c4cdd2}.fn-input-field:focus{outline:none;background-color:#fff;border-bottom-color:#03182b}.fn-input-field.error{border-bottom-color:#ef3e42}.fn-input-field.success{border-bottom-color:#6cc24a}.fn-input-field.disabled{background-color:#eef0f2;color:#c4cdd2;cursor:not-allowed}.fn-input-field.pl-10{padding-left:40px!important}.fn-input-field.has-prefix-text{padding-left:100px!important}.fn-input-field.pr-10{padding-right:40px!important}.fn-input-field.has-suffix-text{padding-right:120px!important}.fn-input-field.pr-12{padding-right:44px!important}.fn-input-field.pr-20{padding-right:72px!important}.fn-input-field.pr-24{padding-right:88px!important}.fn-input-field.pr-30{padding-right:104px!important}.fn-input-field.pr-32{padding-right:120px!important}.fn-field-message-container{margin-top:4px}.fn-field-message-text{font-family:Outfit,sans-serif;font-size:12px;color:#c4cdd2}.fn-field-message-text.error{color:#ef3e42}.fn-field-message-text.success{color:#6cc24a}.icon-container{display:inline-flex;align-items:center;justify-content:center;width:24px;height:24px}.relative{position:relative}.absolute{position:absolute}.right-0{right:0}.left-0{left:0}.top-1\\/2{top:50%}.-translate-y-1\\/2{transform:translateY(-50%)}.flex{display:flex}.items-center{align-items:center}.justify-between{justify-content:space-between}.gap-1{gap:.25rem}.gap-2{gap:.5rem}.pr-3{padding-right:.75rem}.pl-3{padding-left:.75rem}.mt-2{margin-top:.5rem}.space-y-1>:not([hidden])~:not([hidden]){margin-top:.25rem}.text-xs{font-size:.75rem}.font-medium{font-weight:500}.h-1\\.5{height:.375rem}.w-full{width:100%}.bg-\\[\\#eef0f2\\]{background-color:#eef0f2}.rounded-full{border-radius:9999px}.overflow-hidden{overflow:hidden}.transition-all{transition-property:all}.duration-300{transition-duration:.3s}.cursor-pointer{cursor:pointer}.opacity-50{opacity:.5}.cursor-not-allowed{cursor:not-allowed}.text-ellipsis{text-overflow:ellipsis}.whitespace-nowrap{white-space:nowrap}.toast-container{position:fixed;top:24px;left:50%;transform:translate(-50%);z-index:9999;width:100%;max-width:336px;display:flex;flex-direction:column;gap:8px;pointer-events:none}.toast{display:flex;align-items:center;background:#25333f;border-radius:8px;padding:12px;width:336px;max-width:100%;opacity:0;transform:translateY(-20px);transition:all .3s cubic-bezier(.4,0,.2,1);pointer-events:auto}.toast.toast-enter{opacity:1;transform:translateY(0)}.toast.toast-success{border-left:4px solid #6cc24a}.toast.toast-warn{border-left:4px solid #ff9f00}.toast.toast-error{border-left:4px solid #ef3e42}.toast.toast-info{border-left:4px solid #3b82f6}.toast .toast-icon{width:24px;height:24px;display:flex;align-items:center;justify-content:center;margin-right:8px;flex-shrink:0}.toast .toast-content{flex:1;min-width:0}.toast .toast-message{font-size:14px;line-height:20px;color:#fff;display:block}.toast .toast-close{background:none;border:none;font-size:20px;color:#fff;opacity:.7;cursor:pointer;padding:0;margin-left:8px;width:24px;height:24px;display:flex;align-items:center;justify-content:center;flex-shrink:0;transition:opacity .2s;pointer-events:auto}.toast .toast-close:hover{opacity:1}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }, { kind: "component", type: FNLabel, selector: "fn-label", inputs: ["field", "label", "required", "className", "class", "for", "color", "variant", "statusLabel", "ngClass", "hideOptional"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
1165
|
-
}
|
|
1166
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FNInput, decorators: [{
|
|
1167
|
-
type: Component,
|
|
1168
|
-
args: [{ selector: 'fn-input', standalone: true, imports: [FormsModule, CommonModule, ReactiveFormsModule, TranslateModule, FNLabel], changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (field.name && isVisible() && !field.hidden) {\r\n <div class=\"fn-input-container\">\r\n @if (field.type! !== 'hidden') {\r\n <fn-label\r\n [for]=\"uniqueId\"\r\n [label]=\"field.label\"\r\n [required]=\"field.required || false\"\r\n [hideOptional]=\"field.hideOptional || false\"\r\n [color]=\"field.color || '#03182b'\"\r\n [variant]=\"field.labelVariant || 'p1'\"\r\n [statusLabel]=\"field.statusLabel\"\r\n ></fn-label>\r\n }\r\n\r\n <ng-container>\r\n @switch (field.type) {\r\n <!-- Textarea Field -->\r\n @case ('textarea') {\r\n <textarea\r\n #fnTextarea\r\n style=\"resize: none\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"field.disabled || false\"\r\n [rows]=\"field.rows || 1\"\r\n [readonly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n (input)=\"handleTextArea($event)\"\r\n (blur)=\"handleBlur($event)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n ></textarea>\r\n }\r\n\r\n <!-- Password Field -->\r\n @case ('password') {\r\n <div class=\"relative w-full\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"showPassword ? 'text' : 'password'\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [disabled]=\"field.disabled || false\"\r\n [formControl]=\"control\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [readonly]=\"field.readOnly || false\"\r\n (input)=\"isAlphanumeric ? handleAlphanumericInput($event) : handleInput($event)\"\r\n (focus)=\"onPasswordFocus()\"\r\n (blur)=\"handlePasswordBlur($event)\"\r\n class=\"fn-input-field overflow-hidden text-ellipsis\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-12':\r\n (field.toggleMask && !field.isCopyText) ||\r\n (!field.toggleMask && field.isCopyText),\r\n 'pr-24': field.toggleMask && field.isCopyText && !field.hasGenerateKey,\r\n 'pr-32': field.toggleMask && field.isCopyText && field.hasGenerateKey,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n />\r\n\r\n <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n @if (field.toggleMask && !field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"togglePasswordVisibility()\"\r\n (keydown)=\"togglePasswordVisibility()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n } @else if (field.isCopyText && !field.toggleMask) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n } @else if (field.toggleMask && field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"togglePasswordVisibility()\"\r\n (keydown)=\"togglePasswordVisibility()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"large\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n [ngClass]=\"{ 'opacity-50 cursor-not-allowed': !control.value }\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n }\r\n </div>\r\n </div>\r\n\r\n @if (field.feedback && isPasswordFocused) {\r\n <div class=\"mt-2 space-y-1\">\r\n <div class=\"flex items-center justify-between text-xs\">\r\n <span class=\"font-medium\" [ngClass]=\"strengthClass\">\r\n {{ passwordStrengthLabel | translate }}\r\n </span>\r\n <div class=\"flex items-center gap-1\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"\r\n passwordStrengthPercent > 50 ? 'check-circle' : 'info-circle'\r\n \"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"extrasmall\"\r\n [attr.data-icon-color]=\"\r\n passwordStrengthPercent <= 25\r\n ? '#ef3e42'\r\n : passwordStrengthPercent <= 75\r\n ? '#ff9f00'\r\n : '#6cc24a'\r\n \"\r\n ></div>\r\n </div>\r\n </div>\r\n <div class=\"h-1.5 w-full bg-[#eef0f2] rounded-full overflow-hidden\">\r\n <div\r\n class=\"h-full transition-all duration-300 rounded-full\"\r\n [ngClass]=\"{\r\n 'bg-[#ef3e42]': passwordStrengthPercent <= 25,\r\n 'bg-[#ff9f00]': passwordStrengthPercent > 25 && passwordStrengthPercent <= 75,\r\n 'bg-[#6cc24a]': passwordStrengthPercent > 75,\r\n }\"\r\n [style.width.%]=\"passwordStrengthPercent\"\r\n ></div>\r\n </div>\r\n </div>\r\n }\r\n }\r\n\r\n <!-- Number Field -->\r\n @case ('number') {\r\n <div class=\"relative w-full\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"field.type === 'number' && field.isCurrency ? 'text' : 'number'\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"isDisabled || false\"\r\n [readOnly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n [min]=\"field.type === 'number' && !field.isCurrency ? 0 : null\"\r\n [max]=\"field.type === 'number' && !field.isCurrency ? field.max : null\"\r\n [step]=\"field.type === 'number' && !field.isCurrency ? field.step || 1 : null\"\r\n (input)=\"handleNumberInput($event, field.minFractionDigits || 2)\"\r\n (paste)=\"handleNumberPaste($event, field.minFractionDigits || 2)\"\r\n (blur)=\"handleBlur($event, field.minFractionDigits || 2)\"\r\n (keydown)=\"handleNumberKeydown($event, field.isCurrency || false)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-10': field.suffix?.icon && !field.suffix?.text,\r\n 'has-suffix-text': !!field.suffix?.text,\r\n 'text-[24px] font-bold': field.isCurrency,\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n />\r\n </div>\r\n }\r\n\r\n <!-- Default Input Field (text, email, etc.) -->\r\n @default {\r\n <div class=\"relative z-0\">\r\n @if (field.prefix?.icon || field.prefix?.text) {\r\n <div\r\n class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n style=\"z-index: 1\"\r\n >\r\n @if (field.prefix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.prefix!.icon\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n @if (field.prefix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"prefixClick($event)\"\r\n (keydown.enter)=\"prefixClick($event)\"\r\n [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n >{{ field.prefix!.text }}</span\r\n >\r\n }\r\n </div>\r\n }\r\n <input\r\n [type]=\"field.type\"\r\n [name]=\"field.name\"\r\n [id]=\"uniqueId\"\r\n [required]=\"field.required || false\"\r\n [placeholder]=\"field.placeholder || '' | translate\"\r\n [disabled]=\"field.disabled || false\"\r\n [readOnly]=\"field.readOnly || false\"\r\n [formControl]=\"control\"\r\n (input)=\"\r\n isEmailField\r\n ? handleEmailInput($event)\r\n : field.type === 'text' && isAlphanumeric\r\n ? handleAlphanumericInput($event)\r\n : handleInput($event)\r\n \"\r\n (blur)=\"handleBlur($event)\"\r\n class=\"fn-input-field\"\r\n [ngClass]=\"{\r\n disabled: control.disabled || field.readOnly,\r\n error: control.touched && control.errors,\r\n success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n 'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n 'has-prefix-text': !!field.prefix?.text,\r\n 'pr-10':\r\n (field.icon || field.suffix?.icon) && !field.isCopyText && !field.suffix?.text,\r\n 'has-suffix-text': !!field.suffix?.text,\r\n 'pr-20':\r\n field.isCopyText && !field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n 'pr-30':\r\n field.isCopyText && field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n 'pr-32':\r\n field.isCopyText && field.hasGenerateKey && (field.icon || field.suffix?.icon),\r\n }\"\r\n [ngStyle]=\"{\r\n color: field.valueColor || 'inherit',\r\n 'font-size': field.valueSize || '',\r\n }\"\r\n (focus)=\"onFocus()\"\r\n />\r\n\r\n <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n @if (field.isCopyText) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"copyText($event)\"\r\n (keydown)=\"copyText($event)\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"two-square\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n @if (field.hasGenerateKey) {\r\n <span\r\n class=\"cursor-pointer\"\r\n (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n >\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"round-arrow-top-left\"\r\n data-icon-color=\"#03182b\"\r\n ></div>\r\n </span>\r\n }\r\n }\r\n\r\n @if (field.icon || field.suffix?.icon || field.suffix?.text) {\r\n @if (field.suffix?.text) {\r\n <span\r\n class=\"text-sm font-medium whitespace-nowrap\"\r\n [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n (click)=\"suffixClick($event)\"\r\n (keydown.enter)=\"suffixClick($event)\"\r\n [attr.tabindex]=\"field.suffix?.onClick ? 0 : null\"\r\n [class.cursor-pointer]=\"!!field.suffix?.onClick\"\r\n >{{ field.suffix!.text }}</span\r\n >\r\n }\r\n @if (field.icon || field.suffix?.icon) {\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"field.suffix?.icon || field.icon!.name\"\r\n [attr.data-icon-variant]=\"field.icon?.variant || 'Line'\"\r\n [attr.data-icon-size]=\"field.icon ? getIconSizeName(field.icon.size) : 'medium'\"\r\n [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n ></div>\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n }\r\n </ng-container>\r\n\r\n @if (showFormFieldMessage(control, helperText)) {\r\n <div class=\"fn-field-message-container\">\r\n <span\r\n class=\"fn-field-message-text\"\r\n [ngClass]=\"{\r\n error: isError,\r\n success: isSuccess,\r\n }\"\r\n >\r\n {{ getFieldMessage() | translate }}\r\n </span>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Integrated Toast Notifications -->\r\n <div class=\"toast-container\">\r\n @for (toast of toasts(); track toast.id) {\r\n <div class=\"toast toast-{{ toast.type }}\" [class.toast-enter]=\"toast.isVisible\">\r\n <span class=\"toast-icon\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n [attr.data-icon-name]=\"getToastIconName(toast)\"\r\n data-icon-variant=\"Colour\"\r\n data-icon-size=\"medium\"\r\n [attr.data-icon-color]=\"\r\n toast.type === 'success'\r\n ? '#6cc24a'\r\n : toast.type === 'warn'\r\n ? '#ff9f00'\r\n : toast.type === 'error'\r\n ? '#ef3e42'\r\n : '#03182b'\r\n \"\r\n ></div>\r\n </span>\r\n <div class=\"toast-content\">\r\n <span class=\"toast-message\">\r\n {{ getTranslatedMessage(toast.message) | translate }}\r\n </span>\r\n </div>\r\n <button class=\"toast-close\" (click)=\"removeToast(toast.id)\" aria-label=\"Close\">\r\n <div\r\n #iconContainer\r\n class=\"icon-container\"\r\n data-icon-name=\"cross\"\r\n data-icon-variant=\"Line\"\r\n data-icon-size=\"medium\"\r\n data-icon-color=\"#ffffff\"\r\n ></div>\r\n </button>\r\n </div>\r\n }\r\n </div>\r\n}\r\n", styles: [".fn-input-container{display:flex;flex-direction:column;width:100%}.fn-label{font-family:Outfit,sans-serif;font-weight:500;margin-bottom:8px;display:flex;align-items:center}.fn-label span{color:#c4cdd2;font-size:14px;font-weight:400}.fn-input-field{width:100%;height:48px;padding:0 16px;background-color:#f7f8f9;border-bottom:1.5px solid #c4cdd2;font-family:Outfit,sans-serif;font-size:16px;color:#03182b;transition:all .2s ease;border-radius:4px 4px 0 0}.fn-input-field::placeholder{color:#c4cdd2}.fn-input-field:focus{outline:none;background-color:#fff;border-bottom-color:#03182b}.fn-input-field.error{border-bottom-color:#ef3e42}.fn-input-field.success{border-bottom-color:#6cc24a}.fn-input-field.disabled{background-color:#eef0f2;color:#c4cdd2;cursor:not-allowed}.fn-input-field.pl-10{padding-left:40px!important}.fn-input-field.has-prefix-text{padding-left:100px!important}.fn-input-field.pr-10{padding-right:40px!important}.fn-input-field.has-suffix-text{padding-right:120px!important}.fn-input-field.pr-12{padding-right:44px!important}.fn-input-field.pr-20{padding-right:72px!important}.fn-input-field.pr-24{padding-right:88px!important}.fn-input-field.pr-30{padding-right:104px!important}.fn-input-field.pr-32{padding-right:120px!important}.fn-field-message-container{margin-top:4px}.fn-field-message-text{font-family:Outfit,sans-serif;font-size:12px;color:#c4cdd2}.fn-field-message-text.error{color:#ef3e42}.fn-field-message-text.success{color:#6cc24a}.icon-container{display:inline-flex;align-items:center;justify-content:center;width:24px;height:24px}.relative{position:relative}.absolute{position:absolute}.right-0{right:0}.left-0{left:0}.top-1\\/2{top:50%}.-translate-y-1\\/2{transform:translateY(-50%)}.flex{display:flex}.items-center{align-items:center}.justify-between{justify-content:space-between}.gap-1{gap:.25rem}.gap-2{gap:.5rem}.pr-3{padding-right:.75rem}.pl-3{padding-left:.75rem}.mt-2{margin-top:.5rem}.space-y-1>:not([hidden])~:not([hidden]){margin-top:.25rem}.text-xs{font-size:.75rem}.font-medium{font-weight:500}.h-1\\.5{height:.375rem}.w-full{width:100%}.bg-\\[\\#eef0f2\\]{background-color:#eef0f2}.rounded-full{border-radius:9999px}.overflow-hidden{overflow:hidden}.transition-all{transition-property:all}.duration-300{transition-duration:.3s}.cursor-pointer{cursor:pointer}.opacity-50{opacity:.5}.cursor-not-allowed{cursor:not-allowed}.text-ellipsis{text-overflow:ellipsis}.whitespace-nowrap{white-space:nowrap}.toast-container{position:fixed;top:24px;left:50%;transform:translate(-50%);z-index:9999;width:100%;max-width:336px;display:flex;flex-direction:column;gap:8px;pointer-events:none}.toast{display:flex;align-items:center;background:#25333f;border-radius:8px;padding:12px;width:336px;max-width:100%;opacity:0;transform:translateY(-20px);transition:all .3s cubic-bezier(.4,0,.2,1);pointer-events:auto}.toast.toast-enter{opacity:1;transform:translateY(0)}.toast.toast-success{border-left:4px solid #6cc24a}.toast.toast-warn{border-left:4px solid #ff9f00}.toast.toast-error{border-left:4px solid #ef3e42}.toast.toast-info{border-left:4px solid #3b82f6}.toast .toast-icon{width:24px;height:24px;display:flex;align-items:center;justify-content:center;margin-right:8px;flex-shrink:0}.toast .toast-content{flex:1;min-width:0}.toast .toast-message{font-size:14px;line-height:20px;color:#fff;display:block}.toast .toast-close{background:none;border:none;font-size:20px;color:#fff;opacity:.7;cursor:pointer;padding:0;margin-left:8px;width:24px;height:24px;display:flex;align-items:center;justify-content:center;flex-shrink:0;transition:opacity .2s;pointer-events:auto}.toast .toast-close:hover{opacity:1}\n"] }]
|
|
1169
|
-
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { field: [{
|
|
1170
|
-
type: Input
|
|
1171
|
-
}], helperHandle: [{
|
|
1172
|
-
type: Input
|
|
1173
|
-
}], toastService: [{
|
|
1174
|
-
type: Input
|
|
1175
|
-
}], currencyMeta: [{
|
|
1176
|
-
type: Input
|
|
1177
|
-
}], defaultLocale: [{
|
|
1178
|
-
type: Input
|
|
1179
|
-
}], form: [{
|
|
1180
|
-
type: Input
|
|
1181
|
-
}], valueChange: [{
|
|
1182
|
-
type: Output
|
|
1183
|
-
}], fieldBlur: [{
|
|
1184
|
-
type: Output
|
|
1185
|
-
}], textareaElement: [{
|
|
1186
|
-
type: ViewChild,
|
|
1187
|
-
args: ['fnTextarea']
|
|
1188
|
-
}], iconContainers: [{
|
|
1189
|
-
type: ViewChildren,
|
|
1190
|
-
args: ['iconContainer']
|
|
1191
|
-
}] } });
|
|
1192
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fn-input.component.js","sourceRoot":"","sources":["../../../../projects/fn-input/src/lib/fn-input.component.ts","../../../../projects/fn-input/src/lib/fn-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  AfterViewInit,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  Output,\r\n  signal,\r\n  ViewChild,\r\n  OnDestroy,\r\n  OnChanges,\r\n  SimpleChanges,\r\n  ViewChildren,\r\n  QueryList,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\n// Removed HttpClient import\r\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';\r\nimport { TranslateModule, TranslateService } from '@ngx-translate/core';\r\nimport { Subscription, merge } from 'rxjs';\r\nimport { startWith } from 'rxjs/operators';\r\nimport { FNLabel } from 'fn-label';\r\nimport {\r\n  FNInputBase,\r\n  FNInputHelper,\r\n  FNInputToast,\r\n  DEFAULT_CURRENCY_META,\r\n  TypeIconVariant,\r\n  TypeIconSize,\r\n  FN_TOAST_SERVICE,\r\n} from './fn-input.types';\r\n\r\nexport type TypeIconSizeName = 'extrasmall' | 'small' | 'medium' | 'large' | 'x-large' | 'xxlarge';\r\n\r\nconst CORE_ICONS: Record<string, string> = {\r\n  'eye-open': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2 12s4-8 10-8 10 8 10 8-4 8-10 8-10-8-10-8z\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><circle cx=\"12\" cy=\"12\" r=\"3\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'eye-close': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2 10a11 11 0 0 0 20 0\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M5 13l-1 2M12 14v2M19 13l1 2\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'two-square': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><rect x=\"8\" y=\"4\" width=\"12\" height=\"12\" rx=\"2\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4 8v10a2 2 0 0 0 2 2h10\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'round-arrow-top-left': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M23 4v6h-6M1 20v-6h6M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'check-circle': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"2\"/><path d=\"M8 12l3 3 5-5\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'info-circle': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"16\" x2=\"12\" y2=\"12\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"8\" x2=\"12.01\" y2=\"8\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'alert-circle': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  'exclamation-circle': `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n  cross: `<svg viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M18 6L6 18M6 6l12 12\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>`,\r\n};\r\n\r\n@Component({\r\n  selector: 'fn-input',\r\n  standalone: true,\r\n  imports: [FormsModule, CommonModule, ReactiveFormsModule, TranslateModule, FNLabel],\r\n  templateUrl: './fn-input.component.html',\r\n  styleUrls: ['./fn-input.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n})\r\nexport class FNInput implements OnInit, AfterViewInit, OnDestroy, OnChanges {\r\n  @Input() field = {} as FNInputBase;\r\n  @Input() helperHandle?: FNInputHelper;\r\n  @Input() toastService?: FNInputToast;\r\n  private readonly injectedToastService = inject(FN_TOAST_SERVICE, { optional: true });\r\n\r\n  get effectiveToastService(): FNInputToast | undefined {\r\n    return this.toastService || this.injectedToastService || undefined;\r\n  }\r\n  @Input() currencyMeta = DEFAULT_CURRENCY_META;\r\n  @Input() defaultLocale = 'en-US';\r\n  @Input() form!: FormGroup;\r\n  @Output() valueChange = new EventEmitter();\r\n  @Output() fieldBlur = new EventEmitter();\r\n\r\n  translateService = inject(TranslateService);\r\n  hasFocus = signal(false);\r\n  @ViewChild('fnTextarea') textareaElement!: ElementRef<HTMLTextAreaElement>;\r\n  @ViewChildren('iconContainer') iconContainers!: QueryList<ElementRef<HTMLElement>>;\r\n\r\n  // Internal Toast State\r\n  toasts = signal<any[]>([]);\r\n\r\n  // Removed HttpClient injection\r\n  private readonly subs = new Subscription();\r\n\r\n  private readonly sizeMap: Record<TypeIconSizeName, TypeIconSize> = {\r\n    extrasmall: 10,\r\n    small: 16,\r\n    medium: 20,\r\n    large: 24,\r\n    'x-large': 32,\r\n    xxlarge: 48,\r\n  };\r\n\r\n  getIconSizeName(size: number): 'small' | 'medium' | 'large' | 'x-large' {\r\n    switch (size) {\r\n      case 16:\r\n        return 'small';\r\n      case 20:\r\n        return 'medium';\r\n      case 24:\r\n        return 'large';\r\n      case 32:\r\n        return 'x-large';\r\n      default:\r\n        return 'medium';\r\n    }\r\n  }\r\n\r\n  get isAlphanumeric(): boolean {\r\n    return this.field.isAlphanumeric !== false;\r\n  }\r\n\r\n  get isEmailField(): boolean {\r\n    return this.field.type === 'email' || this.field.name?.toLowerCase().includes('email');\r\n  }\r\n\r\n  private get alphanumericPattern(): RegExp {\r\n    if (this.field.isAddressLine) {\r\n      return /[^A-Za-z0-9 \\-_&(),/]/g;\r\n    }\r\n\r\n    return /[^A-Za-z0-9 \\-_&()]/g;\r\n  }\r\n\r\n  isDisabled = false;\r\n  control!: FormControl;\r\n  isVisible = signal(true); // Track visibility state\r\n  helperText = '';\r\n\r\n  // Use a counter-based approach for unique IDs (safer than Math.random())\r\n  private static idCounter = 0;\r\n  uniqueId = `fn-input-${++FNInput.idCounter}`;\r\n\r\n  constructor(private readonly cdr: ChangeDetectorRef) {}\r\n\r\n  ngOnInit(): void {\r\n    if (this.field.hidden) return;\r\n    this.helperText = this.field.helperText ?? '';\r\n    this.initFormControl();\r\n\r\n    if (this.field.value) this.control.setValue(this.field.value);\r\n    this.setupVisibilityCondition();\r\n    this.setupFieldMessageListener();\r\n    this.setupToastSubscription();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subs.unsubscribe();\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['field'] || changes['form']) {\r\n      this.updateAllIcons();\r\n    }\r\n  }\r\n\r\n  private setupFieldMessageListener(): void {\r\n    if (!this.control) return;\r\n\r\n    const val$ = this.control.valueChanges.pipe(startWith(this.control.value));\r\n    const status$ = this.control.statusChanges.pipe(startWith(this.control.status));\r\n\r\n    this.subs.add(\r\n      merge(val$, status$).subscribe(() => {\r\n        try {\r\n          this.cdr.detectChanges();\r\n        } catch (e) {\r\n          console.warn('CDR detectChanges failed during field message update:', e);\r\n          this.cdr.markForCheck();\r\n        }\r\n      }),\r\n    );\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    // Access the textarea element by ViewChild or native element reference\r\n    this.autoResizeInitial();\r\n    if (this.field.type === 'textarea' && this.control?.value) {\r\n      setTimeout(() => this.autoResizeInitial(), 30);\r\n    }\r\n    // Apply initial formatting after view is initialized\r\n    if (this.field.type === 'number' && this.field.isCurrency && this.control?.value) {\r\n      this.formatInitialCurrencyValue();\r\n    }\r\n    this.setupIconContainersListener();\r\n  }\r\n\r\n  private setupIconContainersListener(): void {\r\n    this.subs.add(\r\n      this.iconContainers.changes.subscribe(() => {\r\n        this.updateAllIcons();\r\n      }),\r\n    );\r\n  }\r\n\r\n  enforceLowercase(event: Event) {\r\n    const input = event.target as HTMLInputElement;\r\n    const start = input.selectionStart;\r\n    const end = input.selectionEnd;\r\n\r\n    input.value = input.value.toLowerCase();\r\n    if (this.control) {\r\n      this.control.setValue(input.value, { emitEvent: false });\r\n    }\r\n    this.valueChange.emit({ name: this.field.name, value: input.value });\r\n\r\n    // restore cursor position so typing is smooth\r\n    input.setSelectionRange(start, end);\r\n  }\r\n\r\n  /**\r\n   * Handle email input - allows only valid email characters and enforces lowercase.\r\n   * Allowed: a-z, 0-9, . _ + - @\r\n   * Rules:\r\n   * - Only one @ allowed\r\n   * - No consecutive dots (..)\r\n   * - Automatic lowercase\r\n   */\r\n  handleEmailInput(event: Event): void {\r\n    const input = event.target as HTMLInputElement;\r\n    const cursorPosition = input.selectionStart || 0;\r\n    const originalValue = input.value;\r\n\r\n    // 1. Convert to lowercase\r\n    const lowerValue = originalValue.toLowerCase();\r\n\r\n    // 2. Split at the first @ to handle local and domain parts separately\r\n    const parts = lowerValue.split('@');\r\n    let localPart = parts[0];\r\n    let domainPart = parts.length > 1 ? parts.slice(1).join('') : null;\r\n\r\n    // 3. Filter local part: a-z, 0-9, ., _, +, -\r\n    // Rule: Cannot start with a dot (.)\r\n    localPart = localPart.replaceAll(/[^a-z0-9._+-]/g, '');\r\n    while (localPart.startsWith('.')) {\r\n      localPart = localPart.substring(1);\r\n    }\r\n    while (localPart.includes('..')) {\r\n      localPart = localPart.replaceAll('..', '.');\r\n    }\r\n\r\n    // 4. Filter domain part: a-z, 0-9, ., -\r\n    // Rule: No underscores (_), no plus signs (+), cannot start with a hyphen (-)\r\n    let filteredValue = localPart;\r\n    if (domainPart !== null) {\r\n      domainPart = domainPart.replaceAll(/[^a-z0-9.-]/g, '');\r\n      while (domainPart.startsWith('-')) {\r\n        domainPart = domainPart.substring(1);\r\n      }\r\n      while (domainPart.includes('..')) {\r\n        domainPart = domainPart.replaceAll('..', '.');\r\n      }\r\n      filteredValue += '@' + domainPart;\r\n    }\r\n\r\n    if (originalValue !== filteredValue) {\r\n      const charsRemovedBeforeCursor = this.countRemovedCharsBeforeCursor(\r\n        originalValue.toLowerCase(),\r\n        filteredValue,\r\n        cursorPosition,\r\n      );\r\n      input.value = filteredValue;\r\n      this.control.setValue(filteredValue, { emitEvent: false });\r\n      const newCursorPos = Math.max(0, cursorPosition - charsRemovedBeforeCursor);\r\n      input.setSelectionRange(newCursorPos, newCursorPos);\r\n    } else if (originalValue !== lowerValue) {\r\n      // If only case changed (e.g., typed Uppercase)\r\n      input.value = filteredValue;\r\n      this.control.setValue(filteredValue, { emitEvent: false });\r\n      input.setSelectionRange(cursorPosition, cursorPosition);\r\n    }\r\n\r\n    if (this.field.type === 'password' && this.field.feedback) {\r\n      this.checkPasswordStrength(filteredValue);\r\n    }\r\n\r\n    this.valueChange.emit({ name: this.field.name, value: filteredValue });\r\n    this.control.markAsTouched();\r\n  }\r\n\r\n  onFocus(): void {\r\n    this.hasFocus.set(true);\r\n  }\r\n\r\n  private formatInitialCurrencyValue(): void {\r\n    const numericValue = Number(this.control.value);\r\n    if (!Number.isNaN(numericValue)) {\r\n      setTimeout(() => {\r\n        const inputElement = document.getElementById(this.uniqueId) as HTMLInputElement;\r\n        if (inputElement) {\r\n          inputElement.value = this.formatCurrencyWithCommas(numericValue);\r\n        }\r\n      }, 50);\r\n    }\r\n  }\r\n\r\n  initFormControl() {\r\n    const controller = this.form.controls?.[this.field.name];\r\n    this.control = controller as FormControl;\r\n    if (this.field.value) this.control.setValue(this.field.value);\r\n\r\n    if (this.control) {\r\n      // Handle disabled state\r\n      if (this.field.disabled ?? false) {\r\n        this.control?.disable();\r\n      } else {\r\n        this.control?.enable();\r\n      }\r\n    }\r\n  }\r\n\r\n  autoResizeInitial() {\r\n    const textarea = this.textareaElement?.nativeElement;\r\n    if (textarea) {\r\n      // Get the number of rows (default to 2 if not specified)\r\n      const rows = this.field.rows || 1;\r\n\r\n      // Calculate line height from computed styles\r\n      const computedStyle = globalThis.getComputedStyle(textarea);\r\n      const lineHeight = Number.parseFloat(computedStyle.lineHeight) || 32; // Default to 24px if not set\r\n\r\n      // Calculate minimum height based on rows\r\n      const minHeight = rows * lineHeight;\r\n\r\n      // Reset height to auto to get accurate scrollHeight\r\n      textarea.style.height = 'auto';\r\n\r\n      // Use the larger of scrollHeight or minHeight\r\n      const newHeight = Math.max(textarea.scrollHeight, minHeight);\r\n      textarea.style.height = newHeight + 'px';\r\n      textarea.style.overflow = 'hidden';\r\n    }\r\n  }\r\n\r\n  handleTextArea(event: Event): void {\r\n    const element = event.target as HTMLTextAreaElement;\r\n    const cursorPosition = element.selectionStart;\r\n    const originalValue = element.value;\r\n\r\n    // Apply alphanumeric filtering if enabled\r\n    if (this.field.type === 'textarea' && this.isAlphanumeric) {\r\n      // Only allow characters based on the selected pattern\r\n      const alphanumericPattern = this.alphanumericPattern;\r\n      let filteredValue = originalValue.replaceAll(alphanumericPattern, '');\r\n\r\n      // Prevent leading spaces\r\n      if (filteredValue.startsWith(' ')) {\r\n        filteredValue = filteredValue.trimStart();\r\n      }\r\n\r\n      // Replace multiple consecutive spaces with single space\r\n      filteredValue = filteredValue.replaceAll(/\\s{2,}/g, ' ');\r\n\r\n      if (originalValue !== filteredValue) {\r\n        // Calculate the number of characters removed before the cursor position\r\n        const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;\r\n        element.value = filteredValue;\r\n        this.control.setValue(filteredValue, { emitEvent: false });\r\n        // Restore cursor position - account for removed characters\r\n        const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);\r\n        element.setSelectionRange(newCursorPos, newCursorPos);\r\n      }\r\n\r\n      // increase height\r\n      this.autoResizeInitial();\r\n      this.valueChange.emit({ name: this.field.name, value: filteredValue });\r\n    } else {\r\n      let filteredValue = originalValue;\r\n\r\n      // Strict filtering if !isAlphanumeric\r\n      if (!this.isAlphanumeric) {\r\n        console.log('isAlphanumeric', this.isAlphanumeric);\r\n\r\n        const allowedPattern = /[^a-zA-Z0-9\\s(){}[\\];,.\"='+\\-*/<>!&|%_@$?:]/g;\r\n        filteredValue = filteredValue.replaceAll(allowedPattern, '');\r\n      }\r\n\r\n      // Prevent leading spaces\r\n      if (filteredValue.startsWith(' ')) {\r\n        filteredValue = filteredValue.trimStart();\r\n      }\r\n\r\n      // Replace multiple consecutive spaces with single space\r\n      filteredValue = filteredValue.replaceAll(/\\s{2,}/g, ' ');\r\n\r\n      if (originalValue !== filteredValue) {\r\n        // Calculate the number of characters removed before the cursor position\r\n        const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;\r\n        element.value = filteredValue;\r\n        this.control.setValue(filteredValue, { emitEvent: false });\r\n        // Restore cursor position - account for removed characters\r\n        const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);\r\n        element.setSelectionRange(newCursorPos, newCursorPos);\r\n      }\r\n\r\n      // increase height\r\n      this.autoResizeInitial();\r\n      this.valueChange.emit({ name: this.field.name, value: filteredValue });\r\n    }\r\n\r\n    this.control.markAsTouched();\r\n  }\r\n\r\n  togglePasswordVisibility(): void {\r\n    this.showPassword = !this.showPassword;\r\n    this.updateAllIcons();\r\n  }\r\n\r\n  handleInput(event: Event): void {\r\n    const target = event.target as HTMLInputElement;\r\n    const cursorPosition = target.selectionStart;\r\n    const originalValue = target.value;\r\n\r\n    let filteredValue = originalValue;\r\n\r\n    // IF NOT alphanumeric, user wants strict filtering:\r\n    // \"only accept tihs.) {} [] ; , . \" ' = + - * / < > ! & | % &_( )-/.;@!, $% : dont allow emoji\"\r\n    // Interpretation: Allow alphanumeric + specified special chars. Ban emojis.\r\n    // Actually, the user said: \"if isAlphanumeric false only accept...\"\r\n    // So if isAlphanumeric is TRUE, we use the existing handleAlphanumericInput logic (or similar).\r\n    // The current logic in handleInput is generic.\r\n    // The existing code has a separate `handleAlphanumericInput` method but it's not called here?\r\n    // Wait, the template likely calls `handleAlphanumericInput` if `isAlphanumeric` is true, or `handleInput` otherwise?\r\n    // Let's check the template.\r\n    // BUT assuming `handleInput` is the generic handler:\r\n\r\n    if (!this.isAlphanumeric) {\r\n      // Regex to allow: a-z, A-Z, 0-9, and: ) {} [] ; , . \" ' = + - * / < > ! & | % _ ( - @ $ ? :\r\n      // Note: User list: ) {} [] ; , . \" ' = + - * / < > ! & | % &_( )-/.;@!, $%\r\n      // Combined unique special chars: ) { } [ ] ; , . \" ' = + - * / < > ! & | % _ ( @ $ ? :\r\n      // Also space is implied? usually yes.\r\n      // And NO EMOJIS.\r\n      // easier to just replace anything NOT in the allowed set.\r\n\r\n      // Allowed chars regex pattern:\r\n      // A-Za-z0-9\r\n      // Space: \\s\r\n      // Special: (){}[\\];,.\"='+\\-*/<>!&|%_@$?:\r\n      // Note: - inside [] needs to be escaped or at end. ] needs escape. \\ needs escape? (not in list).\r\n      // User list has: `.` `\"` `'` `?` (wait, ? not in list?)\r\n      // User list: `) {} [] ; , . \" ' = + - * / < > ! & | % &_( )-/.;@!, $%`\r\n      // Uniques: ) { } [ ] ; , . \" ' = + - * / < > ! & | % _ ( @ $\r\n      // (The `&` is repeated, `.` repeated, `,` repeated etc.)\r\n      // I will create a regex that matches anything NOT in this list and replace with empty string.\r\n\r\n      // [^a-zA-Z0-9\\s(){}[\\];,.\"='+\\-*/<>!&|%_@$]\r\n      // Note: `*`, `+`, `?` etc need escaping in regex if outside [], but inside [] they are literals mostly.\r\n      // `-` needs to be last or escaped. `]` needs escape. `^` needs escape if first? No, invalid here.\r\n      // Let's verify valid chars:\r\n      // a-z A-Z 0-9\r\n      // space\r\n      // ) { } [ ] ; , . \" ' = + - * / < > ! & | % _ ( @ $ :\r\n\r\n      const allowedPattern = /[^a-zA-Z0-9\\s(){}[\\];,.\"='+\\-*/<>!&|%_@$?:]/g;\r\n      filteredValue = filteredValue.replaceAll(allowedPattern, '');\r\n    }\r\n\r\n    // Prevent leading spaces\r\n    if (filteredValue.startsWith(' ')) {\r\n      filteredValue = filteredValue.trimStart();\r\n    }\r\n\r\n    // replaceAll multiple consecutive spaces with single space\r\n    filteredValue = filteredValue.replaceAll(/\\s{2,}/g, ' ');\r\n\r\n    if (originalValue !== filteredValue) {\r\n      // Calculate the number of characters removed before the cursor position\r\n      const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;\r\n      target.value = filteredValue;\r\n      this.control.setValue(filteredValue, { emitEvent: false });\r\n      // Restore cursor position - account for removed characters\r\n      const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);\r\n      target.setSelectionRange(newCursorPos, newCursorPos);\r\n    }\r\n\r\n    // Trailing spaces will be trimmed on blur, not during input\r\n    if (this.field.type === 'password' && this.field.feedback) {\r\n      this.checkPasswordStrength(filteredValue);\r\n    }\r\n\r\n    this.valueChange.emit({ name: this.field.name, value: filteredValue });\r\n    this.control.markAsTouched();\r\n  }\r\n\r\n  /**\r\n   * Handle alphanumeric input - allows only letters, numbers, and specific special characters\r\n   * Allowed: A-Z, a-z, 0-9, space, hyphen, underscore, ampersand, and parentheses\r\n   * Also prevents leading spaces and multiple consecutive spaces\r\n   */\r\n  handleAlphanumericInput(event: Event): void {\r\n    const target = event.target as HTMLInputElement;\r\n    const cursorPosition = target.selectionStart;\r\n    const originalValue = target.value;\r\n\r\n    // Only allow characters based on the selected pattern\r\n    const alphanumericPattern = this.alphanumericPattern;\r\n    let filteredValue = originalValue.replaceAll(alphanumericPattern, '');\r\n\r\n    // Prevent leading spaces\r\n    if (filteredValue.startsWith(' ')) {\r\n      filteredValue = filteredValue.trimStart();\r\n    }\r\n\r\n    // Replace multiple consecutive spaces with single space\r\n    filteredValue = filteredValue.replaceAll(/\\s{2,}/g, ' ');\r\n\r\n    if (originalValue !== filteredValue) {\r\n      // Calculate the number of characters removed before the cursor position\r\n      const charsRemovedBeforeCursor = originalValue.length - filteredValue.length;\r\n      target.value = filteredValue;\r\n      this.control.setValue(filteredValue, { emitEvent: false });\r\n      // Restore cursor position - account for removed characters\r\n      const newCursorPos = Math.max(0, (cursorPosition || 0) - charsRemovedBeforeCursor);\r\n      target.setSelectionRange(newCursorPos, newCursorPos);\r\n    }\r\n\r\n    if (this.field.type === 'password' && this.field.feedback) {\r\n      this.checkPasswordStrength(filteredValue);\r\n    }\r\n\r\n    this.valueChange.emit({ name: this.field.name, value: filteredValue });\r\n    this.control.markAsTouched();\r\n  }\r\n\r\n  handleBlur(e: Event, maxDecimals: number = 2): void {\r\n    this.hasFocus.set(false);\r\n    const target = e.target as HTMLInputElement | HTMLTextAreaElement;\r\n    const cleanValueString = target.value;\r\n\r\n    if (!cleanValueString) {\r\n      this.fieldBlur.emit({ name: this.field.name, value: '' });\r\n      this.cdr.detectChanges();\r\n      return;\r\n    }\r\n\r\n    // Delegate to type-specific handlers\r\n    if (this.field.type === 'text' || this.field.type === 'textarea') {\r\n      this.handleTextFieldBlur(target, cleanValueString);\r\n    } else if (this.field.type === 'number') {\r\n      this.handleNumberFieldBlur(target, cleanValueString, maxDecimals);\r\n    }\r\n\r\n    // Emit final value\r\n    this.fieldBlur.emit({\r\n      name: this.field.name,\r\n      value: target.value,\r\n    });\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /**\r\n   * Handle blur for text and textarea fields - trim trailing spaces\r\n   */\r\n  private handleTextFieldBlur(\r\n    target: HTMLInputElement | HTMLTextAreaElement,\r\n    cleanValueString: string,\r\n  ): void {\r\n    const trimmed = cleanValueString.trim();\r\n    if (cleanValueString !== trimmed) {\r\n      target.value = trimmed;\r\n      this.control.setValue(trimmed, { emitEvent: false });\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle blur for number fields - validate and format\r\n   */\r\n  private handleNumberFieldBlur(\r\n    target: HTMLInputElement | HTMLTextAreaElement,\r\n    cleanValueString: string,\r\n    maxDecimals: number,\r\n  ): void {\r\n    // Remove commas for parsing/validation\r\n    const numString = cleanValueString.replaceAll(',', '');\r\n\r\n    // Check if the string is a valid number format\r\n    const isValidNumberFormat = /^-?\\d+(\\.\\d+)?$/.test(numString);\r\n\r\n    // If not a valid number format, clear the field\r\n    if (!isValidNumberFormat) {\r\n      target.value = '';\r\n      this.control.setValue('');\r\n      return;\r\n    }\r\n\r\n    // Format currency fields or set raw value for non-currency\r\n    if (this.field.type === 'number' && this.field.isCurrency) {\r\n      this.processCurrencyValue(target, numString, maxDecimals);\r\n    } else {\r\n      // Non-currency: preserve the raw string value\r\n      target.value = numString;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Process currency value - handle formatting and decimal places\r\n   */\r\n  private processCurrencyValue(\r\n    target: HTMLInputElement | HTMLTextAreaElement,\r\n    numString: string,\r\n    maxDecimals: number,\r\n  ): void {\r\n    // Handle negative values - convert to positive\r\n    let processedValue = numString.startsWith('-') ? numString.substring(1) : numString;\r\n\r\n    // Handle edge case of \"-0\" or just \"-\"\r\n    if (processedValue === '' || processedValue === '0') {\r\n      processedValue = '0';\r\n    }\r\n\r\n    // Format the value with proper decimal places\r\n    const cleanValueString = this.formatDecimalPlaces(processedValue, maxDecimals);\r\n\r\n    const formattedValue = this.formatCurrencyWithCommas(cleanValueString, maxDecimals);\r\n\r\n    // Update the control value with the numeric string (not formatted)\r\n    this.control.setValue(cleanValueString, { emitEvent: false });\r\n\r\n    // Trigger validators and mark as touched\r\n    this.control.updateValueAndValidity();\r\n    this.control.markAsTouched();\r\n\r\n    // Set display value with commas after Angular's form control update\r\n    setTimeout(() => {\r\n      target.value = formattedValue;\r\n    }, 0);\r\n  }\r\n\r\n  /**\r\n   * Format decimal places - pad or truncate to maxDecimals\r\n   */\r\n  private formatDecimalPlaces(value: string, maxDecimals: number): string {\r\n    let [intPart, decPart] = value.includes('.') ? value.split('.') : [value, ''];\r\n\r\n    // Pad or truncate decimal part to maxDecimals\r\n    if (decPart.length < maxDecimals) {\r\n      decPart = decPart.padEnd(maxDecimals, '0');\r\n    } else if (decPart.length > maxDecimals) {\r\n      decPart = decPart.slice(0, maxDecimals);\r\n    }\r\n\r\n    // Build the clean value string (without commas)\r\n    return maxDecimals > 0 ? `${intPart}.${decPart}` : intPart;\r\n  }\r\n\r\n  handleNumberInput(event: Event, maxDecimals: number = 2): void {\r\n    const target = event.target as HTMLInputElement;\r\n    const originalValue = target.value;\r\n    const cursorPosition = target.selectionStart ?? 0;\r\n\r\n    // Step 1: Filter and format the input value\r\n    let value = this.filterNumericInput(target.value, maxDecimals);\r\n\r\n    // Step 2: Update cursor position if value changed\r\n    this.updateCursorPositionAfterFilter(target, originalValue, value, cursorPosition);\r\n\r\n    // Step 3: Update form control and validate\r\n    this.updateFormControlValue(value, maxDecimals);\r\n\r\n    // Step 4: Emit changes\r\n    this.valueChange.emit({ name: this.field.name, value: target.value });\r\n    this.control.markAsTouched();\r\n  }\r\n\r\n  /**\r\n   * Filter numeric input to only allow digits and decimal point\r\n   * Ensures only one decimal point and enforces max decimal places\r\n   */\r\n  private filterNumericInput(value: string, maxDecimals: number): string {\r\n    // Remove ALL non-numeric characters except decimal point\r\n    let filtered = value.replaceAll(/[^0-9.]/g, '');\r\n\r\n    // Ensure only one decimal point\r\n    filtered = this.filterDecimalPoints(filtered);\r\n\r\n    // Restrict to maxDecimals decimal places if decimal exists\r\n    filtered = this.enforceDecimalLimit(filtered, maxDecimals);\r\n\r\n    return filtered;\r\n  }\r\n\r\n  /**\r\n   * Ensure only one decimal point exists in the value\r\n   */\r\n  private filterDecimalPoints(value: string): string {\r\n    const parts = value.split('.');\r\n    if (parts.length > 2) {\r\n      return parts[0] + '.' + parts.slice(1).join('');\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /**\r\n   * Enforce maximum decimal places limit\r\n   */\r\n  private enforceDecimalLimit(value: string, maxDecimals: number): string {\r\n    if (!value.includes('.')) {\r\n      return value;\r\n    }\r\n\r\n    const [integerPart, decimalPart] = value.split('.');\r\n    if (decimalPart && decimalPart.length > maxDecimals) {\r\n      return integerPart + '.' + decimalPart.slice(0, maxDecimals);\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /**\r\n   * Update cursor position after filtering input\r\n   */\r\n  private updateCursorPositionAfterFilter(\r\n    target: HTMLInputElement,\r\n    originalValue: string,\r\n    newValue: string,\r\n    cursorPosition: number,\r\n  ): void {\r\n    if (target.value === newValue) {\r\n      return;\r\n    }\r\n\r\n    const charsRemovedBeforeCursor = this.countRemovedCharsBeforeCursor(\r\n      originalValue,\r\n      newValue,\r\n      cursorPosition,\r\n    );\r\n    target.value = newValue;\r\n    const newPosition = Math.max(0, cursorPosition - charsRemovedBeforeCursor);\r\n    target.setSelectionRange(newPosition, newPosition);\r\n  }\r\n\r\n  /**\r\n   * Update form control value and perform validation\r\n   */\r\n  private updateFormControlValue(value: string, maxDecimals: number): void {\r\n    this.control.setValue(value, { emitEvent: false });\r\n    this.validateIntegerDigits(value, maxDecimals);\r\n    this.clearNumericError();\r\n  }\r\n\r\n  /**\r\n   * Count how many characters were removed before the cursor position\r\n   * This helps maintain correct cursor position after filtering input\r\n   */\r\n  private countRemovedCharsBeforeCursor(\r\n    original: string,\r\n    filtered: string,\r\n    cursorPos: number,\r\n  ): number {\r\n    let removedCount = 0;\r\n    let filteredIndex = 0;\r\n\r\n    for (let i = 0; i < cursorPos && i < original.length; i++) {\r\n      if (filteredIndex < filtered.length && original[i] === filtered[filteredIndex]) {\r\n        filteredIndex++;\r\n      } else {\r\n        removedCount++;\r\n      }\r\n    }\r\n\r\n    return removedCount;\r\n  }\r\n\r\n  /**\r\n   * Handle paste event for number inputs\r\n   * Prevents pasting of non-numeric characters and enforces digit limits\r\n   */\r\n  handleNumberPaste(event: ClipboardEvent, maxDecimals: number = 2): void {\r\n    event.preventDefault();\r\n    const target = event.target as HTMLInputElement;\r\n    const pastedText = event.clipboardData?.getData('text') || '';\r\n    const isCurrency = this.field.type === 'number' && !!this.field.isCurrency;\r\n\r\n    const filteredText = isCurrency\r\n      ? this.getFilteredCurrencyPastedText(pastedText)\r\n      : this.getFilteredIntegerPastedText(pastedText);\r\n    const finalValue = this.buildFinalPasteValue(target, filteredText, isCurrency, maxDecimals);\r\n\r\n    // Update the input\r\n    target.value = finalValue;\r\n    this.control.setValue(finalValue, { emitEvent: false });\r\n\r\n    this.validateIntegerDigits(finalValue, maxDecimals);\r\n    this.clearNumericError();\r\n\r\n    this.valueChange.emit({ name: this.field.name, value: finalValue });\r\n    this.control.markAsTouched();\r\n\r\n    this.setCursorAfterPaste(target, filteredText, finalValue);\r\n  }\r\n\r\n  /**\r\n   * Filter pasted text for currency input (allows digits and decimal point)\r\n   */\r\n  private getFilteredCurrencyPastedText(pastedText: string): string {\r\n    return pastedText.replaceAll(/[^0-9.]/g, '');\r\n  }\r\n\r\n  /**\r\n   * Filter pasted text for integer-only input (allows digits only)\r\n   */\r\n  private getFilteredIntegerPastedText(pastedText: string): string {\r\n    return pastedText.replaceAll(/\\D/g, '');\r\n  }\r\n\r\n  /**\r\n   * Build final value after paste operation\r\n   */\r\n  private buildFinalPasteValue(\r\n    target: HTMLInputElement,\r\n    filteredText: string,\r\n    isCurrency: boolean,\r\n    maxDecimals: number,\r\n  ): string {\r\n    const currentValue = target.value;\r\n    const start = target.selectionStart;\r\n    const end = target.selectionEnd;\r\n\r\n    let finalValue: string;\r\n\r\n    // If selection API is not supported or no selection, replace entire content\r\n    if (start === null || end === null || (start === 0 && end === 0)) {\r\n      finalValue = filteredText;\r\n    } else {\r\n      finalValue = currentValue.substring(0, start) + filteredText + currentValue.substring(end);\r\n    }\r\n\r\n    return isCurrency ? this.applyDecimalRestrictions(finalValue, maxDecimals) : finalValue;\r\n  }\r\n\r\n  /**\r\n   * Apply decimal restrictions for currency fields\r\n   */\r\n  private applyDecimalRestrictions(value: string, maxDecimals: number): string {\r\n    let result = value;\r\n\r\n    // Ensure only one decimal point\r\n    const parts = result.split('.');\r\n    if (parts.length > 2) {\r\n      result = parts[0] + '.' + parts.slice(1).join('');\r\n    }\r\n\r\n    // Restrict to maxDecimals decimal places\r\n    if (result.includes('.')) {\r\n      const [intPart, decPart] = result.split('.');\r\n      if (decPart.length > maxDecimals) {\r\n        result = intPart + '.' + decPart.slice(0, maxDecimals);\r\n      }\r\n    }\r\n\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Validate integer digit limit and set/clear errors\r\n   */\r\n  private validateIntegerDigits(value: string, maxDecimals: number): void {\r\n    if (this.field.type !== 'number' || !this.field.maxIntegerDigits) {\r\n      return;\r\n    }\r\n\r\n    const maxIntDigits = this.field.maxIntegerDigits ?? 15;\r\n    const [intPart] = value.includes('.') ? value.split('.') : [value];\r\n\r\n    if (intPart.length > maxIntDigits) {\r\n      this.setMaxIntegerDigitsError(intPart.length, maxIntDigits, maxDecimals);\r\n    } else {\r\n      this.clearMaxIntegerDigitsError();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Set max integer digits error\r\n   */\r\n  private setMaxIntegerDigitsError(actual: number, max: number, maxDecimals: number): void {\r\n    const currentErrors = this.control.errors || {};\r\n    this.control.setErrors({\r\n      ...currentErrors,\r\n      maxIntegerDigits: {\r\n        actual,\r\n        max,\r\n        maxDecimals,\r\n        message: `Please shorten your input to ${max} whole digits and ${maxDecimals} decimal or fewer.`,\r\n      },\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Clear max integer digits error if present\r\n   */\r\n  private clearMaxIntegerDigitsError(): void {\r\n    if (!this.control.hasError('maxIntegerDigits')) {\r\n      return;\r\n    }\r\n\r\n    const errors = { ...this.control.errors };\r\n    delete errors['maxIntegerDigits'];\r\n    this.control.setErrors(Object.keys(errors).length ? errors : null);\r\n  }\r\n\r\n  /**\r\n   * Clear numeric error if present\r\n   */\r\n  private clearNumericError(): void {\r\n    if (!this.control.hasError('numeric')) {\r\n      return;\r\n    }\r\n\r\n    const errors = { ...this.control.errors };\r\n    delete errors['numeric'];\r\n    this.control.setErrors(Object.keys(errors).length ? errors : null);\r\n  }\r\n\r\n  /**\r\n   * Set cursor position after paste operation\r\n   */\r\n  private setCursorAfterPaste(\r\n    target: HTMLInputElement,\r\n    filteredText: string,\r\n    finalValue: string,\r\n  ): void {\r\n    // Input type=\"number\" does not support selection APIs\r\n    if (target.type === 'number') {\r\n      return;\r\n    }\r\n\r\n    const start = target.selectionStart || 0;\r\n    const newCursorPos = start + filteredText.length;\r\n    const actualPos = Math.min(newCursorPos, finalValue.length);\r\n    target.setSelectionRange(actualPos, actualPos);\r\n  }\r\n\r\n  // Format number with comma separators\r\n  // Uses string-based formatting for large numbers to preserve precision\r\n  formatCurrencyWithCommas(value: number | string, maxDecimals: number = 2): string {\r\n    if (this.field.type !== 'number' || !this.field.isCurrency) return value.toString();\r\n\r\n    const strValue = value.toString();\r\n\r\n    // Return original value for empty or invalid inputs\r\n    if (!strValue || strValue === 'NaN') return '';\r\n\r\n    // Use string-based formatting to preserve precision for large numbers\r\n    const [integerPart, decimalPart] = strValue.includes('.')\r\n      ? strValue.split('.')\r\n      : [strValue, ''];\r\n\r\n    // Format integer part with comma separators (every 3 digits from right)\r\n    // Using iterative approach instead of regex to avoid ReDoS concerns\r\n    let formattedInteger = '';\r\n    const len = integerPart.length;\r\n    for (let i = 0; i < len; i++) {\r\n      if (i > 0 && (len - i) % 3 === 0) {\r\n        formattedInteger += ',';\r\n      }\r\n      formattedInteger += integerPart[i];\r\n    }\r\n    // Format decimal part (pad or truncate to maxDecimals)\r\n    let formattedDecimal = decimalPart || '';\r\n    if (formattedDecimal.length < maxDecimals) {\r\n      formattedDecimal = formattedDecimal.padEnd(maxDecimals, '0');\r\n    } else if (formattedDecimal.length > maxDecimals) {\r\n      formattedDecimal = formattedDecimal.slice(0, maxDecimals);\r\n    }\r\n\r\n    return maxDecimals > 0 ? `${formattedInteger}.${formattedDecimal}` : formattedInteger;\r\n  }\r\n\r\n  handleNumberKeydown(event: KeyboardEvent, isCurrency: boolean) {\r\n    if (isCurrency) return;\r\n\r\n    // Allow: Backspace, Delete, Tab, Arrow keys\r\n    if (['Backspace', 'Delete', 'Tab', 'ArrowLeft', 'ArrowRight'].includes(event.key)) {\r\n      return;\r\n    }\r\n\r\n    // Allow clipboard operations (Ctrl+V, Ctrl+C, Ctrl+X, Cmd+V, Cmd+C, Cmd+X)\r\n    // and selection (Ctrl+A, Cmd+A)\r\n    if (event.ctrlKey || event.metaKey) {\r\n      return;\r\n    }\r\n\r\n    // Block everything that is not 0-9\r\n    if (!/^\\d$/.test(event.key)) {\r\n      event.preventDefault();\r\n    }\r\n  }\r\n\r\n  // Get locale from currency code using currencyMeta\r\n  getLocaleFromCurrency(currencyCode?: string): string {\r\n    const currencyObj = this.currencyMeta.find(\r\n      (c: { code: string; locale: string }) => c.code === currencyCode,\r\n    );\r\n\r\n    return currencyObj?.locale || this.defaultLocale;\r\n  }\r\n\r\n  showPassword = false;\r\n  passwordStrengthLabel = '';\r\n  strengthClass = '';\r\n  passwordStrengthPercent = 0;\r\n  isPasswordFocused = false;\r\n  onPasswordFocus(): void {\r\n    this.hasFocus.set(true);\r\n    this.isPasswordFocused = true;\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  onPasswordBlur(): void {\r\n    this.isPasswordFocused = false;\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  handlePasswordBlur(e: Event): void {\r\n    this.handleBlur(e); // Original blur functionality (sets hasFocus to false)\r\n    this.onPasswordBlur(); // Hide password feedback\r\n  }\r\n\r\n  checkPasswordStrength(value: string) {\r\n    if (!value || this.field.type !== 'password') {\r\n      this.passwordStrengthLabel = '';\r\n      this.strengthClass = '';\r\n      this.passwordStrengthPercent = 0;\r\n\r\n      return;\r\n    }\r\n    let score = 0;\r\n    if (value.length >= 8) score++;\r\n    if (/[A-Z]/.test(value)) score++;\r\n    if (/\\d/.test(value)) score++;\r\n    if (/[\\W_]/.test(value)) score++;\r\n\r\n    if (score <= 1) {\r\n      this.passwordStrengthLabel = this.field.weakLabel || 'Weak';\r\n      this.strengthClass = 'text-[var(--RHB-Red-100)]';\r\n      this.passwordStrengthPercent = 25;\r\n    } else if (score === 2 || score === 3) {\r\n      this.passwordStrengthLabel = this.field.mediumLabel || 'Medium';\r\n      this.strengthClass = 'text-[var(--Orange-100)]';\r\n      this.passwordStrengthPercent = score === 2 ? 50 : 75;\r\n    } else {\r\n      this.passwordStrengthLabel = this.field.strongLabel || 'Strong';\r\n      this.strengthClass = 'text-[var(--Green-100)]';\r\n      this.passwordStrengthPercent = 100;\r\n    }\r\n  }\r\n\r\n  increment(): void {\r\n    if (this.field.type !== 'number') return;\r\n    const step = this.field.step || 1;\r\n    const current = Number(this.control.value) || 0;\r\n    let nextValue = current + step;\r\n\r\n    if (typeof this.field.max === 'number') {\r\n      nextValue = Math.min(nextValue, this.field.max);\r\n    }\r\n\r\n    this.control.setValue(nextValue);\r\n    this.control.markAsTouched();\r\n    this.valueChange.emit({ name: this.field.name, value: nextValue });\r\n\r\n    // Update display value for currency inputs\r\n    if (this.field.isCurrency) {\r\n      setTimeout(() => {\r\n        const inputElement = document.getElementById(this.uniqueId) as HTMLInputElement;\r\n        if (inputElement) {\r\n          inputElement.value = this.formatCurrencyWithCommas(nextValue);\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  decrement(): void {\r\n    if (this.field.type !== 'number') return;\r\n    const step = this.field.step || 1;\r\n    const current = Number(this.control.value) || 0;\r\n    let nextValue = current - step;\r\n\r\n    // Ensure minimum value is 0 (no negative values)\r\n    const minValue = Math.max(this.field.min || 0, 0);\r\n    nextValue = Math.max(nextValue, minValue);\r\n\r\n    this.control.setValue(nextValue);\r\n    this.control.markAsTouched();\r\n    this.valueChange.emit({ name: this.field.name, value: nextValue });\r\n\r\n    // Update display value for currency inputs\r\n    if (this.field.isCurrency) {\r\n      setTimeout(() => {\r\n        const inputElement = document.getElementById(this.uniqueId) as HTMLInputElement;\r\n        if (inputElement) {\r\n          inputElement.value = this.formatCurrencyWithCommas(nextValue);\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  copyText(e: Event): void {\r\n    e.stopPropagation();\r\n    if (!this.control.value) return;\r\n    navigator.clipboard.writeText(this.control.value).then(() => {\r\n      const service = this.effectiveToastService;\r\n      if (service) {\r\n        // Position set to top as requested\r\n        service.success('Copied to clipboard', 'Success', 2000);\r\n      }\r\n    });\r\n  }\r\n\r\n  prefixClick(e: Event): void {\r\n    if (this.field.prefix?.onClick) {\r\n      this.field.prefix.onClick(e);\r\n    }\r\n  }\r\n\r\n  suffixClick(e: Event): void {\r\n    if (this.field.suffix?.onClick) {\r\n      this.field.suffix.onClick(e);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Set up visibility condition listener\r\n   * Watches the dependent field and shows/hides this field based on conditions\r\n   */\r\n  setupVisibilityCondition(): void {\r\n    if (!this.field.visibilityCondition || !this.form) return;\r\n\r\n    const { dependsOn } = this.field.visibilityCondition;\r\n    const dependentControl = this.form.get(dependsOn);\r\n\r\n    if (!dependentControl) {\r\n      return;\r\n    }\r\n\r\n    // Initial visibility check\r\n    this.updateVisibility(dependentControl.value);\r\n\r\n    // Subscribe to value changes\r\n    dependentControl.valueChanges.subscribe((value) => {\r\n      this.updateVisibility(value);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Update field visibility based on dependent field value\r\n   */\r\n  private updateVisibility(dependentValue: any): void {\r\n    if (!this.field.visibilityCondition) return;\r\n\r\n    const { showWhen, hideWhen } = this.field.visibilityCondition;\r\n    let shouldBeVisible = false;\r\n\r\n    // Check showWhen condition\r\n    if (showWhen !== undefined) {\r\n      if (Array.isArray(showWhen)) {\r\n        shouldBeVisible = showWhen.includes(dependentValue);\r\n      } else {\r\n        shouldBeVisible = dependentValue === showWhen;\r\n      }\r\n    }\r\n\r\n    // Check hideWhen condition (overrides showWhen)\r\n    if (hideWhen !== undefined) {\r\n      const shouldHide = Array.isArray(hideWhen)\r\n        ? hideWhen.includes(dependentValue)\r\n        : dependentValue === hideWhen;\r\n\r\n      if (shouldHide) {\r\n        shouldBeVisible = false;\r\n      }\r\n    }\r\n\r\n    // Update visibility\r\n    this.isVisible.set(shouldBeVisible);\r\n\r\n    // Only clear value when hiding if there's no initial value\r\n    // This preserves data when editing existing records\r\n    if (!shouldBeVisible && this.control && !this.control.value) {\r\n      this.control.setValue(null);\r\n      this.control.markAsUntouched();\r\n      this.control.updateValueAndValidity();\r\n    }\r\n\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  private setupToastSubscription(): void {\r\n    const service = this.effectiveToastService;\r\n    if (service) {\r\n      this.subs.add(\r\n        service.toasts$.subscribe((toasts: any[]) => {\r\n          this.toasts.set(toasts);\r\n          // Wait for DOM to update then render toast icons\r\n          this.updateAllIcons();\r\n        }),\r\n      );\r\n    }\r\n  }\r\n\r\n  removeToast(id: string): void {\r\n    this.effectiveToastService?.remove(id);\r\n  }\r\n\r\n  getTranslatedMessage(message: string): string {\r\n    if (this.helperHandle && typeof this.helperHandle.translate === 'function') {\r\n      return this.helperHandle.translate(message);\r\n    }\r\n    return message;\r\n  }\r\n\r\n  getToastIconName(toast: any): string {\r\n    switch (toast.type) {\r\n      case 'success':\r\n        return 'check-circle';\r\n      case 'warn':\r\n      case 'info':\r\n        return 'info-circle';\r\n      case 'error':\r\n        return 'exclamation-circle';\r\n      default:\r\n        return 'info-circle';\r\n    }\r\n  }\r\n\r\n  showFormFieldMessage(control: any, helperText: string): boolean {\r\n    if (this.helperHandle) {\r\n      return this.helperHandle.showFormFieldMessage(control, helperText);\r\n    }\r\n    const isError = control?.touched && !!control?.errors;\r\n    const isHelper = !isError && Boolean(helperText ?? '');\r\n    return isError || isHelper;\r\n  }\r\n\r\n  getTranslatedLabel(label: string): string {\r\n    if (!label) return '';\r\n    try {\r\n      const labelStr = label.toString();\r\n      return (\r\n        labelStr\r\n          .split('::')\r\n          .map((part) => part.trim())\r\n          .filter((part) => !!part)\r\n          .map((part) => this.translateService.instant(part))\r\n          .join(' ') + this.getEndSymbol(labelStr)\r\n      );\r\n    } catch (e) {\r\n      console.warn('Translation failed in getTranslatedLabel:', label, e);\r\n      return label;\r\n    }\r\n  }\r\n\r\n  getEndSymbol(label: string): string {\r\n    return label.toString().includes('required') && label.toString().includes('FNFieldMessage')\r\n      ? '.'\r\n      : '';\r\n  }\r\n\r\n  // --- ICON LOGIC ---\r\n  private updateAllIcons(): void {\r\n    // We'll call this after ViewInit or when inputs change\r\n    setTimeout(() => {\r\n      this.iconContainers?.forEach((container) => {\r\n        const ds = container.nativeElement.dataset;\r\n        const name = ds['iconName'];\r\n        const variant = (ds['iconVariant'] as TypeIconVariant) || 'Line';\r\n        const size = (ds['iconSize'] || 'medium') as TypeIconSizeName;\r\n        const color = ds['iconColor'];\r\n        const disabled = ds['iconDisabled'] === 'true';\r\n\r\n        if (name) {\r\n          this.loadIconToContainer(container.nativeElement, name, variant, size, color, disabled);\r\n        }\r\n      });\r\n    }, 0);\r\n  }\r\n\r\n  private loadIconToContainer(\r\n    container: HTMLElement,\r\n    name: string,\r\n    variant: TypeIconVariant,\r\n    sizeName: TypeIconSizeName,\r\n    color?: string | null,\r\n    disabled?: boolean,\r\n  ): void {\r\n    const size = this.sizeMap[sizeName];\r\n\r\n    // 1. Check if raw SVG string is provided\r\n    const trimmedName = name.trim();\r\n    if (trimmedName.startsWith('<svg')) {\r\n      this.renderSvgToContainer(container, trimmedName, color, size, 'custom', disabled);\r\n      return;\r\n    }\r\n\r\n    // 1b. Check if it's just raw SVG inner content (like `<path ...>`)\r\n    if (\r\n      trimmedName.startsWith('<path') ||\r\n      trimmedName.startsWith('<g') ||\r\n      trimmedName.startsWith('<circle') ||\r\n      trimmedName.startsWith('<rect') ||\r\n      trimmedName.startsWith('<polygon')\r\n    ) {\r\n      const wrappedSvg = `<svg viewBox=\"0 0 24 24\" fill=\"currentColor\" xmlns=\"http://www.w3.org/2000/svg\">${trimmedName}</svg>`;\r\n      this.renderSvgToContainer(container, wrappedSvg, color, size, 'custom', disabled);\r\n      return;\r\n    }\r\n\r\n    // 2. Check built-in icons registry\r\n    // Strip possible --size suffix if it was passed by mistake from old logic\r\n    const baseName = trimmedName.split('--')[0].trim();\r\n    if (CORE_ICONS[baseName]) {\r\n      this.renderSvgToContainer(container, CORE_ICONS[baseName], color, size, baseName, disabled);\r\n      return;\r\n    }\r\n\r\n    // No fallback to HTTP\r\n    console.warn(`[fn-input] Icon \"${name}\" not found in core registry and no raw SVG provided.`);\r\n    container.innerHTML = '';\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  private renderSvgToContainer(\r\n    container: HTMLElement,\r\n    raw: string,\r\n    color: string | null | undefined,\r\n    size: number,\r\n    name: string,\r\n    disabled?: boolean,\r\n  ): void {\r\n    try {\r\n      const parser = new DOMParser();\r\n      const doc = parser.parseFromString(raw, 'image/svg+xml');\r\n      const svg = doc.querySelector('svg') as unknown as SVGSVGElement;\r\n\r\n      if (!svg) {\r\n        container.innerHTML = '';\r\n        return;\r\n      }\r\n\r\n      // SVG Normalization logic\r\n      for (const s of svg.querySelectorAll('style')) s.remove();\r\n\r\n      if (color) {\r\n        for (const el of svg.querySelectorAll<SVGElement>('*')) {\r\n          const style = el.getAttribute('style');\r\n          if (style) {\r\n            const cleaned = style\r\n              .replaceAll(/fill\\s*:\\s*[^;]+;?/gi, '')\r\n              .replaceAll(/stroke\\s*:\\s*[^;]+;?/gi, '');\r\n            if (cleaned) el.setAttribute('style', cleaned);\r\n            else el.removeAttribute('style');\r\n          }\r\n\r\n          const fill = el.getAttribute('fill');\r\n          if (fill && fill !== 'none' && !fill.startsWith('url(')) {\r\n            el.setAttribute('fill', 'currentColor');\r\n          }\r\n\r\n          const stroke = el.getAttribute('stroke');\r\n          if (stroke && stroke !== 'none' && !stroke.startsWith('url(')) {\r\n            el.setAttribute('stroke', 'currentColor');\r\n          }\r\n        }\r\n      }\r\n\r\n      svg.removeAttribute('width');\r\n      svg.removeAttribute('height');\r\n      if (!svg.getAttribute('viewBox')) {\r\n        svg.setAttribute('viewBox', `0 0 ${size} ${size}`);\r\n      }\r\n      svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');\r\n      svg.setAttribute('focusable', 'false');\r\n      svg.setAttribute('aria-hidden', 'true');\r\n      svg.setAttribute('width', '100%');\r\n      svg.setAttribute('height', '100%');\r\n\r\n      const style = `;display:block;max-width:100%;max-height:100%;`;\r\n      svg.setAttribute('style', (svg.getAttribute('style') || '') + style);\r\n\r\n      container.innerHTML = '';\r\n      if (color) {\r\n        container.style.color = color;\r\n      }\r\n      if (disabled) {\r\n        container.style.opacity = '0.5';\r\n        container.style.cursor = 'not-allowed';\r\n      } else {\r\n        container.style.opacity = '1';\r\n        container.style.cursor = '';\r\n      }\r\n      container.appendChild(svg);\r\n      this.cdr.detectChanges();\r\n    } catch (error) {\r\n      console.error(`[fn-input] Error parsing SVG for icon \"${name}\":`, error);\r\n      container.innerHTML = '';\r\n    }\r\n  }\r\n\r\n  // Removed getIconPath as HTTP loading is disabled\r\n\r\n  // --- MESSAGE LOGIC ---\r\n  getFieldMessage(): string {\r\n    if (!this.control) return this.field?.helperText || '';\r\n\r\n    if (this.control.touched && this.control.errors) {\r\n      for (const key of Object.keys(this.control.errors)) {\r\n        const errorValue = this.control.errors[key];\r\n        if (errorValue && typeof errorValue === 'object' && errorValue.message) {\r\n          return errorValue.message;\r\n        }\r\n        if (this.field?.errors?.[key]) {\r\n          return this.field.errors[key];\r\n        }\r\n      }\r\n      return (\r\n        this.field?.errors?.['default'] ||\r\n        'Please enter ' + this.translateService.instant(this.field?.label)\r\n      );\r\n    }\r\n\r\n    return this.field?.helperText || '';\r\n  }\r\n\r\n  get isError(): boolean {\r\n    return !!(this.control?.touched && this.control?.errors);\r\n  }\r\n\r\n  get isSuccess(): boolean {\r\n    return !!(this.control?.valid && this.control?.touched);\r\n  }\r\n}\r\n","@if (field.name && isVisible() && !field.hidden) {\r\n  <div class=\"fn-input-container\">\r\n    @if (field.type! !== 'hidden') {\r\n      <fn-label\r\n        [for]=\"uniqueId\"\r\n        [label]=\"field.label\"\r\n        [required]=\"field.required || false\"\r\n        [hideOptional]=\"field.hideOptional || false\"\r\n        [color]=\"field.color || '#03182b'\"\r\n        [variant]=\"field.labelVariant || 'p1'\"\r\n        [statusLabel]=\"field.statusLabel\"\r\n      ></fn-label>\r\n    }\r\n\r\n    <ng-container>\r\n      @switch (field.type) {\r\n        <!-- Textarea Field -->\r\n        @case ('textarea') {\r\n          <textarea\r\n            #fnTextarea\r\n            style=\"resize: none\"\r\n            [name]=\"field.name\"\r\n            [id]=\"uniqueId\"\r\n            [required]=\"field.required || false\"\r\n            [placeholder]=\"field.placeholder || '' | translate\"\r\n            [disabled]=\"field.disabled || false\"\r\n            [rows]=\"field.rows || 1\"\r\n            [readonly]=\"field.readOnly || false\"\r\n            [formControl]=\"control\"\r\n            (input)=\"handleTextArea($event)\"\r\n            (blur)=\"handleBlur($event)\"\r\n            class=\"fn-input-field\"\r\n            [ngClass]=\"{\r\n              disabled: control.disabled || field.readOnly,\r\n              error: control.touched && control.errors,\r\n              success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n            }\"\r\n            [ngStyle]=\"{\r\n              color: field.valueColor || 'inherit',\r\n              'font-size': field.valueSize || '',\r\n            }\"\r\n            (focus)=\"onFocus()\"\r\n          ></textarea>\r\n        }\r\n\r\n        <!-- Password Field -->\r\n        @case ('password') {\r\n          <div class=\"relative w-full\">\r\n            @if (field.prefix?.icon || field.prefix?.text) {\r\n              <div\r\n                class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n                style=\"z-index: 1\"\r\n              >\r\n                @if (field.prefix?.icon) {\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"field.prefix!.icon\"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"medium\"\r\n                    [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n                  ></div>\r\n                }\r\n                @if (field.prefix?.text) {\r\n                  <span\r\n                    class=\"text-sm font-medium whitespace-nowrap\"\r\n                    [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n                    (click)=\"prefixClick($event)\"\r\n                    (keydown.enter)=\"prefixClick($event)\"\r\n                    [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n                    [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n                    >{{ field.prefix!.text }}</span\r\n                  >\r\n                }\r\n              </div>\r\n            }\r\n            <input\r\n              [type]=\"showPassword ? 'text' : 'password'\"\r\n              [id]=\"uniqueId\"\r\n              [required]=\"field.required || false\"\r\n              [disabled]=\"field.disabled || false\"\r\n              [formControl]=\"control\"\r\n              [placeholder]=\"field.placeholder || '' | translate\"\r\n              [readonly]=\"field.readOnly || false\"\r\n              (input)=\"isAlphanumeric ? handleAlphanumericInput($event) : handleInput($event)\"\r\n              (focus)=\"onPasswordFocus()\"\r\n              (blur)=\"handlePasswordBlur($event)\"\r\n              class=\"fn-input-field overflow-hidden text-ellipsis\"\r\n              [ngClass]=\"{\r\n                disabled: control.disabled || field.readOnly,\r\n                error: control.touched && control.errors,\r\n                success: field['hasSuccessBorder'] && control.valid,\r\n                'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n                'has-prefix-text': !!field.prefix?.text,\r\n                'pr-12':\r\n                  (field.toggleMask && !field.isCopyText) ||\r\n                  (!field.toggleMask && field.isCopyText),\r\n                'pr-24': field.toggleMask && field.isCopyText && !field.hasGenerateKey,\r\n                'pr-32': field.toggleMask && field.isCopyText && field.hasGenerateKey,\r\n              }\"\r\n              [ngStyle]=\"{\r\n                color: field.valueColor || 'inherit',\r\n                'font-size': field.valueSize || '',\r\n              }\"\r\n            />\r\n\r\n            <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n              @if (field.toggleMask && !field.isCopyText) {\r\n                <span\r\n                  class=\"cursor-pointer\"\r\n                  (click)=\"togglePasswordVisibility()\"\r\n                  (keydown)=\"togglePasswordVisibility()\"\r\n                >\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"large\"\r\n                    data-icon-color=\"#03182b\"\r\n                  ></div>\r\n                </span>\r\n              } @else if (field.isCopyText && !field.toggleMask) {\r\n                <span\r\n                  class=\"cursor-pointer\"\r\n                  (click)=\"copyText($event)\"\r\n                  (keydown)=\"copyText($event)\"\r\n                >\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    data-icon-name=\"two-square\"\r\n                    data-icon-color=\"#03182b\"\r\n                  ></div>\r\n                </span>\r\n                @if (field.hasGenerateKey) {\r\n                  <span\r\n                    class=\"cursor-pointer\"\r\n                    (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                    (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                  >\r\n                    <div\r\n                      #iconContainer\r\n                      class=\"icon-container\"\r\n                      data-icon-name=\"round-arrow-top-left\"\r\n                      data-icon-variant=\"Line\"\r\n                      data-icon-size=\"large\"\r\n                      data-icon-color=\"#03182b\"\r\n                    ></div>\r\n                  </span>\r\n                }\r\n              } @else if (field.toggleMask && field.isCopyText) {\r\n                <span\r\n                  class=\"cursor-pointer\"\r\n                  (click)=\"togglePasswordVisibility()\"\r\n                  (keydown)=\"togglePasswordVisibility()\"\r\n                >\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"showPassword ? 'eye-open' : 'eye-close'\"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"large\"\r\n                    data-icon-color=\"#03182b\"\r\n                  ></div>\r\n                </span>\r\n                <span\r\n                  class=\"cursor-pointer\"\r\n                  (click)=\"copyText($event)\"\r\n                  (keydown)=\"copyText($event)\"\r\n                >\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    data-icon-name=\"two-square\"\r\n                    data-icon-color=\"#03182b\"\r\n                  ></div>\r\n                </span>\r\n                @if (field.hasGenerateKey) {\r\n                  <span\r\n                    class=\"cursor-pointer\"\r\n                    (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                    (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                    [ngClass]=\"{ 'opacity-50 cursor-not-allowed': !control.value }\"\r\n                  >\r\n                    <div\r\n                      #iconContainer\r\n                      class=\"icon-container\"\r\n                      data-icon-name=\"round-arrow-top-left\"\r\n                      data-icon-color=\"#03182b\"\r\n                    ></div>\r\n                  </span>\r\n                }\r\n              }\r\n            </div>\r\n          </div>\r\n\r\n          @if (field.feedback && isPasswordFocused) {\r\n            <div class=\"mt-2 space-y-1\">\r\n              <div class=\"flex items-center justify-between text-xs\">\r\n                <span class=\"font-medium\" [ngClass]=\"strengthClass\">\r\n                  {{ passwordStrengthLabel | translate }}\r\n                </span>\r\n                <div class=\"flex items-center gap-1\">\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"\r\n                      passwordStrengthPercent > 50 ? 'check-circle' : 'info-circle'\r\n                    \"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"extrasmall\"\r\n                    [attr.data-icon-color]=\"\r\n                      passwordStrengthPercent <= 25\r\n                        ? '#ef3e42'\r\n                        : passwordStrengthPercent <= 75\r\n                          ? '#ff9f00'\r\n                          : '#6cc24a'\r\n                    \"\r\n                  ></div>\r\n                </div>\r\n              </div>\r\n              <div class=\"h-1.5 w-full bg-[#eef0f2] rounded-full overflow-hidden\">\r\n                <div\r\n                  class=\"h-full transition-all duration-300 rounded-full\"\r\n                  [ngClass]=\"{\r\n                    'bg-[#ef3e42]': passwordStrengthPercent <= 25,\r\n                    'bg-[#ff9f00]': passwordStrengthPercent > 25 && passwordStrengthPercent <= 75,\r\n                    'bg-[#6cc24a]': passwordStrengthPercent > 75,\r\n                  }\"\r\n                  [style.width.%]=\"passwordStrengthPercent\"\r\n                ></div>\r\n              </div>\r\n            </div>\r\n          }\r\n        }\r\n\r\n        <!-- Number Field -->\r\n        @case ('number') {\r\n          <div class=\"relative w-full\">\r\n            @if (field.prefix?.icon || field.prefix?.text) {\r\n              <div\r\n                class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n                style=\"z-index: 1\"\r\n              >\r\n                @if (field.prefix?.icon) {\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"field.prefix!.icon\"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"medium\"\r\n                    [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n                  ></div>\r\n                }\r\n                @if (field.prefix?.text) {\r\n                  <span\r\n                    class=\"text-sm font-medium whitespace-nowrap\"\r\n                    [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n                    (click)=\"prefixClick($event)\"\r\n                    (keydown.enter)=\"prefixClick($event)\"\r\n                    [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n                    [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n                    >{{ field.prefix!.text }}</span\r\n                  >\r\n                }\r\n              </div>\r\n            }\r\n            <input\r\n              [type]=\"field.type === 'number' && field.isCurrency ? 'text' : 'number'\"\r\n              [name]=\"field.name\"\r\n              [id]=\"uniqueId\"\r\n              [required]=\"field.required || false\"\r\n              [placeholder]=\"field.placeholder || '' | translate\"\r\n              [disabled]=\"isDisabled || false\"\r\n              [readOnly]=\"field.readOnly || false\"\r\n              [formControl]=\"control\"\r\n              [min]=\"field.type === 'number' && !field.isCurrency ? 0 : null\"\r\n              [max]=\"field.type === 'number' && !field.isCurrency ? field.max : null\"\r\n              [step]=\"field.type === 'number' && !field.isCurrency ? field.step || 1 : null\"\r\n              (input)=\"handleNumberInput($event, field.minFractionDigits || 2)\"\r\n              (paste)=\"handleNumberPaste($event, field.minFractionDigits || 2)\"\r\n              (blur)=\"handleBlur($event, field.minFractionDigits || 2)\"\r\n              (keydown)=\"handleNumberKeydown($event, field.isCurrency || false)\"\r\n              class=\"fn-input-field\"\r\n              [ngClass]=\"{\r\n                disabled: control.disabled || field.readOnly,\r\n                error: control.touched && control.errors,\r\n                success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n                'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n                'has-prefix-text': !!field.prefix?.text,\r\n                'pr-10': field.suffix?.icon && !field.suffix?.text,\r\n                'has-suffix-text': !!field.suffix?.text,\r\n                'text-[24px] font-bold': field.isCurrency,\r\n              }\"\r\n              [ngStyle]=\"{\r\n                color: field.valueColor || 'inherit',\r\n                'font-size': field.valueSize || '',\r\n              }\"\r\n              (focus)=\"onFocus()\"\r\n            />\r\n          </div>\r\n        }\r\n\r\n        <!-- Default Input Field (text, email, etc.) -->\r\n        @default {\r\n          <div class=\"relative z-0\">\r\n            @if (field.prefix?.icon || field.prefix?.text) {\r\n              <div\r\n                class=\"absolute left-0 top-1/2 -translate-y-1/2 pl-3 flex items-center gap-2\"\r\n                style=\"z-index: 1\"\r\n              >\r\n                @if (field.prefix?.icon) {\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"field.prefix!.icon\"\r\n                    data-icon-variant=\"Line\"\r\n                    data-icon-size=\"medium\"\r\n                    [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n                  ></div>\r\n                }\r\n                @if (field.prefix?.text) {\r\n                  <span\r\n                    class=\"text-sm font-medium whitespace-nowrap\"\r\n                    [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n                    (click)=\"prefixClick($event)\"\r\n                    (keydown.enter)=\"prefixClick($event)\"\r\n                    [attr.tabindex]=\"field.prefix?.onClick ? 0 : null\"\r\n                    [class.cursor-pointer]=\"!!field.prefix?.onClick\"\r\n                    >{{ field.prefix!.text }}</span\r\n                  >\r\n                }\r\n              </div>\r\n            }\r\n            <input\r\n              [type]=\"field.type\"\r\n              [name]=\"field.name\"\r\n              [id]=\"uniqueId\"\r\n              [required]=\"field.required || false\"\r\n              [placeholder]=\"field.placeholder || '' | translate\"\r\n              [disabled]=\"field.disabled || false\"\r\n              [readOnly]=\"field.readOnly || false\"\r\n              [formControl]=\"control\"\r\n              (input)=\"\r\n                isEmailField\r\n                  ? handleEmailInput($event)\r\n                  : field.type === 'text' && isAlphanumeric\r\n                    ? handleAlphanumericInput($event)\r\n                    : handleInput($event)\r\n              \"\r\n              (blur)=\"handleBlur($event)\"\r\n              class=\"fn-input-field\"\r\n              [ngClass]=\"{\r\n                disabled: control.disabled || field.readOnly,\r\n                error: control.touched && control.errors,\r\n                success: field['hasSuccessBorder'] && control.valid && !control.errors,\r\n                'pl-10': field.prefix?.icon && !field.prefix?.text,\r\n                'has-prefix-text': !!field.prefix?.text,\r\n                'pr-10':\r\n                  (field.icon || field.suffix?.icon) && !field.isCopyText && !field.suffix?.text,\r\n                'has-suffix-text': !!field.suffix?.text,\r\n                'pr-20':\r\n                  field.isCopyText && !field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n                'pr-30':\r\n                  field.isCopyText && field.hasGenerateKey && !(field.icon || field.suffix?.icon),\r\n                'pr-32':\r\n                  field.isCopyText && field.hasGenerateKey && (field.icon || field.suffix?.icon),\r\n              }\"\r\n              [ngStyle]=\"{\r\n                color: field.valueColor || 'inherit',\r\n                'font-size': field.valueSize || '',\r\n              }\"\r\n              (focus)=\"onFocus()\"\r\n            />\r\n\r\n            <div class=\"absolute right-0 top-1/2 -translate-y-1/2 flex items-center pr-3 gap-2\">\r\n              @if (field.isCopyText) {\r\n                <span\r\n                  class=\"cursor-pointer\"\r\n                  (click)=\"copyText($event)\"\r\n                  (keydown)=\"copyText($event)\"\r\n                >\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    data-icon-name=\"two-square\"\r\n                    data-icon-color=\"#03182b\"\r\n                  ></div>\r\n                </span>\r\n                @if (field.hasGenerateKey) {\r\n                  <span\r\n                    class=\"cursor-pointer\"\r\n                    (click)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                    (keydown)=\"field.onGenerateKey && field.onGenerateKey()\"\r\n                  >\r\n                    <div\r\n                      #iconContainer\r\n                      class=\"icon-container\"\r\n                      data-icon-name=\"round-arrow-top-left\"\r\n                      data-icon-color=\"#03182b\"\r\n                    ></div>\r\n                  </span>\r\n                }\r\n              }\r\n\r\n              @if (field.icon || field.suffix?.icon || field.suffix?.text) {\r\n                @if (field.suffix?.text) {\r\n                  <span\r\n                    class=\"text-sm font-medium whitespace-nowrap\"\r\n                    [ngStyle]=\"{ color: control.disabled ? '#c4cdd2' : '#03182b' }\"\r\n                    (click)=\"suffixClick($event)\"\r\n                    (keydown.enter)=\"suffixClick($event)\"\r\n                    [attr.tabindex]=\"field.suffix?.onClick ? 0 : null\"\r\n                    [class.cursor-pointer]=\"!!field.suffix?.onClick\"\r\n                    >{{ field.suffix!.text }}</span\r\n                  >\r\n                }\r\n                @if (field.icon || field.suffix?.icon) {\r\n                  <div\r\n                    #iconContainer\r\n                    class=\"icon-container\"\r\n                    [attr.data-icon-name]=\"field.suffix?.icon || field.icon!.name\"\r\n                    [attr.data-icon-variant]=\"field.icon?.variant || 'Line'\"\r\n                    [attr.data-icon-size]=\"field.icon ? getIconSizeName(field.icon.size) : 'medium'\"\r\n                    [attr.data-icon-color]=\"control.disabled ? '#c4cdd2' : '#03182b'\"\r\n                  ></div>\r\n                }\r\n              }\r\n            </div>\r\n          </div>\r\n        }\r\n      }\r\n    </ng-container>\r\n\r\n    @if (showFormFieldMessage(control, helperText)) {\r\n      <div class=\"fn-field-message-container\">\r\n        <span\r\n          class=\"fn-field-message-text\"\r\n          [ngClass]=\"{\r\n            error: isError,\r\n            success: isSuccess,\r\n          }\"\r\n        >\r\n          {{ getFieldMessage() | translate }}\r\n        </span>\r\n      </div>\r\n    }\r\n  </div>\r\n\r\n  <!-- Integrated Toast Notifications -->\r\n  <div class=\"toast-container\">\r\n    @for (toast of toasts(); track toast.id) {\r\n      <div class=\"toast toast-{{ toast.type }}\" [class.toast-enter]=\"toast.isVisible\">\r\n        <span class=\"toast-icon\">\r\n          <div\r\n            #iconContainer\r\n            class=\"icon-container\"\r\n            [attr.data-icon-name]=\"getToastIconName(toast)\"\r\n            data-icon-variant=\"Colour\"\r\n            data-icon-size=\"medium\"\r\n            [attr.data-icon-color]=\"\r\n              toast.type === 'success'\r\n                ? '#6cc24a'\r\n                : toast.type === 'warn'\r\n                  ? '#ff9f00'\r\n                  : toast.type === 'error'\r\n                    ? '#ef3e42'\r\n                    : '#03182b'\r\n            \"\r\n          ></div>\r\n        </span>\r\n        <div class=\"toast-content\">\r\n          <span class=\"toast-message\">\r\n            {{ getTranslatedMessage(toast.message) | translate }}\r\n          </span>\r\n        </div>\r\n        <button class=\"toast-close\" (click)=\"removeToast(toast.id)\" aria-label=\"Close\">\r\n          <div\r\n            #iconContainer\r\n            class=\"icon-container\"\r\n            data-icon-name=\"cross\"\r\n            data-icon-variant=\"Line\"\r\n            data-icon-size=\"medium\"\r\n            data-icon-color=\"#ffffff\"\r\n          ></div>\r\n        </button>\r\n      </div>\r\n    }\r\n  </div>\r\n}\r\n"]}
|