@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.
Files changed (183) hide show
  1. package/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
  2. package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
  3. package/fesm2022/ecodev-natural.mjs +436 -425
  4. package/fesm2022/ecodev-natural.mjs.map +1 -1
  5. package/lib/classes/network-activity.service.d.ts +54 -0
  6. package/lib/classes/validators.d.ts +1 -1
  7. package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
  8. package/lib/modules/common/services/seo.provider.d.ts +2 -2
  9. package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
  10. package/lib/modules/file/abstract-file.d.ts +6 -3
  11. package/lib/modules/file/component/file.component.d.ts +2 -2
  12. package/lib/modules/file/file-drop.directive.d.ts +2 -3
  13. package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
  14. package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
  15. package/lib/modules/icon/icon.module.d.ts +2 -2
  16. package/lib/modules/panels/panels.service.d.ts +1 -2
  17. package/lib/modules/relations/relations.component.d.ts +3 -3
  18. package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
  19. package/lib/modules/search/group/group.component.d.ts +2 -3
  20. package/lib/modules/search/input/input.component.d.ts +5 -5
  21. package/lib/modules/search/search/search.component.d.ts +2 -2
  22. package/lib/modules/select/abstract-select.component.d.ts +3 -3
  23. package/lib/modules/select/select/select.component.d.ts +1 -1
  24. package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
  25. package/lib/modules/table-button/table-button.component.d.ts +4 -2
  26. package/package.json +16 -14
  27. package/public-api.d.ts +1 -0
  28. package/src/lib/_natural.theme.scss +1 -2
  29. package/vanilla/index.d.ts +5 -0
  30. package/vanilla/package.json +3 -0
  31. package/vanilla/public-api.d.ts +11 -0
  32. package/vanilla/src/lib/classes/crypto.d.ts +8 -0
  33. package/vanilla/src/lib/classes/data-source.d.ts +32 -0
  34. package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
  35. package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
  36. package/vanilla/src/lib/classes/signing.d.ts +7 -0
  37. package/vanilla/src/lib/classes/utility.d.ts +77 -0
  38. package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
  39. package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
  40. package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
  41. package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
  42. package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
  43. package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
  44. package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
  45. package/vanilla/src/lib/types/types.d.ts +100 -0
  46. package/esm2022/ecodev-natural.mjs +0 -5
  47. package/esm2022/lib/classes/abstract-detail.mjs +0 -229
  48. package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
  49. package/esm2022/lib/classes/abstract-list.mjs +0 -461
  50. package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
  51. package/esm2022/lib/classes/apollo-utils.mjs +0 -59
  52. package/esm2022/lib/classes/crypto.mjs +0 -23
  53. package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
  54. package/esm2022/lib/classes/data-source.mjs +0 -71
  55. package/esm2022/lib/classes/providers.mjs +0 -13
  56. package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
  57. package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
  58. package/esm2022/lib/classes/rxjs.mjs +0 -54
  59. package/esm2022/lib/classes/signing.mjs +0 -38
  60. package/esm2022/lib/classes/tld.mjs +0 -1476
  61. package/esm2022/lib/classes/utility.mjs +0 -234
  62. package/esm2022/lib/classes/validators.mjs +0 -179
  63. package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
  64. package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
  65. package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
  66. package/esm2022/lib/modules/alert/public-api.mjs +0 -6
  67. package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
  68. package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
  69. package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
  70. package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
  71. package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
  72. package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
  73. package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
  74. package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
  75. package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
  76. package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
  77. package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
  78. package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
  79. package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
  80. package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
  81. package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
  82. package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
  83. package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
  84. package/esm2022/lib/modules/common/public-api.mjs +0 -14
  85. package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
  86. package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
  87. package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
  88. package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
  89. package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
  90. package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
  91. package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
  92. package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
  93. package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
  94. package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
  95. package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
  96. package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
  97. package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
  98. package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
  99. package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
  100. package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
  101. package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
  102. package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
  103. package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
  104. package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
  105. package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
  106. package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
  107. package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
  108. package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
  109. package/esm2022/lib/modules/file/file.service.mjs +0 -43
  110. package/esm2022/lib/modules/file/public-api.mjs +0 -9
  111. package/esm2022/lib/modules/file/types.mjs +0 -2
  112. package/esm2022/lib/modules/file/utils.mjs +0 -129
  113. package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
  114. package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
  115. package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
  116. package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
  117. package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
  118. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
  119. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
  120. package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
  121. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
  122. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
  123. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
  124. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
  125. package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
  126. package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
  127. package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
  128. package/esm2022/lib/modules/icon/public-api.mjs +0 -6
  129. package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
  130. package/esm2022/lib/modules/logger/error.module.mjs +0 -22
  131. package/esm2022/lib/modules/logger/public-api.mjs +0 -6
  132. package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
  133. package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
  134. package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
  135. package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
  136. package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
  137. package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
  138. package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
  139. package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
  140. package/esm2022/lib/modules/panels/public-api.mjs +0 -11
  141. package/esm2022/lib/modules/panels/types.mjs +0 -3
  142. package/esm2022/lib/modules/relations/public-api.mjs +0 -5
  143. package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
  144. package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
  145. package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
  146. package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
  147. package/esm2022/lib/modules/search/classes/url.mjs +0 -53
  148. package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
  149. package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
  150. package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
  151. package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
  152. package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
  153. package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
  154. package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
  155. package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
  156. package/esm2022/lib/modules/search/public-api.mjs +0 -7
  157. package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
  158. package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
  159. package/esm2022/lib/modules/search/types/facet.mjs +0 -2
  160. package/esm2022/lib/modules/search/types/values.mjs +0 -2
  161. package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
  162. package/esm2022/lib/modules/select/public-api.mjs +0 -7
  163. package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
  164. package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
  165. package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
  166. package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
  167. package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
  168. package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
  169. package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
  170. package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
  171. package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
  172. package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
  173. package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
  174. package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
  175. package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
  176. package/esm2022/lib/services/abstract-model.service.mjs +0 -526
  177. package/esm2022/lib/services/debounce.service.mjs +0 -149
  178. package/esm2022/lib/services/enum.service.mjs +0 -64
  179. package/esm2022/lib/services/link-mutation.service.mjs +0 -154
  180. package/esm2022/lib/services/persistence.service.mjs +0 -115
  181. package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
  182. package/esm2022/lib/types/types.mjs +0 -2
  183. package/esm2022/public-api.mjs +0 -46
@@ -1,149 +0,0 @@
1
- import { catchError, debounceTime, EMPTY, forkJoin, map, mergeMap, Observable, of, raceWith, ReplaySubject, shareReplay, Subject, take, } from 'rxjs';
2
- import { Injectable } from '@angular/core';
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 class NaturalDebounceService {
12
- /**
13
- * Stores the debounced update function
14
- */
15
- allDebouncedUpdateCache = new Map();
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(modelService, id, object) {
26
- const debouncedUpdateCache = this.getMap(modelService);
27
- let debounced = debouncedUpdateCache.get(id);
28
- if (debounced) {
29
- debounced.object = {
30
- ...debounced.object,
31
- ...object,
32
- };
33
- }
34
- else {
35
- const debouncer = new ReplaySubject(1);
36
- let wasCancelled = false;
37
- const canceller = new Subject();
38
- canceller.subscribe(() => {
39
- wasCancelled = true;
40
- debouncer.complete();
41
- canceller.complete();
42
- this.delete(modelService, id);
43
- });
44
- const flusher = new Subject();
45
- debounced = {
46
- object,
47
- debouncer,
48
- canceller,
49
- flusher,
50
- modelService: modelService,
51
- result: debouncer.pipe(debounceTime(2000), // Wait 2 seconds...
52
- raceWith(flusher), // ...unless flusher is triggered
53
- take(1), mergeMap(() => {
54
- this.delete(modelService, id);
55
- if (wasCancelled || !debounced) {
56
- return EMPTY;
57
- }
58
- return modelService.updateNow(debounced.object);
59
- }), shareReplay()),
60
- };
61
- debouncedUpdateCache.set(id, debounced);
62
- }
63
- // Notify our debounced update each time we ask to update
64
- debounced.debouncer.next();
65
- // Return and observable that is updated when mutation is done
66
- return debounced.result;
67
- }
68
- cancelOne(modelService, id) {
69
- const debounced = this.allDebouncedUpdateCache.get(modelService)?.get(id);
70
- debounced?.canceller.next();
71
- }
72
- /**
73
- * Immediately execute the pending update, if any.
74
- *
75
- * It should typically be called before resolving the object, to mutate it before re-fetching it from server.
76
- *
77
- * The returned observable will complete when the update completes, even if it errors.
78
- */
79
- flushOne(modelService, id) {
80
- const debounced = this.allDebouncedUpdateCache.get(modelService)?.get(id);
81
- return this.internalFlush(debounced ? [debounced] : []);
82
- }
83
- /**
84
- * Immediately execute all pending updates.
85
- *
86
- * It should typically be called before login out.
87
- *
88
- * The returned observable will complete when all updates complete, even if some of them error.
89
- */
90
- flush() {
91
- const all = [];
92
- this.allDebouncedUpdateCache.forEach(map => map.forEach(debounced => all.push(debounced)));
93
- return this.internalFlush(all);
94
- }
95
- internalFlush(debounceds) {
96
- const all = [];
97
- const allFlusher = [];
98
- debounceds.forEach(debounced => {
99
- all.push(debounced.result.pipe(catchError(() => of(undefined))));
100
- allFlusher.push(debounced.flusher);
101
- });
102
- if (!all.length) {
103
- all.push(of(undefined));
104
- }
105
- return new Observable(subscriber => {
106
- const subscription = forkJoin(all)
107
- .pipe(map(() => undefined))
108
- .subscribe(subscriber);
109
- // Flush only after subscription process is finished
110
- allFlusher.forEach(flusher => flusher.next());
111
- return subscription;
112
- });
113
- }
114
- /**
115
- * Count of pending updates
116
- */
117
- get count() {
118
- let count = 0;
119
- this.allDebouncedUpdateCache.forEach(map => (count += map.size));
120
- return count;
121
- }
122
- getMap(modelService) {
123
- let debouncedUpdateCache = this.allDebouncedUpdateCache.get(modelService);
124
- if (!debouncedUpdateCache) {
125
- debouncedUpdateCache = new Map();
126
- this.allDebouncedUpdateCache.set(modelService, debouncedUpdateCache);
127
- }
128
- return debouncedUpdateCache;
129
- }
130
- delete(modelService, id) {
131
- const map = this.allDebouncedUpdateCache.get(modelService);
132
- if (!map) {
133
- return;
134
- }
135
- map.delete(id);
136
- if (!map.size) {
137
- this.allDebouncedUpdateCache.delete(modelService);
138
- }
139
- }
140
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalDebounceService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
141
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalDebounceService, providedIn: 'root' });
142
- }
143
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalDebounceService, decorators: [{
144
- type: Injectable,
145
- args: [{
146
- providedIn: 'root',
147
- }]
148
- }] });
149
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"debounce.service.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/services/debounce.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    catchError,\n    debounceTime,\n    EMPTY,\n    forkJoin,\n    map,\n    mergeMap,\n    Observable,\n    of,\n    raceWith,\n    ReplaySubject,\n    shareReplay,\n    Subject,\n    take,\n} from 'rxjs';\nimport {Injectable} from '@angular/core';\nimport {UntypedModelService} from '../types/types';\n\ntype Debounced<T extends UntypedModelService> = {\n    object: Parameters<T['updateNow']>[0];\n    modelService: T;\n    debouncer: Subject<void>;\n    canceller: Subject<void>;\n    flusher: Subject<void>;\n    result: ReturnType<T['updateNow']>;\n};\n\n/**\n * Debounce subscriptions to update mutations, with the possibility to cancel one, flush one, or flush all of them.\n *\n * `modelService` is also used to separate objects by their types. So User with ID 1 is not confused with Product with ID 1.\n *\n * `id` must be the ID of the object that will be updated.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalDebounceService {\n    /**\n     * Stores the debounced update function\n     */\n    private readonly allDebouncedUpdateCache = new Map<\n        UntypedModelService,\n        Map<string, Debounced<UntypedModelService>>\n    >();\n\n    /**\n     * Debounce the `modelService.updateNow()` mutation for a short time. If called multiple times with the same\n     * modelService and id, it will postpone the subscription to the mutation.\n     *\n     * All input variables for the same object (same service and ID) will be cumulated over time. So it is possible\n     * to update `field1`, then `field2`, and they will be batched into a single XHR including `field1` and `field2`.\n     *\n     * But it will always keep the same debouncing timeline.\n     */\n    public debounce<T extends UntypedModelService>(\n        modelService: UntypedModelService,\n        id: string,\n        object: Parameters<T['updateNow']>[0],\n    ): ReturnType<T['updateNow']> {\n        const debouncedUpdateCache = this.getMap(modelService);\n        let debounced = debouncedUpdateCache.get(id) as Debounced<T> | undefined;\n\n        if (debounced) {\n            debounced.object = {\n                ...debounced.object,\n                ...object,\n            };\n        } else {\n            const debouncer = new ReplaySubject<void>(1);\n            let wasCancelled = false;\n            const canceller = new Subject<void>();\n            canceller.subscribe(() => {\n                wasCancelled = true;\n                debouncer.complete();\n                canceller.complete();\n                this.delete(modelService, id);\n            });\n\n            const flusher = new Subject<void>();\n\n            debounced = {\n                object,\n                debouncer,\n                canceller,\n                flusher,\n                modelService: modelService as T,\n                result: debouncer.pipe(\n                    debounceTime(2000), // Wait 2 seconds...\n                    raceWith(flusher), // ...unless flusher is triggered\n                    take(1),\n                    mergeMap(() => {\n                        this.delete(modelService, id);\n\n                        if (wasCancelled || !debounced) {\n                            return EMPTY;\n                        }\n\n                        return modelService.updateNow(debounced.object);\n                    }),\n                    shareReplay(), // All attempts to update will share the exact same single result from API\n                ) as ReturnType<T['updateNow']>,\n            };\n\n            debouncedUpdateCache.set(id, debounced);\n        }\n\n        // Notify our debounced update each time we ask to update\n        debounced.debouncer.next();\n\n        // Return and observable that is updated when mutation is done\n        return debounced.result;\n    }\n\n    public cancelOne(modelService: UntypedModelService, id: string): void {\n        const debounced = this.allDebouncedUpdateCache.get(modelService)?.get(id);\n        debounced?.canceller.next();\n    }\n\n    /**\n     * Immediately execute the pending update, if any.\n     *\n     * It should typically be called before resolving the object, to mutate it before re-fetching it from server.\n     *\n     * The returned observable will complete when the update completes, even if it errors.\n     */\n    public flushOne(modelService: UntypedModelService, id: string): Observable<void> {\n        const debounced = this.allDebouncedUpdateCache.get(modelService)?.get(id);\n\n        return this.internalFlush(debounced ? [debounced] : []);\n    }\n\n    /**\n     * Immediately execute all pending updates.\n     *\n     * It should typically be called before login out.\n     *\n     * The returned observable will complete when all updates complete, even if some of them error.\n     */\n    public flush(): Observable<void> {\n        const all: Debounced<UntypedModelService>[] = [];\n        this.allDebouncedUpdateCache.forEach(map => map.forEach(debounced => all.push(debounced)));\n\n        return this.internalFlush(all);\n    }\n\n    private internalFlush(debounceds: Debounced<UntypedModelService>[]): Observable<void> {\n        const all: Observable<unknown>[] = [];\n        const allFlusher: Subject<void>[] = [];\n\n        debounceds.forEach(debounced => {\n            all.push(debounced.result.pipe(catchError(() => of(undefined))));\n            allFlusher.push(debounced.flusher);\n        });\n\n        if (!all.length) {\n            all.push(of(undefined));\n        }\n\n        return new Observable(subscriber => {\n            const subscription = forkJoin(all)\n                .pipe(map(() => undefined))\n                .subscribe(subscriber);\n\n            // Flush only after subscription process is finished\n            allFlusher.forEach(flusher => flusher.next());\n\n            return subscription;\n        });\n    }\n\n    /**\n     * Count of pending updates\n     */\n    public get count(): number {\n        let count = 0;\n        this.allDebouncedUpdateCache.forEach(map => (count += map.size));\n\n        return count;\n    }\n\n    private getMap<T extends UntypedModelService>(modelService: T): Map<string, Debounced<T>> {\n        let debouncedUpdateCache = this.allDebouncedUpdateCache.get(modelService);\n        if (!debouncedUpdateCache) {\n            debouncedUpdateCache = new Map<string, Debounced<UntypedModelService>>();\n            this.allDebouncedUpdateCache.set(modelService, debouncedUpdateCache);\n        }\n\n        return debouncedUpdateCache as Map<string, Debounced<T>>;\n    }\n\n    private delete(modelService: UntypedModelService, id: string): void {\n        const map = this.allDebouncedUpdateCache.get(modelService);\n        if (!map) {\n            return;\n        }\n\n        map.delete(id);\n\n        if (!map.size) {\n            this.allDebouncedUpdateCache.delete(modelService);\n        }\n    }\n}\n"]}
@@ -1,64 +0,0 @@
1
- import { Apollo, gql } from 'apollo-angular';
2
- import { inject, Injectable } from '@angular/core';
3
- import { map } from 'rxjs/operators';
4
- import * as i0 from "@angular/core";
5
- const enumTypeQuery = gql `
6
- query EnumType($name: String!) {
7
- __type(name: $name) {
8
- __typename
9
- enumValues {
10
- name
11
- description
12
- }
13
- }
14
- }
15
- `;
16
- export class NaturalEnumService {
17
- apollo = inject(Apollo);
18
- /**
19
- * Return a list of observable enumerables considering the given name
20
- */
21
- get(name) {
22
- // Load possible action statuses
23
- return this.apollo
24
- .query({
25
- query: enumTypeQuery,
26
- variables: { name: name },
27
- fetchPolicy: 'cache-first',
28
- })
29
- .pipe(map(result => {
30
- const values = [];
31
- if (result.data.__type?.enumValues) {
32
- for (const enumValue of result.data.__type.enumValues) {
33
- values.push({
34
- value: enumValue.name,
35
- name: enumValue.description || '',
36
- });
37
- }
38
- }
39
- return values;
40
- }));
41
- }
42
- /**
43
- * Returns the enum user-friendly name, instead of its value.
44
- */
45
- getValueName(value, enumName) {
46
- return this.get(enumName).pipe(map(values => {
47
- for (const v of values) {
48
- if (v.value === value) {
49
- return v.name;
50
- }
51
- }
52
- return '';
53
- }));
54
- }
55
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalEnumService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
56
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalEnumService, providedIn: 'root' });
57
- }
58
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalEnumService, decorators: [{
59
- type: Injectable,
60
- args: [{
61
- providedIn: 'root',
62
- }]
63
- }] });
64
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW51bS5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL3NlcnZpY2VzL2VudW0uc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsTUFBTSxFQUFFLEdBQUcsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBQzNDLE9BQU8sRUFBQyxNQUFNLEVBQUUsVUFBVSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRWpELE9BQU8sRUFBQyxHQUFHLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQzs7QUFFbkMsTUFBTSxhQUFhLEdBQUcsR0FBRyxDQUFBOzs7Ozs7Ozs7O0NBVXhCLENBQUM7QUFzQkYsTUFBTSxPQUFPLGtCQUFrQjtJQUNWLE1BQU0sR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUM7SUFFekM7O09BRUc7SUFDSSxHQUFHLENBQUMsSUFBWTtRQUNuQixnQ0FBZ0M7UUFDaEMsT0FBTyxJQUFJLENBQUMsTUFBTTthQUNiLEtBQUssQ0FBVztZQUNiLEtBQUssRUFBRSxhQUFhO1lBQ3BCLFNBQVMsRUFBRSxFQUFDLElBQUksRUFBRSxJQUFJLEVBQUM7WUFDdkIsV0FBVyxFQUFFLGFBQWE7U0FDN0IsQ0FBQzthQUNELElBQUksQ0FDRCxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUU7WUFDVCxNQUFNLE1BQU0sR0FBWSxFQUFFLENBQUM7WUFDM0IsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxVQUFVLEVBQUUsQ0FBQztnQkFDakMsS0FBSyxNQUFNLFNBQVMsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLEVBQUUsQ0FBQztvQkFDcEQsTUFBTSxDQUFDLElBQUksQ0FBQzt3QkFDUixLQUFLLEVBQUUsU0FBUyxDQUFDLElBQUk7d0JBQ3JCLElBQUksRUFBRSxTQUFTLENBQUMsV0FBVyxJQUFJLEVBQUU7cUJBQ3BDLENBQUMsQ0FBQztnQkFDUCxDQUFDO1lBQ0wsQ0FBQztZQUVELE9BQU8sTUFBTSxDQUFDO1FBQ2xCLENBQUMsQ0FBQyxDQUNMLENBQUM7SUFDVixDQUFDO0lBRUQ7O09BRUc7SUFDSSxZQUFZLENBQUMsS0FBYSxFQUFFLFFBQWdCO1FBQy9DLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxJQUFJLENBQzFCLEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFBRTtZQUNULEtBQUssTUFBTSxDQUFDLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ3JCLElBQUksQ0FBQyxDQUFDLEtBQUssS0FBSyxLQUFLLEVBQUUsQ0FBQztvQkFDcEIsT0FBTyxDQUFDLENBQUMsSUFBSSxDQUFDO2dCQUNsQixDQUFDO1lBQ0wsQ0FBQztZQUVELE9BQU8sRUFBRSxDQUFDO1FBQ2QsQ0FBQyxDQUFDLENBQ0wsQ0FBQztJQUNOLENBQUM7dUdBOUNRLGtCQUFrQjsyR0FBbEIsa0JBQWtCLGNBRmYsTUFBTTs7MkZBRVQsa0JBQWtCO2tCQUg5QixVQUFVO21CQUFDO29CQUNSLFVBQVUsRUFBRSxNQUFNO2lCQUNyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7QXBvbGxvLCBncWx9IGZyb20gJ2Fwb2xsby1hbmd1bGFyJztcbmltcG9ydCB7aW5qZWN0LCBJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7T2JzZXJ2YWJsZX0gZnJvbSAncnhqcyc7XG5pbXBvcnQge21hcH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuXG5jb25zdCBlbnVtVHlwZVF1ZXJ5ID0gZ3FsYFxuICAgIHF1ZXJ5IEVudW1UeXBlKCRuYW1lOiBTdHJpbmchKSB7XG4gICAgICAgIF9fdHlwZShuYW1lOiAkbmFtZSkge1xuICAgICAgICAgICAgX190eXBlbmFtZVxuICAgICAgICAgICAgZW51bVZhbHVlcyB7XG4gICAgICAgICAgICAgICAgbmFtZVxuICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uXG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICB9XG5gO1xuXG50eXBlIEVudW1UeXBlID0ge1xuICAgIF9fdHlwZTogbnVsbCB8IHtcbiAgICAgICAgX190eXBlbmFtZTogc3RyaW5nO1xuICAgICAgICBlbnVtVmFsdWVzOlxuICAgICAgICAgICAgfCBudWxsXG4gICAgICAgICAgICB8IHtcbiAgICAgICAgICAgICAgICAgIG5hbWU6IHN0cmluZztcbiAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBudWxsIHwgc3RyaW5nO1xuICAgICAgICAgICAgICB9W107XG4gICAgfTtcbn07XG5cbmV4cG9ydCB0eXBlIElFbnVtID0ge1xuICAgIHZhbHVlOiBzdHJpbmc7XG4gICAgbmFtZTogc3RyaW5nO1xufTtcblxuQEluamVjdGFibGUoe1xuICAgIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbEVudW1TZXJ2aWNlIHtcbiAgICBwcml2YXRlIHJlYWRvbmx5IGFwb2xsbyA9IGluamVjdChBcG9sbG8pO1xuXG4gICAgLyoqXG4gICAgICogUmV0dXJuIGEgbGlzdCBvZiBvYnNlcnZhYmxlIGVudW1lcmFibGVzIGNvbnNpZGVyaW5nIHRoZSBnaXZlbiBuYW1lXG4gICAgICovXG4gICAgcHVibGljIGdldChuYW1lOiBzdHJpbmcpOiBPYnNlcnZhYmxlPElFbnVtW10+IHtcbiAgICAgICAgLy8gTG9hZCBwb3NzaWJsZSBhY3Rpb24gc3RhdHVzZXNcbiAgICAgICAgcmV0dXJuIHRoaXMuYXBvbGxvXG4gICAgICAgICAgICAucXVlcnk8RW51bVR5cGU+KHtcbiAgICAgICAgICAgICAgICBxdWVyeTogZW51bVR5cGVRdWVyeSxcbiAgICAgICAgICAgICAgICB2YXJpYWJsZXM6IHtuYW1lOiBuYW1lfSxcbiAgICAgICAgICAgICAgICBmZXRjaFBvbGljeTogJ2NhY2hlLWZpcnN0JyxcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAucGlwZShcbiAgICAgICAgICAgICAgICBtYXAocmVzdWx0ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgdmFsdWVzOiBJRW51bVtdID0gW107XG4gICAgICAgICAgICAgICAgICAgIGlmIChyZXN1bHQuZGF0YS5fX3R5cGU/LmVudW1WYWx1ZXMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGZvciAoY29uc3QgZW51bVZhbHVlIG9mIHJlc3VsdC5kYXRhLl9fdHlwZS5lbnVtVmFsdWVzKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWVzLnB1c2goe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZTogZW51bVZhbHVlLm5hbWUsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5hbWU6IGVudW1WYWx1ZS5kZXNjcmlwdGlvbiB8fCAnJyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgICAgIHJldHVybiB2YWx1ZXM7XG4gICAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICApO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgdGhlIGVudW0gdXNlci1mcmllbmRseSBuYW1lLCBpbnN0ZWFkIG9mIGl0cyB2YWx1ZS5cbiAgICAgKi9cbiAgICBwdWJsaWMgZ2V0VmFsdWVOYW1lKHZhbHVlOiBzdHJpbmcsIGVudW1OYW1lOiBzdHJpbmcpOiBPYnNlcnZhYmxlPHN0cmluZz4ge1xuICAgICAgICByZXR1cm4gdGhpcy5nZXQoZW51bU5hbWUpLnBpcGUoXG4gICAgICAgICAgICBtYXAodmFsdWVzID0+IHtcbiAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IHYgb2YgdmFsdWVzKSB7XG4gICAgICAgICAgICAgICAgICAgIGlmICh2LnZhbHVlID09PSB2YWx1ZSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHYubmFtZTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgIHJldHVybiAnJztcbiAgICAgICAgICAgIH0pLFxuICAgICAgICApO1xuICAgIH1cbn1cbiJdfQ==
@@ -1,154 +0,0 @@
1
- import { Apollo, gql } from 'apollo-angular';
2
- import { inject, Injectable } from '@angular/core';
3
- import { clone } from 'lodash-es';
4
- import { forkJoin, of } from 'rxjs';
5
- import { map, switchMap } from 'rxjs/operators';
6
- import { upperCaseFirstLetter } from '../classes/utility';
7
- import * as i0 from "@angular/core";
8
- /**
9
- * Query to get list of mutations
10
- */
11
- const queriesQuery = gql `
12
- query Mutations {
13
- __type(name: "Mutation") {
14
- fields {
15
- name
16
- args {
17
- name
18
- type {
19
- ofType {
20
- name
21
- }
22
- }
23
- }
24
- }
25
- }
26
- }
27
- `;
28
- export class NaturalLinkMutationService {
29
- apollo = inject(Apollo);
30
- /**
31
- * Receives the list of available mutations
32
- */
33
- allMutations;
34
- /**
35
- * Link two objects together
36
- */
37
- link(obj1, obj2, otherName = null, variables = {}) {
38
- // clone prevents to affect the original reference
39
- const clonedVariables = clone(variables);
40
- return this.getMutation('link', obj1, obj2, otherName, clonedVariables).pipe(switchMap(mutation => this.execute(mutation)));
41
- }
42
- /**
43
- * Link many objects
44
- */
45
- linkMany(obj1, objects, otherName = null, variables = {}) {
46
- return forkJoin(objects.map(obj2 => this.link(obj1, obj2, otherName, variables)));
47
- }
48
- /**
49
- * Unlink two objects
50
- */
51
- unlink(obj1, obj2, otherName = null) {
52
- return this.getMutation('unlink', obj1, obj2, otherName).pipe(switchMap(mutation => this.execute(mutation)));
53
- }
54
- /**
55
- * Return the list of all available mutation names
56
- */
57
- getAllMutationNames() {
58
- if (this.allMutations) {
59
- return of(this.allMutations);
60
- }
61
- const mapArg = (arg) => {
62
- return {
63
- name: arg.name,
64
- type: arg.type?.ofType?.name?.replace(/ID$/, '') || 'should-never-happen',
65
- };
66
- };
67
- return this.apollo
68
- .query({
69
- query: queriesQuery,
70
- fetchPolicy: 'cache-first',
71
- })
72
- .pipe(map(({ data }) => {
73
- if (data.__type?.fields) {
74
- this.allMutations = data.__type.fields
75
- .filter(v => /^(link|unlink)/.exec(v.name))
76
- .map(v => {
77
- return {
78
- name: v.name,
79
- arg1: mapArg(v.args[0]),
80
- arg2: mapArg(v.args[1]),
81
- };
82
- });
83
- }
84
- else {
85
- this.allMutations = [];
86
- }
87
- return this.allMutations;
88
- }));
89
- }
90
- /**
91
- * Generate mutation using patterns and replacing variables
92
- */
93
- getMutation(action, obj1, obj2, otherName, variables = {}) {
94
- otherName = otherName ? upperCaseFirstLetter(otherName) : otherName;
95
- const mutationName = action + obj1.__typename + (otherName || obj2.__typename);
96
- const reversedMutationName = action + obj2.__typename + (otherName || obj1.__typename);
97
- return this.getAllMutationNames().pipe(map(allMutationNames => {
98
- const mutation = allMutationNames.find(mut => mut.name === mutationName) ||
99
- allMutationNames.find(mut => mut.name === reversedMutationName);
100
- if (mutation) {
101
- return this.buildTemplate(mutation, obj1, obj2, variables);
102
- }
103
- throw TypeError('API does not allow to ' + action + ' ' + obj1.__typename + ' and ' + obj2.__typename);
104
- }));
105
- }
106
- /**
107
- * Execute mutation
108
- */
109
- execute(mutation) {
110
- return this.apollo
111
- .mutate({
112
- mutation: gql(mutation),
113
- })
114
- .pipe(map(r => {
115
- this.apollo.client.reFetchObservableQueries();
116
- return r;
117
- }));
118
- }
119
- /**
120
- * Build the actual mutation string
121
- */
122
- buildTemplate(mutation, obj1, obj2, variables = {}) {
123
- let name1;
124
- let name2;
125
- if (obj1.__typename === mutation.arg1.type) {
126
- name1 = mutation.arg1.name;
127
- name2 = mutation.arg2.name;
128
- }
129
- else {
130
- name1 = mutation.arg2.name;
131
- name2 = mutation.arg1.name;
132
- }
133
- variables[name1] = obj1.id;
134
- variables[name2] = obj2.id;
135
- let serializedVariables = '';
136
- for (const key of Object.keys(variables)) {
137
- serializedVariables += key + ': ' + JSON.stringify(variables[key]) + ' ';
138
- }
139
- return `mutation linkAndUnlink {
140
- ${mutation.name}(${serializedVariables}) {
141
- id
142
- }
143
- }`;
144
- }
145
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalLinkMutationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
146
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalLinkMutationService, providedIn: 'root' });
147
- }
148
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalLinkMutationService, decorators: [{
149
- type: Injectable,
150
- args: [{
151
- providedIn: 'root',
152
- }]
153
- }] });
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-mutation.service.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/services/link-mutation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Apollo, gql} from 'apollo-angular';\nimport {FetchResult} from '@apollo/client/core';\nimport {inject, Injectable} from '@angular/core';\nimport {clone} from 'lodash-es';\nimport {forkJoin, Observable, of} from 'rxjs';\nimport {map, switchMap} from 'rxjs/operators';\nimport {Literal} from '../types/types';\nimport {upperCaseFirstLetter} from '../classes/utility';\n\n/**\n * Query to get list of mutations\n */\nconst queriesQuery = gql`\n    query Mutations {\n        __type(name: \"Mutation\") {\n            fields {\n                name\n                args {\n                    name\n                    type {\n                        ofType {\n                            name\n                        }\n                    }\n                }\n            }\n        }\n    }\n`;\n\ntype IntrospectedArg = {\n    name: string;\n    type: {\n        ofType: null | {\n            name: null | string;\n        };\n    };\n};\n\ntype IntrospectedMutations = {\n    __type: null | {\n        fields:\n            | null\n            | {\n                  name: string;\n                  args: IntrospectedArg[];\n              }[];\n    };\n};\n\nexport type LinkableObject = {\n    id: string;\n    __typename: string;\n};\n\ntype MutationArg = {\n    name: string;\n    type: string;\n};\n\ntype Mutation = {\n    name: string;\n    arg1: MutationArg;\n    arg2: MutationArg;\n};\n\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalLinkMutationService {\n    private readonly apollo = inject(Apollo);\n\n    /**\n     * Receives the list of available mutations\n     */\n    private allMutations?: Mutation[];\n\n    /**\n     * Link two objects together\n     */\n    public link(\n        obj1: LinkableObject,\n        obj2: LinkableObject,\n        otherName: string | null = null,\n        variables: Literal = {},\n    ): Observable<FetchResult<{id: string}>> {\n        // clone prevents to affect the original reference\n        const clonedVariables = clone(variables);\n\n        return this.getMutation('link', obj1, obj2, otherName, clonedVariables).pipe(\n            switchMap(mutation => this.execute(mutation)),\n        );\n    }\n\n    /**\n     * Link many objects\n     */\n    public linkMany(\n        obj1: LinkableObject,\n        objects: LinkableObject[],\n        otherName: string | null = null,\n        variables: Literal = {},\n    ): Observable<FetchResult<{id: string}>[]> {\n        return forkJoin(objects.map(obj2 => this.link(obj1, obj2, otherName, variables)));\n    }\n\n    /**\n     * Unlink two objects\n     */\n    public unlink(\n        obj1: LinkableObject,\n        obj2: LinkableObject,\n        otherName: string | null = null,\n    ): Observable<FetchResult<{id: string}>> {\n        return this.getMutation('unlink', obj1, obj2, otherName).pipe(switchMap(mutation => this.execute(mutation)));\n    }\n\n    /**\n     * Return the list of all available mutation names\n     */\n    private getAllMutationNames(): Observable<Mutation[]> {\n        if (this.allMutations) {\n            return of(this.allMutations);\n        }\n\n        const mapArg = (arg: IntrospectedArg): MutationArg => {\n            return {\n                name: arg.name,\n                type: arg.type?.ofType?.name?.replace(/ID$/, '') || 'should-never-happen',\n            };\n        };\n\n        return this.apollo\n            .query<IntrospectedMutations>({\n                query: queriesQuery,\n                fetchPolicy: 'cache-first',\n            })\n            .pipe(\n                map(({data}) => {\n                    if (data.__type?.fields) {\n                        this.allMutations = data.__type.fields\n                            .filter(v => /^(link|unlink)/.exec(v.name))\n                            .map(v => {\n                                return {\n                                    name: v.name,\n                                    arg1: mapArg(v.args[0]),\n                                    arg2: mapArg(v.args[1]),\n                                };\n                            });\n                    } else {\n                        this.allMutations = [];\n                    }\n\n                    return this.allMutations;\n                }),\n            );\n    }\n\n    /**\n     * Generate mutation using patterns and replacing variables\n     */\n    private getMutation(\n        action: string,\n        obj1: LinkableObject,\n        obj2: LinkableObject,\n        otherName: string | null,\n        variables: Literal = {},\n    ): Observable<string> {\n        otherName = otherName ? upperCaseFirstLetter(otherName) : otherName;\n        const mutationName = action + obj1.__typename + (otherName || obj2.__typename);\n        const reversedMutationName = action + obj2.__typename + (otherName || obj1.__typename);\n\n        return this.getAllMutationNames().pipe(\n            map(allMutationNames => {\n                const mutation =\n                    allMutationNames.find(mut => mut.name === mutationName) ||\n                    allMutationNames.find(mut => mut.name === reversedMutationName);\n\n                if (mutation) {\n                    return this.buildTemplate(mutation, obj1, obj2, variables);\n                }\n\n                throw TypeError('API does not allow to ' + action + ' ' + obj1.__typename + ' and ' + obj2.__typename);\n            }),\n        );\n    }\n\n    /**\n     * Execute mutation\n     */\n    private execute(mutation: string): Observable<FetchResult<{id: string}>> {\n        return this.apollo\n            .mutate<{id: string}>({\n                mutation: gql(mutation),\n            })\n            .pipe(\n                map(r => {\n                    this.apollo.client.reFetchObservableQueries();\n                    return r;\n                }),\n            );\n    }\n\n    /**\n     * Build the actual mutation string\n     */\n    private buildTemplate(\n        mutation: Mutation,\n        obj1: LinkableObject,\n        obj2: LinkableObject,\n        variables: Literal = {},\n    ): string {\n        let name1;\n        let name2;\n        if (obj1.__typename === mutation.arg1.type) {\n            name1 = mutation.arg1.name;\n            name2 = mutation.arg2.name;\n        } else {\n            name1 = mutation.arg2.name;\n            name2 = mutation.arg1.name;\n        }\n        variables[name1] = obj1.id;\n        variables[name2] = obj2.id;\n\n        let serializedVariables = '';\n        for (const key of Object.keys(variables)) {\n            serializedVariables += key + ': ' + JSON.stringify(variables[key]) + ' ';\n        }\n\n        return `mutation linkAndUnlink {\n            ${mutation.name}(${serializedVariables}) {\n                id\n            }\n        }`;\n    }\n}\n"]}
@@ -1,115 +0,0 @@
1
- import { inject, Injectable, InjectionToken } from '@angular/core';
2
- import { Router } from '@angular/router';
3
- import { clone } from 'lodash-es';
4
- import { SESSION_STORAGE } from '../modules/common/services/memory-storage';
5
- import * as i0 from "@angular/core";
6
- export const NATURAL_PERSISTENCE_VALIDATOR = new InjectionToken('Validator for persisted value retrieved from NaturalPersistenceService. If returns false, the persisted value will never be returned.');
7
- export class NaturalPersistenceService {
8
- router = inject(Router);
9
- sessionStorage = inject(SESSION_STORAGE);
10
- isValid = inject(NATURAL_PERSISTENCE_VALIDATOR, { optional: true }) ?? (() => true); // By default, anything is valid
11
- /**
12
- * Persist in url and local storage the given value with the given key.
13
- * When stored in storage, we need more "key" to identify the controller.
14
- */
15
- persist(key, value, route, storageKey, navigationExtras) {
16
- this.persistInStorage(key, value, storageKey);
17
- return this.persistInUrl(key, value, route, navigationExtras);
18
- }
19
- /**
20
- * Return object with persisted data in url or in session storage
21
- * Url has priority over session storage because of url sharing. When url is provided, session storage is ignored.
22
- * Url and storage are synced when arriving in a component :
23
- * - When loading with url parameters, storage is updated to stay synced
24
- * - When loading without url, but with storage data, the url is updated
25
- */
26
- // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
27
- get(key, route, storageKey) {
28
- // From url
29
- let params = this.getFromUrl(key, route);
30
- if (!this.isFalseyValue(params)) {
31
- this.persistInStorage(key, params, storageKey);
32
- return params;
33
- }
34
- // From storage
35
- params = this.getFromStorage(key, storageKey);
36
- if (!this.isFalseyValue(params)) {
37
- this.persistInUrl(key, params, route);
38
- return params;
39
- }
40
- return null;
41
- }
42
- /**
43
- * Get given key from the url parameters
44
- */
45
- // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
46
- getFromUrl(key, route) {
47
- const value = route.snapshot.paramMap.get(key);
48
- return this.deserialize(key, null, value);
49
- }
50
- /**
51
- * Add/override given pair key-value in the url
52
- * Always JSON.stringify() the given value
53
- * If the value is falsey, the pair key-value is removed from the url.
54
- */
55
- persistInUrl(key, value, route, navigationExtras) {
56
- const params = clone(route.snapshot.url[route.snapshot.url.length - 1].parameters);
57
- if (this.isFalseyValue(value)) {
58
- delete params[key];
59
- }
60
- else {
61
- params[key] = JSON.stringify(value);
62
- }
63
- navigationExtras = Object.assign(navigationExtras || {}, { relativeTo: route });
64
- return this.router.navigate(['.', params], navigationExtras);
65
- }
66
- // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
67
- getFromStorage(key, storageKey) {
68
- const value = this.sessionStorage.getItem(this.getStorageKey(key, storageKey));
69
- return this.deserialize(key, storageKey, value);
70
- }
71
- /**
72
- * Store value in session storage.
73
- * If value is falsy, the entry is removed
74
- */
75
- persistInStorage(key, value, storageKey) {
76
- if (this.isFalseyValue(value)) {
77
- this.sessionStorage.removeItem(this.getStorageKey(key, storageKey));
78
- }
79
- else {
80
- this.sessionStorage.setItem(this.getStorageKey(key, storageKey), JSON.stringify(value));
81
- }
82
- }
83
- getStorageKey(key, storageKey) {
84
- return storageKey + '-' + key;
85
- }
86
- // Returns if the given value is falsey
87
- // Falsey values are : null, undefined and empty string.
88
- // This cause usually the parameter to be removed from url/storage instead of being stored with no value. Url would be polluted.
89
- isFalseyValue(value) {
90
- return value == null || value === ''; // == means null or undefined;
91
- }
92
- // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
93
- deserialize(key, storageKey, value) {
94
- if (!value) {
95
- return null;
96
- }
97
- let result = null;
98
- try {
99
- result = JSON.parse(value);
100
- }
101
- catch (e) {
102
- // noop
103
- }
104
- return this.isValid(key, storageKey, result) ? result : null;
105
- }
106
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPersistenceService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
107
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPersistenceService, providedIn: 'root' });
108
- }
109
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPersistenceService, decorators: [{
110
- type: Injectable,
111
- args: [{
112
- providedIn: 'root',
113
- }]
114
- }] });
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"persistence.service.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/services/persistence.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {inject, Injectable, InjectionToken} from '@angular/core';\nimport {ActivatedRoute, NavigationExtras, Router} from '@angular/router';\nimport {clone} from 'lodash-es';\nimport {SESSION_STORAGE} from '../modules/common/services/memory-storage';\n\n/**\n * Validator for persisted values retrieved from NaturalPersistenceService. If returns false, the persisted value\n * will be ignored, and instead `null` will be returned.\n *\n * `storageKey` is only given if the value is coming from session storage (and not from URL).\n */\nexport type PersistenceValidator = (key: string, storageKey: string | null, value: unknown) => boolean;\n\nexport const NATURAL_PERSISTENCE_VALIDATOR = new InjectionToken<PersistenceValidator>(\n    'Validator for persisted value retrieved from NaturalPersistenceService. If returns false, the persisted value will never be returned.',\n);\n\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalPersistenceService {\n    private readonly router = inject(Router);\n    private readonly sessionStorage = inject(SESSION_STORAGE);\n    private readonly isValid = inject(NATURAL_PERSISTENCE_VALIDATOR, {optional: true}) ?? (() => true); // By default, anything is valid\n\n    /**\n     * Persist in url and local storage the given value with the given key.\n     * When stored in storage, we need more \"key\" to identify the controller.\n     */\n    public persist(\n        key: string,\n        value: unknown,\n        route: ActivatedRoute,\n        storageKey: string,\n        navigationExtras?: NavigationExtras,\n    ): Promise<boolean> {\n        this.persistInStorage(key, value, storageKey);\n        return this.persistInUrl(key, value, route, navigationExtras);\n    }\n\n    /**\n     * Return object with persisted data in url or in session storage\n     * Url has priority over session storage because of url sharing. When url is provided, session storage is ignored.\n     * Url and storage are synced when arriving in a component :\n     *  - When loading with url parameters, storage is updated to stay synced\n     *  - When loading without url, but with storage data, the url is updated\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    public get(key: string, route: ActivatedRoute, storageKey: string): any | null {\n        // From url\n        let params = this.getFromUrl(key, route);\n        if (!this.isFalseyValue(params)) {\n            this.persistInStorage(key, params, storageKey);\n            return params;\n        }\n\n        // From storage\n        params = this.getFromStorage(key, storageKey);\n        if (!this.isFalseyValue(params)) {\n            this.persistInUrl(key, params, route);\n            return params;\n        }\n\n        return null;\n    }\n\n    /**\n     * Get given key from the url parameters\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    public getFromUrl(key: string, route: ActivatedRoute): any | null {\n        const value = route.snapshot.paramMap.get(key);\n\n        return this.deserialize(key, null, value);\n    }\n\n    /**\n     * Add/override given pair key-value in the url\n     * Always JSON.stringify() the given value\n     * If the value is falsey, the pair key-value is removed from the url.\n     */\n    public persistInUrl(\n        key: string,\n        value: unknown,\n        route: ActivatedRoute,\n        navigationExtras?: NavigationExtras,\n    ): Promise<boolean> {\n        const params = clone(route.snapshot.url[route.snapshot.url.length - 1].parameters);\n\n        if (this.isFalseyValue(value)) {\n            delete params[key];\n        } else {\n            params[key] = JSON.stringify(value);\n        }\n\n        navigationExtras = Object.assign(navigationExtras || {}, {relativeTo: route});\n\n        return this.router.navigate(['.', params], navigationExtras);\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    public getFromStorage(key: string, storageKey: string): any | null {\n        const value = this.sessionStorage.getItem(this.getStorageKey(key, storageKey));\n\n        return this.deserialize(key, storageKey, value);\n    }\n\n    /**\n     * Store value in session storage.\n     * If value is falsy, the entry is removed\n     */\n    public persistInStorage(key: string, value: unknown, storageKey: string): void {\n        if (this.isFalseyValue(value)) {\n            this.sessionStorage.removeItem(this.getStorageKey(key, storageKey));\n        } else {\n            this.sessionStorage.setItem(this.getStorageKey(key, storageKey), JSON.stringify(value));\n        }\n    }\n\n    private getStorageKey(key: string, storageKey: string): string {\n        return storageKey + '-' + key;\n    }\n\n    // Returns if the given value is falsey\n    // Falsey values are : null, undefined and empty string.\n    // This cause usually the parameter to be removed from url/storage instead of being stored with no value. Url would be polluted.\n    private isFalseyValue(value: unknown): boolean {\n        return value == null || value === ''; // == means null or undefined;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    private deserialize(key: string, storageKey: string | null, value: string | null): unknown | null {\n        if (!value) {\n            return null;\n        }\n\n        let result = null;\n        try {\n            result = JSON.parse(value);\n        } catch (e) {\n            // noop\n        }\n\n        return this.isValid(key, storageKey, result) ? result : null;\n    }\n}\n"]}
@@ -1,63 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import { NativeDateAdapter } from '@angular/material/core';
3
- import * as i0 from "@angular/core";
4
- const patterns = [
5
- /^(?<day>\d{1,2})\.(?<month>\d{1,2})\.(?<year>\d{4}|\d{2})$/,
6
- /^(?<day>\d{1,2})-(?<month>\d{1,2})-(?<year>\d{4}|\d{2})$/,
7
- /^(?<day>\d{1,2})\/(?<month>\d{1,2})\/(?<year>\d{4}|\d{2})$/,
8
- /^(?<day>\d{1,2})\\(?<month>\d{1,2})\\(?<year>\d{4}|\d{2})$/,
9
- // strict ISO format
10
- /^(?<year>\d{4})-(?<month>\d{2})-(?<day>\d{2})$/,
11
- ];
12
- export class NaturalSwissParsingDateAdapter extends NativeDateAdapter {
13
- /**
14
- * Parse commonly accepted swiss format, such as:
15
- *
16
- * - 24.12.2018
17
- * - 1.4.18
18
- * - 2018-12-24
19
- */
20
- parse(value) {
21
- if (typeof value === 'number') {
22
- return new Date(value);
23
- }
24
- if (typeof value === 'string') {
25
- const trimmed = value.trim();
26
- for (const pattern of patterns) {
27
- const m = trimmed.match(pattern);
28
- if (m?.groups) {
29
- const year = +m.groups.year;
30
- const month = +m.groups.month;
31
- const day = +m.groups.day;
32
- return this.createDateIfValid(year, month, day);
33
- }
34
- }
35
- }
36
- return null;
37
- }
38
- createDateIfValid(year, month, date) {
39
- // Assume year 2000 if only two digits
40
- if (year < 100) {
41
- year += 2000;
42
- }
43
- month = month - 1;
44
- if (month >= 0 && month <= 11 && date >= 1 && date <= 31) {
45
- return this.createDate(year, month, date);
46
- }
47
- return null;
48
- }
49
- getFirstDayOfWeek() {
50
- // Always starts on Monday, even though it is not true for Canada, U.S., Mexico and many more
51
- // Also see https://github.com/tc39/ecma402/issues/6
52
- return 1;
53
- }
54
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSwissParsingDateAdapter, deps: null, target: i0.ɵɵFactoryTarget.Injectable });
55
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSwissParsingDateAdapter, providedIn: 'root' });
56
- }
57
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSwissParsingDateAdapter, decorators: [{
58
- type: Injectable,
59
- args: [{
60
- providedIn: 'root',
61
- }]
62
- }] });
63
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3dpc3MtcGFyc2luZy1kYXRlLWFkYXB0ZXIuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9zZXJ2aWNlcy9zd2lzcy1wYXJzaW5nLWRhdGUtYWRhcHRlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFDLGlCQUFpQixFQUFDLE1BQU0sd0JBQXdCLENBQUM7O0FBRXpELE1BQU0sUUFBUSxHQUFzQjtJQUNoQyw0REFBNEQ7SUFDNUQsMERBQTBEO0lBQzFELDREQUE0RDtJQUM1RCw0REFBNEQ7SUFDNUQsb0JBQW9CO0lBQ3BCLGdEQUFnRDtDQUNuRCxDQUFDO0FBS0YsTUFBTSxPQUFPLDhCQUErQixTQUFRLGlCQUFpQjtJQUNqRTs7Ozs7O09BTUc7SUFDYSxLQUFLLENBQUMsS0FBYztRQUNoQyxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzVCLE9BQU8sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDM0IsQ0FBQztRQUVELElBQUksT0FBTyxLQUFLLEtBQUssUUFBUSxFQUFFLENBQUM7WUFDNUIsTUFBTSxPQUFPLEdBQUcsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO1lBRTdCLEtBQUssTUFBTSxPQUFPLElBQUksUUFBUSxFQUFFLENBQUM7Z0JBQzdCLE1BQU0sQ0FBQyxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7Z0JBQ2pDLElBQUksQ0FBQyxFQUFFLE1BQU0sRUFBRSxDQUFDO29CQUNaLE1BQU0sSUFBSSxHQUFHLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUM7b0JBQzVCLE1BQU0sS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUM7b0JBQzlCLE1BQU0sR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUM7b0JBRTFCLE9BQU8sSUFBSSxDQUFDLGlCQUFpQixDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUM7Z0JBQ3BELENBQUM7WUFDTCxDQUFDO1FBQ0wsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDO0lBQ2hCLENBQUM7SUFFTyxpQkFBaUIsQ0FBQyxJQUFZLEVBQUUsS0FBYSxFQUFFLElBQVk7UUFDL0Qsc0NBQXNDO1FBQ3RDLElBQUksSUFBSSxHQUFHLEdBQUcsRUFBRSxDQUFDO1lBQ2IsSUFBSSxJQUFJLElBQUksQ0FBQztRQUNqQixDQUFDO1FBRUQsS0FBSyxHQUFHLEtBQUssR0FBRyxDQUFDLENBQUM7UUFDbEIsSUFBSSxLQUFLLElBQUksQ0FBQyxJQUFJLEtBQUssSUFBSSxFQUFFLElBQUksSUFBSSxJQUFJLENBQUMsSUFBSSxJQUFJLElBQUksRUFBRSxFQUFFLENBQUM7WUFDdkQsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDOUMsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDO0lBQ2hCLENBQUM7SUFFZSxpQkFBaUI7UUFDN0IsNkZBQTZGO1FBQzdGLG9EQUFvRDtRQUNwRCxPQUFPLENBQUMsQ0FBQztJQUNiLENBQUM7dUdBakRRLDhCQUE4QjsyR0FBOUIsOEJBQThCLGNBRjNCLE1BQU07OzJGQUVULDhCQUE4QjtrQkFIMUMsVUFBVTttQkFBQztvQkFDUixVQUFVLEVBQUUsTUFBTTtpQkFDckIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0luamVjdGFibGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtOYXRpdmVEYXRlQWRhcHRlcn0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY29yZSc7XG5cbmNvbnN0IHBhdHRlcm5zOiByZWFkb25seSBSZWdFeHBbXSA9IFtcbiAgICAvXig/PGRheT5cXGR7MSwyfSlcXC4oPzxtb250aD5cXGR7MSwyfSlcXC4oPzx5ZWFyPlxcZHs0fXxcXGR7Mn0pJC8sXG4gICAgL14oPzxkYXk+XFxkezEsMn0pLSg/PG1vbnRoPlxcZHsxLDJ9KS0oPzx5ZWFyPlxcZHs0fXxcXGR7Mn0pJC8sXG4gICAgL14oPzxkYXk+XFxkezEsMn0pXFwvKD88bW9udGg+XFxkezEsMn0pXFwvKD88eWVhcj5cXGR7NH18XFxkezJ9KSQvLFxuICAgIC9eKD88ZGF5PlxcZHsxLDJ9KVxcXFwoPzxtb250aD5cXGR7MSwyfSlcXFxcKD88eWVhcj5cXGR7NH18XFxkezJ9KSQvLFxuICAgIC8vIHN0cmljdCBJU08gZm9ybWF0XG4gICAgL14oPzx5ZWFyPlxcZHs0fSktKD88bW9udGg+XFxkezJ9KS0oPzxkYXk+XFxkezJ9KSQvLFxuXTtcblxuQEluamVjdGFibGUoe1xuICAgIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbFN3aXNzUGFyc2luZ0RhdGVBZGFwdGVyIGV4dGVuZHMgTmF0aXZlRGF0ZUFkYXB0ZXIge1xuICAgIC8qKlxuICAgICAqIFBhcnNlIGNvbW1vbmx5IGFjY2VwdGVkIHN3aXNzIGZvcm1hdCwgc3VjaCBhczpcbiAgICAgKlxuICAgICAqIC0gMjQuMTIuMjAxOFxuICAgICAqIC0gMS40LjE4XG4gICAgICogLSAyMDE4LTEyLTI0XG4gICAgICovXG4gICAgcHVibGljIG92ZXJyaWRlIHBhcnNlKHZhbHVlOiB1bmtub3duKTogRGF0ZSB8IG51bGwge1xuICAgICAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnbnVtYmVyJykge1xuICAgICAgICAgICAgcmV0dXJuIG5ldyBEYXRlKHZhbHVlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgICAgICBjb25zdCB0cmltbWVkID0gdmFsdWUudHJpbSgpO1xuXG4gICAgICAgICAgICBmb3IgKGNvbnN0IHBhdHRlcm4gb2YgcGF0dGVybnMpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBtID0gdHJpbW1lZC5tYXRjaChwYXR0ZXJuKTtcbiAgICAgICAgICAgICAgICBpZiAobT8uZ3JvdXBzKSB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHllYXIgPSArbS5ncm91cHMueWVhcjtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbW9udGggPSArbS5ncm91cHMubW9udGg7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheSA9ICttLmdyb3Vwcy5kYXk7XG5cbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuY3JlYXRlRGF0ZUlmVmFsaWQoeWVhciwgbW9udGgsIGRheSk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBjcmVhdGVEYXRlSWZWYWxpZCh5ZWFyOiBudW1iZXIsIG1vbnRoOiBudW1iZXIsIGRhdGU6IG51bWJlcik6IERhdGUgfCBudWxsIHtcbiAgICAgICAgLy8gQXNzdW1lIHllYXIgMjAwMCBpZiBvbmx5IHR3byBkaWdpdHNcbiAgICAgICAgaWYgKHllYXIgPCAxMDApIHtcbiAgICAgICAgICAgIHllYXIgKz0gMjAwMDtcbiAgICAgICAgfVxuXG4gICAgICAgIG1vbnRoID0gbW9udGggLSAxO1xuICAgICAgICBpZiAobW9udGggPj0gMCAmJiBtb250aCA8PSAxMSAmJiBkYXRlID49IDEgJiYgZGF0ZSA8PSAzMSkge1xuICAgICAgICAgICAgcmV0dXJuIHRoaXMuY3JlYXRlRGF0ZSh5ZWFyLCBtb250aCwgZGF0ZSk7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBwdWJsaWMgb3ZlcnJpZGUgZ2V0Rmlyc3REYXlPZldlZWsoKTogbnVtYmVyIHtcbiAgICAgICAgLy8gQWx3YXlzIHN0YXJ0cyBvbiBNb25kYXksIGV2ZW4gdGhvdWdoIGl0IGlzIG5vdCB0cnVlIGZvciBDYW5hZGEsIFUuUy4sIE1leGljbyBhbmQgbWFueSBtb3JlXG4gICAgICAgIC8vIEFsc28gc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS90YzM5L2VjbWE0MDIvaXNzdWVzLzZcbiAgICAgICAgcmV0dXJuIDE7XG4gICAgfVxufVxuIl19