@ecodev/natural 62.1.2 → 63.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/fesm2022/ecodev-natural-vanilla.mjs +1193 -0
- package/fesm2022/ecodev-natural-vanilla.mjs.map +1 -0
- package/fesm2022/ecodev-natural.mjs +436 -425
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/classes/network-activity.service.d.ts +54 -0
- package/lib/classes/validators.d.ts +1 -1
- package/lib/modules/columns-picker/columns-picker.component.d.ts +2 -2
- package/lib/modules/common/services/seo.provider.d.ts +2 -2
- package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
- package/lib/modules/file/abstract-file.d.ts +6 -3
- package/lib/modules/file/component/file.component.d.ts +2 -2
- package/lib/modules/file/file-drop.directive.d.ts +2 -3
- package/lib/modules/fixed-button-detail/fixed-button-detail.component.d.ts +2 -3
- package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.d.ts +3 -3
- package/lib/modules/icon/icon.module.d.ts +2 -2
- package/lib/modules/panels/panels.service.d.ts +1 -2
- package/lib/modules/relations/relations.component.d.ts +3 -3
- package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +2 -3
- package/lib/modules/search/group/group.component.d.ts +2 -3
- package/lib/modules/search/input/input.component.d.ts +5 -5
- package/lib/modules/search/search/search.component.d.ts +2 -2
- package/lib/modules/select/abstract-select.component.d.ts +3 -3
- package/lib/modules/select/select/select.component.d.ts +1 -1
- package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +1 -1
- package/lib/modules/table-button/table-button.component.d.ts +4 -2
- package/package.json +16 -14
- package/public-api.d.ts +1 -0
- package/src/lib/_natural.theme.scss +1 -2
- package/vanilla/index.d.ts +5 -0
- package/vanilla/package.json +3 -0
- package/vanilla/public-api.d.ts +11 -0
- package/vanilla/src/lib/classes/crypto.d.ts +8 -0
- package/vanilla/src/lib/classes/data-source.d.ts +32 -0
- package/vanilla/src/lib/classes/query-variable-manager-utils.d.ts +2 -0
- package/vanilla/src/lib/classes/query-variable-manager.d.ts +91 -0
- package/vanilla/src/lib/classes/signing.d.ts +7 -0
- package/vanilla/src/lib/classes/utility.d.ts +77 -0
- package/vanilla/src/lib/modules/search/classes/graphql-doctrine.types.d.ts +83 -0
- package/vanilla/src/lib/modules/search/classes/utils.d.ts +17 -0
- package/vanilla/src/lib/modules/search/types/dropdown-component.d.ts +20 -0
- package/vanilla/src/lib/modules/search/types/facet.d.ts +75 -0
- package/vanilla/src/lib/modules/search/types/values.d.ts +32 -0
- package/vanilla/src/lib/services/abstract-model.service.d.ts +244 -0
- package/vanilla/src/lib/services/debounce.service.d.ts +52 -0
- package/vanilla/src/lib/types/types.d.ts +100 -0
- package/esm2022/ecodev-natural.mjs +0 -5
- package/esm2022/lib/classes/abstract-detail.mjs +0 -229
- package/esm2022/lib/classes/abstract-editable-list.mjs +0 -99
- package/esm2022/lib/classes/abstract-list.mjs +0 -461
- package/esm2022/lib/classes/abstract-navigable-list.mjs +0 -133
- package/esm2022/lib/classes/apollo-utils.mjs +0 -59
- package/esm2022/lib/classes/crypto.mjs +0 -23
- package/esm2022/lib/classes/cumulative-changes.mjs +0 -50
- package/esm2022/lib/classes/data-source.mjs +0 -71
- package/esm2022/lib/classes/providers.mjs +0 -13
- package/esm2022/lib/classes/query-variable-manager-utils.mjs +0 -14
- package/esm2022/lib/classes/query-variable-manager.mjs +0 -172
- package/esm2022/lib/classes/rxjs.mjs +0 -54
- package/esm2022/lib/classes/signing.mjs +0 -38
- package/esm2022/lib/classes/tld.mjs +0 -1476
- package/esm2022/lib/classes/utility.mjs +0 -234
- package/esm2022/lib/classes/validators.mjs +0 -179
- package/esm2022/lib/directives/http-prefix.directive.mjs +0 -47
- package/esm2022/lib/modules/alert/alert.service.mjs +0 -53
- package/esm2022/lib/modules/alert/confirm.component.mjs +0 -16
- package/esm2022/lib/modules/alert/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/component/avatar.component.mjs +0 -203
- package/esm2022/lib/modules/avatar/public-api.mjs +0 -6
- package/esm2022/lib/modules/avatar/service/avatar.service.mjs +0 -63
- package/esm2022/lib/modules/avatar/sources/gravatar.mjs +0 -29
- package/esm2022/lib/modules/avatar/sources/image.mjs +0 -13
- package/esm2022/lib/modules/avatar/sources/initials.mjs +0 -39
- package/esm2022/lib/modules/avatar/sources/source.mjs +0 -16
- package/esm2022/lib/modules/columns-picker/columns-picker.component.mjs +0 -145
- package/esm2022/lib/modules/columns-picker/public-api.mjs +0 -5
- package/esm2022/lib/modules/columns-picker/types.mjs +0 -2
- package/esm2022/lib/modules/common/directives/background-density.directive.mjs +0 -63
- package/esm2022/lib/modules/common/directives/linkable-tab.directive.mjs +0 -93
- package/esm2022/lib/modules/common/directives/src-density.directive.mjs +0 -72
- package/esm2022/lib/modules/common/pipes/capitalize.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/ellipsis.pipe.mjs +0 -17
- package/esm2022/lib/modules/common/pipes/enum.pipe.mjs +0 -24
- package/esm2022/lib/modules/common/pipes/time-ago.pipe.mjs +0 -140
- package/esm2022/lib/modules/common/public-api.mjs +0 -14
- package/esm2022/lib/modules/common/services/memory-storage.mjs +0 -110
- package/esm2022/lib/modules/common/services/seo.provider.mjs +0 -23
- package/esm2022/lib/modules/common/services/seo.service.mjs +0 -235
- package/esm2022/lib/modules/detail-header/detail-header.component.mjs +0 -84
- package/esm2022/lib/modules/detail-header/public-api.mjs +0 -5
- package/esm2022/lib/modules/dialog-trigger/dialog-trigger.component.mjs +0 -72
- package/esm2022/lib/modules/dialog-trigger/public-api.mjs +0 -5
- package/esm2022/lib/modules/dropdown-components/abstract-association-select-component.directive.mjs +0 -100
- package/esm2022/lib/modules/dropdown-components/public-api.mjs +0 -14
- package/esm2022/lib/modules/dropdown-components/type-boolean/type-boolean.component.mjs +0 -39
- package/esm2022/lib/modules/dropdown-components/type-date/type-date.component.mjs +0 -173
- package/esm2022/lib/modules/dropdown-components/type-date-range/type-date-range.component.mjs +0 -134
- package/esm2022/lib/modules/dropdown-components/type-hierarchic-selector/type-hierarchic-selector.component.mjs +0 -80
- package/esm2022/lib/modules/dropdown-components/type-natural-select/type-natural-select.component.mjs +0 -48
- package/esm2022/lib/modules/dropdown-components/type-number/type-number.component.mjs +0 -110
- package/esm2022/lib/modules/dropdown-components/type-options/type-options.component.mjs +0 -64
- package/esm2022/lib/modules/dropdown-components/type-select/type-select.component.mjs +0 -175
- package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +0 -62
- package/esm2022/lib/modules/dropdown-components/types.mjs +0 -41
- package/esm2022/lib/modules/dropdown-components/utils.mjs +0 -35
- package/esm2022/lib/modules/file/abstract-file.mjs +0 -230
- package/esm2022/lib/modules/file/component/file.component.mjs +0 -172
- package/esm2022/lib/modules/file/file-drop.directive.mjs +0 -111
- package/esm2022/lib/modules/file/file-select.directive.mjs +0 -26
- package/esm2022/lib/modules/file/file.service.mjs +0 -43
- package/esm2022/lib/modules/file/public-api.mjs +0 -9
- package/esm2022/lib/modules/file/types.mjs +0 -2
- package/esm2022/lib/modules/file/utils.mjs +0 -129
- package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +0 -30
- package/esm2022/lib/modules/fixed-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/fixed-button-detail/fixed-button-detail.component.mjs +0 -56
- package/esm2022/lib/modules/fixed-button-detail/public-api.mjs +0 -5
- package/esm2022/lib/modules/hierarchic-selector/classes/flat-node.mjs +0 -18
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-filters-configuration.mjs +0 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/model-node.mjs +0 -14
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +0 -398
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +0 -243
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component.mjs +0 -38
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service.mjs +0 -22
- package/esm2022/lib/modules/hierarchic-selector/public-api.mjs +0 -10
- package/esm2022/lib/modules/icon/icon.directive.mjs +0 -96
- package/esm2022/lib/modules/icon/icon.module.mjs +0 -33
- package/esm2022/lib/modules/icon/public-api.mjs +0 -6
- package/esm2022/lib/modules/logger/error-handler.mjs +0 -87
- package/esm2022/lib/modules/logger/error.module.mjs +0 -22
- package/esm2022/lib/modules/logger/public-api.mjs +0 -6
- package/esm2022/lib/modules/matomo/matomo.service.mjs +0 -96
- package/esm2022/lib/modules/matomo/public-api.mjs +0 -5
- package/esm2022/lib/modules/panels/abstract-panel.mjs +0 -76
- package/esm2022/lib/modules/panels/fallback-if-no-opened-panels.urlmatcher.mjs +0 -12
- package/esm2022/lib/modules/panels/panels.component.mjs +0 -27
- package/esm2022/lib/modules/panels/panels.module.mjs +0 -10
- package/esm2022/lib/modules/panels/panels.service.mjs +0 -329
- package/esm2022/lib/modules/panels/panels.urlmatcher.mjs +0 -75
- package/esm2022/lib/modules/panels/public-api.mjs +0 -11
- package/esm2022/lib/modules/panels/types.mjs +0 -3
- package/esm2022/lib/modules/relations/public-api.mjs +0 -5
- package/esm2022/lib/modules/relations/relations.component.mjs +0 -254
- package/esm2022/lib/modules/search/classes/graphql-doctrine.mjs +0 -111
- package/esm2022/lib/modules/search/classes/graphql-doctrine.types.mjs +0 -14
- package/esm2022/lib/modules/search/classes/transformers.mjs +0 -142
- package/esm2022/lib/modules/search/classes/url.mjs +0 -53
- package/esm2022/lib/modules/search/classes/utils.mjs +0 -25
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container-animations.mjs +0 -44
- package/esm2022/lib/modules/search/dropdown-container/dropdown-container.component.mjs +0 -87
- package/esm2022/lib/modules/search/dropdown-container/dropdown-ref.mjs +0 -24
- package/esm2022/lib/modules/search/dropdown-container/dropdown.service.mjs +0 -90
- package/esm2022/lib/modules/search/facet-selector/facet-selector.component.mjs +0 -45
- package/esm2022/lib/modules/search/group/group.component.mjs +0 -53
- package/esm2022/lib/modules/search/input/input.component.mjs +0 -365
- package/esm2022/lib/modules/search/public-api.mjs +0 -7
- package/esm2022/lib/modules/search/search/search.component.mjs +0 -102
- package/esm2022/lib/modules/search/types/dropdown-component.mjs +0 -2
- package/esm2022/lib/modules/search/types/facet.mjs +0 -2
- package/esm2022/lib/modules/search/types/values.mjs +0 -2
- package/esm2022/lib/modules/select/abstract-select.component.mjs +0 -232
- package/esm2022/lib/modules/select/public-api.mjs +0 -7
- package/esm2022/lib/modules/select/select/select.component.mjs +0 -310
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +0 -57
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +0 -155
- package/esm2022/lib/modules/sidenav/public-api.mjs +0 -9
- package/esm2022/lib/modules/sidenav/sidenav/sidenav.component.mjs +0 -15
- package/esm2022/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +0 -90
- package/esm2022/lib/modules/sidenav/sidenav-content/sidenav-content.component.mjs +0 -11
- package/esm2022/lib/modules/sidenav/sidenav-stack.service.mjs +0 -50
- package/esm2022/lib/modules/sidenav/sidenav.service.mjs +0 -196
- package/esm2022/lib/modules/stamp/public-api.mjs +0 -5
- package/esm2022/lib/modules/stamp/stamp.component.mjs +0 -23
- package/esm2022/lib/modules/table-button/public-api.mjs +0 -5
- package/esm2022/lib/modules/table-button/table-button.component.mjs +0 -78
- package/esm2022/lib/services/abstract-model.service.mjs +0 -526
- package/esm2022/lib/services/debounce.service.mjs +0 -149
- package/esm2022/lib/services/enum.service.mjs +0 -64
- package/esm2022/lib/services/link-mutation.service.mjs +0 -154
- package/esm2022/lib/services/persistence.service.mjs +0 -115
- package/esm2022/lib/services/swiss-parsing-date-adapter.service.mjs +0 -63
- package/esm2022/lib/types/types.mjs +0 -2
- package/esm2022/public-api.mjs +0 -46
|
@@ -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"]}
|