@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,398 +0,0 @@
1
- import { SelectionModel } from '@angular/cdk/collections';
2
- import { FlatTreeControl } from '@angular/cdk/tree';
3
- import { Component, DestroyRef, EventEmitter, inject, Input, Output, } from '@angular/core';
4
- import { MatTreeFlatDataSource, MatTreeFlattener, MatTreeModule } from '@angular/material/tree';
5
- import { finalize } from 'rxjs/operators';
6
- import { toGraphQLDoctrineFilter } from '../../search/classes/graphql-doctrine';
7
- import { HierarchicFlatNode } from '../classes/flat-node';
8
- import { NaturalHierarchicSelectorService } from './hierarchic-selector.service';
9
- import { replaceObjectKeepingReference } from '../../../classes/utility';
10
- import { MatChipsModule } from '@angular/material/chips';
11
- import { MatCheckboxModule } from '@angular/material/checkbox';
12
- import { NaturalIconDirective } from '../../icon/icon.directive';
13
- import { MatIconModule } from '@angular/material/icon';
14
- import { MatButtonModule } from '@angular/material/button';
15
- import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
16
- import { CommonModule } from '@angular/common';
17
- import { NaturalSearchComponent } from '../../search/search/search.component';
18
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
19
- import * as i0 from "@angular/core";
20
- import * as i1 from "@angular/common";
21
- import * as i2 from "@angular/material/progress-spinner";
22
- import * as i3 from "@angular/material/tree";
23
- import * as i4 from "@angular/material/button";
24
- import * as i5 from "@angular/material/icon";
25
- import * as i6 from "@angular/material/checkbox";
26
- import * as i7 from "@angular/material/chips";
27
- export class NaturalHierarchicSelectorComponent {
28
- destroyRef = inject(DestroyRef);
29
- hierarchicSelectorService = inject(NaturalHierarchicSelectorService);
30
- /**
31
- * Function that receives a model and returns a string for display value
32
- */
33
- displayWith;
34
- /**
35
- * Config for items and relations arrangement
36
- */
37
- config;
38
- /**
39
- * If multiple or single item selection
40
- */
41
- multiple = false;
42
- /**
43
- * Selected items
44
- * Organized by key, containing each an array of selected items of same type
45
- */
46
- selected = {};
47
- /**
48
- * Whether selectable elements can be unselected
49
- */
50
- allowUnselect = true;
51
- /**
52
- * Filters that apply to each query
53
- */
54
- filters;
55
- /**
56
- * Search facets
57
- */
58
- searchFacets = [];
59
- /**
60
- * Selections to apply on natural-search on component initialisation
61
- */
62
- searchSelections = [];
63
- /**
64
- * Emits when natural-search selections change
65
- */
66
- searchSelectionChange = new EventEmitter();
67
- /**
68
- * Inner representation of selected @Input() to allow flat listing as mat-chip.
69
- */
70
- selectedNodes = [];
71
- /**
72
- * Emits selection change
73
- * Returns a Literal where selected models are organized by key
74
- */
75
- selectionChange = new EventEmitter();
76
- /**
77
- * Controller for nodes selection
78
- */
79
- flatNodesSelection;
80
- treeControl;
81
- treeFlattener;
82
- dataSource;
83
- loading = false;
84
- /**
85
- * Cache for transformed nodes
86
- */
87
- flatNodeMap = new Map();
88
- /**
89
- * Angular OnChange implementation
90
- */
91
- ngOnChanges(changes) {
92
- if (changes.selected && !changes.selected.firstChange) {
93
- this.updateInnerSelection(this.selected);
94
- }
95
- if (changes.filters && !changes.filters.firstChange) {
96
- this.loadRoots();
97
- }
98
- }
99
- /**
100
- * Angular OnInit implementation
101
- */
102
- ngOnInit() {
103
- // Init tree checkbox selectors
104
- this.flatNodesSelection = new SelectionModel(this.multiple);
105
- // Tree controllers and manipulators
106
- this.treeFlattener = new MatTreeFlattener(this.transformer(), this.getLevel(), this.isExpandable(), this.getChildren());
107
- this.treeControl = new FlatTreeControl(this.getLevel(), this.isExpandable());
108
- // The dataSource contains a nested ModelNodes list. Each ModelNode has a child attribute that returns an observable.
109
- // The dataSource contains a flat representation of the nested ModelNodes that is generated by the treeFlattener related functions
110
- this.dataSource = new MatTreeFlatDataSource(this.treeControl, this.treeFlattener);
111
- // Update dataSource when receiving new list -> we assign the whole tree
112
- // The treeControl and treeFlattener will generate the displayed tree
113
- this.hierarchicSelectorService.dataChange
114
- .pipe(takeUntilDestroyed(this.destroyRef))
115
- .subscribe(data => (this.dataSource.data = data));
116
- // Prevent empty screen on first load and init NaturalHierarchicSelectorService with inputted configuration
117
- let variables;
118
- if (this.searchSelections.some(s => s.length)) {
119
- variables = { filter: toGraphQLDoctrineFilter(this.searchFacets, this.searchSelections) };
120
- }
121
- this.loadRoots(variables);
122
- // OrganizedSelection into list usable by template
123
- this.updateInnerSelection(this.selected);
124
- }
125
- /**
126
- * Toggle selection of a FlatNode, considering if multiple selection is activated or not
127
- */
128
- toggleFlatNode(flatNode) {
129
- if (this.multiple) {
130
- // Is multiple allowed, just toggle element
131
- if (this.flatNodesSelection.isSelected(flatNode)) {
132
- this.unselectFlatNode(flatNode);
133
- }
134
- else {
135
- this.selectFlatNode(flatNode);
136
- }
137
- }
138
- else if (!this.multiple) {
139
- if (this.flatNodesSelection.isSelected(flatNode)) {
140
- this.unselectSingleFlatNode();
141
- }
142
- else {
143
- // If not multiple, and we want to select an element, unselect everything before to keep a single selection
144
- this.selectSingleFlatNode(flatNode);
145
- }
146
- }
147
- }
148
- /**
149
- * When unselecting an element from the mat-chips, it can be deep in the hierarchy, and the tree element may not exist...
150
- * ... but we still need to remove the element from the mat-chips list.
151
- */
152
- unselectModelNode(node) {
153
- const flatNode = this.getFlatNode(node);
154
- if (flatNode) {
155
- this.unselectFlatNode(flatNode);
156
- }
157
- else {
158
- // Remove from chips list only if no flatNode, because unselectFlatNode() already deals with it.
159
- this.removeModelNode(node);
160
- this.updateSelection(this.selectedNodes);
161
- }
162
- }
163
- isNodeTogglable(flatNode) {
164
- if (this.isNodeSelected(flatNode.node)) {
165
- return flatNode.deselectable;
166
- }
167
- else {
168
- return flatNode.selectable;
169
- }
170
- }
171
- getDisplayFn(config) {
172
- if (config.displayWith) {
173
- return config.displayWith;
174
- }
175
- if (this.displayWith) {
176
- return this.displayWith;
177
- }
178
- return item => (item ? item.fullName || item.name : '');
179
- }
180
- loadChildren(flatNode) {
181
- if (this.treeControl.isExpanded(flatNode)) {
182
- this.hierarchicSelectorService.loadChildren(flatNode, this.filters);
183
- }
184
- }
185
- /**
186
- * Created to collapse all children when closing a parent, but not sure it's good.
187
- */
188
- // public loadChildren(flatNode: HierarchicFlatNode) {
189
- // if (this.treeControl.isExpanded(flatNode)) {
190
- //
191
- // const cachedFlatNode = this.getFlatNode(flatNode.node);
192
- // if (cachedFlatNode) {
193
- // this.hierarchicSelectorService.loadChildren(cachedFlatNode, this.filters);
194
- //
195
- // // Close children
196
- // cachedFlatNode.node.children.forEach(child => {
197
- // const childNode = this.getFlatNode(child);
198
- // if (childNode) {
199
- // this.treeControl.collapse(childNode);
200
- // }
201
- // });
202
- // }
203
- // }
204
- // }
205
- getChildren() {
206
- return (node) => {
207
- return node.childrenChange;
208
- };
209
- }
210
- /**
211
- * Transforms a HierarchicModelNode into a FlatNode
212
- */
213
- transformer() {
214
- return (node, level) => {
215
- return this.getOrCreateFlatNode(node, level);
216
- };
217
- }
218
- /**
219
- * Return deep of the node in the tree
220
- */
221
- getLevel() {
222
- return (node) => {
223
- return node.level;
224
- };
225
- }
226
- /**
227
- * Is always expandable because we load on demand, we don't know if there are children yet
228
- */
229
- isExpandable() {
230
- return (node) => {
231
- return node.expandable;
232
- };
233
- }
234
- getOrCreateFlatNode(node, level) {
235
- // Return FlatNode if exists
236
- const flatNode = this.getFlatNode(node);
237
- if (flatNode) {
238
- return flatNode;
239
- }
240
- // Return new FlatNode
241
- return this.createFlatNode(node, level);
242
- }
243
- search(selections) {
244
- this.searchSelectionChange.emit(selections);
245
- if (selections.some(s => s.length)) {
246
- const variables = { filter: toGraphQLDoctrineFilter(this.searchFacets, selections) };
247
- this.hierarchicSelectorService.search(variables, this.filters);
248
- }
249
- else {
250
- this.loadRoots();
251
- }
252
- }
253
- loadRoots(searchVariables) {
254
- this.loading = true;
255
- this.flatNodeMap = new Map();
256
- this.hierarchicSelectorService
257
- .init(this.config, this.filters, searchVariables || null)
258
- .pipe(finalize(() => (this.loading = false)))
259
- .subscribe();
260
- }
261
- /**
262
- * Sync inner selection (tree and mat-chips) according to selected input attribute
263
- */
264
- updateInnerSelection(selected) {
265
- // Transform an OrganizedModelSelection into a ModelNode list that is used in the selected zone of the component (see template)
266
- this.selectedNodes = this.hierarchicSelectorService.fromOrganizedSelection(selected);
267
- this.flatNodesSelection.clear();
268
- for (const node of this.selectedNodes) {
269
- const flatNode = this.getFlatNode(node);
270
- if (flatNode) {
271
- this.flatNodesSelection.select(flatNode);
272
- }
273
- }
274
- }
275
- /**
276
- * Unselect a node, keeping the rest of the selected untouched
277
- */
278
- unselectFlatNode(flatNode) {
279
- this.flatNodesSelection.deselect(flatNode);
280
- this.removeModelNode(flatNode.node);
281
- this.updateSelection(this.selectedNodes);
282
- }
283
- /**
284
- * Remove a node from chip lists
285
- */
286
- removeModelNode(node) {
287
- const key = this.getMapKey(node.model);
288
- const selectionIndex = this.selectedNodes.findIndex(n => this.getMapKey(n.model) === key);
289
- this.selectedNodes.splice(selectionIndex, 1);
290
- }
291
- /**
292
- * Select a node, keeping th rest of the selected untouched
293
- */
294
- selectFlatNode(flatNode) {
295
- this.flatNodesSelection.select(flatNode);
296
- this.selectedNodes.push(flatNode.node);
297
- this.updateSelection(this.selectedNodes);
298
- }
299
- /**
300
- * Clear all selected and select the given node
301
- */
302
- selectSingleFlatNode(flatNode) {
303
- this.flatNodesSelection.clear();
304
- this.flatNodesSelection.select(flatNode);
305
- this.selectedNodes = [flatNode.node];
306
- this.updateSelection(this.selectedNodes);
307
- }
308
- /**
309
- * Clear all selected and select the given node
310
- */
311
- unselectSingleFlatNode() {
312
- this.flatNodesSelection.clear();
313
- this.selectedNodes = [];
314
- this.updateSelection(this.selectedNodes);
315
- }
316
- /**
317
- * Transform the given elements into the organized selection that is emitted to output
318
- */
319
- updateSelection(selected) {
320
- const organizedFlatNodesSelection = this.hierarchicSelectorService.toOrganizedSelection(selected);
321
- replaceObjectKeepingReference(this.selected, organizedFlatNodesSelection);
322
- this.selectionChange.emit(organizedFlatNodesSelection);
323
- }
324
- isNodeSelected(node) {
325
- const key = this.getMapKey(node.model);
326
- return this.selectedNodes.some(n => this.getMapKey(n.model) === key);
327
- }
328
- getFlatNode(node) {
329
- const key = this.getMapKey(node.model);
330
- return this.flatNodeMap.get(key) || null;
331
- }
332
- createFlatNode(node, level) {
333
- const key = this.getMapKey(node.model);
334
- const name = this.getDisplayFn(node.config)(node.model);
335
- const expandable = false;
336
- const isCustomSelectable = node.config.isSelectableCallback
337
- ? node.config.isSelectableCallback(node.model)
338
- : true;
339
- const isSelectable = !!node.config.selectableAtKey && isCustomSelectable;
340
- const flatNode = new HierarchicFlatNode(node, name, level, expandable, isSelectable);
341
- this.hierarchicSelectorService.countItems(flatNode, this.filters);
342
- // Mark node as selected if needed (checks the selected processed input)
343
- if (this.isNodeSelected(node)) {
344
- if (!this.allowUnselect) {
345
- flatNode.deselectable = false;
346
- }
347
- this.flatNodesSelection.select(flatNode);
348
- }
349
- // Cache FlatNode
350
- this.flatNodeMap.set(key, flatNode);
351
- return flatNode;
352
- }
353
- /**
354
- * Returns an identifier key for map cache
355
- * As many object types can be used, this function considers typename and ID to return something like document-123
356
- */
357
- getMapKey(model) {
358
- return model.__typename + '-' + model.id;
359
- }
360
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalHierarchicSelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
361
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.3", type: NaturalHierarchicSelectorComponent, isStandalone: true, selector: "natural-hierarchic-selector", inputs: { displayWith: "displayWith", config: "config", multiple: "multiple", selected: "selected", allowUnselect: "allowUnselect", filters: "filters", searchFacets: "searchFacets", searchSelections: "searchSelections" }, outputs: { searchSelectionChange: "searchSelectionChange", selectionChange: "selectionChange" }, providers: [NaturalHierarchicSelectorService], usesOnChanges: true, ngImport: i0, template: "<div [style.margin-bottom.px]=\"20\">\n <natural-search (selectionChange)=\"search($event)\" [facets]=\"searchFacets\" [selections]=\"searchSelections\" />\n</div>\n\n<div class=\"body\">\n @if (loading) {\n <mat-progress-spinner [diameter]=\"36\" mode=\"indeterminate\" style=\"margin: 10px\" />\n }\n\n <mat-tree [dataSource]=\"dataSource\" [treeControl]=\"treeControl\">\n <mat-tree-node *matTreeNodeDef=\"let node\" [ngClass]=\"{leaf: !node.expandable}\" matTreeNodePadding>\n @if (node.expandable) {\n <button\n (click)=\"loadChildren(node)\"\n [attr.aria-label]=\"'toggle ' + node.name\"\n mat-icon-button\n matTreeNodeToggle\n >\n @if (node.loading) {\n <mat-progress-spinner [diameter]=\"24\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n }\n @if (!node.loading) {\n <mat-icon [naturalIcon]=\"treeControl.isExpanded(node) ? 'expand_more' : 'chevron_right'\" />\n }\n </button>\n }\n\n <mat-checkbox\n (change)=\"toggleFlatNode(node)\"\n [checked]=\"flatNodesSelection.isSelected(node)\"\n [disabled]=\"!isNodeTogglable(node)\"\n style=\"margin-right: 10px\"\n >\n @if (node.node.config.icon) {\n <mat-icon [naturalIcon]=\"node.node.config.icon\" style=\"margin-right: 10px\" />\n }\n <span>{{ node.name }}</span>\n </mat-checkbox>\n </mat-tree-node>\n </mat-tree>\n\n <mat-chip-listbox aria-orientation=\"vertical\" class=\"mat-mdc-chip-set-stacked\">\n @for (node of selectedNodes; track node.model.id) {\n <mat-chip-option (removed)=\"unselectModelNode(node)\" [removable]=\"true\" [selectable]=\"false\">\n @if (node.config.icon) {\n <mat-icon matChipAvatar [naturalIcon]=\"node.config.icon\" />\n }\n {{ node.model.name || node.model.fullName }}\n <button matChipRemove>\n <mat-icon naturalIcon=\"cancel\" />\n </button>\n </mat-chip-option>\n } @empty {\n <p class=\"mat-body nat-padding-horizontal\" i18n>Aucune s\u00E9lection</p>\n }\n </mat-chip-listbox>\n</div>\n\n@if (!loading && !dataSource.data.length) {\n <div i18n>Aucun r\u00E9sultat</div>\n}\n", styles: [":host{display:block}:host ul,:host li{-webkit-margin-before:0;-webkit-margin-after:0;list-style-type:none}:host mat-icon{width:18px;height:18px;font-size:18px}:host .mat-tree-node.leaf{margin-left:48px}:host .body{display:flex;flex-direction:row;justify-content:space-between}:host .body mat-tree{flex:66}:host .body mat-chip-listbox{flex:33}:host mat-tree{flex-shrink:0}:host mat-chip-listbox{margin-left:10px}\n"], dependencies: [{ kind: "component", type: NaturalSearchComponent, selector: "natural-search", inputs: ["placeholder", "facets", "multipleGroups", "dropdownTitle", "selections"], outputs: ["selectionChange"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i2.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatTreeModule }, { kind: "directive", type: i3.MatTreeNodeDef, selector: "[matTreeNodeDef]", inputs: ["matTreeNodeDefWhen", "matTreeNode"] }, { kind: "directive", type: i3.MatTreeNodePadding, selector: "[matTreeNodePadding]", inputs: ["matTreeNodePadding", "matTreeNodePaddingIndent"] }, { kind: "directive", type: i3.MatTreeNodeToggle, selector: "[matTreeNodeToggle]", inputs: ["matTreeNodeToggleRecursive"] }, { kind: "component", type: i3.MatTree, selector: "mat-tree", exportAs: ["matTree"] }, { kind: "directive", type: i3.MatTreeNode, selector: "mat-tree-node", inputs: ["tabIndex", "disabled"], outputs: ["activation", "expandedChange"], exportAs: ["matTreeNode"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "component", type: i6.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "ngmodule", type: MatChipsModule }, { kind: "directive", type: i7.MatChipAvatar, selector: "mat-chip-avatar, [matChipAvatar]" }, { kind: "component", type: i7.MatChipListbox, selector: "mat-chip-listbox", inputs: ["multiple", "aria-orientation", "selectable", "compareWith", "required", "hideSingleSelectionIndicator", "value"], outputs: ["change"] }, { kind: "component", type: i7.MatChipOption, selector: "mat-basic-chip-option, [mat-basic-chip-option], mat-chip-option, [mat-chip-option]", inputs: ["selectable", "selected"], outputs: ["selectionChange"] }, { kind: "directive", type: i7.MatChipRemove, selector: "[matChipRemove]" }] });
362
- }
363
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalHierarchicSelectorComponent, decorators: [{
364
- type: Component,
365
- args: [{ selector: 'natural-hierarchic-selector', providers: [NaturalHierarchicSelectorService], standalone: true, imports: [
366
- NaturalSearchComponent,
367
- CommonModule,
368
- MatProgressSpinnerModule,
369
- MatTreeModule,
370
- MatButtonModule,
371
- MatIconModule,
372
- NaturalIconDirective,
373
- MatCheckboxModule,
374
- MatChipsModule,
375
- ], template: "<div [style.margin-bottom.px]=\"20\">\n <natural-search (selectionChange)=\"search($event)\" [facets]=\"searchFacets\" [selections]=\"searchSelections\" />\n</div>\n\n<div class=\"body\">\n @if (loading) {\n <mat-progress-spinner [diameter]=\"36\" mode=\"indeterminate\" style=\"margin: 10px\" />\n }\n\n <mat-tree [dataSource]=\"dataSource\" [treeControl]=\"treeControl\">\n <mat-tree-node *matTreeNodeDef=\"let node\" [ngClass]=\"{leaf: !node.expandable}\" matTreeNodePadding>\n @if (node.expandable) {\n <button\n (click)=\"loadChildren(node)\"\n [attr.aria-label]=\"'toggle ' + node.name\"\n mat-icon-button\n matTreeNodeToggle\n >\n @if (node.loading) {\n <mat-progress-spinner [diameter]=\"24\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n }\n @if (!node.loading) {\n <mat-icon [naturalIcon]=\"treeControl.isExpanded(node) ? 'expand_more' : 'chevron_right'\" />\n }\n </button>\n }\n\n <mat-checkbox\n (change)=\"toggleFlatNode(node)\"\n [checked]=\"flatNodesSelection.isSelected(node)\"\n [disabled]=\"!isNodeTogglable(node)\"\n style=\"margin-right: 10px\"\n >\n @if (node.node.config.icon) {\n <mat-icon [naturalIcon]=\"node.node.config.icon\" style=\"margin-right: 10px\" />\n }\n <span>{{ node.name }}</span>\n </mat-checkbox>\n </mat-tree-node>\n </mat-tree>\n\n <mat-chip-listbox aria-orientation=\"vertical\" class=\"mat-mdc-chip-set-stacked\">\n @for (node of selectedNodes; track node.model.id) {\n <mat-chip-option (removed)=\"unselectModelNode(node)\" [removable]=\"true\" [selectable]=\"false\">\n @if (node.config.icon) {\n <mat-icon matChipAvatar [naturalIcon]=\"node.config.icon\" />\n }\n {{ node.model.name || node.model.fullName }}\n <button matChipRemove>\n <mat-icon naturalIcon=\"cancel\" />\n </button>\n </mat-chip-option>\n } @empty {\n <p class=\"mat-body nat-padding-horizontal\" i18n>Aucune s\u00E9lection</p>\n }\n </mat-chip-listbox>\n</div>\n\n@if (!loading && !dataSource.data.length) {\n <div i18n>Aucun r\u00E9sultat</div>\n}\n", styles: [":host{display:block}:host ul,:host li{-webkit-margin-before:0;-webkit-margin-after:0;list-style-type:none}:host mat-icon{width:18px;height:18px;font-size:18px}:host .mat-tree-node.leaf{margin-left:48px}:host .body{display:flex;flex-direction:row;justify-content:space-between}:host .body mat-tree{flex:66}:host .body mat-chip-listbox{flex:33}:host mat-tree{flex-shrink:0}:host mat-chip-listbox{margin-left:10px}\n"] }]
376
- }], propDecorators: { displayWith: [{
377
- type: Input
378
- }], config: [{
379
- type: Input,
380
- args: [{ required: true }]
381
- }], multiple: [{
382
- type: Input
383
- }], selected: [{
384
- type: Input
385
- }], allowUnselect: [{
386
- type: Input
387
- }], filters: [{
388
- type: Input
389
- }], searchFacets: [{
390
- type: Input
391
- }], searchSelections: [{
392
- type: Input
393
- }], searchSelectionChange: [{
394
- type: Output
395
- }], selectionChange: [{
396
- type: Output
397
- }] } });
398
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchic-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.ts","../../../../../../../projects/natural/src/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import {SelectionModel} from '@angular/cdk/collections';\nimport {FlatTreeControl} from '@angular/cdk/tree';\nimport {\n    Component,\n    DestroyRef,\n    EventEmitter,\n    inject,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {MatTreeFlatDataSource, MatTreeFlattener, MatTreeModule} from '@angular/material/tree';\nimport {Observable} from 'rxjs';\nimport {finalize} from 'rxjs/operators';\nimport {QueryVariables} from '../../../classes/query-variable-manager';\nimport {Literal} from '../../../types/types';\nimport {toGraphQLDoctrineFilter} from '../../search/classes/graphql-doctrine';\nimport {NaturalSearchFacets} from '../../search/types/facet';\nimport {NaturalSearchSelections} from '../../search/types/values';\nimport {HierarchicFlatNode} from '../classes/flat-node';\nimport {NaturalHierarchicConfiguration} from '../classes/hierarchic-configuration';\nimport {HierarchicFiltersConfiguration} from '../classes/hierarchic-filters-configuration';\nimport {HierarchicModelNode} from '../classes/model-node';\nimport {NaturalHierarchicSelectorService, OrganizedModelSelection} from './hierarchic-selector.service';\nimport {replaceObjectKeepingReference} from '../../../classes/utility';\nimport {MatChipsModule} from '@angular/material/chips';\nimport {MatCheckboxModule} from '@angular/material/checkbox';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {CommonModule} from '@angular/common';\nimport {NaturalSearchComponent} from '../../search/search/search.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n@Component({\n    selector: 'natural-hierarchic-selector',\n    templateUrl: './hierarchic-selector.component.html',\n    styleUrl: './hierarchic-selector.component.scss',\n    providers: [NaturalHierarchicSelectorService],\n    standalone: true,\n    imports: [\n        NaturalSearchComponent,\n        CommonModule,\n        MatProgressSpinnerModule,\n        MatTreeModule,\n        MatButtonModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatCheckboxModule,\n        MatChipsModule,\n    ],\n})\nexport class NaturalHierarchicSelectorComponent implements OnInit, OnChanges {\n    private readonly destroyRef = inject(DestroyRef);\n    private readonly hierarchicSelectorService = inject(NaturalHierarchicSelectorService);\n\n    /**\n     * Function that receives a model and returns a string for display value\n     */\n    @Input() public displayWith?: (item: any) => string;\n\n    /**\n     * Config for items and relations arrangement\n     */\n    @Input({required: true}) public config!: NaturalHierarchicConfiguration[];\n\n    /**\n     * If multiple or single item selection\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Selected items\n     * Organized by key, containing each an array of selected items of same type\n     */\n    @Input() public selected: OrganizedModelSelection = {};\n\n    /**\n     * Whether selectable elements can be unselected\n     */\n    @Input() public allowUnselect = true;\n\n    /**\n     * Filters that apply to each query\n     */\n    @Input() public filters?: HierarchicFiltersConfiguration | null;\n\n    /**\n     * Search facets\n     */\n    @Input() public searchFacets: NaturalSearchFacets = [];\n\n    /**\n     * Selections to apply on natural-search on component initialisation\n     */\n    @Input() public searchSelections: NaturalSearchSelections = [];\n\n    /**\n     * Emits when natural-search selections change\n     */\n    @Output() public readonly searchSelectionChange = new EventEmitter<NaturalSearchSelections>();\n\n    /**\n     * Inner representation of selected @Input() to allow flat listing as mat-chip.\n     */\n    public selectedNodes: HierarchicModelNode[] = [];\n\n    /**\n     * Emits selection change\n     * Returns a Literal where selected models are organized by key\n     */\n    @Output() public readonly selectionChange = new EventEmitter<OrganizedModelSelection>();\n\n    /**\n     * Controller for nodes selection\n     */\n    public flatNodesSelection!: SelectionModel<HierarchicFlatNode>;\n\n    public treeControl!: FlatTreeControl<HierarchicFlatNode>;\n    public treeFlattener!: MatTreeFlattener<HierarchicModelNode, HierarchicFlatNode>;\n    public dataSource!: MatTreeFlatDataSource<HierarchicModelNode, HierarchicFlatNode>;\n\n    public loading = false;\n\n    /**\n     * Cache for transformed nodes\n     */\n    private flatNodeMap: Map<string, HierarchicFlatNode> = new Map<string, HierarchicFlatNode>();\n\n    /**\n     * Angular OnChange implementation\n     */\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.selected && !changes.selected.firstChange) {\n            this.updateInnerSelection(this.selected);\n        }\n\n        if (changes.filters && !changes.filters.firstChange) {\n            this.loadRoots();\n        }\n    }\n\n    /**\n     * Angular OnInit implementation\n     */\n    public ngOnInit(): void {\n        // Init tree checkbox selectors\n        this.flatNodesSelection = new SelectionModel<any>(this.multiple);\n\n        // Tree controllers and manipulators\n        this.treeFlattener = new MatTreeFlattener(\n            this.transformer(),\n            this.getLevel(),\n            this.isExpandable(),\n            this.getChildren(),\n        );\n        this.treeControl = new FlatTreeControl<HierarchicFlatNode>(this.getLevel(), this.isExpandable());\n\n        // The dataSource contains a nested ModelNodes list. Each ModelNode has a child attribute that returns an observable.\n        // The dataSource contains a flat representation of the nested ModelNodes that is generated by the treeFlattener related functions\n        this.dataSource = new MatTreeFlatDataSource(this.treeControl, this.treeFlattener);\n\n        // Update dataSource when receiving new list -> we assign the whole tree\n        // The treeControl and treeFlattener will generate the displayed tree\n        this.hierarchicSelectorService.dataChange\n            .pipe(takeUntilDestroyed(this.destroyRef))\n            .subscribe(data => (this.dataSource.data = data));\n\n        // Prevent empty screen on first load and init NaturalHierarchicSelectorService with inputted configuration\n        let variables;\n        if (this.searchSelections.some(s => s.length)) {\n            variables = {filter: toGraphQLDoctrineFilter(this.searchFacets, this.searchSelections)};\n        }\n        this.loadRoots(variables);\n\n        // OrganizedSelection into list usable by template\n        this.updateInnerSelection(this.selected);\n    }\n\n    /**\n     * Toggle selection of a FlatNode, considering if multiple selection is activated or not\n     */\n    public toggleFlatNode(flatNode: HierarchicFlatNode): void {\n        if (this.multiple) {\n            // Is multiple allowed, just toggle element\n            if (this.flatNodesSelection.isSelected(flatNode)) {\n                this.unselectFlatNode(flatNode);\n            } else {\n                this.selectFlatNode(flatNode);\n            }\n        } else if (!this.multiple) {\n            if (this.flatNodesSelection.isSelected(flatNode)) {\n                this.unselectSingleFlatNode();\n            } else {\n                // If not multiple, and we want to select an element, unselect everything before to keep a single selection\n                this.selectSingleFlatNode(flatNode);\n            }\n        }\n    }\n\n    /**\n     * When unselecting an element from the mat-chips, it can be deep in the hierarchy, and the tree element may not exist...\n     * ... but we still need to remove the element from the mat-chips list.\n     */\n    public unselectModelNode(node: HierarchicModelNode): void {\n        const flatNode = this.getFlatNode(node);\n        if (flatNode) {\n            this.unselectFlatNode(flatNode);\n        } else {\n            // Remove from chips list only if no flatNode, because unselectFlatNode() already deals with it.\n            this.removeModelNode(node);\n            this.updateSelection(this.selectedNodes);\n        }\n    }\n\n    public isNodeTogglable(flatNode: HierarchicFlatNode): boolean {\n        if (this.isNodeSelected(flatNode.node)) {\n            return flatNode.deselectable;\n        } else {\n            return flatNode.selectable;\n        }\n    }\n\n    private getDisplayFn(config: NaturalHierarchicConfiguration): (item: any) => string {\n        if (config.displayWith) {\n            return config.displayWith;\n        }\n\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return item => (item ? item.fullName || item.name : '');\n    }\n\n    public loadChildren(flatNode: HierarchicFlatNode): void {\n        if (this.treeControl.isExpanded(flatNode)) {\n            this.hierarchicSelectorService.loadChildren(flatNode, this.filters);\n        }\n    }\n\n    /**\n     * Created to collapse all children when closing a parent, but not sure it's good.\n     */\n    // public loadChildren(flatNode: HierarchicFlatNode) {\n    //     if (this.treeControl.isExpanded(flatNode)) {\n    //\n    //         const cachedFlatNode = this.getFlatNode(flatNode.node);\n    //         if (cachedFlatNode) {\n    //             this.hierarchicSelectorService.loadChildren(cachedFlatNode, this.filters);\n    //\n    //             // Close children\n    //             cachedFlatNode.node.children.forEach(child => {\n    //                 const childNode = this.getFlatNode(child);\n    //                 if (childNode) {\n    //                     this.treeControl.collapse(childNode);\n    //                 }\n    //             });\n    //         }\n    //     }\n    // }\n\n    private getChildren(): (node: HierarchicModelNode) => Observable<HierarchicModelNode[]> {\n        return (node: HierarchicModelNode): Observable<HierarchicModelNode[]> => {\n            return node.childrenChange;\n        };\n    }\n\n    /**\n     * Transforms a HierarchicModelNode into a FlatNode\n     */\n    private transformer(): (node: HierarchicModelNode, level: number) => HierarchicFlatNode {\n        return (node: HierarchicModelNode, level: number) => {\n            return this.getOrCreateFlatNode(node, level);\n        };\n    }\n\n    /**\n     * Return deep of the node in the tree\n     */\n    private getLevel(): (node: HierarchicFlatNode) => number {\n        return (node: HierarchicFlatNode) => {\n            return node.level;\n        };\n    }\n\n    /**\n     * Is always expandable because we load on demand, we don't know if there are children yet\n     */\n    private isExpandable(): (node: HierarchicFlatNode) => boolean {\n        return (node: HierarchicFlatNode) => {\n            return node.expandable;\n        };\n    }\n\n    private getOrCreateFlatNode(node: HierarchicModelNode, level: number): HierarchicFlatNode {\n        // Return FlatNode if exists\n        const flatNode = this.getFlatNode(node);\n        if (flatNode) {\n            return flatNode;\n        }\n\n        // Return new FlatNode\n        return this.createFlatNode(node, level);\n    }\n\n    public search(selections: NaturalSearchSelections): void {\n        this.searchSelectionChange.emit(selections);\n        if (selections.some(s => s.length)) {\n            const variables = {filter: toGraphQLDoctrineFilter(this.searchFacets, selections)};\n            this.hierarchicSelectorService.search(variables, this.filters);\n        } else {\n            this.loadRoots();\n        }\n    }\n\n    private loadRoots(searchVariables?: QueryVariables): void {\n        this.loading = true;\n        this.flatNodeMap = new Map<string, HierarchicFlatNode>();\n        this.hierarchicSelectorService\n            .init(this.config, this.filters, searchVariables || null)\n            .pipe(finalize(() => (this.loading = false)))\n            .subscribe();\n    }\n\n    /**\n     * Sync inner selection (tree and mat-chips) according to selected input attribute\n     */\n    private updateInnerSelection(selected: OrganizedModelSelection): void {\n        // Transform an OrganizedModelSelection into a ModelNode list that is used in the selected zone of the component (see template)\n        this.selectedNodes = this.hierarchicSelectorService.fromOrganizedSelection(selected);\n\n        this.flatNodesSelection.clear();\n        for (const node of this.selectedNodes) {\n            const flatNode = this.getFlatNode(node);\n            if (flatNode) {\n                this.flatNodesSelection.select(flatNode);\n            }\n        }\n    }\n\n    /**\n     * Unselect a node, keeping the rest of the selected untouched\n     */\n    private unselectFlatNode(flatNode: HierarchicFlatNode): void {\n        this.flatNodesSelection.deselect(flatNode);\n        this.removeModelNode(flatNode.node);\n        this.updateSelection(this.selectedNodes);\n    }\n\n    /**\n     * Remove a node from chip lists\n     */\n    private removeModelNode(node: HierarchicModelNode): void {\n        const key = this.getMapKey(node.model);\n        const selectionIndex = this.selectedNodes.findIndex(n => this.getMapKey(n.model) === key);\n        this.selectedNodes.splice(selectionIndex, 1);\n    }\n\n    /**\n     * Select a node, keeping th rest of the selected untouched\n     */\n    private selectFlatNode(flatNode: HierarchicFlatNode): void {\n        this.flatNodesSelection.select(flatNode);\n        this.selectedNodes.push(flatNode.node);\n        this.updateSelection(this.selectedNodes);\n    }\n\n    /**\n     * Clear all selected and select the given node\n     */\n    private selectSingleFlatNode(flatNode: HierarchicFlatNode): void {\n        this.flatNodesSelection.clear();\n        this.flatNodesSelection.select(flatNode);\n        this.selectedNodes = [flatNode.node];\n        this.updateSelection(this.selectedNodes);\n    }\n\n    /**\n     * Clear all selected and select the given node\n     */\n    private unselectSingleFlatNode(): void {\n        this.flatNodesSelection.clear();\n        this.selectedNodes = [];\n        this.updateSelection(this.selectedNodes);\n    }\n\n    /**\n     * Transform the given elements into the organized selection that is emitted to output\n     */\n    private updateSelection(selected: HierarchicModelNode[]): void {\n        const organizedFlatNodesSelection = this.hierarchicSelectorService.toOrganizedSelection(selected);\n        replaceObjectKeepingReference(this.selected, organizedFlatNodesSelection);\n        this.selectionChange.emit(organizedFlatNodesSelection);\n    }\n\n    private isNodeSelected(node: HierarchicModelNode): boolean {\n        const key = this.getMapKey(node.model);\n\n        return this.selectedNodes.some(n => this.getMapKey(n.model) === key);\n    }\n\n    private getFlatNode(node: HierarchicModelNode): HierarchicFlatNode | null {\n        const key = this.getMapKey(node.model);\n        return this.flatNodeMap.get(key) || null;\n    }\n\n    private createFlatNode(node: HierarchicModelNode, level: number): HierarchicFlatNode {\n        const key = this.getMapKey(node.model);\n        const name = this.getDisplayFn(node.config)(node.model);\n        const expandable = false;\n        const isCustomSelectable = node.config.isSelectableCallback\n            ? node.config.isSelectableCallback(node.model)\n            : true;\n        const isSelectable = !!node.config.selectableAtKey && isCustomSelectable;\n\n        const flatNode = new HierarchicFlatNode(node, name, level, expandable, isSelectable);\n\n        this.hierarchicSelectorService.countItems(flatNode, this.filters);\n\n        // Mark node as selected if needed (checks the selected processed input)\n        if (this.isNodeSelected(node)) {\n            if (!this.allowUnselect) {\n                flatNode.deselectable = false;\n            }\n            this.flatNodesSelection.select(flatNode);\n        }\n\n        // Cache FlatNode\n        this.flatNodeMap.set(key, flatNode);\n\n        return flatNode;\n    }\n\n    /**\n     * Returns an identifier key for map cache\n     * As many object types can be used, this function considers typename and ID to return something like document-123\n     */\n    private getMapKey(model: Literal): string {\n        return model.__typename + '-' + model.id;\n    }\n}\n","<div [style.margin-bottom.px]=\"20\">\n    <natural-search (selectionChange)=\"search($event)\" [facets]=\"searchFacets\" [selections]=\"searchSelections\" />\n</div>\n\n<div class=\"body\">\n    @if (loading) {\n        <mat-progress-spinner [diameter]=\"36\" mode=\"indeterminate\" style=\"margin: 10px\" />\n    }\n\n    <mat-tree [dataSource]=\"dataSource\" [treeControl]=\"treeControl\">\n        <mat-tree-node *matTreeNodeDef=\"let node\" [ngClass]=\"{leaf: !node.expandable}\" matTreeNodePadding>\n            @if (node.expandable) {\n                <button\n                    (click)=\"loadChildren(node)\"\n                    [attr.aria-label]=\"'toggle ' + node.name\"\n                    mat-icon-button\n                    matTreeNodeToggle\n                >\n                    @if (node.loading) {\n                        <mat-progress-spinner [diameter]=\"24\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n                    }\n                    @if (!node.loading) {\n                        <mat-icon [naturalIcon]=\"treeControl.isExpanded(node) ? 'expand_more' : 'chevron_right'\" />\n                    }\n                </button>\n            }\n\n            <mat-checkbox\n                (change)=\"toggleFlatNode(node)\"\n                [checked]=\"flatNodesSelection.isSelected(node)\"\n                [disabled]=\"!isNodeTogglable(node)\"\n                style=\"margin-right: 10px\"\n            >\n                @if (node.node.config.icon) {\n                    <mat-icon [naturalIcon]=\"node.node.config.icon\" style=\"margin-right: 10px\" />\n                }\n                <span>{{ node.name }}</span>\n            </mat-checkbox>\n        </mat-tree-node>\n    </mat-tree>\n\n    <mat-chip-listbox aria-orientation=\"vertical\" class=\"mat-mdc-chip-set-stacked\">\n        @for (node of selectedNodes; track node.model.id) {\n            <mat-chip-option (removed)=\"unselectModelNode(node)\" [removable]=\"true\" [selectable]=\"false\">\n                @if (node.config.icon) {\n                    <mat-icon matChipAvatar [naturalIcon]=\"node.config.icon\" />\n                }\n                {{ node.model.name || node.model.fullName }}\n                <button matChipRemove>\n                    <mat-icon naturalIcon=\"cancel\" />\n                </button>\n            </mat-chip-option>\n        } @empty {\n            <p class=\"mat-body nat-padding-horizontal\" i18n>Aucune sélection</p>\n        }\n    </mat-chip-listbox>\n</div>\n\n@if (!loading && !dataSource.data.length) {\n    <div i18n>Aucun résultat</div>\n}\n"]}