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.
- package/LICENSE +21 -21
- package/README.md +206 -206
- package/esm2020/form-draft-banner.component.mjs +84 -84
- package/esm2020/form-draft.directive.mjs +267 -267
- package/esm2020/form-draft.service.mjs +92 -92
- package/esm2020/index.mjs +4 -4
- package/esm2020/ngx-form-draft.mjs +4 -4
- package/esm2020/ngx-form-draft.module.mjs +19 -19
- package/fesm2015/ngx-form-draft.mjs +446 -446
- package/fesm2015/ngx-form-draft.mjs.map +1 -1
- package/fesm2020/ngx-form-draft.mjs +439 -439
- package/fesm2020/ngx-form-draft.mjs.map +1 -1
- package/form-draft-banner.component.d.ts +14 -14
- package/form-draft.directive.d.ts +56 -56
- package/form-draft.service.d.ts +35 -35
- package/index.d.ts +4 -4
- package/ngx-form-draft.module.d.ts +9 -9
- package/package.json +1 -1
|
@@ -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"]}
|