@ecodev/natural 62.1.2 → 63.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
- package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
- package/fesm2022/ecodev-natural.mjs +436 -425
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/classes/network-activity.service.d.ts +54 -0
- package/lib/classes/validators.d.ts +1 -1
- package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
- package/lib/modules/common/services/seo.provider.d.ts +2 -2
- package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
- package/lib/modules/file/abstract-file.d.ts +6 -3
- package/lib/modules/file/component/file.component.d.ts +2 -2
- package/lib/modules/file/file-drop.directive.d.ts +2 -3
- package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
- package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
- package/lib/modules/icon/icon.module.d.ts +2 -2
- package/lib/modules/panels/panels.service.d.ts +1 -2
- package/lib/modules/relations/relations.component.d.ts +3 -3
- package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
- package/lib/modules/search/group/group.component.d.ts +2 -3
- package/lib/modules/search/input/input.component.d.ts +5 -5
- package/lib/modules/search/search/search.component.d.ts +2 -2
- package/lib/modules/select/abstract-select.component.d.ts +3 -3
- package/lib/modules/select/select/select.component.d.ts +1 -1
- package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
- package/lib/modules/table-button/table-button.component.d.ts +4 -2
- package/package.json +16 -14
- package/public-api.d.ts +1 -0
- package/src/lib/_natural.theme.scss +1 -2
- package/vanilla/index.d.ts +5 -0
- package/vanilla/package.json +3 -0
- package/vanilla/public-api.d.ts +11 -0
- package/vanilla/src/lib/classes/crypto.d.ts +8 -0
- package/vanilla/src/lib/classes/data-source.d.ts +32 -0
- package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
- package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
- package/vanilla/src/lib/classes/signing.d.ts +7 -0
- package/vanilla/src/lib/classes/utility.d.ts +77 -0
- package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
- package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
- package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
- package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
- package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
- package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
- package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
- package/vanilla/src/lib/types/types.d.ts +100 -0
- package/esm2022/ecodev-natural.mjs +0 -5
- package/esm2022/lib/classes/abstract-detail.mjs +0 -229
- package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
- package/esm2022/lib/classes/abstract-list.mjs +0 -461
- package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
- package/esm2022/lib/classes/apollo-utils.mjs +0 -59
- package/esm2022/lib/classes/crypto.mjs +0 -23
- package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
- package/esm2022/lib/classes/data-source.mjs +0 -71
- package/esm2022/lib/classes/providers.mjs +0 -13
- package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
- package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
- package/esm2022/lib/classes/rxjs.mjs +0 -54
- package/esm2022/lib/classes/signing.mjs +0 -38
- package/esm2022/lib/classes/tld.mjs +0 -1476
- package/esm2022/lib/classes/utility.mjs +0 -234
- package/esm2022/lib/classes/validators.mjs +0 -179
- package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
- package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
- package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
- package/esm2022/lib/modules/alert/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
- package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
- package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
- package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
- package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
- package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
- package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
- package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
- package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
- package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
- package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
- package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
- package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
- package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
- package/esm2022/lib/modules/common/public-api.mjs +0 -14
- package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
- package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
- package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
- package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
- package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
- package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
- package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
- package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
- package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
- package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
- package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
- package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
- package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
- package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
- package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
- package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
- package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
- package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
- package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
- package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
- package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
- package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
- package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
- package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
- package/esm2022/lib/modules/file/file.service.mjs +0 -43
- package/esm2022/lib/modules/file/public-api.mjs +0 -9
- package/esm2022/lib/modules/file/types.mjs +0 -2
- package/esm2022/lib/modules/file/utils.mjs +0 -129
- package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
- package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
- package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
- package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
- package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
- package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
- package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
- package/esm2022/lib/modules/icon/public-api.mjs +0 -6
- package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
- package/esm2022/lib/modules/logger/error.module.mjs +0 -22
- package/esm2022/lib/modules/logger/public-api.mjs +0 -6
- package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
- package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
- package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
- package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
- package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
- package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
- package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
- package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
- package/esm2022/lib/modules/panels/public-api.mjs +0 -11
- package/esm2022/lib/modules/panels/types.mjs +0 -3
- package/esm2022/lib/modules/relations/public-api.mjs +0 -5
- package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
- package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
- package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
- package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
- package/esm2022/lib/modules/search/classes/url.mjs +0 -53
- package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
- package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
- package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
- package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
- package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
- package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
- package/esm2022/lib/modules/search/public-api.mjs +0 -7
- package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
- package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
- package/esm2022/lib/modules/search/types/facet.mjs +0 -2
- package/esm2022/lib/modules/search/types/values.mjs +0 -2
- package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
- package/esm2022/lib/modules/select/public-api.mjs +0 -7
- package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
- package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
- package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
- package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
- package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
- package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
- package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
- package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
- package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
- package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
- package/esm2022/lib/services/abstract-model.service.mjs +0 -526
- package/esm2022/lib/services/debounce.service.mjs +0 -149
- package/esm2022/lib/services/enum.service.mjs +0 -64
- package/esm2022/lib/services/link-mutation.service.mjs +0 -154
- package/esm2022/lib/services/persistence.service.mjs +0 -115
- package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
- package/esm2022/lib/types/types.mjs +0 -2
- package/esm2022/public-api.mjs +0 -46
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { Observable } from 'rxjs';
|
|
2
|
+
import { UntypedModelService } from '../types/types';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Debounce subscriptions to update mutations, with the possibility to cancel one, flush one, or flush all of them.
|
|
6
|
+
*
|
|
7
|
+
* `modelService` is also used to separate objects by their types. So User with ID 1 is not confused with Product with ID 1.
|
|
8
|
+
*
|
|
9
|
+
* `id` must be the ID of the object that will be updated.
|
|
10
|
+
*/
|
|
11
|
+
export declare class NaturalDebounceService {
|
|
12
|
+
/**
|
|
13
|
+
* Stores the debounced update function
|
|
14
|
+
*/
|
|
15
|
+
private readonly allDebouncedUpdateCache;
|
|
16
|
+
/**
|
|
17
|
+
* Debounce the `modelService.updateNow()` mutation for a short time. If called multiple times with the same
|
|
18
|
+
* modelService and id, it will postpone the subscription to the mutation.
|
|
19
|
+
*
|
|
20
|
+
* All input variables for the same object (same service and ID) will be cumulated over time. So it is possible
|
|
21
|
+
* to update `field1`, then `field2`, and they will be batched into a single XHR including `field1` and `field2`.
|
|
22
|
+
*
|
|
23
|
+
* But it will always keep the same debouncing timeline.
|
|
24
|
+
*/
|
|
25
|
+
debounce<T extends UntypedModelService>(modelService: UntypedModelService, id: string, object: Parameters<T['updateNow']>[0]): ReturnType<T['updateNow']>;
|
|
26
|
+
cancelOne(modelService: UntypedModelService, id: string): void;
|
|
27
|
+
/**
|
|
28
|
+
* Immediately execute the pending update, if any.
|
|
29
|
+
*
|
|
30
|
+
* It should typically be called before resolving the object, to mutate it before re-fetching it from server.
|
|
31
|
+
*
|
|
32
|
+
* The returned observable will complete when the update completes, even if it errors.
|
|
33
|
+
*/
|
|
34
|
+
flushOne(modelService: UntypedModelService, id: string): Observable<void>;
|
|
35
|
+
/**
|
|
36
|
+
* Immediately execute all pending updates.
|
|
37
|
+
*
|
|
38
|
+
* It should typically be called before login out.
|
|
39
|
+
*
|
|
40
|
+
* The returned observable will complete when all updates complete, even if some of them error.
|
|
41
|
+
*/
|
|
42
|
+
flush(): Observable<void>;
|
|
43
|
+
private internalFlush;
|
|
44
|
+
/**
|
|
45
|
+
* Count of pending updates
|
|
46
|
+
*/
|
|
47
|
+
get count(): number;
|
|
48
|
+
private getMap;
|
|
49
|
+
private delete;
|
|
50
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<NaturalDebounceService, never>;
|
|
51
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<NaturalDebounceService>;
|
|
52
|
+
}
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import { PaginatedData } from '../classes/data-source';
|
|
2
|
+
import { NaturalAbstractModelService, VariablesWithInput } from '../services/abstract-model.service';
|
|
3
|
+
import { QueryVariables } from '../classes/query-variable-manager';
|
|
4
|
+
import { ObservedValueOf } from 'rxjs';
|
|
5
|
+
import { ResolveData } from '@angular/router';
|
|
6
|
+
/**
|
|
7
|
+
* An object literal with any keys and values
|
|
8
|
+
*/
|
|
9
|
+
export type Literal = Record<string, any>;
|
|
10
|
+
/**
|
|
11
|
+
* An object with either a name or a fullName (or maybe both)
|
|
12
|
+
*/
|
|
13
|
+
export type NameOrFullName = {
|
|
14
|
+
id: string;
|
|
15
|
+
name: string;
|
|
16
|
+
fullName?: string;
|
|
17
|
+
} | {
|
|
18
|
+
id: string;
|
|
19
|
+
name?: string;
|
|
20
|
+
fullName: string;
|
|
21
|
+
};
|
|
22
|
+
/**
|
|
23
|
+
* Extract the Tone type from a NaturalAbstractModelService
|
|
24
|
+
*/
|
|
25
|
+
export type ExtractTone<P> = P extends NaturalAbstractModelService<infer Tone, any, any, any, any, any, any, any, any, any> ? Tone : never;
|
|
26
|
+
/**
|
|
27
|
+
* Extract the Vone type from a NaturalAbstractModelService
|
|
28
|
+
*/
|
|
29
|
+
export type ExtractVone<P> = P extends NaturalAbstractModelService<any, infer Vone, any, any, any, any, any, any, any, any> ? Vone extends {
|
|
30
|
+
id: string;
|
|
31
|
+
} ? Vone : never : never;
|
|
32
|
+
/**
|
|
33
|
+
* Extract the Tall type from a NaturalAbstractModelService
|
|
34
|
+
*/
|
|
35
|
+
export type ExtractTall<P> = P extends NaturalAbstractModelService<any, any, infer Tall, any, any, any, any, any, any, any> ? Tall extends PaginatedData<Literal> ? Tall : never : never;
|
|
36
|
+
/**
|
|
37
|
+
* Extract the TallOne type for a single item coming from a list of items from a NaturalAbstractModelService
|
|
38
|
+
*/
|
|
39
|
+
export type ExtractTallOne<P> = P extends NaturalAbstractModelService<any, any, PaginatedData<infer TallOne extends Literal>, any, any, any, any, any, any, any> ? TallOne extends Literal ? TallOne : never : never;
|
|
40
|
+
/**
|
|
41
|
+
* Extract the Vall type from a NaturalAbstractModelService
|
|
42
|
+
*/
|
|
43
|
+
export type ExtractVall<P> = P extends NaturalAbstractModelService<any, any, any, infer Vall, any, any, any, any, any, any> ? Vall extends QueryVariables ? Vall : never : never;
|
|
44
|
+
/**
|
|
45
|
+
* Extract the Tcreate type from a NaturalAbstractModelService
|
|
46
|
+
*/
|
|
47
|
+
export type ExtractTcreate<P> = P extends NaturalAbstractModelService<any, any, any, any, infer Tcreate, any, any, any, any, any> ? Tcreate : never;
|
|
48
|
+
/**
|
|
49
|
+
* Extract the Vcreate type from a NaturalAbstractModelService
|
|
50
|
+
*/
|
|
51
|
+
export type ExtractVcreate<P> = P extends NaturalAbstractModelService<any, any, any, any, any, infer Vcreate, any, any, any, any> ? Vcreate extends VariablesWithInput ? Vcreate : never : never;
|
|
52
|
+
/**
|
|
53
|
+
* Extract the Tupdate type from a NaturalAbstractModelService
|
|
54
|
+
*/
|
|
55
|
+
export type ExtractTupdate<P> = P extends NaturalAbstractModelService<any, any, any, any, any, any, infer Tupdate, any, any, any> ? Tupdate : never;
|
|
56
|
+
/**
|
|
57
|
+
* Extract the Vupdate type from a NaturalAbstractModelService
|
|
58
|
+
*/
|
|
59
|
+
export type ExtractVupdate<P> = P extends NaturalAbstractModelService<any, any, any, any, any, any, any, infer Vupdate, any, any> ? Vupdate extends {
|
|
60
|
+
id: string;
|
|
61
|
+
input: Literal;
|
|
62
|
+
} ? Vupdate : never : never;
|
|
63
|
+
/**
|
|
64
|
+
* Extract the Tdelete type from a NaturalAbstractModelService
|
|
65
|
+
*/
|
|
66
|
+
export type ExtractTdelete<P> = P extends NaturalAbstractModelService<any, any, any, any, any, any, any, any, infer Tdelete, any> ? Tdelete : never;
|
|
67
|
+
/**
|
|
68
|
+
* Extract the Vdelete type from a NaturalAbstractModelService
|
|
69
|
+
*/
|
|
70
|
+
export type ExtractVdelete<P> = P extends NaturalAbstractModelService<any, any, any, any, any, any, any, any, any, infer Vdelete> ? Vdelete extends {
|
|
71
|
+
ids: string[];
|
|
72
|
+
} ? Vdelete : never : never;
|
|
73
|
+
/**
|
|
74
|
+
* Extract the resolve type from a NaturalAbstractModelService
|
|
75
|
+
*/
|
|
76
|
+
export type ExtractResolve<P> = P extends NaturalAbstractModelService<any, any, any, any, any, any, any, any, any, any> ? ObservedValueOf<ObservedValueOf<ReturnType<P['resolve']>>> : never;
|
|
77
|
+
/**
|
|
78
|
+
* This should be avoided if possible, and instead use a more precise type with some constraints on it to ensure that the model
|
|
79
|
+
* service is able to fulfill its requirements.
|
|
80
|
+
*/
|
|
81
|
+
export type UntypedModelService = NaturalAbstractModelService<any, any, any, any, any, any, any, any, any, any>;
|
|
82
|
+
/**
|
|
83
|
+
* Returns the resolved data type, as available in components, from the given resolvers
|
|
84
|
+
*
|
|
85
|
+
* Eg:
|
|
86
|
+
*
|
|
87
|
+
* ```ts
|
|
88
|
+
* const actionResolvers = {
|
|
89
|
+
* model: resolveAction,
|
|
90
|
+
* statuses: () => inject(NaturalEnumService).get('Status'),
|
|
91
|
+
* } as const;
|
|
92
|
+
*
|
|
93
|
+
* // In action.component.ts
|
|
94
|
+
* const data: ResolvedData<typeof actionResolvers>;
|
|
95
|
+
* data.statuses.forEach(...);
|
|
96
|
+
* ```
|
|
97
|
+
*/
|
|
98
|
+
export type ResolvedData<T extends ResolveData> = {
|
|
99
|
+
readonly [KeyType in keyof Pick<T, keyof T>]: ObservedValueOf<ReturnType<Pick<T, keyof T>[KeyType]>>;
|
|
100
|
+
};
|
|
@@ -1,5 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Generated bundle index. Do not edit.
|
|
3
|
-
*/
|
|
4
|
-
export * from './public-api';
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWNvZGV2LW5hdHVyYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9lY29kZXYtbmF0dXJhbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
|
|
@@ -1,229 +0,0 @@
|
|
|
1
|
-
import { Directive, inject } from '@angular/core';
|
|
2
|
-
import { UntypedFormGroup } from '@angular/forms';
|
|
3
|
-
import { ActivatedRoute, Router } from '@angular/router';
|
|
4
|
-
import { kebabCase } from 'lodash-es';
|
|
5
|
-
import { NaturalAlertService } from '../modules/alert/alert.service';
|
|
6
|
-
import { NaturalAbstractPanel } from '../modules/panels/abstract-panel';
|
|
7
|
-
import { EMPTY, endWith, finalize, last, Observable, switchMap, tap } from 'rxjs';
|
|
8
|
-
import { ifValid, validateAllFormControls } from './validators';
|
|
9
|
-
import { CumulativeChanges } from './cumulative-changes';
|
|
10
|
-
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
11
|
-
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
12
|
-
import * as i0 from "@angular/core";
|
|
13
|
-
function isNaturalDialogTriggerProvidedData(dialogData) {
|
|
14
|
-
return (!!dialogData &&
|
|
15
|
-
typeof dialogData === 'object' &&
|
|
16
|
-
'activatedRoute' in dialogData &&
|
|
17
|
-
dialogData.activatedRoute instanceof ActivatedRoute);
|
|
18
|
-
}
|
|
19
|
-
// @dynamic
|
|
20
|
-
export class NaturalAbstractDetail extends NaturalAbstractPanel {
|
|
21
|
-
key;
|
|
22
|
-
service;
|
|
23
|
-
/**
|
|
24
|
-
* Data retrieved by the server via route resolvers.
|
|
25
|
-
*
|
|
26
|
-
* The key `model` is special. It is readonly and represents the model being updated
|
|
27
|
-
* as it exists on server. The value is kept up to date when Apollo mutates it on server.
|
|
28
|
-
*
|
|
29
|
-
* The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.
|
|
30
|
-
*
|
|
31
|
-
* Other keys, if present, are whatever is returned from route resolvers as-is.
|
|
32
|
-
*/
|
|
33
|
-
data;
|
|
34
|
-
/**
|
|
35
|
-
* Form that manages the data from the controller
|
|
36
|
-
*/
|
|
37
|
-
form = new UntypedFormGroup({});
|
|
38
|
-
/**
|
|
39
|
-
* Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...
|
|
40
|
-
* semantic of data on other tab, like relations that list other objects)
|
|
41
|
-
*/
|
|
42
|
-
showFabButton = true;
|
|
43
|
-
alertService = inject(NaturalAlertService);
|
|
44
|
-
router = inject(Router);
|
|
45
|
-
route = inject(ActivatedRoute);
|
|
46
|
-
_dialogData = inject(MAT_DIALOG_DATA, { optional: true });
|
|
47
|
-
/**
|
|
48
|
-
* Once set, this must not change anymore, especially not right after the creation mutation,
|
|
49
|
-
* so the form does not switch from creation mode to update mode without an actual reload of
|
|
50
|
-
* model from DB (by navigating to update page).
|
|
51
|
-
*/
|
|
52
|
-
_isUpdatePage = false;
|
|
53
|
-
changes = new CumulativeChanges();
|
|
54
|
-
constructor(key, service) {
|
|
55
|
-
super();
|
|
56
|
-
this.key = key;
|
|
57
|
-
this.service = service;
|
|
58
|
-
this.data = {
|
|
59
|
-
model: this.service.getDefaultForServer(),
|
|
60
|
-
};
|
|
61
|
-
}
|
|
62
|
-
/**
|
|
63
|
-
* You probably should not override this method. Instead, consider overriding `initForm()`.
|
|
64
|
-
*/
|
|
65
|
-
ngOnInit() {
|
|
66
|
-
if (this.isPanel) {
|
|
67
|
-
this.initForm();
|
|
68
|
-
}
|
|
69
|
-
else {
|
|
70
|
-
const route = isNaturalDialogTriggerProvidedData(this._dialogData)
|
|
71
|
-
? this._dialogData.activatedRoute
|
|
72
|
-
: this.route;
|
|
73
|
-
this.#subscribeToModelFromResolvedData(route);
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
changeTab(index) {
|
|
77
|
-
this.showFabButton = index === 0;
|
|
78
|
-
}
|
|
79
|
-
#subscribeToModelFromResolvedData(route) {
|
|
80
|
-
let previousId = -1;
|
|
81
|
-
route.data
|
|
82
|
-
.pipe(switchMap(data => {
|
|
83
|
-
if (!(data.model instanceof Observable)) {
|
|
84
|
-
throw new Error('Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).');
|
|
85
|
-
}
|
|
86
|
-
// Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`
|
|
87
|
-
return data.model.pipe(tap((model) => {
|
|
88
|
-
this.data = {
|
|
89
|
-
...data,
|
|
90
|
-
model: model,
|
|
91
|
-
};
|
|
92
|
-
if (previousId !== model.id) {
|
|
93
|
-
previousId = model.id;
|
|
94
|
-
this.initForm();
|
|
95
|
-
}
|
|
96
|
-
}));
|
|
97
|
-
}), takeUntilDestroyed(this.destroyRef))
|
|
98
|
-
.subscribe();
|
|
99
|
-
}
|
|
100
|
-
/**
|
|
101
|
-
* Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object
|
|
102
|
-
* with (only) default values, so we are on a creation page.
|
|
103
|
-
*
|
|
104
|
-
* This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing
|
|
105
|
-
*/
|
|
106
|
-
isUpdatePage() {
|
|
107
|
-
return this._isUpdatePage;
|
|
108
|
-
}
|
|
109
|
-
/**
|
|
110
|
-
* Update the object on the server with the values from the form fields that were modified since
|
|
111
|
-
* the initialization, or since the previous successful update.
|
|
112
|
-
*
|
|
113
|
-
* Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.
|
|
114
|
-
*/
|
|
115
|
-
update(now = false, submitAllFields = false) {
|
|
116
|
-
if (!this.isUpdatePage()) {
|
|
117
|
-
return;
|
|
118
|
-
}
|
|
119
|
-
validateAllFormControls(this.form);
|
|
120
|
-
ifValid(this.form).subscribe(() => {
|
|
121
|
-
const newValues = this.form.getRawValue();
|
|
122
|
-
if (submitAllFields) {
|
|
123
|
-
this.changes.initialize({});
|
|
124
|
-
}
|
|
125
|
-
const toSubmit = {
|
|
126
|
-
id: this.data.model.id,
|
|
127
|
-
...this.changes.differences(newValues),
|
|
128
|
-
};
|
|
129
|
-
const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);
|
|
130
|
-
update.subscribe(model => {
|
|
131
|
-
this.changes.commit(newValues);
|
|
132
|
-
this.alertService.info($localize `Mis à jour`);
|
|
133
|
-
this.postUpdate(model);
|
|
134
|
-
});
|
|
135
|
-
});
|
|
136
|
-
}
|
|
137
|
-
create(redirect = true) {
|
|
138
|
-
validateAllFormControls(this.form);
|
|
139
|
-
if (!this.form.valid) {
|
|
140
|
-
return;
|
|
141
|
-
}
|
|
142
|
-
const newValues = this.form.getRawValue();
|
|
143
|
-
this.form.disable();
|
|
144
|
-
this.service
|
|
145
|
-
.create(newValues)
|
|
146
|
-
.pipe(switchMap(model => {
|
|
147
|
-
this.alertService.info($localize `Créé`);
|
|
148
|
-
return this.postCreate(model).pipe(endWith(model), last());
|
|
149
|
-
}), switchMap((model) => {
|
|
150
|
-
if (redirect) {
|
|
151
|
-
if (this.isPanel) {
|
|
152
|
-
const oldUrl = this.router.url;
|
|
153
|
-
const nextUrl = this.panelData?.config.params.nextRoute;
|
|
154
|
-
const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');
|
|
155
|
-
return this.router.navigateByUrl(newUrl); // replace /new by /123
|
|
156
|
-
}
|
|
157
|
-
else {
|
|
158
|
-
const queryParams = this.route.snapshot.queryParams;
|
|
159
|
-
return this.router.navigate(['..', model.id], { relativeTo: this.route, queryParams });
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
return EMPTY;
|
|
163
|
-
}), finalize(() => this.form.enable()))
|
|
164
|
-
.subscribe();
|
|
165
|
-
}
|
|
166
|
-
/**
|
|
167
|
-
* `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog
|
|
168
|
-
*/
|
|
169
|
-
delete(redirectionRoute, confirmer) {
|
|
170
|
-
this.form.disable();
|
|
171
|
-
(confirmer ??
|
|
172
|
-
this.alertService.confirm($localize `Suppression`, $localize `Voulez-vous supprimer définitivement cet élément ?`, $localize `Supprimer définitivement`))
|
|
173
|
-
.pipe(switchMap(confirmed => {
|
|
174
|
-
if (!confirmed || !this.isUpdatePage()) {
|
|
175
|
-
return EMPTY;
|
|
176
|
-
}
|
|
177
|
-
this.preDelete(this.data.model);
|
|
178
|
-
return this.service.delete([this.data.model]).pipe(switchMap(() => {
|
|
179
|
-
this.alertService.info($localize `Supprimé`);
|
|
180
|
-
if (this.isPanel) {
|
|
181
|
-
this.panelService?.goToPenultimatePanel();
|
|
182
|
-
return EMPTY;
|
|
183
|
-
}
|
|
184
|
-
else {
|
|
185
|
-
const defaultRoute = ['../../' + kebabCase(this.key)];
|
|
186
|
-
return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {
|
|
187
|
-
relativeTo: this.route,
|
|
188
|
-
});
|
|
189
|
-
}
|
|
190
|
-
}));
|
|
191
|
-
}), finalize(() => this.form.enable()))
|
|
192
|
-
.subscribe();
|
|
193
|
-
}
|
|
194
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
195
|
-
postUpdate(model) {
|
|
196
|
-
// noop
|
|
197
|
-
}
|
|
198
|
-
/**
|
|
199
|
-
* Returns an observable that will be subscribed to immediately and the
|
|
200
|
-
* redirect navigation will only happen after the observable completes.
|
|
201
|
-
*/
|
|
202
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
203
|
-
postCreate(model) {
|
|
204
|
-
return EMPTY;
|
|
205
|
-
}
|
|
206
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
207
|
-
preDelete(model) {
|
|
208
|
-
// noop
|
|
209
|
-
}
|
|
210
|
-
/**
|
|
211
|
-
* Initialize the form whenever it is needed.
|
|
212
|
-
*
|
|
213
|
-
* You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will
|
|
214
|
-
* correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`
|
|
215
|
-
* will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.
|
|
216
|
-
*/
|
|
217
|
-
initForm() {
|
|
218
|
-
this._isUpdatePage = !!this.data.model.id;
|
|
219
|
-
this.form = this.service.getFormGroup(this.data.model);
|
|
220
|
-
this.changes.initialize(this.form.getRawValue());
|
|
221
|
-
}
|
|
222
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractDetail, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive });
|
|
223
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: NaturalAbstractDetail, isStandalone: true, usesInheritance: true, ngImport: i0 });
|
|
224
|
-
}
|
|
225
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractDetail, decorators: [{
|
|
226
|
-
type: Directive,
|
|
227
|
-
args: [{ standalone: true }]
|
|
228
|
-
}], ctorParameters: () => [{ type: undefined }, { type: undefined }] });
|
|
229
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-detail.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-detail.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase} from 'lodash-es';\nimport {NaturalAlertService} from '../modules/alert/alert.service';\nimport {NaturalAbstractPanel} from '../modules/panels/abstract-panel';\nimport {NaturalAbstractModelService, WithId} from '../services/abstract-model.service';\nimport {ExtractResolve, ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {EMPTY, endWith, finalize, last, Observable, switchMap, tap} from 'rxjs';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {CumulativeChanges} from './cumulative-changes';\nimport {MAT_DIALOG_DATA} from '@angular/material/dialog';\nimport {NaturalDialogTriggerProvidedData} from '../modules/dialog-trigger/dialog-trigger.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * `Data` contains in `model` either the model fetched from DB or default values (without ID). And besides `model`,\n * any other extra keys defined by Extra.\n */\ntype Data<TService, Extra> = {model: {id?: string} & ExtractResolve<TService>} & Extra;\n\nfunction isNaturalDialogTriggerProvidedData(\n    dialogData: unknown,\n): dialogData is NaturalDialogTriggerProvidedData<never> {\n    return (\n        !!dialogData &&\n        typeof dialogData === 'object' &&\n        'activatedRoute' in dialogData &&\n        dialogData.activatedRoute instanceof ActivatedRoute\n    );\n}\n\n// @dynamic\n@Directive({standalone: true})\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            {id: string},\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n        ExtraResolve extends Literal = Record<never, never>,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Data retrieved by the server via route resolvers.\n     *\n     * The key `model` is special. It is readonly and represents the model being updated\n     * as it exists on server. The value is kept up to date when Apollo mutates it on server.\n     *\n     * The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.\n     *\n     * Other keys, if present, are whatever is returned from route resolvers as-is.\n     */\n    public override data: Data<TService, ExtraResolve>;\n\n    /**\n     * Form that manages the data from the controller\n     */\n    public form: UntypedFormGroup = new UntypedFormGroup({});\n\n    /**\n     * Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...\n     * semantic of data on other tab, like relations that list other objects)\n     */\n    public showFabButton = true;\n\n    protected readonly alertService = inject(NaturalAlertService);\n\n    protected readonly router = inject(Router);\n\n    protected readonly route = inject(ActivatedRoute);\n\n    private readonly _dialogData: unknown = inject(MAT_DIALOG_DATA, {optional: true});\n\n    /**\n     * Once set, this must not change anymore, especially not right after the creation mutation,\n     * so the form does not switch from creation mode to update mode without an actual reload of\n     * model from DB (by navigating to update page).\n     */\n    private _isUpdatePage = false;\n    private readonly changes = new CumulativeChanges<typeof this.form.getRawValue>();\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n    ) {\n        super();\n\n        this.data = {\n            model: this.service.getDefaultForServer(),\n        } as Data<TService, ExtraResolve>;\n    }\n\n    /**\n     * You probably should not override this method. Instead, consider overriding `initForm()`.\n     */\n    public ngOnInit(): void {\n        if (this.isPanel) {\n            this.initForm();\n        } else {\n            const route = isNaturalDialogTriggerProvidedData(this._dialogData)\n                ? this._dialogData.activatedRoute\n                : this.route;\n            this.#subscribeToModelFromResolvedData(route);\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    #subscribeToModelFromResolvedData(route: ActivatedRoute): void {\n        let previousId = -1;\n        route.data\n            .pipe(\n                switchMap(data => {\n                    if (!(data.model instanceof Observable)) {\n                        throw new Error(\n                            'Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).',\n                        );\n                    }\n\n                    // Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`\n                    return data.model.pipe(\n                        tap((model: ExtractResolve<TService>) => {\n                            this.data = {\n                                ...data,\n                                model: model,\n                            } as Data<TService, ExtraResolve>;\n\n                            if (previousId !== model.id) {\n                                previousId = model.id;\n                                this.initForm();\n                            }\n                        }),\n                    );\n                }),\n                takeUntilDestroyed(this.destroyRef),\n            )\n            .subscribe();\n    }\n\n    /**\n     * Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object\n     * with (only) default values, so we are on a creation page.\n     *\n     * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing\n     */\n    protected isUpdatePage(): this is {data: {model: ExtractTone<TService>}} {\n        return this._isUpdatePage;\n    }\n\n    /**\n     * Update the object on the server with the values from the form fields that were modified since\n     * the initialization, or since the previous successful update.\n     *\n     * Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.\n     */\n    public update(now = false, submitAllFields = false): void {\n        if (!this.isUpdatePage()) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            const newValues = this.form.getRawValue();\n            if (submitAllFields) {\n                this.changes.initialize({});\n            }\n\n            const toSubmit = {\n                id: this.data.model.id,\n                ...this.changes.differences(newValues),\n            };\n\n            const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);\n            update.subscribe(model => {\n                this.changes.commit(newValues);\n                this.alertService.info($localize`Mis à jour`);\n                this.postUpdate(model);\n            });\n        });\n    }\n\n    public create(redirect = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        const newValues = this.form.getRawValue();\n        this.form.disable();\n\n        this.service\n            .create(newValues)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n\n                    return this.postCreate(model).pipe(endWith(model), last());\n                }),\n                switchMap((model: WithId<object>) => {\n                    if (redirect) {\n                        if (this.isPanel) {\n                            const oldUrl = this.router.url;\n                            const nextUrl: string = this.panelData?.config.params.nextRoute;\n                            const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');\n                            return this.router.navigateByUrl(newUrl); // replace /new by /123\n                        } else {\n                            const queryParams = this.route.snapshot.queryParams;\n                            return this.router.navigate(['..', model.id], {relativeTo: this.route, queryParams});\n                        }\n                    }\n\n                    return EMPTY;\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    /**\n     * `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog\n     */\n    public delete(redirectionRoute?: unknown[], confirmer?: Observable<boolean | undefined>): void {\n        this.form.disable();\n\n        (\n            confirmer ??\n            this.alertService.confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n        )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed || !this.isUpdatePage()) {\n                        return EMPTY;\n                    }\n\n                    this.preDelete(this.data.model);\n\n                    return this.service.delete([this.data.model]).pipe(\n                        switchMap(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (this.isPanel) {\n                                this.panelService?.goToPenultimatePanel();\n\n                                return EMPTY;\n                            } else {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            }\n                        }),\n                    );\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postUpdate(model: ExtractTupdate<TService>): void {\n        // noop\n    }\n\n    /**\n     * Returns an observable that will be subscribed to immediately and the\n     * redirect navigation will only happen after the observable completes.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected preDelete(model: ExtractTone<TService>): void {\n        // noop\n    }\n\n    /**\n     * Initialize the form whenever it is needed.\n     *\n     * You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will\n     * correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`\n     * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.\n     */\n    protected initForm(): void {\n        this._isUpdatePage = !!this.data.model.id;\n        this.form = this.service.getFormGroup(this.data.model);\n        this.changes.initialize(this.form.getRawValue());\n    }\n}\n"]}
|
|
@@ -1,99 +0,0 @@
|
|
|
1
|
-
import { UntypedFormArray, UntypedFormGroup } from '@angular/forms';
|
|
2
|
-
import { MatTableDataSource } from '@angular/material/table';
|
|
3
|
-
import { merge } from 'lodash-es';
|
|
4
|
-
import { NaturalQueryVariablesManager } from './query-variable-manager';
|
|
5
|
-
import { validateAllFormControls } from './validators';
|
|
6
|
-
import { Directive } from '@angular/core';
|
|
7
|
-
import * as i0 from "@angular/core";
|
|
8
|
-
/**
|
|
9
|
-
* This class helps managing non-paginated rows of items that can be edited in-place, typically in a <mat-table>.
|
|
10
|
-
* But it does **not** mutate anything to persist the edits on the server. It is up to the consuming component to implement
|
|
11
|
-
* custom mutation mechanism.
|
|
12
|
-
*
|
|
13
|
-
* To access data of this component from a parent component, use:
|
|
14
|
-
*
|
|
15
|
-
* ```
|
|
16
|
-
* @ViewChildren(ComponentType) cmp: ComponentType;
|
|
17
|
-
* this.cmp.getItems();
|
|
18
|
-
* ```
|
|
19
|
-
*
|
|
20
|
-
* To add empty line, call:
|
|
21
|
-
*
|
|
22
|
-
* ```
|
|
23
|
-
* this.cmp.addEmpty();
|
|
24
|
-
* ```
|
|
25
|
-
*
|
|
26
|
-
* @dynamic
|
|
27
|
-
*/
|
|
28
|
-
export class NaturalAbstractEditableList {
|
|
29
|
-
service;
|
|
30
|
-
form;
|
|
31
|
-
formArray = new UntypedFormArray([]);
|
|
32
|
-
variablesManager = new NaturalQueryVariablesManager();
|
|
33
|
-
dataSource = new MatTableDataSource();
|
|
34
|
-
constructor(service) {
|
|
35
|
-
this.service = service;
|
|
36
|
-
// Create a form group with a line attributes that contain an array of formGroups (one by line = one by model)
|
|
37
|
-
this.form = new UntypedFormGroup({ rows: this.formArray });
|
|
38
|
-
this.dataSource.data = this.formArray.controls;
|
|
39
|
-
this.variablesManager.set('pagination', { pagination: { pageSize: 999, pageIndex: 0 } });
|
|
40
|
-
}
|
|
41
|
-
/**
|
|
42
|
-
* Set the list of items (overwriting what may have existed)
|
|
43
|
-
*/
|
|
44
|
-
setItems(items) {
|
|
45
|
-
this.formArray.clear(); // reset list
|
|
46
|
-
this.addItems(items);
|
|
47
|
-
}
|
|
48
|
-
/**
|
|
49
|
-
* Add given items to the list
|
|
50
|
-
* Reproduces the model data loading the same way as it would be on a detail page (via AbstractDetail controller) but without resolving
|
|
51
|
-
*/
|
|
52
|
-
addItems(items) {
|
|
53
|
-
items.forEach(item => {
|
|
54
|
-
const completedItem = merge(this.service.getDefaultForServer(), item);
|
|
55
|
-
const lineFormGroup = this.service.getFormGroup(completedItem);
|
|
56
|
-
this.formArray.push(lineFormGroup);
|
|
57
|
-
});
|
|
58
|
-
this.dataSource.data = this.formArray.controls;
|
|
59
|
-
}
|
|
60
|
-
removeAt(index) {
|
|
61
|
-
this.formArray.removeAt(index);
|
|
62
|
-
this.dataSource.data = this.formArray.controls;
|
|
63
|
-
}
|
|
64
|
-
/**
|
|
65
|
-
* Add empty item at the end of the list
|
|
66
|
-
*/
|
|
67
|
-
addEmpty() {
|
|
68
|
-
this.addItems([{}]);
|
|
69
|
-
}
|
|
70
|
-
/**
|
|
71
|
-
* Return a list of models without any treatment.
|
|
72
|
-
*
|
|
73
|
-
* To mutate models, it would be required to map them using :
|
|
74
|
-
* - AbstractModelService.getInput()
|
|
75
|
-
* - AbstractModelService.getPartialVariablesForCreation()
|
|
76
|
-
* - AbstractModelService.getPartialVariablesForUpdate()
|
|
77
|
-
* - some other required treatment.
|
|
78
|
-
*
|
|
79
|
-
* TODO return type is incorrect and should be closer to `Partial<T>[]` or an even looser type, because we don't really know what fields exists in the form. When we fix this, we should also remove type coercing in unit tests.
|
|
80
|
-
*/
|
|
81
|
-
getItems() {
|
|
82
|
-
return this.formArray.getRawValue();
|
|
83
|
-
}
|
|
84
|
-
/**
|
|
85
|
-
* Force the form validation.
|
|
86
|
-
*
|
|
87
|
-
* The valid state can then be read via `this.form.valid`
|
|
88
|
-
*/
|
|
89
|
-
validateForm() {
|
|
90
|
-
validateAllFormControls(this.form);
|
|
91
|
-
}
|
|
92
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractEditableList, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive });
|
|
93
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: NaturalAbstractEditableList, isStandalone: true, ngImport: i0 });
|
|
94
|
-
}
|
|
95
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalAbstractEditableList, decorators: [{
|
|
96
|
-
type: Directive,
|
|
97
|
-
args: [{ standalone: true }]
|
|
98
|
-
}], ctorParameters: () => [{ type: undefined }] });
|
|
99
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-editable-list.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-editable-list.ts"],"names":[],"mappings":"AAAA,OAAO,EAAkB,gBAAgB,EAAE,gBAAgB,EAAC,MAAM,gBAAgB,CAAC;AACnF,OAAO,EAAC,kBAAkB,EAAC,MAAM,yBAAyB,CAAC;AAC3D,OAAO,EAAC,KAAK,EAAC,MAAM,WAAW,CAAC;AAEhC,OAAO,EAAC,4BAA4B,EAAiB,MAAM,0BAA0B,CAAC;AAEtF,OAAO,EAAC,uBAAuB,EAAC,MAAM,cAAc,CAAC;AACrD,OAAO,EAAC,SAAS,EAAC,MAAM,eAAe,CAAC;;AAGxC;;;;;;;;;;;;;;;;;;;GAmBG;AAEH,MAAM,OAAO,2BAA2B;IAsBE;IALtB,IAAI,CAAmB;IACvB,SAAS,GAAG,IAAI,gBAAgB,CAAC,EAAE,CAAC,CAAC;IACrC,gBAAgB,GAAG,IAAI,4BAA4B,EAAyB,CAAC;IAC7E,UAAU,GAAG,IAAI,kBAAkB,EAAmB,CAAC;IAEvE,YAAsC,OAAiB;QAAjB,YAAO,GAAP,OAAO,CAAU;QACnD,8GAA8G;QAC9G,IAAI,CAAC,IAAI,GAAG,IAAI,gBAAgB,CAAC,EAAC,IAAI,EAAE,IAAI,CAAC,SAAS,EAAC,CAAC,CAAC;QACzD,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;QAC/C,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,YAAY,EAAE,EAAC,UAAU,EAAE,EAAC,QAAQ,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,EAAC,EAA0B,CAAC,CAAC;IAClH,CAAC;IAED;;OAEG;IACI,QAAQ,CAAC,KAAmB;QAC/B,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC,CAAC,aAAa;QACrC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACzB,CAAC;IAED;;;OAGG;IACI,QAAQ,CAAC,KAAmB;QAC/B,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACjB,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,mBAAmB,EAAE,EAAE,IAAI,CAAC,CAAC;YACtE,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC;YAC/D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;IACnD,CAAC;IAEM,QAAQ,CAAC,KAAa;QACzB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC/B,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;IACnD,CAAC;IAED;;OAEG;IACI,QAAQ;QACX,IAAI,CAAC,QAAQ,CAAC,CAAC,EAAO,CAAC,CAAC,CAAC;IAC7B,CAAC;IAED;;;;;;;;;;OAUG;IACI,QAAQ;QACX,OAAO,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;IACxC,CAAC;IAED;;;;OAIG;IACI,YAAY;QACf,uBAAuB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACvC,CAAC;uGArFQ,2BAA2B;2FAA3B,2BAA2B;;2FAA3B,2BAA2B;kBADvC,SAAS;mBAAC,EAAC,UAAU,EAAE,IAAI,EAAC","sourcesContent":["import {AbstractControl, UntypedFormArray, UntypedFormGroup} from '@angular/forms';\nimport {MatTableDataSource} from '@angular/material/table';\nimport {merge} from 'lodash-es';\nimport {NaturalAbstractModelService} from '../services/abstract-model.service';\nimport {NaturalQueryVariablesManager, QueryVariables} from './query-variable-manager';\nimport {ExtractTallOne, ExtractVall, Literal} from '../types/types';\nimport {validateAllFormControls} from './validators';\nimport {Directive} from '@angular/core';\nimport {PaginatedData} from './data-source';\n\n/**\n * This class helps managing non-paginated rows of items that can be edited in-place, typically in a <mat-table>.\n * But it does **not** mutate anything to persist the edits on the server. It is up to the consuming component to implement\n * custom mutation mechanism.\n *\n * To access data of this component from a parent component, use:\n *\n * ```\n * @ViewChildren(ComponentType) cmp: ComponentType;\n * this.cmp.getItems();\n * ```\n *\n * To add empty line, call:\n *\n * ```\n * this.cmp.addEmpty();\n * ```\n *\n * @dynamic\n */\n@Directive({standalone: true})\nexport class NaturalAbstractEditableList<\n    TService extends NaturalAbstractModelService<\n        any,\n        any,\n        PaginatedData<Literal>,\n        QueryVariables,\n        any,\n        any,\n        any,\n        any,\n        any,\n        any\n    >,\n    // The Literal here is a bit too loose. Ideally we would like to express\n    // \"it must be a union and one of the type in the union must be ExtractTallOne<TService>\"\n    T extends Literal = ExtractTallOne<TService>,\n> {\n    public readonly form: UntypedFormGroup;\n    public readonly formArray = new UntypedFormArray([]);\n    public readonly variablesManager = new NaturalQueryVariablesManager<ExtractVall<TService>>();\n    public readonly dataSource = new MatTableDataSource<AbstractControl>();\n\n    public constructor(protected readonly service: TService) {\n        // Create a form group with a line attributes that contain an array of formGroups (one by line = one by model)\n        this.form = new UntypedFormGroup({rows: this.formArray});\n        this.dataSource.data = this.formArray.controls;\n        this.variablesManager.set('pagination', {pagination: {pageSize: 999, pageIndex: 0}} as ExtractVall<TService>);\n    }\n\n    /**\n     * Set the list of items (overwriting what may have existed)\n     */\n    public setItems(items: readonly T[]): void {\n        this.formArray.clear(); // reset list\n        this.addItems(items);\n    }\n\n    /**\n     * Add given items to the list\n     * Reproduces the model data loading the same way as it would be on a detail page (via AbstractDetail controller) but without resolving\n     */\n    public addItems(items: readonly T[]): void {\n        items.forEach(item => {\n            const completedItem = merge(this.service.getDefaultForServer(), item);\n            const lineFormGroup = this.service.getFormGroup(completedItem);\n            this.formArray.push(lineFormGroup);\n        });\n\n        this.dataSource.data = this.formArray.controls;\n    }\n\n    public removeAt(index: number): void {\n        this.formArray.removeAt(index);\n        this.dataSource.data = this.formArray.controls;\n    }\n\n    /**\n     * Add empty item at the end of the list\n     */\n    public addEmpty(): void {\n        this.addItems([{} as T]);\n    }\n\n    /**\n     * Return a list of models without any treatment.\n     *\n     * To mutate models, it would be required to map them using :\n     *  - AbstractModelService.getInput()\n     *  - AbstractModelService.getPartialVariablesForCreation()\n     *  - AbstractModelService.getPartialVariablesForUpdate()\n     *  - some other required treatment.\n     *\n     * TODO return type is incorrect and should be closer to `Partial<T>[]` or an even looser type, because we don't really know what fields exists in the form. When we fix this, we should also remove type coercing in unit tests.\n     */\n    public getItems(): T[] {\n        return this.formArray.getRawValue();\n    }\n\n    /**\n     * Force the form validation.\n     *\n     * The valid state can then be read via `this.form.valid`\n     */\n    public validateForm(): void {\n        validateAllFormControls(this.form);\n    }\n}\n"]}
|