ngx-form-draft 2.2.11 → 2.2.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,268 +1,268 @@
1
- import { Directive, Input, Optional, } from '@angular/core';
2
- import { FormArray, FormGroup, FormControl } from '@angular/forms';
3
- import { Subject } from 'rxjs';
4
- import { debounceTime, takeUntil, filter } from 'rxjs/operators';
5
- import { FormDraftBannerComponent } from './form-draft-banner.component';
6
- import * as i0 from "@angular/core";
7
- import * as i1 from "@angular/forms";
8
- import * as i2 from "./form-draft.service";
9
- /**
10
- * Auto-saves and restores form drafts
11
- *
12
- * @example
13
- * <form [formGroup]="myForm" ngxFormDraft="myFormId">
14
- *
15
- * @example
16
- * <form [formGroup]="myForm" [ngxFormDraft]="'edit_' + entityId" [draftExcludeFields]="['password']">
17
- */
18
- export class FormDraftDirective {
19
- constructor(formGroupDir, ngForm, draftService, viewContainerRef, cdRef, elRef, renderer) {
20
- this.formGroupDir = formGroupDir;
21
- this.ngForm = ngForm;
22
- this.draftService = draftService;
23
- this.viewContainerRef = viewContainerRef;
24
- this.cdRef = cdRef;
25
- this.elRef = elRef;
26
- this.renderer = renderer;
27
- this.draftDebounce = 800;
28
- this.draftExcludeFields = [];
29
- this.draftShowOnChange = false;
30
- this.draftRestoredText = 'Draft restored';
31
- this.draftSavedText = 'Draft saved';
32
- this.draftSavedLabel = 'saved';
33
- this.draftDiscardText = 'Discard';
34
- this.destroy$ = new Subject();
35
- this.bannerRef = null;
36
- this.formControl = null;
37
- this.initialValues = {};
38
- this.isRestoring = false;
39
- }
40
- ngOnInit() {
41
- this.formControl = this.formGroupDir?.form || this.ngForm?.form || null;
42
- if (!this.formControl || !this.formId)
43
- return;
44
- this.draftService.registerReset(this.formId, () => this.performResetAndDestroyBanner());
45
- // For reactive forms, capture initial values and restore draft immediately
46
- if (this.formGroupDir) {
47
- this.initialValues = JSON.parse(JSON.stringify(this.formControl.value));
48
- const draft = this.draftService.load(this.formId);
49
- if (draft) {
50
- this.restoreDraft(draft.values);
51
- this.showBanner(draft.savedAt, true);
52
- }
53
- }
54
- let hasUserInteraction = false;
55
- this.formControl.valueChanges
56
- .pipe(filter(() => {
57
- if (this.isRestoring)
58
- return false;
59
- if (this.ngForm && !hasUserInteraction) {
60
- const currentValues = this.formControl?.value || {};
61
- const isDifferent = JSON.stringify(currentValues) !== JSON.stringify(this.initialValues);
62
- if (isDifferent) {
63
- hasUserInteraction = true;
64
- return true;
65
- }
66
- return false;
67
- }
68
- return true;
69
- }), debounceTime(this.draftDebounce), takeUntil(this.destroy$))
70
- .subscribe((values) => {
71
- this.saveDraft(values);
72
- });
73
- }
74
- ngAfterViewInit() {
75
- if (this.ngForm && this.formControl) {
76
- const draft = this.draftService.load(this.formId);
77
- if (draft) {
78
- this.isRestoring = true;
79
- }
80
- setTimeout(() => {
81
- if (draft) {
82
- this.restoreDraft(draft.values);
83
- this.showBanner(draft.savedAt, true);
84
- this.initialValues = {};
85
- }
86
- else {
87
- this.initialValues = JSON.parse(JSON.stringify(this.formControl.value));
88
- }
89
- }, 0);
90
- }
91
- }
92
- ngOnDestroy() {
93
- this.draftService.unregisterReset(this.formId);
94
- this.destroy$.next();
95
- this.destroy$.complete();
96
- this.destroyBanner();
97
- }
98
- saveDraft(values) {
99
- const filtered = this.filterFields(values);
100
- // Don't save if empty
101
- if (this.isAllEmpty(filtered)) {
102
- return;
103
- }
104
- // Don't save if matches initial values (even if initial is empty)
105
- if (this.matchesInitialValues(filtered)) {
106
- return;
107
- }
108
- this.draftService.save(this.formId, filtered);
109
- if (this.draftShowOnChange && !this.bannerRef) {
110
- this.showBanner(Date.now(), false);
111
- }
112
- }
113
- filterFields(values) {
114
- if (!this.draftExcludeFields.length)
115
- return values;
116
- const result = { ...values };
117
- this.draftExcludeFields.forEach(field => delete result[field]);
118
- return result;
119
- }
120
- isAllEmpty(values) {
121
- return Object.values(values).every(v => v === null || v === undefined || v === '' || (Array.isArray(v) && v.length === 0));
122
- }
123
- matchesInitialValues(values) {
124
- return JSON.stringify(values) === JSON.stringify(this.initialValues);
125
- }
126
- restoreDraft(values) {
127
- if (!this.formControl)
128
- return;
129
- this.isRestoring = true;
130
- const form = this.formGroupDir?.form || this.ngForm?.form;
131
- if (form) {
132
- this.prepareFormArrays(form, values);
133
- form.patchValue(values);
134
- }
135
- setTimeout(() => this.isRestoring = false, 100);
136
- }
137
- prepareFormArrays(control, value) {
138
- if (!control || value == null)
139
- return;
140
- if (control instanceof FormGroup && value && typeof value === 'object' && !Array.isArray(value)) {
141
- Object.keys(value).forEach(key => {
142
- const childControl = control.get(key);
143
- if (childControl)
144
- this.prepareFormArrays(childControl, value[key]);
145
- });
146
- return;
147
- }
148
- if (control instanceof FormArray && Array.isArray(value)) {
149
- const formArray = control;
150
- while (formArray.length < value.length) {
151
- const template = formArray.at(0);
152
- if (template instanceof FormGroup) {
153
- const newGroup = new FormGroup({});
154
- Object.keys(template.controls).forEach(ctrlName => {
155
- const existing = template.get(ctrlName);
156
- if (existing instanceof FormArray) {
157
- newGroup.addControl(ctrlName, new FormArray([]));
158
- }
159
- else if (existing instanceof FormGroup) {
160
- newGroup.addControl(ctrlName, new FormGroup({}));
161
- }
162
- else {
163
- newGroup.addControl(ctrlName, new FormControl(null));
164
- }
165
- });
166
- formArray.push(newGroup);
167
- }
168
- else if (template) {
169
- formArray.push(new FormControl(null));
170
- }
171
- else {
172
- const firstValue = value[0];
173
- if (firstValue && typeof firstValue === 'object' && !Array.isArray(firstValue)) {
174
- const group = new FormGroup({});
175
- Object.keys(firstValue).forEach(key => group.addControl(key, new FormControl(null)));
176
- formArray.push(group);
177
- }
178
- else {
179
- formArray.push(new FormControl(null));
180
- }
181
- }
182
- }
183
- while (formArray.length > value.length) {
184
- formArray.removeAt(formArray.length - 1);
185
- }
186
- value.forEach((childValue, index) => {
187
- this.prepareFormArrays(formArray.at(index), childValue);
188
- });
189
- }
190
- }
191
- discardDraft() {
192
- this.draftService.clear(this.formId);
193
- this.performResetAndDestroyBanner();
194
- }
195
- /**
196
- * Resets the form to initial values and destroys the draft banner.
197
- * Used by the service when clearAndReset(formId) is called (e.g. on submit).
198
- */
199
- performResetAndDestroyBanner() {
200
- if (this.formControl) {
201
- this.isRestoring = true;
202
- const form = this.formGroupDir?.form || this.ngForm?.form;
203
- if (form) {
204
- this.prepareFormArrays(form, this.initialValues);
205
- form.reset(this.initialValues);
206
- }
207
- setTimeout(() => {
208
- this.isRestoring = false;
209
- }, this.draftDebounce + 200);
210
- }
211
- this.destroyBanner();
212
- }
213
- showBanner(savedAt, isRestored = false) {
214
- this.bannerRef = this.viewContainerRef.createComponent(FormDraftBannerComponent);
215
- this.bannerRef.setInput('visible', true);
216
- this.bannerRef.setInput('isRestored', isRestored);
217
- this.bannerRef.setInput('timeLabel', isRestored ? this.draftService.formatTimestamp(savedAt) : '');
218
- this.bannerRef.setInput('restoredText', this.draftRestoredText);
219
- this.bannerRef.setInput('savedText', this.draftSavedText);
220
- this.bannerRef.setInput('savedLabel', this.draftSavedLabel);
221
- this.bannerRef.setInput('discardText', this.draftDiscardText);
222
- this.bannerRef.instance.discard.subscribe(() => this.discardDraft());
223
- const bannerEl = this.bannerRef.location.nativeElement;
224
- const formEl = this.elRef.nativeElement;
225
- this.renderer.insertBefore(formEl, bannerEl, formEl.firstChild);
226
- this.cdRef.detectChanges();
227
- }
228
- destroyBanner() {
229
- if (this.bannerRef) {
230
- this.bannerRef.destroy();
231
- this.bannerRef = null;
232
- }
233
- }
234
- clearDraft() {
235
- this.draftService.clear(this.formId);
236
- this.destroyBanner();
237
- }
238
- }
239
- FormDraftDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FormDraftDirective, deps: [{ token: i1.FormGroupDirective, optional: true }, { token: i1.NgForm, optional: true }, { token: i2.FormDraftService }, { token: i0.ViewContainerRef }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive });
240
- FormDraftDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.3.0", type: FormDraftDirective, selector: "[ngxFormDraft]", inputs: { formId: ["ngxFormDraft", "formId"], draftDebounce: "draftDebounce", draftExcludeFields: "draftExcludeFields", draftShowOnChange: "draftShowOnChange", draftRestoredText: "draftRestoredText", draftSavedText: "draftSavedText", draftSavedLabel: "draftSavedLabel", draftDiscardText: "draftDiscardText" }, ngImport: i0 });
241
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FormDraftDirective, decorators: [{
242
- type: Directive,
243
- args: [{
244
- selector: '[ngxFormDraft]',
245
- }]
246
- }], ctorParameters: function () { return [{ type: i1.FormGroupDirective, decorators: [{
247
- type: Optional
248
- }] }, { type: i1.NgForm, decorators: [{
249
- type: Optional
250
- }] }, { type: i2.FormDraftService }, { type: i0.ViewContainerRef }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i0.Renderer2 }]; }, propDecorators: { formId: [{
251
- type: Input,
252
- args: ['ngxFormDraft']
253
- }], draftDebounce: [{
254
- type: Input
255
- }], draftExcludeFields: [{
256
- type: Input
257
- }], draftShowOnChange: [{
258
- type: Input
259
- }], draftRestoredText: [{
260
- type: Input
261
- }], draftSavedText: [{
262
- type: Input
263
- }], draftSavedLabel: [{
264
- type: Input
265
- }], draftDiscardText: [{
266
- type: Input
267
- }] } });
1
+ import { Directive, Input, Optional, } from '@angular/core';
2
+ import { FormArray, FormGroup, FormControl } from '@angular/forms';
3
+ import { Subject } from 'rxjs';
4
+ import { debounceTime, takeUntil, filter } from 'rxjs/operators';
5
+ import { FormDraftBannerComponent } from './form-draft-banner.component';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/forms";
8
+ import * as i2 from "./form-draft.service";
9
+ /**
10
+ * Auto-saves and restores form drafts
11
+ *
12
+ * @example
13
+ * <form [formGroup]="myForm" ngxFormDraft="myFormId">
14
+ *
15
+ * @example
16
+ * <form [formGroup]="myForm" [ngxFormDraft]="'edit_' + entityId" [draftExcludeFields]="['password']">
17
+ */
18
+ export class FormDraftDirective {
19
+ constructor(formGroupDir, ngForm, draftService, viewContainerRef, cdRef, elRef, renderer) {
20
+ this.formGroupDir = formGroupDir;
21
+ this.ngForm = ngForm;
22
+ this.draftService = draftService;
23
+ this.viewContainerRef = viewContainerRef;
24
+ this.cdRef = cdRef;
25
+ this.elRef = elRef;
26
+ this.renderer = renderer;
27
+ this.draftDebounce = 800;
28
+ this.draftExcludeFields = [];
29
+ this.draftShowOnChange = false;
30
+ this.draftRestoredText = 'Draft restored';
31
+ this.draftSavedText = 'Draft saved';
32
+ this.draftSavedLabel = 'saved';
33
+ this.draftDiscardText = 'Discard';
34
+ this.destroy$ = new Subject();
35
+ this.bannerRef = null;
36
+ this.formControl = null;
37
+ this.initialValues = {};
38
+ this.isRestoring = false;
39
+ }
40
+ ngOnInit() {
41
+ this.formControl = this.formGroupDir?.form || this.ngForm?.form || null;
42
+ if (!this.formControl || !this.formId)
43
+ return;
44
+ this.draftService.registerReset(this.formId, () => this.performResetAndDestroyBanner());
45
+ // For reactive forms, capture initial values and restore draft immediately
46
+ if (this.formGroupDir) {
47
+ this.initialValues = JSON.parse(JSON.stringify(this.formControl.value));
48
+ const draft = this.draftService.load(this.formId);
49
+ if (draft) {
50
+ this.restoreDraft(draft.values);
51
+ this.showBanner(draft.savedAt, true);
52
+ }
53
+ }
54
+ let hasUserInteraction = false;
55
+ this.formControl.valueChanges
56
+ .pipe(filter(() => {
57
+ if (this.isRestoring)
58
+ return false;
59
+ if (this.ngForm && !hasUserInteraction) {
60
+ const currentValues = this.formControl?.value || {};
61
+ const isDifferent = JSON.stringify(currentValues) !== JSON.stringify(this.initialValues);
62
+ if (isDifferent) {
63
+ hasUserInteraction = true;
64
+ return true;
65
+ }
66
+ return false;
67
+ }
68
+ return true;
69
+ }), debounceTime(this.draftDebounce), takeUntil(this.destroy$))
70
+ .subscribe((values) => {
71
+ this.saveDraft(values);
72
+ });
73
+ }
74
+ ngAfterViewInit() {
75
+ if (this.ngForm && this.formControl) {
76
+ const draft = this.draftService.load(this.formId);
77
+ if (draft) {
78
+ this.isRestoring = true;
79
+ }
80
+ setTimeout(() => {
81
+ if (draft) {
82
+ this.restoreDraft(draft.values);
83
+ this.showBanner(draft.savedAt, true);
84
+ this.initialValues = {};
85
+ }
86
+ else {
87
+ this.initialValues = JSON.parse(JSON.stringify(this.formControl.value));
88
+ }
89
+ }, 0);
90
+ }
91
+ }
92
+ ngOnDestroy() {
93
+ this.draftService.unregisterReset(this.formId);
94
+ this.destroy$.next();
95
+ this.destroy$.complete();
96
+ this.destroyBanner();
97
+ }
98
+ saveDraft(values) {
99
+ const filtered = this.filterFields(values);
100
+ // Don't save if empty
101
+ if (this.isAllEmpty(filtered)) {
102
+ return;
103
+ }
104
+ // Don't save if matches initial values (even if initial is empty)
105
+ if (this.matchesInitialValues(filtered)) {
106
+ return;
107
+ }
108
+ this.draftService.save(this.formId, filtered);
109
+ if (this.draftShowOnChange && !this.bannerRef) {
110
+ this.showBanner(Date.now(), false);
111
+ }
112
+ }
113
+ filterFields(values) {
114
+ if (!this.draftExcludeFields.length)
115
+ return values;
116
+ const result = { ...values };
117
+ this.draftExcludeFields.forEach(field => delete result[field]);
118
+ return result;
119
+ }
120
+ isAllEmpty(values) {
121
+ return Object.values(values).every(v => v === null || v === undefined || v === '' || (Array.isArray(v) && v.length === 0));
122
+ }
123
+ matchesInitialValues(values) {
124
+ return JSON.stringify(values) === JSON.stringify(this.initialValues);
125
+ }
126
+ restoreDraft(values) {
127
+ if (!this.formControl)
128
+ return;
129
+ this.isRestoring = true;
130
+ const form = this.formGroupDir?.form || this.ngForm?.form;
131
+ if (form) {
132
+ this.prepareFormArrays(form, values);
133
+ form.patchValue(values);
134
+ }
135
+ setTimeout(() => this.isRestoring = false, 100);
136
+ }
137
+ prepareFormArrays(control, value) {
138
+ if (!control || value == null)
139
+ return;
140
+ if (control instanceof FormGroup && value && typeof value === 'object' && !Array.isArray(value)) {
141
+ Object.keys(value).forEach(key => {
142
+ const childControl = control.get(key);
143
+ if (childControl)
144
+ this.prepareFormArrays(childControl, value[key]);
145
+ });
146
+ return;
147
+ }
148
+ if (control instanceof FormArray && Array.isArray(value)) {
149
+ const formArray = control;
150
+ while (formArray.length < value.length) {
151
+ const template = formArray.at(0);
152
+ if (template instanceof FormGroup) {
153
+ const newGroup = new FormGroup({});
154
+ Object.keys(template.controls).forEach(ctrlName => {
155
+ const existing = template.get(ctrlName);
156
+ if (existing instanceof FormArray) {
157
+ newGroup.addControl(ctrlName, new FormArray([]));
158
+ }
159
+ else if (existing instanceof FormGroup) {
160
+ newGroup.addControl(ctrlName, new FormGroup({}));
161
+ }
162
+ else {
163
+ newGroup.addControl(ctrlName, new FormControl(null));
164
+ }
165
+ });
166
+ formArray.push(newGroup);
167
+ }
168
+ else if (template) {
169
+ formArray.push(new FormControl(null));
170
+ }
171
+ else {
172
+ const firstValue = value[0];
173
+ if (firstValue && typeof firstValue === 'object' && !Array.isArray(firstValue)) {
174
+ const group = new FormGroup({});
175
+ Object.keys(firstValue).forEach(key => group.addControl(key, new FormControl(null)));
176
+ formArray.push(group);
177
+ }
178
+ else {
179
+ formArray.push(new FormControl(null));
180
+ }
181
+ }
182
+ }
183
+ while (formArray.length > value.length) {
184
+ formArray.removeAt(formArray.length - 1);
185
+ }
186
+ value.forEach((childValue, index) => {
187
+ this.prepareFormArrays(formArray.at(index), childValue);
188
+ });
189
+ }
190
+ }
191
+ discardDraft() {
192
+ this.draftService.clear(this.formId);
193
+ this.performResetAndDestroyBanner();
194
+ }
195
+ /**
196
+ * Resets the form to initial values and destroys the draft banner.
197
+ * Used by the service when clearAndReset(formId) is called (e.g. on submit).
198
+ */
199
+ performResetAndDestroyBanner() {
200
+ if (this.formControl) {
201
+ this.isRestoring = true;
202
+ const form = this.formGroupDir?.form || this.ngForm?.form;
203
+ if (form) {
204
+ this.prepareFormArrays(form, this.initialValues);
205
+ form.reset(this.initialValues);
206
+ }
207
+ setTimeout(() => {
208
+ this.isRestoring = false;
209
+ }, this.draftDebounce + 200);
210
+ }
211
+ this.destroyBanner();
212
+ }
213
+ showBanner(savedAt, isRestored = false) {
214
+ this.bannerRef = this.viewContainerRef.createComponent(FormDraftBannerComponent);
215
+ this.bannerRef.setInput('visible', true);
216
+ this.bannerRef.setInput('isRestored', isRestored);
217
+ this.bannerRef.setInput('timeLabel', isRestored ? this.draftService.formatTimestamp(savedAt) : '');
218
+ this.bannerRef.setInput('restoredText', this.draftRestoredText);
219
+ this.bannerRef.setInput('savedText', this.draftSavedText);
220
+ this.bannerRef.setInput('savedLabel', this.draftSavedLabel);
221
+ this.bannerRef.setInput('discardText', this.draftDiscardText);
222
+ this.bannerRef.instance.discard.subscribe(() => this.discardDraft());
223
+ const bannerEl = this.bannerRef.location.nativeElement;
224
+ const formEl = this.elRef.nativeElement;
225
+ this.renderer.insertBefore(formEl, bannerEl, formEl.firstChild);
226
+ this.cdRef.detectChanges();
227
+ }
228
+ destroyBanner() {
229
+ if (this.bannerRef) {
230
+ this.bannerRef.destroy();
231
+ this.bannerRef = null;
232
+ }
233
+ }
234
+ clearDraft() {
235
+ this.draftService.clear(this.formId);
236
+ this.destroyBanner();
237
+ }
238
+ }
239
+ FormDraftDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FormDraftDirective, deps: [{ token: i1.FormGroupDirective, optional: true }, { token: i1.NgForm, optional: true }, { token: i2.FormDraftService }, { token: i0.ViewContainerRef }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive });
240
+ FormDraftDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.3.0", type: FormDraftDirective, selector: "[ngxFormDraft]", inputs: { formId: ["ngxFormDraft", "formId"], draftDebounce: "draftDebounce", draftExcludeFields: "draftExcludeFields", draftShowOnChange: "draftShowOnChange", draftRestoredText: "draftRestoredText", draftSavedText: "draftSavedText", draftSavedLabel: "draftSavedLabel", draftDiscardText: "draftDiscardText" }, ngImport: i0 });
241
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FormDraftDirective, decorators: [{
242
+ type: Directive,
243
+ args: [{
244
+ selector: '[ngxFormDraft]',
245
+ }]
246
+ }], ctorParameters: function () { return [{ type: i1.FormGroupDirective, decorators: [{
247
+ type: Optional
248
+ }] }, { type: i1.NgForm, decorators: [{
249
+ type: Optional
250
+ }] }, { type: i2.FormDraftService }, { type: i0.ViewContainerRef }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i0.Renderer2 }]; }, propDecorators: { formId: [{
251
+ type: Input,
252
+ args: ['ngxFormDraft']
253
+ }], draftDebounce: [{
254
+ type: Input
255
+ }], draftExcludeFields: [{
256
+ type: Input
257
+ }], draftShowOnChange: [{
258
+ type: Input
259
+ }], draftRestoredText: [{
260
+ type: Input
261
+ }], draftSavedText: [{
262
+ type: Input
263
+ }], draftSavedLabel: [{
264
+ type: Input
265
+ }], draftDiscardText: [{
266
+ type: Input
267
+ }] } });
268
268
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-draft.directive.js","sourceRoot":"","sources":["../../src/form-draft.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive, Input, OnInit, AfterViewInit, OnDestroy, Optional, ComponentRef,\n  ViewContainerRef, ChangeDetectorRef, ElementRef, Renderer2,\n} from '@angular/core';\nimport { NgForm, FormGroupDirective, AbstractControl, FormArray, FormGroup, FormControl } from '@angular/forms';\nimport { Subject } from 'rxjs';\nimport { debounceTime, takeUntil, skip, filter } from 'rxjs/operators';\nimport { FormDraftService } from './form-draft.service';\nimport { FormDraftBannerComponent } from './form-draft-banner.component';\n\n/**\n * Auto-saves and restores form drafts\n * \n * @example\n * <form [formGroup]=\"myForm\" ngxFormDraft=\"myFormId\">\n * \n * @example\n * <form [formGroup]=\"myForm\" [ngxFormDraft]=\"'edit_' + entityId\" [draftExcludeFields]=\"['password']\">\n */\n@Directive({\n  selector: '[ngxFormDraft]',\n})\nexport class FormDraftDirective implements OnInit, AfterViewInit, OnDestroy {\n  @Input('ngxFormDraft') formId!: string;\n  @Input() draftDebounce = 800;\n  @Input() draftExcludeFields: string[] = [];\n  @Input() draftShowOnChange = false;\n  @Input() draftRestoredText = 'Draft restored';\n  @Input() draftSavedText = 'Draft saved';\n  @Input() draftSavedLabel = 'saved';\n  @Input() draftDiscardText = 'Discard';\n\n  private destroy$ = new Subject<void>();\n  private bannerRef: ComponentRef<FormDraftBannerComponent> | null = null;\n  private formControl: AbstractControl | null = null;\n  private initialValues: Record<string, any> = {};\n  private isRestoring = false;\n\n  constructor(\n    @Optional() private formGroupDir: FormGroupDirective,\n    @Optional() private ngForm: NgForm,\n    private draftService: FormDraftService,\n    private viewContainerRef: ViewContainerRef,\n    private cdRef: ChangeDetectorRef,\n    private elRef: ElementRef,\n    private renderer: Renderer2,\n  ) {}\n\n  ngOnInit(): void {\n    this.formControl = this.formGroupDir?.form || this.ngForm?.form || null;\n    if (!this.formControl || !this.formId) return;\n\n    this.draftService.registerReset(this.formId, () => this.performResetAndDestroyBanner());\n\n    // For reactive forms, capture initial values and restore draft immediately\n    if (this.formGroupDir) {\n      this.initialValues = JSON.parse(JSON.stringify(this.formControl.value));\n      \n      const draft = this.draftService.load(this.formId);\n      if (draft) {\n        this.restoreDraft(draft.values);\n        this.showBanner(draft.savedAt, true);\n      }\n    }\n\n    let hasUserInteraction = false;\n    \n    this.formControl.valueChanges\n      .pipe(\n        filter(() => {\n          if (this.isRestoring) return false;\n          if (this.ngForm && !hasUserInteraction) {\n            const currentValues = this.formControl?.value || {};\n            const isDifferent = JSON.stringify(currentValues) !== JSON.stringify(this.initialValues);\n            if (isDifferent) {\n              hasUserInteraction = true;\n              return true;\n            }\n            return false;\n          }\n          return true;\n        }),\n        debounceTime(this.draftDebounce),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((values) => {\n        this.saveDraft(values);\n      });\n  }\n\n  ngAfterViewInit(): void {\n    if (this.ngForm && this.formControl) {\n      const draft = this.draftService.load(this.formId);\n      \n      if (draft) {\n        this.isRestoring = true;\n      }\n      \n      setTimeout(() => {\n        if (draft) {\n          this.restoreDraft(draft.values);\n          this.showBanner(draft.savedAt, true);\n          this.initialValues = {};\n        } else {\n          this.initialValues = JSON.parse(JSON.stringify(this.formControl!.value));\n        }\n      }, 0);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.draftService.unregisterReset(this.formId);\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.destroyBanner();\n  }\n\n  private saveDraft(values: Record<string, any>): void {\n    const filtered = this.filterFields(values);\n    \n    // Don't save if empty\n    if (this.isAllEmpty(filtered)) {\n      return;\n    }\n\n    // Don't save if matches initial values (even if initial is empty)\n    if (this.matchesInitialValues(filtered)) {\n      return;\n    }\n\n    this.draftService.save(this.formId, filtered);\n    if (this.draftShowOnChange && !this.bannerRef) {\n      this.showBanner(Date.now(), false);\n    }\n  }\n\n  private filterFields(values: Record<string, any>): Record<string, any> {\n    if (!this.draftExcludeFields.length) return values;\n    const result = { ...values };\n    this.draftExcludeFields.forEach(field => delete result[field]);\n    return result;\n  }\n\n  private isAllEmpty(values: Record<string, any>): boolean {\n    return Object.values(values).every(\n      v => v === null || v === undefined || v === '' || (Array.isArray(v) && v.length === 0)\n    );\n  }\n\n  private matchesInitialValues(values: Record<string, any>): boolean {\n    return JSON.stringify(values) === JSON.stringify(this.initialValues);\n  }\n\n  private restoreDraft(values: Record<string, any>): void {\n    if (!this.formControl) return;\n    this.isRestoring = true;\n\n    const form = this.formGroupDir?.form || this.ngForm?.form;\n    if (form) {\n      this.prepareFormArrays(form, values);\n      form.patchValue(values);\n    }\n\n    setTimeout(() => this.isRestoring = false, 100);\n  }\n\n  private prepareFormArrays(control: AbstractControl, value: any): void {\n    if (!control || value == null) return;\n\n    if (control instanceof FormGroup && value && typeof value === 'object' && !Array.isArray(value)) {\n      Object.keys(value).forEach(key => {\n        const childControl = control.get(key);\n        if (childControl) this.prepareFormArrays(childControl, value[key]);\n      });\n      return;\n    }\n\n    if (control instanceof FormArray && Array.isArray(value)) {\n      const formArray = control as FormArray;\n\n      while (formArray.length < value.length) {\n        const template = formArray.at(0);\n        if (template instanceof FormGroup) {\n          const newGroup = new FormGroup({});\n          Object.keys(template.controls).forEach(ctrlName => {\n            const existing = template.get(ctrlName) as AbstractControl;\n            if (existing instanceof FormArray) {\n              newGroup.addControl(ctrlName, new FormArray([]));\n            } else if (existing instanceof FormGroup) {\n              newGroup.addControl(ctrlName, new FormGroup({}));\n            } else {\n              newGroup.addControl(ctrlName, new FormControl(null));\n            }\n          });\n          formArray.push(newGroup);\n        } else if (template) {\n          formArray.push(new FormControl(null));\n        } else {\n          const firstValue = value[0];\n          if (firstValue && typeof firstValue === 'object' && !Array.isArray(firstValue)) {\n            const group = new FormGroup({});\n            Object.keys(firstValue).forEach(key => group.addControl(key, new FormControl(null)));\n            formArray.push(group);\n          } else {\n            formArray.push(new FormControl(null));\n          }\n        }\n      }\n\n      while (formArray.length > value.length) {\n        formArray.removeAt(formArray.length - 1);\n      }\n\n      value.forEach((childValue, index) => {\n        this.prepareFormArrays(formArray.at(index), childValue);\n      });\n    }\n  }\n\n  private discardDraft(): void {\n    this.draftService.clear(this.formId);\n    this.performResetAndDestroyBanner();\n  }\n\n  /**\n   * Resets the form to initial values and destroys the draft banner.\n   * Used by the service when clearAndReset(formId) is called (e.g. on submit).\n   */\n  public performResetAndDestroyBanner(): void {\n    if (this.formControl) {\n      this.isRestoring = true;\n      const form = this.formGroupDir?.form || this.ngForm?.form;\n      if (form) {\n        this.prepareFormArrays(form, this.initialValues);\n        form.reset(this.initialValues);\n      }\n      setTimeout(() => {\n        this.isRestoring = false;\n      }, this.draftDebounce + 200);\n    }\n    this.destroyBanner();\n  }\n\n  private showBanner(savedAt: number, isRestored = false): void {\n    this.bannerRef = this.viewContainerRef.createComponent(FormDraftBannerComponent);\n    this.bannerRef.setInput('visible', true);\n    this.bannerRef.setInput('isRestored', isRestored);\n    this.bannerRef.setInput('timeLabel', isRestored ? this.draftService.formatTimestamp(savedAt) : '');\n    this.bannerRef.setInput('restoredText', this.draftRestoredText);\n    this.bannerRef.setInput('savedText', this.draftSavedText);\n    this.bannerRef.setInput('savedLabel', this.draftSavedLabel);\n    this.bannerRef.setInput('discardText', this.draftDiscardText);\n\n    this.bannerRef.instance.discard.subscribe(() => this.discardDraft());\n\n    const bannerEl = this.bannerRef.location.nativeElement;\n    const formEl = this.elRef.nativeElement;\n    this.renderer.insertBefore(formEl, bannerEl, formEl.firstChild);\n\n    this.cdRef.detectChanges();\n  }\n\n  private destroyBanner(): void {\n    if (this.bannerRef) {\n      this.bannerRef.destroy();\n      this.bannerRef = null;\n    }\n  }\n\n  public clearDraft(): void {\n    this.draftService.clear(this.formId);\n    this.destroyBanner();\n  }\n}\n"]}