@ecodev/natural 62.1.3 → 62.3.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 (188) hide show
  1. package/fesm2022/ecodev-natural-vanilla.mjs +1199 -0
  2. package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
  3. package/fesm2022/ecodev-natural.mjs +606 -644
  4. package/fesm2022/ecodev-natural.mjs.map +1 -1
  5. package/lib/classes/abstract-editable-list.d.ts +3 -2
  6. package/lib/classes/network-activity.service.d.ts +54 -0
  7. package/lib/classes/validators.d.ts +1 -1
  8. package/lib/modules/avatar/component/avatar.component.d.ts +21 -36
  9. package/lib/modules/avatar/service/avatar.service.d.ts +3 -2
  10. package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
  11. package/lib/modules/common/services/seo.provider.d.ts +2 -2
  12. package/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.d.ts +1 -0
  13. package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
  14. package/lib/modules/file/abstract-file.d.ts +8 -5
  15. package/lib/modules/file/component/file.component.d.ts +9 -4
  16. package/lib/modules/file/file-drop.directive.d.ts +2 -3
  17. package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
  18. package/lib/modules/hierarchic-selector/classes/model-node.d.ts +1 -1
  19. package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
  20. package/lib/modules/icon/icon.module.d.ts +2 -2
  21. package/lib/modules/panels/panels.service.d.ts +1 -2
  22. package/lib/modules/relations/relations.component.d.ts +4 -4
  23. package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
  24. package/lib/modules/search/group/group.component.d.ts +2 -3
  25. package/lib/modules/search/input/input.component.d.ts +5 -5
  26. package/lib/modules/search/search/search.component.d.ts +5 -4
  27. package/lib/modules/select/abstract-select.component.d.ts +3 -3
  28. package/lib/modules/select/select/select.component.d.ts +4 -4
  29. package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
  30. package/lib/modules/table-button/table-button.component.d.ts +18 -18
  31. package/package.json +16 -14
  32. package/public-api.d.ts +1 -0
  33. package/src/lib/_natural.theme.scss +1 -2
  34. package/vanilla/index.d.ts +5 -0
  35. package/vanilla/package.json +3 -0
  36. package/vanilla/public-api.d.ts +11 -0
  37. package/vanilla/src/lib/classes/crypto.d.ts +8 -0
  38. package/vanilla/src/lib/classes/data-source.d.ts +32 -0
  39. package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
  40. package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
  41. package/vanilla/src/lib/classes/signing.d.ts +7 -0
  42. package/vanilla/src/lib/classes/utility.d.ts +77 -0
  43. package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
  44. package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
  45. package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
  46. package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
  47. package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
  48. package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
  49. package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
  50. package/vanilla/src/lib/types/types.d.ts +100 -0
  51. package/esm2022/ecodev-natural.mjs +0 -5
  52. package/esm2022/lib/classes/abstract-detail.mjs +0 -229
  53. package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
  54. package/esm2022/lib/classes/abstract-list.mjs +0 -461
  55. package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
  56. package/esm2022/lib/classes/apollo-utils.mjs +0 -59
  57. package/esm2022/lib/classes/crypto.mjs +0 -23
  58. package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
  59. package/esm2022/lib/classes/data-source.mjs +0 -71
  60. package/esm2022/lib/classes/providers.mjs +0 -13
  61. package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
  62. package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
  63. package/esm2022/lib/classes/rxjs.mjs +0 -54
  64. package/esm2022/lib/classes/signing.mjs +0 -38
  65. package/esm2022/lib/classes/tld.mjs +0 -1476
  66. package/esm2022/lib/classes/utility.mjs +0 -234
  67. package/esm2022/lib/classes/validators.mjs +0 -179
  68. package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
  69. package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
  70. package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
  71. package/esm2022/lib/modules/alert/public-api.mjs +0 -6
  72. package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
  73. package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
  74. package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
  75. package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
  76. package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
  77. package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
  78. package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
  79. package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
  80. package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
  81. package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
  82. package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
  83. package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
  84. package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
  85. package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
  86. package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
  87. package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
  88. package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
  89. package/esm2022/lib/modules/common/public-api.mjs +0 -14
  90. package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
  91. package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
  92. package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
  93. package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
  94. package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
  95. package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
  96. package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
  97. package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
  98. package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
  99. package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
  100. package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
  101. package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
  102. package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
  103. package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
  104. package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
  105. package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
  106. package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
  107. package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
  108. package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
  109. package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
  110. package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
  111. package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
  112. package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
  113. package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
  114. package/esm2022/lib/modules/file/file.service.mjs +0 -43
  115. package/esm2022/lib/modules/file/public-api.mjs +0 -9
  116. package/esm2022/lib/modules/file/types.mjs +0 -2
  117. package/esm2022/lib/modules/file/utils.mjs +0 -129
  118. package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
  119. package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
  120. package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
  121. package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
  122. package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
  123. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
  124. package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
  125. package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
  126. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
  127. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
  128. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
  129. package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
  130. package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
  131. package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
  132. package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
  133. package/esm2022/lib/modules/icon/public-api.mjs +0 -6
  134. package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
  135. package/esm2022/lib/modules/logger/error.module.mjs +0 -22
  136. package/esm2022/lib/modules/logger/public-api.mjs +0 -6
  137. package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
  138. package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
  139. package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
  140. package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
  141. package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
  142. package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
  143. package/esm2022/lib/modules/panels/panels.service.mjs +0 -348
  144. package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
  145. package/esm2022/lib/modules/panels/public-api.mjs +0 -11
  146. package/esm2022/lib/modules/panels/types.mjs +0 -3
  147. package/esm2022/lib/modules/relations/public-api.mjs +0 -5
  148. package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
  149. package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
  150. package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
  151. package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
  152. package/esm2022/lib/modules/search/classes/url.mjs +0 -53
  153. package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
  154. package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
  155. package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
  156. package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
  157. package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
  158. package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
  159. package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
  160. package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
  161. package/esm2022/lib/modules/search/public-api.mjs +0 -7
  162. package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
  163. package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
  164. package/esm2022/lib/modules/search/types/facet.mjs +0 -2
  165. package/esm2022/lib/modules/search/types/values.mjs +0 -2
  166. package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
  167. package/esm2022/lib/modules/select/public-api.mjs +0 -7
  168. package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
  169. package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
  170. package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
  171. package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
  172. package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
  173. package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
  174. package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
  175. package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
  176. package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
  177. package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
  178. package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
  179. package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
  180. package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
  181. package/esm2022/lib/services/abstract-model.service.mjs +0 -526
  182. package/esm2022/lib/services/debounce.service.mjs +0 -149
  183. package/esm2022/lib/services/enum.service.mjs +0 -64
  184. package/esm2022/lib/services/link-mutation.service.mjs +0 -154
  185. package/esm2022/lib/services/persistence.service.mjs +0 -115
  186. package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
  187. package/esm2022/lib/types/types.mjs +0 -2
  188. package/esm2022/public-api.mjs +0 -46
@@ -1,348 +0,0 @@
1
- import { BreakpointObserver, Breakpoints } from '@angular/cdk/layout';
2
- import { inject, Injectable, Injector, runInInjectionContext } from '@angular/core';
3
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
4
- import { MatDialog } from '@angular/material/dialog';
5
- import { DefaultUrlSerializer, NavigationError, Router } from '@angular/router';
6
- import { differenceWith, flatten, isEqual } from 'lodash-es';
7
- import { forkJoin, of, Subject } from 'rxjs';
8
- import { getStackConfig } from './panels.urlmatcher';
9
- import { PanelsHooksConfig, } from './types';
10
- import * as i0 from "@angular/core";
11
- function segmentsToString(segments) {
12
- return segments.map(s => s.toString()).join('/');
13
- }
14
- function compareConfigs(a, b) {
15
- return a.route.path === b.route.path && a.rule === b.rule && isEqual(a.params, b.params);
16
- }
17
- /**
18
- * TODO: implement route update when closing dialog with escape
19
- * @dynamic
20
- */
21
- export class NaturalPanelsService {
22
- router = inject(Router);
23
- dialog = inject(MatDialog);
24
- injector = inject(Injector);
25
- hooksConfig = inject(PanelsHooksConfig);
26
- panelWidth = '960px';
27
- /**
28
- * Because of this static property Panels are **not** compatible with SSR.
29
- * And we cannot make it non-static, because `UrlMatcher` cannot be injected.
30
- */
31
- static _opened = false;
32
- static get opened() {
33
- return this._opened;
34
- }
35
- /**
36
- * Stream that emits when all open dialog have finished closing
37
- */
38
- afterAllClosed = new Subject();
39
- /**
40
- * Cache for panels counter. Works more like an ID.
41
- * Is used to give an unique identifier to multiple similar panels configurations
42
- */
43
- counter = 1;
44
- /**
45
- * Class applied to dialog overlay related with panels
46
- * If change, change CSS too
47
- */
48
- panelClass = 'panel';
49
- /**
50
- * Cache for panels setup before navigation change.
51
- * Used to detect panels openings/closings and adapt for new configuration
52
- */
53
- oldFullConfig = [];
54
- /**
55
- * Cache for subscription stop
56
- */
57
- routeSub;
58
- /**
59
- * Cache for subscription stop
60
- */
61
- navSub;
62
- /**
63
- * Horizontal gaps between panels
64
- */
65
- panelsOffsetH = 35;
66
- /**
67
- * Vertical gaps between panels
68
- */
69
- panelsOffsetV = 40;
70
- /**
71
- * Cache of previous screen size
72
- * Used to change panels stack orientation on small screens
73
- */
74
- isVertical = false;
75
- constructor() {
76
- const breakpointObserver = inject(BreakpointObserver);
77
- // Watch media to know if display panels horizontally or vertically
78
- breakpointObserver
79
- .observe(Breakpoints.XSmall)
80
- .pipe(takeUntilDestroyed())
81
- .subscribe(result => {
82
- this.isVertical = result.matches;
83
- this.updateComponentsPosition();
84
- });
85
- }
86
- /**
87
- * Notify the service to start listening to route changes to open panels
88
- *
89
- * @internal
90
- */
91
- start(route) {
92
- NaturalPanelsService._opened = true;
93
- this.routeSub = route.url.subscribe(segments => {
94
- this.updatePanels(segments, route.snapshot.data.panelsRoutes);
95
- });
96
- this.navSub = this.router.events.subscribe(ev => {
97
- if (!(ev instanceof NavigationError)) {
98
- return;
99
- }
100
- // Abort and propagate navigation error
101
- if (ev.url.endsWith('/')) {
102
- this.stop();
103
- // This is a bit hackish: we hardcode an impossible url to trigger normal navigation error mechanism
104
- this.router.navigateByUrl('panels-reached-invalid-url');
105
- return;
106
- }
107
- this.counter++;
108
- // pc stands for "panel counter", required to give an identification to panels with exact same config
109
- // E.g /new
110
- const wantedUrl = ev.url + ';pc=' + this.counter;
111
- // Segments matching from wanted url. E.g ~ ['new']
112
- const wantedUrSegments = new DefaultUrlSerializer().parse(wantedUrl).root.children.primary.segments;
113
- // Don't match any config
114
- const wantedConfig = getStackConfig(wantedUrSegments, route.snapshot.data.panelsRoutes, this.injector);
115
- if (wantedConfig.length) {
116
- return this.appendConfigToCurrentUrl(wantedConfig);
117
- }
118
- // Currently activated routes. E.g ['objective', 'objective', 123, 'risk']
119
- const currentSegments = flatten(this.dialog.openDialogs.map(d => d.componentInstance.panelData.config.route.segments));
120
- // Last segment. E.g ['risk']
121
- const lastOfCurrentSegments = currentSegments.slice(-1);
122
- // Config for ['risk', 'new']
123
- const currentAndWantedConfig = getStackConfig(lastOfCurrentSegments.concat(wantedUrSegments), route.snapshot.data.panelsRoutes, this.injector);
124
- return this.appendConfigToCurrentUrl(currentAndWantedConfig);
125
- });
126
- }
127
- /**
128
- * Uses given configuration to add in the end of current url
129
- * Neutralizes router error handling
130
- */
131
- appendConfigToCurrentUrl(config) {
132
- const originalErrorHandler = this.router.errorHandler;
133
- // Nullify error handler (will be de-neutralized after route redirection)
134
- if (config) {
135
- this.router.errorHandler = () => undefined;
136
- }
137
- // Navigate to same url + /risk/new Result : /risk/risk/new
138
- const newUrl = config.map(conf => segmentsToString(conf.route.segments)).join('/');
139
- this.router.navigateByUrl(this.router.url + '/' + newUrl).then(() => {
140
- // After navigation has ended, restore original error handler because he's not a bad guy
141
- this.router.errorHandler = originalErrorHandler;
142
- });
143
- }
144
- /**
145
- * Notify the service that all panels were closed
146
- *
147
- * @internal
148
- */
149
- stop() {
150
- NaturalPanelsService._opened = false;
151
- this.routeSub?.unsubscribe();
152
- this.navSub?.unsubscribe();
153
- this.dialog.closeAll();
154
- this.oldFullConfig = [];
155
- this.afterAllClosed.next();
156
- }
157
- /**
158
- * Go to panel matching given component. Causes an url change.
159
- *
160
- * @internal
161
- */
162
- goToPanelByComponent(component) {
163
- this.goToPanelByIndex(this.getPanelIndex(component));
164
- }
165
- /**
166
- * Go to panel matching given component. Causes an url change.
167
- */
168
- goToPenultimatePanel() {
169
- this.goToPanelByIndex(this.dialog.openDialogs.length - 2);
170
- }
171
- /**
172
- * Calls the new url that only includes the segments from the panels we want to stay open
173
- */
174
- goToPanelByIndex(index) {
175
- // Extracts url segments from next panel until last one
176
- const url = this.dialog.openDialogs
177
- .slice(index + 1)
178
- .map(dialog => {
179
- return segmentsToString(dialog.componentInstance.panelData.config.route.segments);
180
- })
181
- .join('/');
182
- // Remove extra segments and redirects to root
183
- this.router.navigateByUrl(this.router.url.replace('/' + url, ''));
184
- }
185
- /**
186
- * Selecting a panel is equivalent to close all those that are in front of him
187
- * @param index of panel in stack. The most behind (the first one) is 0.
188
- */
189
- selectPanelByIndex(index) {
190
- const lastDialog = this.dialog.openDialogs[this.dialog.openDialogs.length - 1];
191
- // Update new panels set positions
192
- this.updateComponentsPosition();
193
- for (let i = this.dialog.openDialogs.length - 1; i >= index + 1; i--) {
194
- this.dialog.openDialogs[i].close();
195
- }
196
- return lastDialog.afterClosed();
197
- }
198
- /**
199
- * Open new panels if url has changed with new segments
200
- */
201
- updatePanels(segments, routes) {
202
- // Transform url segments into a config with component name and ID if provided in next segment
203
- // Returns an array of configs, each config represents the content relative to a panel
204
- const newFullConfig = getStackConfig(segments, routes, this.injector);
205
- const configsToRemove = differenceWith(this.oldFullConfig, newFullConfig, compareConfigs);
206
- const configsToAdd = differenceWith(newFullConfig, this.oldFullConfig, compareConfigs);
207
- const indexOfNextPanel = this.oldFullConfig.length - configsToRemove.length - 1;
208
- if (configsToRemove.length && configsToAdd.length) {
209
- // Add and remove panels
210
- this.selectPanelByIndex(indexOfNextPanel).subscribe(() => {
211
- this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());
212
- });
213
- }
214
- else if (configsToRemove.length && !configsToAdd.length) {
215
- // only remove panels
216
- this.selectPanelByIndex(indexOfNextPanel).subscribe(() => this.updateComponentsPosition());
217
- }
218
- else if (!configsToRemove.length && configsToAdd.length) {
219
- // only add panels
220
- this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());
221
- }
222
- this.oldFullConfig = newFullConfig;
223
- }
224
- /**
225
- * Resolve all services, then open panels
226
- */
227
- openPanels(newItemsConfig, fullConfig) {
228
- const subject = new Subject();
229
- // Resolve everything before opening a single panel
230
- const resolves = newItemsConfig.map(conf => this.getResolvedData(conf));
231
- // ForkJoin emits when all promises are executed;
232
- forkJoin(resolves).subscribe(resolvedResult => {
233
- // For each new config entry, open a new panel
234
- for (let i = 0; i < newItemsConfig.length; i++) {
235
- const config = newItemsConfig[i];
236
- let itemData = {
237
- // Config of actual panel route
238
- config: config,
239
- // Data resolved by service
240
- // Use in component by calling this.panelData.data.xyz
241
- data: resolvedResult[i],
242
- linkableObjects: [],
243
- };
244
- if (this.hooksConfig?.beforeOpenPanel) {
245
- const event = {
246
- itemData: itemData,
247
- panelConfig: config,
248
- resolvedResult: resolvedResult,
249
- fullPanelsConfig: fullConfig,
250
- };
251
- itemData = this.hooksConfig.beforeOpenPanel(this.injector, event);
252
- }
253
- this.openPanel(config.component, itemData);
254
- }
255
- this.dialog.openDialogs[this.dialog.openDialogs.length - 1].afterOpened().subscribe(() => {
256
- subject.next(null);
257
- });
258
- });
259
- return subject;
260
- }
261
- getResolvedData(config) {
262
- if (!config.resolve || (config.resolve && Object.keys(config.resolve).length === 0)) {
263
- return of({});
264
- }
265
- const resolveKeys = Object.keys(config.resolve);
266
- const resolvedData = {};
267
- const injector = config.injector;
268
- if (injector) {
269
- resolveKeys.forEach(key => {
270
- resolvedData[key] = runInInjectionContext(injector, () => config.resolve[key](config));
271
- });
272
- }
273
- return forkJoin(resolvedData);
274
- }
275
- openPanel(componentOrTemplateRef, data) {
276
- const conf = {
277
- panelClass: this.panelClass,
278
- closeOnNavigation: false,
279
- hasBackdrop: this.dialog.openDialogs.length === 0,
280
- height: '100%',
281
- width: this.panelWidth,
282
- position: {
283
- top: '0',
284
- right: '0',
285
- },
286
- };
287
- const dialogRef = this.dialog.open(componentOrTemplateRef, conf);
288
- // Panelisable interface attributes/functions
289
- dialogRef.componentInstance.initPanel(data);
290
- dialogRef.beforeClosed().subscribe(() => {
291
- const index = this.getPanelIndex(dialogRef.componentInstance);
292
- this.goToPanelByIndex(index - 1);
293
- });
294
- }
295
- /**
296
- * Return panel position (index) by searching matching component
297
- */
298
- getPanelIndex(component) {
299
- if (!component) {
300
- return -1;
301
- }
302
- return this.dialog.openDialogs.findIndex(dialog => dialog.componentInstance === component);
303
- }
304
- /**
305
- * Whether the given panel is currently the top, visible, panel. If there are no panels opened at all, then any panel given is considered top, visible, panel.
306
- */
307
- isTopPanel(component) {
308
- const length = this.dialog.openDialogs.length;
309
- return !length || this.dialog.openDialogs[length - 1]?.componentInstance === component;
310
- }
311
- /**
312
- * Repositions panels from start until given index
313
- */
314
- updateComponentsPosition() {
315
- if (!this.dialog.openDialogs.length) {
316
- return;
317
- }
318
- // Select the panels that are still opened, ignore the others because they'll be closed
319
- const affectedElements = this.dialog.openDialogs;
320
- for (let i = 0; i < affectedElements.length; i++) {
321
- const dialog = affectedElements[i];
322
- // Set all panels as "hidden" except last one. IsFrontPanel attribute causes a CSS that hides body via hostbinding
323
- dialog.componentInstance.isFrontPanel = i === affectedElements.length - 1;
324
- dialog.componentInstance.panelService = this;
325
- // Assign offset
326
- const deep = affectedElements.length - 1 - i;
327
- let position = { right: deep * this.panelsOffsetH + 'px' };
328
- if (this.isVertical && affectedElements.length > 1) {
329
- const top = i * this.panelsOffsetV + 'px';
330
- position = { top: top, right: '0px' };
331
- dialog.updateSize(this.panelWidth, `calc(100% - ${top})`); // call before updatePosition
332
- }
333
- else {
334
- dialog.updateSize(this.panelWidth, `100%`);
335
- }
336
- dialog.updatePosition(position);
337
- }
338
- }
339
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPanelsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
340
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPanelsService, providedIn: 'root' });
341
- }
342
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalPanelsService, decorators: [{
343
- type: Injectable,
344
- args: [{
345
- providedIn: 'root',
346
- }]
347
- }], ctorParameters: () => [] });
348
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"panels.service.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/panels/panels.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {BreakpointObserver, Breakpoints} from '@angular/cdk/layout';\nimport {ComponentType} from '@angular/cdk/portal';\nimport {inject, Injectable, Injector, runInInjectionContext} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {MatDialog, MatDialogConfig} from '@angular/material/dialog';\nimport {ActivatedRoute, DefaultUrlSerializer, NavigationError, Router, UrlSegment} from '@angular/router';\nimport {differenceWith, flatten, isEqual} from 'lodash-es';\nimport {forkJoin, Observable, of, Subject, Subscription} from 'rxjs';\nimport {NaturalAbstractPanel} from './abstract-panel';\nimport {getStackConfig} from './panels.urlmatcher';\nimport {\n    NaturalPanelConfig,\n    NaturalPanelData,\n    NaturalPanelsBeforeOpenPanel,\n    NaturalPanelsRouterRule,\n    PanelsHooksConfig,\n} from './types';\n\nfunction segmentsToString(segments: UrlSegment[]): string {\n    return segments.map(s => s.toString()).join('/');\n}\n\nfunction compareConfigs(a: NaturalPanelConfig, b: NaturalPanelConfig): boolean {\n    return a.route.path === b.route.path && a.rule === b.rule && isEqual(a.params, b.params);\n}\n\n/**\n * TODO: implement route update when closing dialog with escape\n * @dynamic\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalPanelsService {\n    private readonly router = inject(Router);\n    private readonly dialog = inject(MatDialog);\n    private readonly injector = inject(Injector);\n    private hooksConfig = inject(PanelsHooksConfig);\n\n    private readonly panelWidth = '960px';\n\n    /**\n     * Because of this static property Panels are **not** compatible with SSR.\n     * And we cannot make it non-static, because `UrlMatcher` cannot be injected.\n     */\n    private static _opened = false;\n    public static get opened(): boolean {\n        return this._opened;\n    }\n\n    /**\n     * Stream that emits when all open dialog have finished closing\n     */\n    public readonly afterAllClosed = new Subject<void>();\n    /**\n     * Cache for panels counter. Works more like an ID.\n     * Is used to give an unique identifier to multiple similar panels configurations\n     */\n    private counter = 1;\n    /**\n     * Class applied to dialog overlay related with panels\n     * If change, change CSS too\n     */\n    private panelClass = 'panel';\n\n    /**\n     * Cache for panels setup before navigation change.\n     * Used to detect panels openings/closings and adapt for new configuration\n     */\n    private oldFullConfig: NaturalPanelConfig[] = [];\n\n    /**\n     * Cache for subscription stop\n     */\n    private routeSub?: Subscription;\n\n    /**\n     * Cache for subscription stop\n     */\n    private navSub?: Subscription;\n\n    /**\n     * Horizontal gaps between panels\n     */\n    private panelsOffsetH = 35;\n\n    /**\n     * Vertical gaps between panels\n     */\n    private panelsOffsetV = 40;\n\n    /**\n     * Cache of previous screen size\n     * Used to change panels stack orientation on small screens\n     */\n    private isVertical = false;\n\n    public constructor() {\n        const breakpointObserver = inject(BreakpointObserver);\n\n        //  Watch media to know if display panels horizontally or vertically\n        breakpointObserver\n            .observe(Breakpoints.XSmall)\n            .pipe(takeUntilDestroyed())\n            .subscribe(result => {\n                this.isVertical = result.matches;\n                this.updateComponentsPosition();\n            });\n    }\n\n    /**\n     * Notify the service to start listening to route changes to open panels\n     *\n     * @internal\n     */\n    public start(route: ActivatedRoute): void {\n        NaturalPanelsService._opened = true;\n        this.routeSub = route.url.subscribe(segments => {\n            this.updatePanels(segments, route.snapshot.data.panelsRoutes);\n        });\n\n        this.navSub = this.router.events.subscribe(ev => {\n            if (!(ev instanceof NavigationError)) {\n                return;\n            }\n\n            // Abort and propagate navigation error\n            if (ev.url.endsWith('/')) {\n                this.stop();\n\n                // This is a bit hackish: we hardcode an impossible url to trigger normal navigation error mechanism\n                this.router.navigateByUrl('panels-reached-invalid-url');\n\n                return;\n            }\n\n            this.counter++;\n\n            // pc stands for \"panel counter\", required to give an identification to panels with exact same config\n            // E.g /new\n            const wantedUrl = ev.url + ';pc=' + this.counter;\n\n            // Segments matching from wanted url. E.g ~ ['new']\n            const wantedUrSegments = new DefaultUrlSerializer().parse(wantedUrl).root.children.primary.segments;\n\n            // Don't match any config\n            const wantedConfig = getStackConfig(wantedUrSegments, route.snapshot.data.panelsRoutes, this.injector);\n\n            if (wantedConfig.length) {\n                return this.appendConfigToCurrentUrl(wantedConfig);\n            }\n\n            // Currently activated routes. E.g ['objective', 'objective', 123, 'risk']\n            const currentSegments = flatten(\n                this.dialog.openDialogs.map(d => d.componentInstance.panelData.config.route.segments),\n            );\n\n            // Last segment. E.g ['risk']\n            const lastOfCurrentSegments = currentSegments.slice(-1);\n\n            // Config for ['risk', 'new']\n            const currentAndWantedConfig = getStackConfig(\n                lastOfCurrentSegments.concat(wantedUrSegments),\n                route.snapshot.data.panelsRoutes,\n                this.injector,\n            );\n\n            return this.appendConfigToCurrentUrl(currentAndWantedConfig);\n        });\n    }\n\n    /**\n     * Uses given configuration to add in the end of current url\n     * Neutralizes router error handling\n     */\n    private appendConfigToCurrentUrl(config: NaturalPanelConfig[]): void {\n        const originalErrorHandler = this.router.errorHandler;\n\n        // Nullify error handler (will be de-neutralized after route redirection)\n        if (config) {\n            this.router.errorHandler = () => undefined;\n        }\n\n        // Navigate to same url + /risk/new Result : /risk/risk/new\n        const newUrl = config.map(conf => segmentsToString(conf.route.segments)).join('/');\n\n        this.router.navigateByUrl(this.router.url + '/' + newUrl).then(() => {\n            // After navigation has ended, restore original error handler because he's not a bad guy\n            this.router.errorHandler = originalErrorHandler;\n        });\n    }\n\n    /**\n     * Notify the service that all panels were closed\n     *\n     * @internal\n     */\n    public stop(): void {\n        NaturalPanelsService._opened = false;\n        this.routeSub?.unsubscribe();\n        this.navSub?.unsubscribe();\n        this.dialog.closeAll();\n        this.oldFullConfig = [];\n        this.afterAllClosed.next();\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     *\n     * @internal\n     */\n    public goToPanelByComponent(component: NaturalAbstractPanel): void {\n        this.goToPanelByIndex(this.getPanelIndex(component));\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     */\n    public goToPenultimatePanel(): void {\n        this.goToPanelByIndex(this.dialog.openDialogs.length - 2);\n    }\n\n    /**\n     * Calls the new url that only includes the segments from the panels we want to stay open\n     */\n    private goToPanelByIndex(index: number): void {\n        // Extracts url segments from next panel until last one\n        const url = this.dialog.openDialogs\n            .slice(index + 1)\n            .map(dialog => {\n                return segmentsToString(dialog.componentInstance.panelData.config.route.segments);\n            })\n            .join('/');\n\n        // Remove extra segments and redirects to root\n        this.router.navigateByUrl(this.router.url.replace('/' + url, ''));\n    }\n\n    /**\n     * Selecting a panel is equivalent to close all those that are in front of him\n     * @param index of panel in stack. The most behind (the first one) is 0.\n     */\n    private selectPanelByIndex(index: number): Observable<void> {\n        const lastDialog = this.dialog.openDialogs[this.dialog.openDialogs.length - 1];\n\n        // Update new panels set positions\n        this.updateComponentsPosition();\n\n        for (let i = this.dialog.openDialogs.length - 1; i >= index + 1; i--) {\n            this.dialog.openDialogs[i].close();\n        }\n\n        return lastDialog.afterClosed();\n    }\n\n    /**\n     * Open new panels if url has changed with new segments\n     */\n    private updatePanels(segments: UrlSegment[], routes: NaturalPanelsRouterRule[]): void {\n        // Transform url segments into a config with component name and ID if provided in next segment\n        // Returns an array of configs, each config represents the content relative to a panel\n        const newFullConfig = getStackConfig(segments, routes, this.injector);\n        const configsToRemove = differenceWith(this.oldFullConfig, newFullConfig, compareConfigs);\n        const configsToAdd = differenceWith(newFullConfig, this.oldFullConfig, compareConfigs);\n\n        const indexOfNextPanel = this.oldFullConfig.length - configsToRemove.length - 1;\n\n        if (configsToRemove.length && configsToAdd.length) {\n            // Add and remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => {\n                this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n            });\n        } else if (configsToRemove.length && !configsToAdd.length) {\n            // only remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => this.updateComponentsPosition());\n        } else if (!configsToRemove.length && configsToAdd.length) {\n            // only add panels\n            this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n        }\n\n        this.oldFullConfig = newFullConfig;\n    }\n\n    /**\n     * Resolve all services, then open panels\n     */\n    private openPanels(\n        newItemsConfig: NaturalPanelConfig[],\n        fullConfig: NaturalPanelConfig[],\n    ): Observable<Observable<any> | null> {\n        const subject = new Subject<Observable<any> | null>();\n\n        // Resolve everything before opening a single panel\n        const resolves = newItemsConfig.map(conf => this.getResolvedData(conf));\n\n        // ForkJoin emits when all promises are executed;\n        forkJoin(resolves).subscribe(resolvedResult => {\n            // For each new config entry, open a new panel\n            for (let i = 0; i < newItemsConfig.length; i++) {\n                const config = newItemsConfig[i];\n                let itemData: NaturalPanelData = {\n                    // Config of actual panel route\n                    config: config,\n\n                    // Data resolved by service\n                    // Use in component by calling this.panelData.data.xyz\n                    data: resolvedResult[i],\n                    linkableObjects: [],\n                };\n\n                if (this.hooksConfig?.beforeOpenPanel) {\n                    const event: NaturalPanelsBeforeOpenPanel = {\n                        itemData: itemData,\n                        panelConfig: config,\n                        resolvedResult: resolvedResult,\n                        fullPanelsConfig: fullConfig,\n                    };\n\n                    itemData = this.hooksConfig.beforeOpenPanel(this.injector, event);\n                }\n\n                this.openPanel(config.component, itemData);\n            }\n\n            this.dialog.openDialogs[this.dialog.openDialogs.length - 1].afterOpened().subscribe(() => {\n                subject.next(null);\n            });\n        });\n\n        return subject;\n    }\n\n    private getResolvedData(config: NaturalPanelConfig): Observable<Record<string, unknown>> {\n        if (!config.resolve || (config.resolve && Object.keys(config.resolve).length === 0)) {\n            return of({});\n        }\n\n        const resolveKeys = Object.keys(config.resolve);\n        const resolvedData: Record<string, Observable<unknown>> = {};\n        const injector = config.injector;\n\n        if (injector) {\n            resolveKeys.forEach(key => {\n                resolvedData[key] = runInInjectionContext(injector, () => config.resolve[key](config));\n            });\n        }\n\n        return forkJoin(resolvedData);\n    }\n\n    private openPanel(componentOrTemplateRef: ComponentType<NaturalAbstractPanel>, data: NaturalPanelData): void {\n        const conf: MatDialogConfig = {\n            panelClass: this.panelClass,\n            closeOnNavigation: false,\n            hasBackdrop: this.dialog.openDialogs.length === 0,\n            height: '100%',\n            width: this.panelWidth,\n            position: {\n                top: '0',\n                right: '0',\n            },\n        };\n\n        const dialogRef = this.dialog.open<NaturalAbstractPanel>(componentOrTemplateRef, conf);\n\n        // Panelisable interface attributes/functions\n        dialogRef.componentInstance.initPanel(data);\n\n        dialogRef.beforeClosed().subscribe(() => {\n            const index = this.getPanelIndex(dialogRef.componentInstance);\n            this.goToPanelByIndex(index - 1);\n        });\n    }\n\n    /**\n     * Return panel position (index) by searching matching component\n     */\n    private getPanelIndex(component: NaturalAbstractPanel): number {\n        if (!component) {\n            return -1;\n        }\n\n        return this.dialog.openDialogs.findIndex(dialog => dialog.componentInstance === component);\n    }\n\n    /**\n     * Whether the given panel is currently the top, visible, panel. If there are no panels opened at all, then any panel given is considered top, visible, panel.\n     */\n    public isTopPanel(component: NaturalAbstractPanel): boolean {\n        const length = this.dialog.openDialogs.length;\n        return !length || this.dialog.openDialogs[length - 1]?.componentInstance === component;\n    }\n\n    /**\n     * Repositions panels from start until given index\n     */\n    private updateComponentsPosition(): void {\n        if (!this.dialog.openDialogs.length) {\n            return;\n        }\n\n        // Select the panels that are still opened, ignore the others because they'll be closed\n        const affectedElements = this.dialog.openDialogs;\n        for (let i = 0; i < affectedElements.length; i++) {\n            const dialog = affectedElements[i];\n\n            // Set all panels as \"hidden\" except last one. IsFrontPanel attribute causes a CSS that hides body via hostbinding\n            dialog.componentInstance.isFrontPanel = i === affectedElements.length - 1;\n            dialog.componentInstance.panelService = this;\n\n            // Assign offset\n            const deep = affectedElements.length - 1 - i;\n\n            let position: any = {right: deep * this.panelsOffsetH + 'px'};\n            if (this.isVertical && affectedElements.length > 1) {\n                const top = i * this.panelsOffsetV + 'px';\n                position = {top: top, right: '0px'};\n                dialog.updateSize(this.panelWidth, `calc(100% - ${top})`); // call before updatePosition\n            } else {\n                dialog.updateSize(this.panelWidth, `100%`);\n            }\n\n            dialog.updatePosition(position);\n        }\n    }\n}\n"]}
@@ -1,75 +0,0 @@
1
- import { flatten, merge } from 'lodash-es';
2
- function getConsumedSegments(segments, routes) {
3
- return flatten(getStackConfig(segments, routes).map(conf => conf.route.segments));
4
- }
5
- /**
6
- * Return a list of items specifying the component, the service and the optional id of url segments
7
- */
8
- export function getStackConfig(segments, routes, injector = null) {
9
- if (!routes) {
10
- return [];
11
- }
12
- const comp = getComponentConfig(segments, routes, injector);
13
- if (comp) {
14
- return [comp].concat(getStackConfig(segments.slice(comp.route.segments.length), routes, injector));
15
- }
16
- return [];
17
- }
18
- /**
19
- * Returns an object with a component class, a service and an optional id from current and next url segments
20
- */
21
- function getComponentConfig(segments, routes, injector = null) {
22
- if (!segments.length) {
23
- return null;
24
- }
25
- // For each config
26
- for (const routeConfig of routes) {
27
- const params = {};
28
- const configSegments = routeConfig.path.split('/');
29
- let match = true;
30
- // For each current url segment
31
- for (let i = 0; i < segments.length; i++) {
32
- if (!configSegments[i]) {
33
- match = false;
34
- break;
35
- }
36
- // If find variable, store it
37
- if (configSegments[i].includes(':') && +segments[i].path > 0) {
38
- params[configSegments[i].replace(':', '')] = segments[i].path;
39
- }
40
- else if (configSegments[i] !== segments[i].path) {
41
- // If segments are different, url does not match
42
- match = false;
43
- break;
44
- }
45
- }
46
- if (match) {
47
- const matrixParams = segments.reduce((groupedParams, segment) => {
48
- return merge(groupedParams, segment.parameters);
49
- }, {});
50
- return {
51
- component: routeConfig.component,
52
- injector: injector,
53
- resolve: routeConfig.resolve ?? {},
54
- params: merge(params, matrixParams),
55
- rule: routeConfig,
56
- route: {
57
- segments: segments,
58
- path: segments.map(s => s.path).join('/'),
59
- },
60
- };
61
- }
62
- }
63
- if (segments.length > 1) {
64
- return getComponentConfig(segments.slice(0, -1), routes, injector);
65
- }
66
- return null;
67
- }
68
- export const naturalPanelsUrlMatcher = (segments, group, route) => {
69
- const matchedSegments = getConsumedSegments(segments, route.data ? route.data.panelsRoutes : []);
70
- if (matchedSegments.length) {
71
- return { consumed: matchedSegments };
72
- }
73
- return null;
74
- };
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"panels.urlmatcher.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/panels/panels.urlmatcher.ts"],"names":[],"mappings":"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","sourcesContent":["import {Injector} from '@angular/core';\nimport {Route, UrlMatcher, UrlMatchResult, UrlSegment, UrlSegmentGroup} from '@angular/router';\nimport {flatten, merge} from 'lodash-es';\nimport {NaturalPanelConfig, NaturalPanelsRouterRule} from './types';\nimport {Literal} from '../../types/types';\n\nfunction getConsumedSegments(segments: UrlSegment[], routes: NaturalPanelsRouterRule[]): UrlSegment[] {\n    return flatten(getStackConfig(segments, routes).map(conf => conf.route.segments));\n}\n\n/**\n * Return a list of items specifying the component, the service and the optional id of url segments\n */\nexport function getStackConfig(\n    segments: UrlSegment[],\n    routes: NaturalPanelsRouterRule[],\n    injector: Injector | null = null,\n): NaturalPanelConfig[] {\n    if (!routes) {\n        return [];\n    }\n\n    const comp = getComponentConfig(segments, routes, injector);\n\n    if (comp) {\n        return [comp].concat(getStackConfig(segments.slice(comp.route.segments.length), routes, injector));\n    }\n\n    return [];\n}\n\n/**\n * Returns an object with a component class, a service and an optional id from current and next url segments\n */\nfunction getComponentConfig(\n    segments: UrlSegment[],\n    routes: NaturalPanelsRouterRule[],\n    injector: Injector | null = null,\n): NaturalPanelConfig | null {\n    if (!segments.length) {\n        return null;\n    }\n\n    // For each config\n    for (const routeConfig of routes) {\n        const params: Literal = {};\n        const configSegments = routeConfig.path.split('/');\n        let match = true;\n\n        // For each current url segment\n        for (let i = 0; i < segments.length; i++) {\n            if (!configSegments[i]) {\n                match = false;\n                break;\n            }\n\n            // If find variable, store it\n            if (configSegments[i].includes(':') && +segments[i].path > 0) {\n                params[configSegments[i].replace(':', '')] = segments[i].path;\n            } else if (configSegments[i] !== segments[i].path) {\n                // If segments are different, url does not match\n                match = false;\n                break;\n            }\n        }\n\n        if (match) {\n            const matrixParams = segments.reduce((groupedParams, segment) => {\n                return merge(groupedParams, segment.parameters);\n            }, {});\n\n            return {\n                component: routeConfig.component,\n                injector: injector,\n                resolve: routeConfig.resolve ?? {},\n                params: merge(params, matrixParams),\n                rule: routeConfig,\n                route: {\n                    segments: segments,\n                    path: segments.map(s => s.path).join('/'),\n                },\n            };\n        }\n    }\n\n    if (segments.length > 1) {\n        return getComponentConfig(segments.slice(0, -1), routes, injector);\n    }\n\n    return null;\n}\n\nexport const naturalPanelsUrlMatcher: UrlMatcher = (\n    segments: UrlSegment[],\n    group: UrlSegmentGroup,\n    route: Route,\n): UrlMatchResult | null => {\n    const matchedSegments = getConsumedSegments(segments, route.data ? route.data.panelsRoutes : []);\n\n    if (matchedSegments.length) {\n        return {consumed: matchedSegments};\n    }\n\n    return null;\n};\n"]}
@@ -1,11 +0,0 @@
1
- /*
2
- * Public API Surface of natural
3
- */
4
- export { providePanels } from './panels.module';
5
- export * from './panels.component';
6
- export * from './panels.service';
7
- export { naturalPanelsUrlMatcher } from './panels.urlmatcher';
8
- export { fallbackIfNoOpenedPanels } from './fallback-if-no-opened-panels.urlmatcher';
9
- export * from './abstract-panel';
10
- export * from './types';
11
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL3BhbmVscy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLGlCQUFpQixDQUFDO0FBQzlDLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxPQUFPLEVBQUMsdUJBQXVCLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUM1RCxPQUFPLEVBQUMsd0JBQXdCLEVBQUMsTUFBTSwyQ0FBMkMsQ0FBQztBQUNuRixjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBuYXR1cmFsXG4gKi9cblxuZXhwb3J0IHtwcm92aWRlUGFuZWxzfSBmcm9tICcuL3BhbmVscy5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9wYW5lbHMuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcGFuZWxzLnNlcnZpY2UnO1xuZXhwb3J0IHtuYXR1cmFsUGFuZWxzVXJsTWF0Y2hlcn0gZnJvbSAnLi9wYW5lbHMudXJsbWF0Y2hlcic7XG5leHBvcnQge2ZhbGxiYWNrSWZOb09wZW5lZFBhbmVsc30gZnJvbSAnLi9mYWxsYmFjay1pZi1uby1vcGVuZWQtcGFuZWxzLnVybG1hdGNoZXInO1xuZXhwb3J0ICogZnJvbSAnLi9hYnN0cmFjdC1wYW5lbCc7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
@@ -1,3 +0,0 @@
1
- import { InjectionToken } from '@angular/core';
2
- export const PanelsHooksConfig = new InjectionToken('NaturalPanelsHooksConfig');
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9wYW5lbHMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFDLGNBQWMsRUFBVyxNQUFNLGVBQWUsQ0FBQztBQXVFdkQsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsSUFBSSxjQUFjLENBQTJCLDBCQUEwQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0NvbXBvbmVudFR5cGV9IGZyb20gJ0Bhbmd1bGFyL2Nkay9wb3J0YWwnO1xuaW1wb3J0IHtJbmplY3Rpb25Ub2tlbiwgSW5qZWN0b3J9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtVcmxTZWdtZW50fSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHtMaXRlcmFsfSBmcm9tICcuLi8uLi90eXBlcy90eXBlcyc7XG5pbXBvcnQge09ic2VydmFibGV9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtOYXR1cmFsQWJzdHJhY3RQYW5lbH0gZnJvbSAnLi9hYnN0cmFjdC1wYW5lbCc7XG5pbXBvcnQge0xpbmthYmxlT2JqZWN0fSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9saW5rLW11dGF0aW9uLnNlcnZpY2UnO1xuXG4vKipcbiAqIEtpbmQgb2Ygc25hcHNob3Qgb2YgdGhlIGluc3RhbmNlIG9mIGEgcGFuZWwgYWN0aXZhdGVkIHJvdXRlXG4gKi9cbmV4cG9ydCB0eXBlIE5hdHVyYWxQYW5lbHNSb3V0ZUNvbmZpZyA9IHtcbiAgICBzZWdtZW50czogVXJsU2VnbWVudFtdO1xuICAgIHBhdGg6IHN0cmluZztcbn07XG5cbi8qKlxuICogQ29uZmlnIHJlcXVpcmVkIHRvIG1hbmFnZSB1cmwgYW5kIGluc3RhbnRpYXRlIGNvbXBvbmVudCBjb3JyZWN0bHlcbiAqL1xuZXhwb3J0IHR5cGUgTmF0dXJhbFBhbmVsQ29uZmlnID0ge1xuICAgIGNvbXBvbmVudDogQ29tcG9uZW50VHlwZTxOYXR1cmFsQWJzdHJhY3RQYW5lbD47XG4gICAgaW5qZWN0b3I6IEluamVjdG9yIHwgbnVsbDtcbiAgICByZXNvbHZlOiBOYXR1cmFsUGFuZWxSZXNvbHZlcztcbiAgICBwYXJhbXM6IExpdGVyYWw7XG4gICAgcnVsZTogTmF0dXJhbFBhbmVsc1JvdXRlclJ1bGU7XG4gICAgcm91dGU6IE5hdHVyYWxQYW5lbHNSb3V0ZUNvbmZpZztcbn07XG5cbi8qKlxuICogRGF0YSBwcm92aWRlZCB0byBpbnN0YW50aWF0ZWQgY29tcG9uZW50cyBpbiBjb250ZXh0IG9mIGEgcGFuZWwvZGlhbG9nXG4gKi9cbmV4cG9ydCB0eXBlIE5hdHVyYWxQYW5lbERhdGEgPSB7XG4gICAgY29uZmlnOiBOYXR1cmFsUGFuZWxDb25maWc7XG4gICAgZGF0YTogTGl0ZXJhbDtcblxuICAgIC8qKlxuICAgICAqIFJlbGF0ZWQgb2JqZWN0cyB0aGF0IHNob3VsZCBiZSBsaW5rZWQgdG8gdGhlIG9iamVjdCBzaG93biBpbiB0aGUgcGFuZWwgYWZ0ZXIgaXRzIGNyZWF0aW9uXG4gICAgICovXG4gICAgbGlua2FibGVPYmplY3RzOiBMaW5rYWJsZU9iamVjdFtdO1xufTtcblxuLyoqXG4gKiBTaW1pbGFyIHRvIEFuZ3VsYXIgZnVuY3Rpb25hbCByZXNvbHZlciBpbnRlcmZhY2UsIGJ1dCBzaW1wbGVyIGZvciBvdXIgcGFuZWxzJyBuZWVkc1xuICovXG50eXBlIE5hdHVyYWxQYW5lbFJlc29sdmU8VD4gPSAocm91dGU6IE5hdHVyYWxQYW5lbENvbmZpZykgPT4gT2JzZXJ2YWJsZTxUPjtcbmV4cG9ydCB0eXBlIE5hdHVyYWxQYW5lbFJlc29sdmVzID0gUmVjb3JkPHN0cmluZywgTmF0dXJhbFBhbmVsUmVzb2x2ZTx1bmtub3duPj47XG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIGEgcm91dGVcbiAqL1xuZXhwb3J0IHR5cGUgTmF0dXJhbFBhbmVsc1JvdXRlclJ1bGUgPSB7XG4gICAgcGF0aDogc3RyaW5nO1xuICAgIGNvbXBvbmVudDogQ29tcG9uZW50VHlwZTxOYXR1cmFsQWJzdHJhY3RQYW5lbD47XG4gICAgcmVzb2x2ZT86IE5hdHVyYWxQYW5lbFJlc29sdmVzO1xufTtcblxuZXhwb3J0IHR5cGUgTmF0dXJhbFBhbmVsc0JlZm9yZU9wZW5QYW5lbCA9IHtcbiAgICBpdGVtRGF0YTogTmF0dXJhbFBhbmVsRGF0YTtcbiAgICBwYW5lbENvbmZpZzogTmF0dXJhbFBhbmVsQ29uZmlnO1xuICAgIGZ1bGxQYW5lbHNDb25maWc6IE5hdHVyYWxQYW5lbENvbmZpZ1tdO1xuICAgIHJlc29sdmVkUmVzdWx0OiBhbnk7IC8vIHRvZG8gOiBHZW5lcmljIG9yIE5hdHVyYWxBYnN0cmFjdE1vZGVsU2VydmljZVxufTtcblxuZXhwb3J0IHR5cGUgTmF0dXJhbFBhbmVsc0hvb2tzQ29uZmlnID0ge1xuICAgIGJlZm9yZU9wZW5QYW5lbD86IChcbiAgICAgICAgaW5qZWN0b3I6IEluamVjdG9yLFxuICAgICAgICBuYXR1cmFsUGFuZWxzQmVmb3JlT3BlblBhbmVsOiBOYXR1cmFsUGFuZWxzQmVmb3JlT3BlblBhbmVsLFxuICAgICkgPT4gTmF0dXJhbFBhbmVsRGF0YTtcbn07XG5cbi8vIEFycmF5IG9mIE5hdHVyYWxQYW5lbHNSb3V0ZXJSdWxlXG5leHBvcnQgdHlwZSBOYXR1cmFsUGFuZWxzUm91dGVzQ29uZmlnID0gTmF0dXJhbFBhbmVsc1JvdXRlclJ1bGVbXTtcblxuZXhwb3J0IGNvbnN0IFBhbmVsc0hvb2tzQ29uZmlnID0gbmV3IEluamVjdGlvblRva2VuPE5hdHVyYWxQYW5lbHNIb29rc0NvbmZpZz4oJ05hdHVyYWxQYW5lbHNIb29rc0NvbmZpZycpO1xuIl19
@@ -1,5 +0,0 @@
1
- /*
2
- * Public API Surface of natural
3
- */
4
- export * from './relations.component';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL3JlbGF0aW9ucy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyx1QkFBdUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgbmF0dXJhbFxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcmVsYXRpb25zLmNvbXBvbmVudCc7XG4iXX0=