@fagon/ngx-intellitoolx 16.0.3 → 16.0.5

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.
Files changed (26) hide show
  1. package/README.md +277 -7
  2. package/esm2022/lib/form-changes/can-component-deactivate.interface.mjs +2 -0
  3. package/esm2022/lib/form-changes/confirm-handler.type.mjs +2 -0
  4. package/esm2022/lib/form-changes/form-changes-tracker.service.mjs +47 -0
  5. package/esm2022/lib/form-changes/form-update-message-config.interface.mjs +2 -0
  6. package/esm2022/lib/form-changes/form-update-message.component.mjs +53 -0
  7. package/esm2022/lib/form-changes/unsaved-changes.guard.mjs +29 -0
  8. package/esm2022/lib/form-errors/form-errors.component.mjs +3 -3
  9. package/esm2022/lib/helpers/intellitoolx.helper.mjs +2 -3
  10. package/esm2022/public-api.mjs +15 -14
  11. package/fesm2022/fagon-ngx-intellitoolx.mjs +188 -136
  12. package/fesm2022/fagon-ngx-intellitoolx.mjs.map +1 -1
  13. package/lib/form-changes/can-component-deactivate.interface.d.ts +5 -0
  14. package/lib/form-changes/form-changes-tracker.service.d.ts +12 -0
  15. package/lib/{form-update → form-changes}/unsaved-changes.guard.d.ts +5 -5
  16. package/lib/form-errors/form-errors.component.d.ts +1 -1
  17. package/lib/helpers/intellitoolx.helper.d.ts +1 -1
  18. package/package.json +1 -1
  19. package/public-api.d.ts +11 -9
  20. package/esm2022/lib/form-update/confirm-handler.type.mjs +0 -2
  21. package/esm2022/lib/form-update/form-update-message-config.interface.mjs +0 -2
  22. package/esm2022/lib/form-update/form-update-message.component.mjs +0 -53
  23. package/esm2022/lib/form-update/unsaved-changes.guard.mjs +0 -19
  24. /package/lib/{form-update → form-changes}/confirm-handler.type.d.ts +0 -0
  25. /package/lib/{form-update → form-changes}/form-update-message-config.interface.d.ts +0 -0
  26. /package/lib/{form-update → form-changes}/form-update-message.component.d.ts +0 -0
package/README.md CHANGED
@@ -1,10 +1,15 @@
1
1
  # IntelliToolx
2
2
 
3
+ ![npm version](https://img.shields.io/npm/v/@fagon/ngx-intellittolx.svg)
4
+ ![npm downloads](https://img.shields.io/npm/dm/@fagon/ngx-intellittolx.svg)
5
+ ![bundle size](https://img.shields.io/bundlephobia/minzip/@fagon/ngx-intellittolx)
3
6
  ![Angular](https://img.shields.io/badge/Angular-14%2B-red)
4
- ![TypeScript](https://img.shields.io/badge/TypeScript-Ready-blue)
5
- ![License](https://img.shields.io/badge/License-MIT-green)
7
+ ![TypeScript](https://img.shields.io/badge/TypeScript-ready-blue)
8
+ ![License](https://img.shields.io/npm/l/@fagon/ngx-chronox)
6
9
  ![Reactive Forms](https://img.shields.io/badge/Reactive%20Forms-Supported-orange)
7
10
  ![Standalone Components](https://img.shields.io/badge/Standalone-Compatible-purple)
11
+ ![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)
12
+ ![Made with Love](https://img.shields.io/badge/made%20with-❤-red)
8
13
 
9
14
  ## Overview
10
15
 
@@ -17,7 +22,7 @@ Built with scalability, accessibility, and maintainability in mind, IntelliToolx
17
22
  ### Installation
18
23
 
19
24
  ```bash
20
- npm install intellitoolx
25
+ npm intall @fagon/ngx-intellitoolx
21
26
  ```
22
27
 
23
28
  # IntelliToolxHelper
@@ -562,7 +567,7 @@ The component includes built-in messages for common validators:
562
567
  | maxMonthYear | Date is later than allowed |
563
568
  | minMonthYear | Date is earlier than allowed |
564
569
  | exceededAllowedDateDifference | Date difference can only be one month |
565
- | exceededLeastDateAllowed | Start date cannot be greater than end date |
570
+ | startDateAfterEndDate | Start date cannot be greater than end date |
566
571
 
567
572
  ### Adding Control Labels (User-Friendly Messages)
568
573
 
@@ -1124,7 +1129,7 @@ These validators return error keys compatible with the error component:
1124
1129
  | passwordMismatchValidator | passwordMismatch |
1125
1130
  | uniqueEmailsValidator | duplicateEmail |
1126
1131
 
1127
- Best Practices
1132
+ ** Best Practices **
1128
1133
 
1129
1134
  - Use with Reactive Forms only
1130
1135
  - Combine with required validators when necessary
@@ -1141,5 +1146,270 @@ this.form = new FormGroup({
1141
1146
  });
1142
1147
  ```
1143
1148
 
1144
- License
1145
- MIT
1149
+ # Unsaved Changes Protection
1150
+
1151
+ Protect Angular routes and modals from accidental navigation or closing when there are unsaved changes.
1152
+
1153
+ Supports:
1154
+
1155
+ - Angular route navigation
1156
+ - Browser back button
1157
+ - Browser refresh / tab close
1158
+ - Modal close (NG Bootstrap / Material / custom)
1159
+ - Parent + child component setups
1160
+ - Nested forms
1161
+
1162
+ ** The system is built around three core pieces: **
1163
+
1164
+ - UnsavedChangesTrackerService
1165
+ - UnsavedChangesGuard
1166
+ - CanComponentDeactivate interface (for route protection)
1167
+
1168
+ ** Architecture: **
1169
+
1170
+ ```
1171
+ Form → TrackerService → Guard → Confirmation → Continue / Block
1172
+ ```
1173
+
1174
+ The library does NOT depend on your UI layer.
1175
+ Your app controls the confirmation modal.
1176
+
1177
+ ### Core Exports
1178
+
1179
+ ```ts
1180
+ import { UnsavedChangesTrackerService, UnsavedChangesGuard, CanComponentDeactivate } from "ngx-intellitoolx";
1181
+ ```
1182
+
1183
+ ### Protecting Angular Routes
1184
+
1185
+ Add Guard to Route
1186
+
1187
+ ```ts
1188
+ {
1189
+ path: 'add-item',
1190
+ component: AddItemPageComponent,
1191
+ canDeactivate: [UnsavedChangesGuard]
1192
+ }
1193
+ ```
1194
+
1195
+ ### Implement Route Component
1196
+
1197
+ Your routed component must implement CanComponentDeactivate.
1198
+
1199
+ ```ts
1200
+ import { UnsavedChangesTrackerService } from "ngx-intellitoolx";
1201
+ import { UtilityService } from "../services/utility.service";
1202
+
1203
+ @Component({
1204
+ selector: "app-add-item-page",
1205
+ template: `<app-item-form></app-item-form>`,
1206
+ })
1207
+ export class AddItemPageComponent implements CanComponentDeactivate {
1208
+ constructor(
1209
+ private tracker: UnsavedChangesTrackerService,
1210
+ private utils: UtilityService,
1211
+ ) {}
1212
+
1213
+ confirmUnsavedChanges(): Promise<boolean> {
1214
+ return this.utils.confirmUnsavedChanges();
1215
+ }
1216
+ }
1217
+ ```
1218
+
1219
+ ### Track Form Changes (Child Component)
1220
+
1221
+ ```ts
1222
+ @Component({
1223
+ selector: "app-item-form",
1224
+ templateUrl: "./item-form.component.html",
1225
+ })
1226
+ export class ItemFormComponent implements OnInit {
1227
+ form!: FormGroup;
1228
+
1229
+ constructor(
1230
+ private fb: FormBuilder,
1231
+ private tracker: UnsavedChangesTrackerService,
1232
+ ) {}
1233
+
1234
+ ngOnInit(): void {
1235
+ this.form = this.fb.group({
1236
+ name: [""],
1237
+ description: [""],
1238
+ });
1239
+
1240
+ this.tracker.register(this.form);
1241
+ }
1242
+
1243
+ save(): void {
1244
+ // API call
1245
+ this.tracker.markAsPristine();
1246
+ this.form.markAsPristine();
1247
+ }
1248
+ }
1249
+ ```
1250
+
1251
+ ### Protecting Browser Refresh / Tab Close
1252
+
1253
+ Add this inside the routed component:
1254
+
1255
+ ```ts
1256
+ private unregister?: () => void;
1257
+
1258
+ constructor(
1259
+ private tracker: FormChangesTrackerService,
1260
+ public utilservice: UtilityService
1261
+ ) {
1262
+ super();
1263
+ }
1264
+
1265
+ ngOnInit() {
1266
+ this.unregister = IntelliToolxHelper.registerBeforeUnload(() =>
1267
+ this.tracker.hasChanges(),
1268
+ );
1269
+ }
1270
+
1271
+ confirmUnsavedChanges(): Promise<boolean> {
1272
+ return this.utilservice.confirmUnsavedChanges();
1273
+ }
1274
+
1275
+ ngOnDestroy() {
1276
+ this.unregister?.();
1277
+ }
1278
+ ```
1279
+
1280
+ ### Protecting Modals (NG Bootstrap Example)
1281
+
1282
+ Route guards do NOT protect modals.
1283
+ You must intercept modal close manually.
1284
+
1285
+ Modal Form Component
1286
+
1287
+ ```ts
1288
+ import { NgbActiveModal } from "@ng-bootstrap/ng-bootstrap";
1289
+ import { UnsavedChangesTrackerService } from "ngx-intellitoolx";
1290
+ import { UtilityService } from "../utility.service";
1291
+
1292
+ @Component({
1293
+ selector: "app-item-modal",
1294
+ templateUrl: "./item-modal.component.html",
1295
+ })
1296
+ export class ItemModalComponent implements OnInit {
1297
+ form!: FormGroup;
1298
+
1299
+ constructor(
1300
+ private fb: FormBuilder,
1301
+ private tracker: UnsavedChangesTrackerService,
1302
+ private utils: UtilityService,
1303
+ public activeModal: NgbActiveModal,
1304
+ ) {}
1305
+
1306
+ ngOnInit(): void {
1307
+ this.form = this.fb.group({
1308
+ name: [""],
1309
+ description: [""],
1310
+ });
1311
+
1312
+ this.form.valueChanges.subscribe(() => {
1313
+ if (this.form.dirty) {
1314
+ this.tracker.markAsDirty();
1315
+ }
1316
+ });
1317
+ }
1318
+
1319
+ async closeModal() {
1320
+ if (this.tracker.hasChanges()) {
1321
+ const confirmed = await this.utils.confirmUnsavedChanges();
1322
+
1323
+ if (!confirmed) {
1324
+ return; // cancel close
1325
+ }
1326
+ }
1327
+
1328
+ this.activeModal.dismiss();
1329
+ }
1330
+
1331
+ save(): void {
1332
+ this.tracker.markAsPristine();
1333
+ this.form.markAsPristine();
1334
+ this.activeModal.close("saved");
1335
+ }
1336
+ }
1337
+ ```
1338
+
1339
+ ### Modal Template
1340
+
1341
+ ```html
1342
+ <div class="modal-header">
1343
+ <h5 class="modal-title">Add Item</h5>
1344
+ <button type="button" class="btn-close" (click)="closeModal()"></button>
1345
+ </div>
1346
+
1347
+ <div class="modal-body">
1348
+ <form [formGroup]="form">
1349
+ <input formControlName="name" />
1350
+ <textarea formControlName="description"></textarea>
1351
+ </form>
1352
+ </div>
1353
+
1354
+ <div class="modal-footer">
1355
+ <button class="btn btn-secondary" (click)="closeModal()">Cancel</button>
1356
+
1357
+ <button class="btn btn-primary" (click)="save()">Save</button>
1358
+ </div>
1359
+ ```
1360
+
1361
+ ** How It Works **
1362
+
1363
+ | Action | What Happens |
1364
+ | -------------------- | --------------------- |
1365
+ | User edits form | Tracker marks dirty |
1366
+ | User navigates route | Guard checks tracker |
1367
+ | User refreshes page | beforeunload triggers |
1368
+ | User closes modal | Manual interception |
1369
+ | User saves form | Tracker resets |
1370
+
1371
+ ** Parent + Child Structure **
1372
+
1373
+ If your form is nested:
1374
+
1375
+ ```
1376
+ RouteComponent
1377
+ └── Modal
1378
+ └── Form
1379
+
1380
+ Child form → marks tracker dirty
1381
+
1382
+ Route component → implements guard interface
1383
+
1384
+ Modal → manually checks tracker before close
1385
+ ```
1386
+
1387
+ ** Best Practices **
1388
+
1389
+ - Always call markAsPristine() after successful save
1390
+ - Always intercept modal close
1391
+ - Keep confirmation UI inside the consuming app
1392
+ - Do not couple library to UI services
1393
+
1394
+ ** Confirmation Modal Example (App Layer) **
1395
+
1396
+ ```ts
1397
+ @Injectable({ providedIn: "root" })
1398
+ export class UtilityService {
1399
+ constructor(private modal: NgbModal) {}
1400
+
1401
+ confirmUnsavedChanges(): Promise<boolean> {
1402
+ const modalRef = this.modal.open(UnsavedConfirmComponent, {
1403
+ backdrop: "static",
1404
+ });
1405
+
1406
+ return modalRef.result.then(() => true).catch(() => false);
1407
+ }
1408
+ }
1409
+ ```
1410
+
1411
+ 📄 License
1412
+
1413
+ ```
1414
+ MIT © Fagon Technologies
1415
+ ```
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FuLWNvbXBvbmVudC1kZWFjdGl2YXRlLmludGVyZmFjZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvZm9ybS1jaGFuZ2VzL2Nhbi1jb21wb25lbnQtZGVhY3RpdmF0ZS5pbnRlcmZhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgQ2FuQ29tcG9uZW50RGVhY3RpdmF0ZSB7XG4gIGhhc1Vuc2F2ZWRDaGFuZ2VzOiAoKSA9PiBib29sZWFuO1xuICBjb25maXJtVW5zYXZlZENoYW5nZXM/OiAoKSA9PiBQcm9taXNlPGJvb2xlYW4+IHwgYm9vbGVhbjtcbiAgZ2V0VW5zYXZlZE1lc3NhZ2U/OiAoKSA9PiBzdHJpbmc7XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlybS1oYW5kbGVyLnR5cGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL2Zvcm0tY2hhbmdlcy9jb25maXJtLWhhbmRsZXIudHlwZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgQ29uZmlybUhhbmRsZXIgPSAoKSA9PiBib29sZWFuIHwgUHJvbWlzZTxib29sZWFuPjtcbiJdfQ==
@@ -0,0 +1,47 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { IntelliToolxHelper } from '../helpers/intellitoolx.helper';
3
+ import * as i0 from "@angular/core";
4
+ export class FormChangesTrackerService {
5
+ constructor() {
6
+ this.trackedForms = new Map();
7
+ }
8
+ register(form) {
9
+ const currentRawValue = form.getRawValue?.() ?? form.value;
10
+ this.trackedForms.set(form, {
11
+ initialValue: IntelliToolxHelper.clone(currentRawValue),
12
+ });
13
+ form.markAsPristine();
14
+ }
15
+ hasChanges() {
16
+ // We iterate over entries [form, metadata]
17
+ for (const [form, { initialValue }] of this.trackedForms.entries()) {
18
+ if (IntelliToolxHelper.formHasChanges(initialValue, form)) {
19
+ return true;
20
+ }
21
+ }
22
+ return false;
23
+ }
24
+ unregister(form) {
25
+ this.trackedForms.delete(form);
26
+ }
27
+ reset(form) {
28
+ const entry = this.trackedForms.get(form);
29
+ if (!entry)
30
+ return;
31
+ const currentRawValue = form.getRawValue?.() ?? form.value;
32
+ entry.initialValue = IntelliToolxHelper.clone(currentRawValue);
33
+ form.markAsPristine();
34
+ }
35
+ clearAll() {
36
+ this.trackedForms.clear();
37
+ }
38
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: FormChangesTrackerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
39
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: FormChangesTrackerService, providedIn: 'root' }); }
40
+ }
41
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: FormChangesTrackerService, decorators: [{
42
+ type: Injectable,
43
+ args: [{
44
+ providedIn: 'root',
45
+ }]
46
+ }] });
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybS1jaGFuZ2VzLXRyYWNrZXIuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvZm9ybS1jaGFuZ2VzL2Zvcm0tY2hhbmdlcy10cmFja2VyLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQzs7QUFNcEUsTUFBTSxPQUFPLHlCQUF5QjtJQUh0QztRQUlVLGlCQUFZLEdBQUcsSUFBSSxHQUFHLEVBSzNCLENBQUM7S0F1Q0w7SUFyQ0MsUUFBUSxDQUFDLElBQXFCO1FBQzVCLE1BQU0sZUFBZSxHQUFJLElBQVksQ0FBQyxXQUFXLEVBQUUsRUFBRSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUM7UUFFcEUsSUFBSSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsSUFBSSxFQUFFO1lBQzFCLFlBQVksRUFBRSxrQkFBa0IsQ0FBQyxLQUFLLENBQUMsZUFBZSxDQUFDO1NBQ3hELENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztJQUN4QixDQUFDO0lBRUQsVUFBVTtRQUNSLDJDQUEyQztRQUMzQyxLQUFLLE1BQU0sQ0FBQyxJQUFJLEVBQUUsRUFBRSxZQUFZLEVBQUUsQ0FBQyxJQUFJLElBQUksQ0FBQyxZQUFZLENBQUMsT0FBTyxFQUFFLEVBQUU7WUFDbEUsSUFBSSxrQkFBa0IsQ0FBQyxjQUFjLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxFQUFFO2dCQUN6RCxPQUFPLElBQUksQ0FBQzthQUNiO1NBQ0Y7UUFDRCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFRCxVQUFVLENBQUMsSUFBcUI7UUFDOUIsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDakMsQ0FBQztJQUVELEtBQUssQ0FBQyxJQUFxQjtRQUN6QixNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMxQyxJQUFJLENBQUMsS0FBSztZQUFFLE9BQU87UUFFbkIsTUFBTSxlQUFlLEdBQUksSUFBWSxDQUFDLFdBQVcsRUFBRSxFQUFFLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQztRQUNwRSxLQUFLLENBQUMsWUFBWSxHQUFHLGtCQUFrQixDQUFDLEtBQUssQ0FBQyxlQUFlLENBQUMsQ0FBQztRQUUvRCxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7SUFDeEIsQ0FBQztJQUVELFFBQVE7UUFDTixJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQzVCLENBQUM7K0dBNUNVLHlCQUF5QjttSEFBekIseUJBQXlCLGNBRnhCLE1BQU07OzRGQUVQLHlCQUF5QjtrQkFIckMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBJbnRlbGxpVG9vbHhIZWxwZXIgfSBmcm9tICcuLi9oZWxwZXJzL2ludGVsbGl0b29seC5oZWxwZXInO1xuaW1wb3J0IHsgQWJzdHJhY3RDb250cm9sIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgRm9ybUNoYW5nZXNUcmFja2VyU2VydmljZSB7XG4gIHByaXZhdGUgdHJhY2tlZEZvcm1zID0gbmV3IE1hcDxcbiAgICBBYnN0cmFjdENvbnRyb2wsXG4gICAge1xuICAgICAgaW5pdGlhbFZhbHVlOiBhbnk7XG4gICAgfVxuICA+KCk7XG5cbiAgcmVnaXN0ZXIoZm9ybTogQWJzdHJhY3RDb250cm9sKSB7XG4gICAgY29uc3QgY3VycmVudFJhd1ZhbHVlID0gKGZvcm0gYXMgYW55KS5nZXRSYXdWYWx1ZT8uKCkgPz8gZm9ybS52YWx1ZTtcblxuICAgIHRoaXMudHJhY2tlZEZvcm1zLnNldChmb3JtLCB7XG4gICAgICBpbml0aWFsVmFsdWU6IEludGVsbGlUb29seEhlbHBlci5jbG9uZShjdXJyZW50UmF3VmFsdWUpLFxuICAgIH0pO1xuXG4gICAgZm9ybS5tYXJrQXNQcmlzdGluZSgpO1xuICB9XG5cbiAgaGFzQ2hhbmdlcygpOiBib29sZWFuIHtcbiAgICAvLyBXZSBpdGVyYXRlIG92ZXIgZW50cmllcyBbZm9ybSwgbWV0YWRhdGFdXG4gICAgZm9yIChjb25zdCBbZm9ybSwgeyBpbml0aWFsVmFsdWUgfV0gb2YgdGhpcy50cmFja2VkRm9ybXMuZW50cmllcygpKSB7XG4gICAgICBpZiAoSW50ZWxsaVRvb2x4SGVscGVyLmZvcm1IYXNDaGFuZ2VzKGluaXRpYWxWYWx1ZSwgZm9ybSkpIHtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIHVucmVnaXN0ZXIoZm9ybTogQWJzdHJhY3RDb250cm9sKSB7XG4gICAgdGhpcy50cmFja2VkRm9ybXMuZGVsZXRlKGZvcm0pO1xuICB9XG5cbiAgcmVzZXQoZm9ybTogQWJzdHJhY3RDb250cm9sKSB7XG4gICAgY29uc3QgZW50cnkgPSB0aGlzLnRyYWNrZWRGb3Jtcy5nZXQoZm9ybSk7XG4gICAgaWYgKCFlbnRyeSkgcmV0dXJuO1xuXG4gICAgY29uc3QgY3VycmVudFJhd1ZhbHVlID0gKGZvcm0gYXMgYW55KS5nZXRSYXdWYWx1ZT8uKCkgPz8gZm9ybS52YWx1ZTtcbiAgICBlbnRyeS5pbml0aWFsVmFsdWUgPSBJbnRlbGxpVG9vbHhIZWxwZXIuY2xvbmUoY3VycmVudFJhd1ZhbHVlKTtcblxuICAgIGZvcm0ubWFya0FzUHJpc3RpbmUoKTtcbiAgfVxuXG4gIGNsZWFyQWxsKCkge1xuICAgIHRoaXMudHJhY2tlZEZvcm1zLmNsZWFyKCk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybS11cGRhdGUtbWVzc2FnZS1jb25maWcuaW50ZXJmYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9mb3JtLWNoYW5nZXMvZm9ybS11cGRhdGUtbWVzc2FnZS1jb25maWcuaW50ZXJmYWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIEludGVsbGl0b29seEZvcm1VcGRhdGVNZXNzYWdlQ29uZmlnIHtcbiAgbWVzc2FnZT86IHN0cmluZztcbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nO1xuICB0ZXh0Q29sb3I/OiBzdHJpbmc7XG4gIGJvcmRlckNvbG9yPzogc3RyaW5nO1xuICBmb250V2VpZ2h0PzogbnVtYmVyO1xuICBib3JkZXJSYWRpdXM/OiBzdHJpbmc7XG4gIHBhZGRpbmc/OiBzdHJpbmc7XG4gIGljb25BbmRNZXNzYWdlR2FwPzogc3RyaW5nO1xuICBpY29uU2l6ZT86IHN0cmluZztcbn1cbiJdfQ==
@@ -0,0 +1,53 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "@angular/common";
5
+ export class IntellitoolxFormUpdateMessage {
6
+ constructor() {
7
+ this.defaultMessage = 'There are no changes to save. Please modify a field to continue.';
8
+ }
9
+ get containerStyles() {
10
+ return {
11
+ padding: this.itxFormUpdateMessageConfig?.padding || '1rem',
12
+ gap: this.itxFormUpdateMessageConfig?.iconAndMessageGap || '0.5rem',
13
+ color: this.itxFormUpdateMessageConfig?.textColor || '#963C00',
14
+ border: '1px solid ' +
15
+ (this.itxFormUpdateMessageConfig?.borderColor || '#f3cd5a'),
16
+ 'background-color': this.itxFormUpdateMessageConfig?.backgroundColor || '#fff3cd',
17
+ 'font-weight': String(this.itxFormUpdateMessageConfig?.fontWeight || 400),
18
+ 'border-radius': this.itxFormUpdateMessageConfig?.borderRadius || '0.5rem',
19
+ };
20
+ }
21
+ get iconStyles() {
22
+ return {
23
+ 'font-size': this.itxFormUpdateMessageConfig?.iconSize || '1rem',
24
+ };
25
+ }
26
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IntellitoolxFormUpdateMessage, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
27
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: IntellitoolxFormUpdateMessage, isStandalone: true, selector: "itx-form-update-message", inputs: { itxFormUpdateMessageConfig: "itxFormUpdateMessageConfig" }, ngImport: i0, template: `
28
+ <div class="form-update-message" role="alert" [ngStyle]="containerStyles">
29
+ <span class="form-update-icon" aria-hidden="true" [ngStyle]="iconStyles"
30
+ >⚠</span
31
+ >
32
+ <span>
33
+ {{ itxFormUpdateMessageConfig?.message || defaultMessage }}
34
+ </span>
35
+ </div>
36
+ `, isInline: true, styles: [".form-update-message{display:flex;margin-top:.5rem}.form-update-icon{flex-shrink:0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
37
+ }
38
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IntellitoolxFormUpdateMessage, decorators: [{
39
+ type: Component,
40
+ args: [{ selector: 'itx-form-update-message', standalone: true, imports: [CommonModule], template: `
41
+ <div class="form-update-message" role="alert" [ngStyle]="containerStyles">
42
+ <span class="form-update-icon" aria-hidden="true" [ngStyle]="iconStyles"
43
+ >⚠</span
44
+ >
45
+ <span>
46
+ {{ itxFormUpdateMessageConfig?.message || defaultMessage }}
47
+ </span>
48
+ </div>
49
+ `, styles: [".form-update-message{display:flex;margin-top:.5rem}.form-update-icon{flex-shrink:0}\n"] }]
50
+ }], propDecorators: { itxFormUpdateMessageConfig: [{
51
+ type: Input
52
+ }] } });
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybS11cGRhdGUtbWVzc2FnZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL2Zvcm0tY2hhbmdlcy9mb3JtLXVwZGF0ZS1tZXNzYWdlLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVqRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7OztBQWtCL0MsTUFBTSxPQUFPLDZCQUE2QjtJQWhCMUM7UUFrQkUsbUJBQWMsR0FDWixrRUFBa0UsQ0FBQztLQXVCdEU7SUFyQkMsSUFBSSxlQUFlO1FBQ2pCLE9BQU87WUFDTCxPQUFPLEVBQUUsSUFBSSxDQUFDLDBCQUEwQixFQUFFLE9BQU8sSUFBSSxNQUFNO1lBQzNELEdBQUcsRUFBRSxJQUFJLENBQUMsMEJBQTBCLEVBQUUsaUJBQWlCLElBQUksUUFBUTtZQUNuRSxLQUFLLEVBQUUsSUFBSSxDQUFDLDBCQUEwQixFQUFFLFNBQVMsSUFBSSxTQUFTO1lBQzlELE1BQU0sRUFDSixZQUFZO2dCQUNaLENBQUMsSUFBSSxDQUFDLDBCQUEwQixFQUFFLFdBQVcsSUFBSSxTQUFTLENBQUM7WUFDN0Qsa0JBQWtCLEVBQ2hCLElBQUksQ0FBQywwQkFBMEIsRUFBRSxlQUFlLElBQUksU0FBUztZQUMvRCxhQUFhLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQywwQkFBMEIsRUFBRSxVQUFVLElBQUksR0FBRyxDQUFDO1lBQ3pFLGVBQWUsRUFDYixJQUFJLENBQUMsMEJBQTBCLEVBQUUsWUFBWSxJQUFJLFFBQVE7U0FDNUQsQ0FBQztJQUNKLENBQUM7SUFFRCxJQUFJLFVBQVU7UUFDWixPQUFPO1lBQ0wsV0FBVyxFQUFFLElBQUksQ0FBQywwQkFBMEIsRUFBRSxRQUFRLElBQUksTUFBTTtTQUNqRSxDQUFDO0lBQ0osQ0FBQzsrR0F6QlUsNkJBQTZCO21HQUE3Qiw2QkFBNkIseUpBWDlCOzs7Ozs7Ozs7R0FTVCw4SkFYUyxZQUFZOzs0RkFhWCw2QkFBNkI7a0JBaEJ6QyxTQUFTOytCQUNFLHlCQUF5QixjQUN2QixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUMsWUFFYjs7Ozs7Ozs7O0dBU1Q7OEJBR1EsMEJBQTBCO3NCQUFsQyxLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSW50ZWxsaXRvb2x4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWcgfSBmcm9tICcuL2Zvcm0tdXBkYXRlLW1lc3NhZ2UtY29uZmlnLmludGVyZmFjZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdpdHgtZm9ybS11cGRhdGUtbWVzc2FnZScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxuICBzdHlsZVVybHM6IFsnLi9mb3JtLXVwZGF0ZS1tZXNzYWdlLmNvbXBvbmVudC5jc3MnXSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGl2IGNsYXNzPVwiZm9ybS11cGRhdGUtbWVzc2FnZVwiIHJvbGU9XCJhbGVydFwiIFtuZ1N0eWxlXT1cImNvbnRhaW5lclN0eWxlc1wiPlxuICAgICAgPHNwYW4gY2xhc3M9XCJmb3JtLXVwZGF0ZS1pY29uXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgW25nU3R5bGVdPVwiaWNvblN0eWxlc1wiXG4gICAgICAgID7imqA8L3NwYW5cbiAgICAgID5cbiAgICAgIDxzcGFuPlxuICAgICAgICB7eyBpdHhGb3JtVXBkYXRlTWVzc2FnZUNvbmZpZz8ubWVzc2FnZSB8fCBkZWZhdWx0TWVzc2FnZSB9fVxuICAgICAgPC9zcGFuPlxuICAgIDwvZGl2PlxuICBgLFxufSlcbmV4cG9ydCBjbGFzcyBJbnRlbGxpdG9vbHhGb3JtVXBkYXRlTWVzc2FnZSB7XG4gIEBJbnB1dCgpIGl0eEZvcm1VcGRhdGVNZXNzYWdlQ29uZmlnPzogSW50ZWxsaXRvb2x4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWc7XG4gIGRlZmF1bHRNZXNzYWdlID1cbiAgICAnVGhlcmUgYXJlIG5vIGNoYW5nZXMgdG8gc2F2ZS4gUGxlYXNlIG1vZGlmeSBhIGZpZWxkIHRvIGNvbnRpbnVlLic7XG5cbiAgZ2V0IGNvbnRhaW5lclN0eWxlcygpOiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB9IHtcbiAgICByZXR1cm4ge1xuICAgICAgcGFkZGluZzogdGhpcy5pdHhGb3JtVXBkYXRlTWVzc2FnZUNvbmZpZz8ucGFkZGluZyB8fCAnMXJlbScsXG4gICAgICBnYXA6IHRoaXMuaXR4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWc/Lmljb25BbmRNZXNzYWdlR2FwIHx8ICcwLjVyZW0nLFxuICAgICAgY29sb3I6IHRoaXMuaXR4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWc/LnRleHRDb2xvciB8fCAnIzk2M0MwMCcsXG4gICAgICBib3JkZXI6XG4gICAgICAgICcxcHggc29saWQgJyArXG4gICAgICAgICh0aGlzLml0eEZvcm1VcGRhdGVNZXNzYWdlQ29uZmlnPy5ib3JkZXJDb2xvciB8fCAnI2YzY2Q1YScpLFxuICAgICAgJ2JhY2tncm91bmQtY29sb3InOlxuICAgICAgICB0aGlzLml0eEZvcm1VcGRhdGVNZXNzYWdlQ29uZmlnPy5iYWNrZ3JvdW5kQ29sb3IgfHwgJyNmZmYzY2QnLFxuICAgICAgJ2ZvbnQtd2VpZ2h0JzogU3RyaW5nKHRoaXMuaXR4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWc/LmZvbnRXZWlnaHQgfHwgNDAwKSxcbiAgICAgICdib3JkZXItcmFkaXVzJzpcbiAgICAgICAgdGhpcy5pdHhGb3JtVXBkYXRlTWVzc2FnZUNvbmZpZz8uYm9yZGVyUmFkaXVzIHx8ICcwLjVyZW0nLFxuICAgIH07XG4gIH1cblxuICBnZXQgaWNvblN0eWxlcygpOiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB9IHtcbiAgICByZXR1cm4ge1xuICAgICAgJ2ZvbnQtc2l6ZSc6IHRoaXMuaXR4Rm9ybVVwZGF0ZU1lc3NhZ2VDb25maWc/Lmljb25TaXplIHx8ICcxcmVtJyxcbiAgICB9O1xuICB9XG59XG4iXX0=
@@ -0,0 +1,29 @@
1
+ import { Injectable } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "./form-changes-tracker.service";
4
+ export class UnsavedChangesGuard {
5
+ constructor(tracker) {
6
+ this.tracker = tracker;
7
+ }
8
+ async canDeactivate(component) {
9
+ if (!this.tracker.hasChanges()) {
10
+ return true;
11
+ }
12
+ // Component provides its own confirmation (modal)
13
+ if (component.confirmUnsavedChanges) {
14
+ const result = component.confirmUnsavedChanges();
15
+ return typeof result === 'boolean' ? result : await result;
16
+ }
17
+ const message = component.getUnsavedMessage?.() ||
18
+ 'You have unsaved changes. Leave without saving?';
19
+ // Fallback browser confirm
20
+ return confirm(message);
21
+ }
22
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: UnsavedChangesGuard, deps: [{ token: i1.FormChangesTrackerService }], target: i0.ɵɵFactoryTarget.Injectable }); }
23
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: UnsavedChangesGuard, providedIn: 'root' }); }
24
+ }
25
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: UnsavedChangesGuard, decorators: [{
26
+ type: Injectable,
27
+ args: [{ providedIn: 'root' }]
28
+ }], ctorParameters: function () { return [{ type: i1.FormChangesTrackerService }]; } });
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5zYXZlZC1jaGFuZ2VzLmd1YXJkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9mb3JtLWNoYW5nZXMvdW5zYXZlZC1jaGFuZ2VzLmd1YXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7OztBQU0zQyxNQUFNLE9BQU8sbUJBQW1CO0lBQzlCLFlBQW9CLE9BQWtDO1FBQWxDLFlBQU8sR0FBUCxPQUFPLENBQTJCO0lBQUcsQ0FBQztJQUUxRCxLQUFLLENBQUMsYUFBYSxDQUFDLFNBQWlDO1FBQ25ELElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxFQUFFO1lBQzlCLE9BQU8sSUFBSSxDQUFDO1NBQ2I7UUFDRCxrREFBa0Q7UUFDbEQsSUFBSSxTQUFTLENBQUMscUJBQXFCLEVBQUU7WUFDbkMsTUFBTSxNQUFNLEdBQUcsU0FBUyxDQUFDLHFCQUFxQixFQUFFLENBQUM7WUFDakQsT0FBTyxPQUFPLE1BQU0sS0FBSyxTQUFTLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsTUFBTSxNQUFNLENBQUM7U0FDNUQ7UUFFRCxNQUFNLE9BQU8sR0FDWCxTQUFTLENBQUMsaUJBQWlCLEVBQUUsRUFBRTtZQUMvQixpREFBaUQsQ0FBQztRQUNwRCwyQkFBMkI7UUFDM0IsT0FBTyxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDMUIsQ0FBQzsrR0FsQlUsbUJBQW1CO21IQUFuQixtQkFBbUIsY0FETixNQUFNOzs0RkFDbkIsbUJBQW1CO2tCQUQvQixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENhbkRlYWN0aXZhdGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgQ2FuQ29tcG9uZW50RGVhY3RpdmF0ZSB9IGZyb20gJy4vY2FuLWNvbXBvbmVudC1kZWFjdGl2YXRlLmludGVyZmFjZSc7XG5pbXBvcnQgeyBGb3JtQ2hhbmdlc1RyYWNrZXJTZXJ2aWNlIH0gZnJvbSAnLi9mb3JtLWNoYW5nZXMtdHJhY2tlci5zZXJ2aWNlJztcblxuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBVbnNhdmVkQ2hhbmdlc0d1YXJkIGltcGxlbWVudHMgQ2FuRGVhY3RpdmF0ZTxDYW5Db21wb25lbnREZWFjdGl2YXRlPiB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgdHJhY2tlcjogRm9ybUNoYW5nZXNUcmFja2VyU2VydmljZSkge31cblxuICBhc3luYyBjYW5EZWFjdGl2YXRlKGNvbXBvbmVudDogQ2FuQ29tcG9uZW50RGVhY3RpdmF0ZSk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGlmICghdGhpcy50cmFja2VyLmhhc0NoYW5nZXMoKSkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIC8vIENvbXBvbmVudCBwcm92aWRlcyBpdHMgb3duIGNvbmZpcm1hdGlvbiAobW9kYWwpXG4gICAgaWYgKGNvbXBvbmVudC5jb25maXJtVW5zYXZlZENoYW5nZXMpIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGNvbXBvbmVudC5jb25maXJtVW5zYXZlZENoYW5nZXMoKTtcbiAgICAgIHJldHVybiB0eXBlb2YgcmVzdWx0ID09PSAnYm9vbGVhbicgPyByZXN1bHQgOiBhd2FpdCByZXN1bHQ7XG4gICAgfVxuXG4gICAgY29uc3QgbWVzc2FnZSA9XG4gICAgICBjb21wb25lbnQuZ2V0VW5zYXZlZE1lc3NhZ2U/LigpIHx8XG4gICAgICAnWW91IGhhdmUgdW5zYXZlZCBjaGFuZ2VzLiBMZWF2ZSB3aXRob3V0IHNhdmluZz8nO1xuICAgIC8vIEZhbGxiYWNrIGJyb3dzZXIgY29uZmlybVxuICAgIHJldHVybiBjb25maXJtKG1lc3NhZ2UpO1xuICB9XG59XG4iXX0=
@@ -16,7 +16,7 @@ export class IntellitoolxFormErrors {
16
16
  passwordMismatch: 'Password and confirm password do not match.',
17
17
  futureDate: 'Future date is not allowed.',
18
18
  exceededAllowedDateDifference: 'Date difference can only be one month.',
19
- exceededLeastDateAllowed: 'Start date cannot be greater than end date',
19
+ startDateAfterEndDate: 'Start date cannot be greater than end date',
20
20
  duplicateEmail: 'Each email must be unique.',
21
21
  maxWords: 'Exceeded maximum number of words.',
22
22
  maxMonthYear: 'Date is later than allowed.',
@@ -69,7 +69,7 @@ export class IntellitoolxFormErrors {
69
69
  case 'maxMonthYear':
70
70
  case 'minMonthYear':
71
71
  case 'exceededAllowedDateDifference':
72
- case 'exceededLeastDateAllowed':
72
+ case 'startDateAfterEndDate':
73
73
  return customMessage || this.errorMessages[key];
74
74
  default:
75
75
  return customMessage || '';
@@ -107,4 +107,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
107
107
  }], controlLabel: [{
108
108
  type: Input
109
109
  }] } });
110
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-errors.component.js","sourceRoot":"","sources":["../../../../src/lib/form-errors/form-errors.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input } from '@angular/core';\nimport {\n  AbstractControl,\n  FormControl,\n  UntypedFormControl,\n} from '@angular/forms';\n\nexport interface ErrorMessages {\n  [key: string]: string;\n  required: string;\n  email: string;\n  minlength: string;\n  maxlength: string;\n  pattern: string;\n  passwordMismatch: string;\n  futureDate: string;\n  exceededAllowedDateDifference: string;\n  exceededLeastDateAllowed: string;\n  duplicateEmail: string;\n  maxWords: string;\n  maxMonthYear: string;\n  minMonthYear: string;\n}\n\n@Component({\n  selector: 'itx-form-errors',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <ng-container *ngIf=\"control?.touched && control?.invalid\">\n      <ng-container *ngFor=\"let error of control?.errors | keyvalue\">\n        {{ getErrorMessage(error?.key ?? '', error?.value) }}\n      </ng-container>\n    </ng-container>\n  `,\n})\nexport class IntellitoolxFormErrors {\n  @Input() control: FormControl | UntypedFormControl | AbstractControl | null =\n    null;\n  @Input() componentValidation: Record<string, { message: string }> = {};\n  @Input() controlLabel: string | null = null;\n\n  errorMessages: ErrorMessages = {\n    required: 'This field is required',\n    email: 'The email entered is invalid',\n    minlength: 'You must enter at least {length} characters',\n    maxlength: 'You must not enter more than {length} characters',\n    pattern: 'Your entry must match the required pattern',\n    passwordMismatch: 'Password and confirm password do not match.',\n    futureDate: 'Future date is not allowed.',\n    exceededAllowedDateDifference: 'Date difference can only be one month.',\n    exceededLeastDateAllowed: 'Start date cannot be greater than end date',\n    duplicateEmail: 'Each email must be unique.',\n    maxWords: 'Exceeded maximum number of words.',\n    maxMonthYear: 'Date is later than allowed.',\n    minMonthYear: 'Date is earlier than allowed.',\n  };\n\n  getErrorMessage(key: string, value: any): string {\n    if (!this.control) return '';\n\n    const customMessage = this.componentValidation?.[key]?.message;\n\n    if (typeof value === 'boolean') {\n      return customMessage || this.errorMessages[key] || '';\n    }\n\n    if (typeof value === 'string') {\n      return customMessage || value;\n    }\n\n    if (typeof value === 'object') {\n      switch (key) {\n        case 'pattern':\n          return (\n            customMessage ||\n            `Please provide a valid ${this.controlLabel || 'data'}.`\n          );\n        case 'min':\n          return customMessage\n            ? customMessage\n            : this.controlLabel\n              ? `${this.controlLabel} cannot be less than ${value.min}.`\n              : `Number cannot be less than ${value.min}`;\n        case 'max':\n          return customMessage\n            ? customMessage\n            : this.controlLabel\n              ? `${this.controlLabel} cannot be greater than ${value.max}.`\n              : `Number cannot be greater than ${value.max}`;\n        case 'minlength':\n          return (\n            customMessage ||\n            this.errorMessages.minlength.replace(\n              '{length}',\n              value?.requiredLength || '',\n            )\n          );\n        case 'maxlength':\n          return (\n            customMessage ||\n            this.errorMessages.maxlength.replace(\n              '{length}',\n              value?.requiredLength || '',\n            )\n          );\n        case 'customMinValue':\n        case 'customMaxValue':\n          return customMessage || value?.message || '';\n        case 'ngbDate':\n          return (\n            customMessage ||\n            (this.control.value ? 'Invalid date format provided' : '')\n          );\n        case 'passwordMismatch':\n        case 'futureDate':\n        case 'duplicateEmail':\n        case 'maxWords':\n        case 'maxMonthYear':\n        case 'minMonthYear':\n        case 'exceededAllowedDateDifference':\n        case 'exceededLeastDateAllowed':\n          return customMessage || this.errorMessages[key];\n        default:\n          return customMessage || '';\n      }\n    }\n\n    return '';\n  }\n}\n"]}
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-errors.component.js","sourceRoot":"","sources":["../../../../src/lib/form-errors/form-errors.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input } from '@angular/core';\nimport {\n  AbstractControl,\n  FormControl,\n  UntypedFormControl,\n} from '@angular/forms';\n\nexport interface ErrorMessages {\n  [key: string]: string;\n  required: string;\n  email: string;\n  minlength: string;\n  maxlength: string;\n  pattern: string;\n  passwordMismatch: string;\n  futureDate: string;\n  exceededAllowedDateDifference: string;\n  startDateAfterEndDate: string;\n  duplicateEmail: string;\n  maxWords: string;\n  maxMonthYear: string;\n  minMonthYear: string;\n}\n\n@Component({\n  selector: 'itx-form-errors',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <ng-container *ngIf=\"control?.touched && control?.invalid\">\n      <ng-container *ngFor=\"let error of control?.errors | keyvalue\">\n        {{ getErrorMessage(error?.key ?? '', error?.value) }}\n      </ng-container>\n    </ng-container>\n  `,\n})\nexport class IntellitoolxFormErrors {\n  @Input() control: FormControl | UntypedFormControl | AbstractControl | null =\n    null;\n  @Input() componentValidation: Record<string, { message: string }> = {};\n  @Input() controlLabel: string | null = null;\n\n  errorMessages: ErrorMessages = {\n    required: 'This field is required',\n    email: 'The email entered is invalid',\n    minlength: 'You must enter at least {length} characters',\n    maxlength: 'You must not enter more than {length} characters',\n    pattern: 'Your entry must match the required pattern',\n    passwordMismatch: 'Password and confirm password do not match.',\n    futureDate: 'Future date is not allowed.',\n    exceededAllowedDateDifference: 'Date difference can only be one month.',\n    startDateAfterEndDate: 'Start date cannot be greater than end date',\n    duplicateEmail: 'Each email must be unique.',\n    maxWords: 'Exceeded maximum number of words.',\n    maxMonthYear: 'Date is later than allowed.',\n    minMonthYear: 'Date is earlier than allowed.',\n  };\n\n  getErrorMessage(key: string, value: any): string {\n    if (!this.control) return '';\n\n    const customMessage = this.componentValidation?.[key]?.message;\n\n    if (typeof value === 'boolean') {\n      return customMessage || this.errorMessages[key] || '';\n    }\n\n    if (typeof value === 'string') {\n      return customMessage || value;\n    }\n\n    if (typeof value === 'object') {\n      switch (key) {\n        case 'pattern':\n          return (\n            customMessage ||\n            `Please provide a valid ${this.controlLabel || 'data'}.`\n          );\n        case 'min':\n          return customMessage\n            ? customMessage\n            : this.controlLabel\n              ? `${this.controlLabel} cannot be less than ${value.min}.`\n              : `Number cannot be less than ${value.min}`;\n        case 'max':\n          return customMessage\n            ? customMessage\n            : this.controlLabel\n              ? `${this.controlLabel} cannot be greater than ${value.max}.`\n              : `Number cannot be greater than ${value.max}`;\n        case 'minlength':\n          return (\n            customMessage ||\n            this.errorMessages.minlength.replace(\n              '{length}',\n              value?.requiredLength || '',\n            )\n          );\n        case 'maxlength':\n          return (\n            customMessage ||\n            this.errorMessages.maxlength.replace(\n              '{length}',\n              value?.requiredLength || '',\n            )\n          );\n        case 'customMinValue':\n        case 'customMaxValue':\n          return customMessage || value?.message || '';\n        case 'ngbDate':\n          return (\n            customMessage ||\n            (this.control.value ? 'Invalid date format provided' : '')\n          );\n        case 'passwordMismatch':\n        case 'futureDate':\n        case 'duplicateEmail':\n        case 'maxWords':\n        case 'maxMonthYear':\n        case 'minMonthYear':\n        case 'exceededAllowedDateDifference':\n        case 'startDateAfterEndDate':\n          return customMessage || this.errorMessages[key];\n        default:\n          return customMessage || '';\n      }\n    }\n\n    return '';\n  }\n}\n"]}
@@ -86,7 +86,6 @@ export class IntelliToolxHelper {
86
86
  const result = confirmHandler();
87
87
  return typeof result === 'boolean' ? result : await result;
88
88
  }
89
- // fallback (only if handler not provided)
90
89
  return confirm('You have unsaved changes. Continue?');
91
90
  }
92
91
  static registerBeforeUnload(shouldBlock, message) {
@@ -94,7 +93,7 @@ export class IntelliToolxHelper {
94
93
  if (!shouldBlock())
95
94
  return;
96
95
  event.preventDefault();
97
- event.returnValue = message || '';
96
+ event.returnValue = message ?? '';
98
97
  };
99
98
  window.addEventListener('beforeunload', handler);
100
99
  return () => window.removeEventListener('beforeunload', handler);
@@ -153,4 +152,4 @@ export class IntelliToolxHelper {
153
152
  return value;
154
153
  }
155
154
  }
156
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"intellitoolx.helper.js","sourceRoot":"","sources":["../../../../src/lib/helpers/intellitoolx.helper.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AbstractControl,\n  FormArray,\n  FormControl,\n  FormGroup,\n} from '@angular/forms';\nimport { ConfirmHandler } from '../form-update/confirm-handler.type';\n\nexport class IntelliToolxHelper {\n  static captureInitialFormValue(form: AbstractControl): any {\n    const initialValue = (form as any).getRawValue?.() ?? form.value;\n    form.markAsPristine();\n    return this.clone(initialValue);\n  }\n\n  static trimFormGroup(control: AbstractControl): void {\n    if (control instanceof FormGroup) {\n      Object.keys(control.controls).forEach((key) =>\n        this.trimFormGroup(control.controls[key]),\n      );\n      return;\n    }\n\n    if (control instanceof FormArray) {\n      control.controls.forEach((ctrl) => this.trimFormGroup(ctrl));\n      return;\n    }\n\n    if (control instanceof FormControl) {\n      const value = control.value;\n      if (typeof value === 'string') {\n        control.setValue(value.trim(), { emitEvent: false });\n      }\n    }\n  }\n\n  static deepEqual(obj1: any, obj2: any): boolean {\n    // treat null, undefined, and empty string as equal\n    if (this.isEmpty(obj1) && this.isEmpty(obj2)) {\n      return true;\n    }\n\n    // normalize numeric values\n    if (typeof obj1 === 'number' && typeof obj2 === 'number') {\n      return Number(obj1) === Number(obj2);\n    }\n\n    // strict equality shortcut\n    if (obj1 === obj2) return true;\n\n    // File comparison\n    if (obj1 instanceof File && obj2 instanceof File) {\n      return (\n        obj1.name === obj2.name &&\n        obj1.size === obj2.size &&\n        obj1.type === obj2.type &&\n        obj1.lastModified === obj2.lastModified\n      );\n    }\n\n    // if not objects → not equal\n    if (\n      typeof obj1 !== 'object' ||\n      typeof obj2 !== 'object' ||\n      obj1 === null ||\n      obj2 === null\n    ) {\n      return false;\n    }\n\n    // arrays\n    if (Array.isArray(obj1) && Array.isArray(obj2)) {\n      if (obj1.length !== obj2.length) return false;\n      return obj1.every((v, i) => this.deepEqual(v, obj2[i]));\n    }\n\n    const keys1 = Object.keys(obj1);\n    const keys2 = Object.keys(obj2);\n\n    if (keys1.length !== keys2.length) return false;\n\n    for (const key of keys1) {\n      if (!keys2.includes(key)) return false;\n      if (!this.deepEqual(obj1[key], obj2[key])) return false;\n    }\n\n    return true;\n  }\n\n  static isEmpty(value: any): boolean {\n    return (\n      value === null ||\n      value === undefined ||\n      (typeof value === 'string' && value.trim() === '')\n    );\n  }\n\n  static clone(value: any): any {\n    return structuredClone(value); // Angular 14+ safe\n  }\n\n  static formHasChanges(initialValue: any, form: AbstractControl): boolean {\n    if (!form) return false;\n\n    const current = form.getRawValue();\n    return !this.deepEqual(initialValue, current);\n  }\n\n  static async confirmIfChanged(\n    hasChanges: boolean,\n    confirmHandler?: ConfirmHandler,\n  ): Promise<boolean> {\n    if (!hasChanges) return true;\n\n    if (confirmHandler) {\n      const result = confirmHandler();\n      return typeof result === 'boolean' ? result : await result;\n    }\n\n    // fallback (only if handler not provided)\n    return confirm('You have unsaved changes. Continue?');\n  }\n\n  static registerBeforeUnload(shouldBlock: () => boolean, message?: string) {\n    const handler = (event: BeforeUnloadEvent) => {\n      if (!shouldBlock()) return;\n\n      event.preventDefault();\n      event.returnValue = message || '';\n    };\n\n    window.addEventListener('beforeunload', handler);\n\n    return () => window.removeEventListener('beforeunload', handler);\n  }\n\n  static getFormControl<T = any>(\n    form: AbstractControl,\n    path: string,\n  ): FormControl<T> | null {\n    if (!form || !path) return null;\n\n    const segments = path.split('.');\n    let current: AbstractControl | null = form;\n\n    for (const segment of segments) {\n      if (!current) return null;\n\n      // Treat numeric segment as FormArray index\n      const index = Number(segment);\n      if (!isNaN(index) && current instanceof FormArray) {\n        current = current.at(index);\n      } else if (current instanceof FormGroup) {\n        current = current.get(segment);\n      } else {\n        return null;\n      }\n    }\n\n    return current instanceof FormControl ? (current as FormControl<T>) : null;\n  }\n\n  static convertImageToBase64(file: File): Promise<string> {\n    return new Promise<string>((resolve, reject) => {\n      const reader = new FileReader();\n\n      reader.readAsDataURL(file);\n\n      reader.onload = () => {\n        const base64String = reader.result as string;\n        const base64StringArray = base64String.split(',');\n        resolve(base64StringArray[1]);\n      };\n\n      reader.onerror = (error) => {\n        reject(error);\n      };\n    });\n  }\n\n  static replaceCharacter(\n    text: any,\n    replaceChar: string,\n    replaceWithChar: string,\n  ): string {\n    return text?.replaceAll(replaceChar, replaceWithChar) || '';\n  }\n\n  static convertJsonStringToJson<T = any>(\n    value: T | string | null | undefined,\n  ): T | null {\n    if (!value) {\n      return null;\n    }\n\n    if (typeof value === 'string') {\n      try {\n        return JSON.parse(value) as T;\n      } catch {\n        return null;\n      }\n    }\n\n    return value as T;\n  }\n}\n"]}
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"intellitoolx.helper.js","sourceRoot":"","sources":["../../../../src/lib/helpers/intellitoolx.helper.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AbstractControl,\n  FormArray,\n  FormControl,\n  FormGroup,\n} from '@angular/forms';\nimport { ConfirmHandler } from '../form-changes/confirm-handler.type';\n\nexport class IntelliToolxHelper {\n  static captureInitialFormValue(form: AbstractControl): any {\n    const initialValue = (form as any).getRawValue?.() ?? form.value;\n    form.markAsPristine();\n    return this.clone(initialValue);\n  }\n\n  static trimFormGroup(control: AbstractControl): void {\n    if (control instanceof FormGroup) {\n      Object.keys(control.controls).forEach((key) =>\n        this.trimFormGroup(control.controls[key]),\n      );\n      return;\n    }\n\n    if (control instanceof FormArray) {\n      control.controls.forEach((ctrl) => this.trimFormGroup(ctrl));\n      return;\n    }\n\n    if (control instanceof FormControl) {\n      const value = control.value;\n      if (typeof value === 'string') {\n        control.setValue(value.trim(), { emitEvent: false });\n      }\n    }\n  }\n\n  static deepEqual(obj1: any, obj2: any): boolean {\n    // treat null, undefined, and empty string as equal\n    if (this.isEmpty(obj1) && this.isEmpty(obj2)) {\n      return true;\n    }\n\n    // normalize numeric values\n    if (typeof obj1 === 'number' && typeof obj2 === 'number') {\n      return Number(obj1) === Number(obj2);\n    }\n\n    // strict equality shortcut\n    if (obj1 === obj2) return true;\n\n    // File comparison\n    if (obj1 instanceof File && obj2 instanceof File) {\n      return (\n        obj1.name === obj2.name &&\n        obj1.size === obj2.size &&\n        obj1.type === obj2.type &&\n        obj1.lastModified === obj2.lastModified\n      );\n    }\n\n    // if not objects → not equal\n    if (\n      typeof obj1 !== 'object' ||\n      typeof obj2 !== 'object' ||\n      obj1 === null ||\n      obj2 === null\n    ) {\n      return false;\n    }\n\n    // arrays\n    if (Array.isArray(obj1) && Array.isArray(obj2)) {\n      if (obj1.length !== obj2.length) return false;\n      return obj1.every((v, i) => this.deepEqual(v, obj2[i]));\n    }\n\n    const keys1 = Object.keys(obj1);\n    const keys2 = Object.keys(obj2);\n\n    if (keys1.length !== keys2.length) return false;\n\n    for (const key of keys1) {\n      if (!keys2.includes(key)) return false;\n      if (!this.deepEqual(obj1[key], obj2[key])) return false;\n    }\n\n    return true;\n  }\n\n  static isEmpty(value: any): boolean {\n    return (\n      value === null ||\n      value === undefined ||\n      (typeof value === 'string' && value.trim() === '')\n    );\n  }\n\n  static clone(value: any): any {\n    return structuredClone(value); // Angular 14+ safe\n  }\n\n  static formHasChanges(initialValue: any, form: AbstractControl): boolean {\n    if (!form) return false;\n\n    const current = form.getRawValue();\n    return !this.deepEqual(initialValue, current);\n  }\n\n  static async confirmIfChanged(\n    hasChanges: boolean,\n    confirmHandler?: ConfirmHandler,\n  ): Promise<boolean> {\n    if (!hasChanges) return true;\n\n    if (confirmHandler) {\n      const result = confirmHandler();\n      return typeof result === 'boolean' ? result : await result;\n    }\n\n    return confirm('You have unsaved changes. Continue?');\n  }\n\n  static registerBeforeUnload(shouldBlock: () => boolean, message?: string) {\n    const handler = (event: BeforeUnloadEvent) => {\n      if (!shouldBlock()) return;\n      event.preventDefault();\n      event.returnValue = message ?? '';\n    };\n\n    window.addEventListener('beforeunload', handler);\n    return () => window.removeEventListener('beforeunload', handler);\n  }\n\n  static getFormControl<T = any>(\n    form: AbstractControl,\n    path: string,\n  ): FormControl<T> | null {\n    if (!form || !path) return null;\n\n    const segments = path.split('.');\n    let current: AbstractControl | null = form;\n\n    for (const segment of segments) {\n      if (!current) return null;\n\n      // Treat numeric segment as FormArray index\n      const index = Number(segment);\n      if (!isNaN(index) && current instanceof FormArray) {\n        current = current.at(index);\n      } else if (current instanceof FormGroup) {\n        current = current.get(segment);\n      } else {\n        return null;\n      }\n    }\n\n    return current instanceof FormControl ? (current as FormControl<T>) : null;\n  }\n\n  static convertImageToBase64(file: File): Promise<string> {\n    return new Promise<string>((resolve, reject) => {\n      const reader = new FileReader();\n\n      reader.readAsDataURL(file);\n\n      reader.onload = () => {\n        const base64String = reader.result as string;\n        const base64StringArray = base64String.split(',');\n        resolve(base64StringArray[1]);\n      };\n\n      reader.onerror = (error) => {\n        reject(error);\n      };\n    });\n  }\n\n  static replaceCharacter(\n    text: any,\n    replaceChar: string,\n    replaceWithChar: string,\n  ): string {\n    return text?.replaceAll(replaceChar, replaceWithChar) || '';\n  }\n\n  static convertJsonStringToJson<T = any>(\n    value: T | string | null | undefined,\n  ): T | null {\n    if (!value) {\n      return null;\n    }\n\n    if (typeof value === 'string') {\n      try {\n        return JSON.parse(value) as T;\n      } catch {\n        return null;\n      }\n    }\n\n    return value as T;\n  }\n}\n"]}