@agorapulse/ui-components 18.0.9 → 18.0.10

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 (67) hide show
  1. package/agorapulse-ui-components-18.0.10.tgz +0 -0
  2. package/datepicker/datepicker.component.d.ts +1 -1
  3. package/esm2022/datepicker/datepicker.component.mjs +2 -2
  4. package/esm2022/index.mjs +2 -1
  5. package/esm2022/input-search/input-search.component.mjs +5 -4
  6. package/esm2022/nav-selector/agorapulse-ui-components-nav-selector.mjs +5 -0
  7. package/esm2022/nav-selector/directives/tree-node-accessibility.directive.mjs +30 -0
  8. package/esm2022/nav-selector/nav-selector-category/nav-selector-category.component.mjs +81 -0
  9. package/esm2022/nav-selector/nav-selector-category/nav-selector-category.presenter.mjs +29 -0
  10. package/esm2022/nav-selector/nav-selector-group/nav-selector-group.component.mjs +142 -0
  11. package/esm2022/nav-selector/nav-selector-group/nav-selector-group.presenter.mjs +31 -0
  12. package/esm2022/nav-selector/nav-selector-leaf/nav-selector-leaf.component.mjs +240 -0
  13. package/esm2022/nav-selector/nav-selector-leaf/nav-selector-leaf.presenter.mjs +35 -0
  14. package/esm2022/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.component.mjs +39 -0
  15. package/esm2022/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.presenter.mjs +18 -0
  16. package/esm2022/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.component.mjs +92 -0
  17. package/esm2022/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.presenter.mjs +31 -0
  18. package/esm2022/nav-selector/nav-selector.component.mjs +123 -0
  19. package/esm2022/nav-selector/nav-selector.mjs +9 -0
  20. package/esm2022/nav-selector/nav-selector.state.mjs +173 -0
  21. package/esm2022/nav-selector/public_api.mjs +2 -0
  22. package/esm2022/nav-selector/utils/leaf.utils.mjs +10 -0
  23. package/esm2022/nav-selector/utils/nav-selector.accessibility.mjs +171 -0
  24. package/esm2022/nav-selector/utils/nav-selector.builder.mjs +263 -0
  25. package/esm2022/nav-selector/utils/nav-selector.filter.mjs +102 -0
  26. package/esm2022/nav-selector/utils/nav-selector.folding.mjs +219 -0
  27. package/esm2022/nav-selector/utils/nav-selector.minifying.mjs +50 -0
  28. package/esm2022/nav-selector/utils/nav-selector.multi-select.mjs +208 -0
  29. package/esm2022/nav-selector/utils/nav-selector.single-select.mjs +91 -0
  30. package/esm2022/nav-selector/utils/nav-selector.view-more.mjs +98 -0
  31. package/fesm2022/agorapulse-ui-components-datepicker.mjs +1 -1
  32. package/fesm2022/agorapulse-ui-components-datepicker.mjs.map +1 -1
  33. package/fesm2022/agorapulse-ui-components-input-search.mjs +4 -3
  34. package/fesm2022/agorapulse-ui-components-input-search.mjs.map +1 -1
  35. package/fesm2022/agorapulse-ui-components-nav-selector.mjs +2198 -0
  36. package/fesm2022/agorapulse-ui-components-nav-selector.mjs.map +1 -0
  37. package/fesm2022/agorapulse-ui-components.mjs +1 -0
  38. package/fesm2022/agorapulse-ui-components.mjs.map +1 -1
  39. package/index.d.ts +1 -0
  40. package/input-search/input-search.component.d.ts +1 -1
  41. package/nav-selector/directives/tree-node-accessibility.directive.d.ts +9 -0
  42. package/nav-selector/index.d.ts +5 -0
  43. package/nav-selector/nav-selector-category/nav-selector-category.component.d.ts +16 -0
  44. package/nav-selector/nav-selector-category/nav-selector-category.presenter.d.ts +14 -0
  45. package/nav-selector/nav-selector-group/nav-selector-group.component.d.ts +29 -0
  46. package/nav-selector/nav-selector-group/nav-selector-group.presenter.d.ts +17 -0
  47. package/nav-selector/nav-selector-leaf/nav-selector-leaf.component.d.ts +51 -0
  48. package/nav-selector/nav-selector-leaf/nav-selector-leaf.presenter.d.ts +19 -0
  49. package/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.component.d.ts +13 -0
  50. package/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.presenter.d.ts +10 -0
  51. package/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.component.d.ts +24 -0
  52. package/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.presenter.d.ts +14 -0
  53. package/nav-selector/nav-selector.component.d.ts +29 -0
  54. package/nav-selector/nav-selector.d.ts +220 -0
  55. package/nav-selector/nav-selector.state.d.ts +47 -0
  56. package/nav-selector/public_api.d.ts +2 -0
  57. package/nav-selector/utils/leaf.utils.d.ts +5 -0
  58. package/nav-selector/utils/nav-selector.accessibility.d.ts +52 -0
  59. package/nav-selector/utils/nav-selector.builder.d.ts +32 -0
  60. package/nav-selector/utils/nav-selector.filter.d.ts +30 -0
  61. package/nav-selector/utils/nav-selector.folding.d.ts +47 -0
  62. package/nav-selector/utils/nav-selector.minifying.d.ts +27 -0
  63. package/nav-selector/utils/nav-selector.multi-select.d.ts +54 -0
  64. package/nav-selector/utils/nav-selector.single-select.d.ts +15 -0
  65. package/nav-selector/utils/nav-selector.view-more.d.ts +30 -0
  66. package/package.json +13 -7
  67. package/agorapulse-ui-components-18.0.9.tgz +0 -0
@@ -0,0 +1,92 @@
1
+ import { animate, keyframes, state, style, transition, trigger } from '@angular/animations';
2
+ import { ChangeDetectionStrategy, Component, computed, effect, input, signal } from '@angular/core';
3
+ import { NavSelectorLeafDetailComponent } from '../nav-selector-leaf-detail/nav-selector-leaf-detail.component';
4
+ import { NavSelectorLeafDetailsPresenter } from '../nav-selector-leaf-details/nav-selector-leaf-details.presenter';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "../nav-selector-leaf-details/nav-selector-leaf-details.presenter";
7
+ export class NavSelectorLeafDetailsComponent {
8
+ el;
9
+ navSelectorLeafDetailsPresenter;
10
+ leaf = input.required();
11
+ details = input.required();
12
+ firstDetails = computed(() => this.details().slice(0, this.leaf().viewMoreDetailsDisplayedLimit));
13
+ lastDetails = computed(() => this.details().slice(this.leaf().viewMoreDetailsDisplayedLimit));
14
+ viewMoreDelay = signal(false);
15
+ animationState = computed(() => (this.leaf().viewMoreDisplayable && this.leaf().viewMoreDisplayed ? 'collapsed' : 'expanded'));
16
+ /**
17
+ * The height when details are all displayed
18
+ */
19
+ maxHeight = signal('0px');
20
+ constructor(el, navSelectorLeafDetailsPresenter) {
21
+ this.el = el;
22
+ this.navSelectorLeafDetailsPresenter = navSelectorLeafDetailsPresenter;
23
+ effect(() => {
24
+ if (this.leaf().viewMoreDisplayed) {
25
+ setTimeout(() => this.viewMoreDelay.set(false), 150);
26
+ }
27
+ else {
28
+ this.viewMoreDelay.set(true);
29
+ }
30
+ }, { allowSignalWrites: true });
31
+ }
32
+ ngAfterViewInit() {
33
+ // scrollHeight = leaf + first details displayed when there is a limit
34
+ const height = this.el.nativeElement.scrollHeight;
35
+ // cross multiplication to determine height of remaining details
36
+ this.maxHeight.set(`${(height * this.lastDetails().length) / this.firstDetails().length}px`);
37
+ }
38
+ toggleViewMore() {
39
+ this.navSelectorLeafDetailsPresenter.toggleViewMore(this.leaf());
40
+ }
41
+ onSpaceOrEnter(event) {
42
+ event.stopImmediatePropagation();
43
+ this.toggleViewMore();
44
+ }
45
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorLeafDetailsComponent, deps: [{ token: i0.ElementRef }, { token: i1.NavSelectorLeafDetailsPresenter }], target: i0.ɵɵFactoryTarget.Component });
46
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.9", type: NavSelectorLeafDetailsComponent, isStandalone: true, selector: "ap-nav-selector-leaf-details", inputs: { leaf: { classPropertyName: "leaf", publicName: "leaf", isSignal: true, isRequired: true, transformFunction: null }, details: { classPropertyName: "details", publicName: "details", isSignal: true, isRequired: true, transformFunction: null } }, host: { properties: { "attr.aria-expanded": "!leaf().viewMoreDisplayable || leaf().viewMoreDisplayed" } }, providers: [NavSelectorLeafDetailsPresenter], ngImport: i0, template: "@for (detail of firstDetails(); track detail.uid) {\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n </div>\n}\n\n@if (leaf().viewMoreDisplayable) {\n <div\n class=\"details-container\"\n [@accordion]=\"{\n value: animationState(),\n params: {\n maxHeight: maxHeight()\n }\n }\">\n @if (viewMoreDelay()) {\n @for (detail of lastDetails(); track detail.uid) {\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n </div>\n }\n }\n </div>\n\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <a\n tabindex=\"0\"\n role=\"button\"\n class=\"standalone view-more\"\n (keydown.space)=\"onSpaceOrEnter($event)\"\n (keydown.enter)=\"onSpaceOrEnter($event)\"\n (click)=\"toggleViewMore()\">\n @if (leaf().viewMoreDisplayed) {\n {{ navSelectorLeafDetailsPresenter.viewMoreText() }}\n } @else {\n {{ navSelectorLeafDetailsPresenter.viewLessText() }}\n }\n </a>\n </div>\n}\n", styles: [":host{display:flex;flex-direction:column;align-items:flex-start;flex:1 0 0}:host .detail{display:flex;align-self:stretch}:host .separator{display:flex;padding:0 11px 0 20px;align-items:center;align-self:stretch}:host .rectangle{width:1px;align-self:stretch;background:var(--ref-color-grey-10)}:host .view-more{display:flex;padding:8px;align-items:center;align-self:stretch}:host .details-container{display:flex;flex-direction:column;align-self:stretch}\n"], dependencies: [{ kind: "component", type: NavSelectorLeafDetailComponent, selector: "ap-nav-selector-leaf-detail", inputs: ["detail", "embedded"] }], animations: [
47
+ /**
48
+ * Overflow hidden is put only during the animation and on the collapsed state because if it is put on the expanded state then children’s border will be cut (hover / focus)
49
+ */
50
+ trigger('accordion', [
51
+ state('collapsed', style({
52
+ maxHeight: '0',
53
+ overflow: 'hidden',
54
+ })),
55
+ state('expanded', style({
56
+ maxHeight: 'initial',
57
+ })),
58
+ transition('collapsed => expanded', [
59
+ animate('250ms cubic-bezier(.4, 0, .3, 1)', style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' })),
60
+ ]),
61
+ transition('expanded => collapsed', [
62
+ animate('250ms cubic-bezier(.4, 0, .3, 1)', keyframes([style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' }), style({ maxHeight: 0 })])),
63
+ ]),
64
+ ]),
65
+ ], changeDetection: i0.ChangeDetectionStrategy.OnPush });
66
+ }
67
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorLeafDetailsComponent, decorators: [{
68
+ type: Component,
69
+ args: [{ selector: 'ap-nav-selector-leaf-details', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [NavSelectorLeafDetailComponent], providers: [NavSelectorLeafDetailsPresenter], host: {
70
+ '[attr.aria-expanded]': '!leaf().viewMoreDisplayable || leaf().viewMoreDisplayed',
71
+ }, animations: [
72
+ /**
73
+ * Overflow hidden is put only during the animation and on the collapsed state because if it is put on the expanded state then children’s border will be cut (hover / focus)
74
+ */
75
+ trigger('accordion', [
76
+ state('collapsed', style({
77
+ maxHeight: '0',
78
+ overflow: 'hidden',
79
+ })),
80
+ state('expanded', style({
81
+ maxHeight: 'initial',
82
+ })),
83
+ transition('collapsed => expanded', [
84
+ animate('250ms cubic-bezier(.4, 0, .3, 1)', style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' })),
85
+ ]),
86
+ transition('expanded => collapsed', [
87
+ animate('250ms cubic-bezier(.4, 0, .3, 1)', keyframes([style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' }), style({ maxHeight: 0 })])),
88
+ ]),
89
+ ]),
90
+ ], template: "@for (detail of firstDetails(); track detail.uid) {\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n </div>\n}\n\n@if (leaf().viewMoreDisplayable) {\n <div\n class=\"details-container\"\n [@accordion]=\"{\n value: animationState(),\n params: {\n maxHeight: maxHeight()\n }\n }\">\n @if (viewMoreDelay()) {\n @for (detail of lastDetails(); track detail.uid) {\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n </div>\n }\n }\n </div>\n\n <div class=\"detail\">\n <div class=\"separator\">\n <div class=\"rectangle\"></div>\n </div>\n <a\n tabindex=\"0\"\n role=\"button\"\n class=\"standalone view-more\"\n (keydown.space)=\"onSpaceOrEnter($event)\"\n (keydown.enter)=\"onSpaceOrEnter($event)\"\n (click)=\"toggleViewMore()\">\n @if (leaf().viewMoreDisplayed) {\n {{ navSelectorLeafDetailsPresenter.viewMoreText() }}\n } @else {\n {{ navSelectorLeafDetailsPresenter.viewLessText() }}\n }\n </a>\n </div>\n}\n", styles: [":host{display:flex;flex-direction:column;align-items:flex-start;flex:1 0 0}:host .detail{display:flex;align-self:stretch}:host .separator{display:flex;padding:0 11px 0 20px;align-items:center;align-self:stretch}:host .rectangle{width:1px;align-self:stretch;background:var(--ref-color-grey-10)}:host .view-more{display:flex;padding:8px;align-items:center;align-self:stretch}:host .details-container{display:flex;flex-direction:column;align-self:stretch}\n"] }]
91
+ }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.NavSelectorLeafDetailsPresenter }] });
92
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nav-selector-leaf-details.component.js","sourceRoot":"","sources":["../../../../../libs/ui-components/nav-selector/src/nav-selector-leaf-details/nav-selector-leaf-details.component.ts","../../../../../libs/ui-components/nav-selector/src/nav-selector-leaf-details/nav-selector-leaf-details.component.html"],"names":[],"mappings":"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","sourcesContent":["import { animate, keyframes, state, style, transition, trigger } from '@angular/animations';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, computed, effect, ElementRef, input, signal } from '@angular/core';\nimport { InternalNavSelectorLeaf, InternalNavSelectorLeafDetails } from '../nav-selector';\nimport { NavSelectorLeafDetailComponent } from '../nav-selector-leaf-detail/nav-selector-leaf-detail.component';\nimport { NavSelectorLeafDetailsPresenter } from '../nav-selector-leaf-details/nav-selector-leaf-details.presenter';\n\n@Component({\n    selector: 'ap-nav-selector-leaf-details',\n    templateUrl: './nav-selector-leaf-details.component.html',\n    styleUrls: ['./nav-selector-leaf-details.component.scss'],\n    standalone: true,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    imports: [NavSelectorLeafDetailComponent],\n    providers: [NavSelectorLeafDetailsPresenter],\n    host: {\n        '[attr.aria-expanded]': '!leaf().viewMoreDisplayable || leaf().viewMoreDisplayed',\n    },\n    animations: [\n        /**\n         * Overflow hidden is put only during the animation and on the collapsed state because if it is put on the expanded state then children’s border will be cut (hover / focus)\n         */\n        trigger('accordion', [\n            state(\n                'collapsed',\n                style({\n                    maxHeight: '0',\n                    overflow: 'hidden',\n                })\n            ),\n            state(\n                'expanded',\n                style({\n                    maxHeight: 'initial',\n                })\n            ),\n            transition('collapsed => expanded', [\n                animate('250ms cubic-bezier(.4, 0, .3, 1)', style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' })),\n            ]),\n            transition('expanded => collapsed', [\n                animate(\n                    '250ms cubic-bezier(.4, 0, .3, 1)',\n                    keyframes([style({ overflow: 'hidden', maxHeight: '{{maxHeight}}' }), style({ maxHeight: 0 })])\n                ),\n            ]),\n        ]),\n    ],\n})\nexport class NavSelectorLeafDetailsComponent implements AfterViewInit {\n    leaf = input.required<InternalNavSelectorLeaf>();\n    details = input.required<InternalNavSelectorLeafDetails[]>();\n\n    firstDetails = computed(() => this.details().slice(0, this.leaf().viewMoreDetailsDisplayedLimit));\n    lastDetails = computed(() => this.details().slice(this.leaf().viewMoreDetailsDisplayedLimit));\n\n    viewMoreDelay = signal(false);\n\n    animationState = computed(() => (this.leaf().viewMoreDisplayable && this.leaf().viewMoreDisplayed ? 'collapsed' : 'expanded'));\n\n    /**\n     * The height when details are all displayed\n     */\n    maxHeight = signal('0px');\n\n    constructor(\n        private el: ElementRef,\n        public navSelectorLeafDetailsPresenter: NavSelectorLeafDetailsPresenter\n    ) {\n        effect(\n            () => {\n                if (this.leaf().viewMoreDisplayed) {\n                    setTimeout(() => this.viewMoreDelay.set(false), 150);\n                } else {\n                    this.viewMoreDelay.set(true);\n                }\n            },\n            { allowSignalWrites: true }\n        );\n    }\n\n    ngAfterViewInit(): void {\n        // scrollHeight = leaf + first details displayed when there is a limit\n        const height = this.el.nativeElement.scrollHeight;\n        // cross multiplication to determine height of remaining details\n        this.maxHeight.set(`${(height * this.lastDetails().length) / this.firstDetails().length}px`);\n    }\n\n    toggleViewMore() {\n        this.navSelectorLeafDetailsPresenter.toggleViewMore(this.leaf());\n    }\n\n    onSpaceOrEnter(event: Event) {\n        event.stopImmediatePropagation();\n        this.toggleViewMore();\n    }\n}\n","@for (detail of firstDetails(); track detail.uid) {\n    <div class=\"detail\">\n        <div class=\"separator\">\n            <div class=\"rectangle\"></div>\n        </div>\n        <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n    </div>\n}\n\n@if (leaf().viewMoreDisplayable) {\n    <div\n        class=\"details-container\"\n        [@accordion]=\"{\n            value: animationState(),\n            params: {\n                maxHeight: maxHeight()\n            }\n        }\">\n        @if (viewMoreDelay()) {\n            @for (detail of lastDetails(); track detail.uid) {\n                <div class=\"detail\">\n                    <div class=\"separator\">\n                        <div class=\"rectangle\"></div>\n                    </div>\n                    <ap-nav-selector-leaf-detail [detail]=\"detail\" />\n                </div>\n            }\n        }\n    </div>\n\n    <div class=\"detail\">\n        <div class=\"separator\">\n            <div class=\"rectangle\"></div>\n        </div>\n        <a\n            tabindex=\"0\"\n            role=\"button\"\n            class=\"standalone view-more\"\n            (keydown.space)=\"onSpaceOrEnter($event)\"\n            (keydown.enter)=\"onSpaceOrEnter($event)\"\n            (click)=\"toggleViewMore()\">\n            @if (leaf().viewMoreDisplayed) {\n                {{ navSelectorLeafDetailsPresenter.viewMoreText() }}\n            } @else {\n                {{ navSelectorLeafDetailsPresenter.viewLessText() }}\n            }\n        </a>\n    </div>\n}\n"]}
@@ -0,0 +1,31 @@
1
+ import { computed, Injectable } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../nav-selector.state";
4
+ export class NavSelectorLeafDetailsPresenter {
5
+ navSelectorState;
6
+ viewMoreText = computed(() => this.navSelectorState.texts().viewMore);
7
+ viewLessText = computed(() => this.navSelectorState.texts().viewLess);
8
+ constructor(navSelectorState) {
9
+ this.navSelectorState = navSelectorState;
10
+ }
11
+ toggleViewMore(leaf) {
12
+ if (leaf.viewMoreDisplayed) {
13
+ this.viewMore(leaf);
14
+ }
15
+ else {
16
+ this.viewLess(leaf);
17
+ }
18
+ }
19
+ viewMore(leaf) {
20
+ this.navSelectorState.viewMore(leaf);
21
+ }
22
+ viewLess(leaf) {
23
+ this.navSelectorState.viewLess(leaf);
24
+ }
25
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorLeafDetailsPresenter, deps: [{ token: i1.NavSelectorState }], target: i0.ɵɵFactoryTarget.Injectable });
26
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorLeafDetailsPresenter });
27
+ }
28
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorLeafDetailsPresenter, decorators: [{
29
+ type: Injectable
30
+ }], ctorParameters: () => [{ type: i1.NavSelectorState }] });
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmF2LXNlbGVjdG9yLWxlYWYtZGV0YWlscy5wcmVzZW50ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpLWNvbXBvbmVudHMvbmF2LXNlbGVjdG9yL3NyYy9uYXYtc2VsZWN0b3ItbGVhZi1kZXRhaWxzL25hdi1zZWxlY3Rvci1sZWFmLWRldGFpbHMucHJlc2VudGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFLckQsTUFBTSxPQUFPLCtCQUErQjtJQUlwQjtJQUhwQixZQUFZLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUN0RSxZQUFZLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUV0RSxZQUFvQixnQkFBa0M7UUFBbEMscUJBQWdCLEdBQWhCLGdCQUFnQixDQUFrQjtJQUFHLENBQUM7SUFFMUQsY0FBYyxDQUFDLElBQTZCO1FBQ3hDLElBQUksSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFDekIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUN4QixDQUFDO2FBQU0sQ0FBQztZQUNKLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDeEIsQ0FBQztJQUNMLENBQUM7SUFFRCxRQUFRLENBQUMsSUFBNkI7UUFDbEMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUN6QyxDQUFDO0lBRUQsUUFBUSxDQUFDLElBQTZCO1FBQ2xDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDekMsQ0FBQzt1R0FwQlEsK0JBQStCOzJHQUEvQiwrQkFBK0I7OzJGQUEvQiwrQkFBK0I7a0JBRDNDLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjb21wdXRlZCwgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSW50ZXJuYWxOYXZTZWxlY3RvckxlYWYgfSBmcm9tICcuLi9uYXYtc2VsZWN0b3InO1xuaW1wb3J0IHsgTmF2U2VsZWN0b3JTdGF0ZSB9IGZyb20gJy4uL25hdi1zZWxlY3Rvci5zdGF0ZSc7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBOYXZTZWxlY3RvckxlYWZEZXRhaWxzUHJlc2VudGVyIHtcbiAgICB2aWV3TW9yZVRleHQgPSBjb21wdXRlZCgoKSA9PiB0aGlzLm5hdlNlbGVjdG9yU3RhdGUudGV4dHMoKS52aWV3TW9yZSk7XG4gICAgdmlld0xlc3NUZXh0ID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5uYXZTZWxlY3RvclN0YXRlLnRleHRzKCkudmlld0xlc3MpO1xuXG4gICAgY29uc3RydWN0b3IocHJpdmF0ZSBuYXZTZWxlY3RvclN0YXRlOiBOYXZTZWxlY3RvclN0YXRlKSB7fVxuXG4gICAgdG9nZ2xlVmlld01vcmUobGVhZjogSW50ZXJuYWxOYXZTZWxlY3RvckxlYWYpIHtcbiAgICAgICAgaWYgKGxlYWYudmlld01vcmVEaXNwbGF5ZWQpIHtcbiAgICAgICAgICAgIHRoaXMudmlld01vcmUobGVhZik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICB0aGlzLnZpZXdMZXNzKGxlYWYpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgdmlld01vcmUobGVhZjogSW50ZXJuYWxOYXZTZWxlY3RvckxlYWYpIHtcbiAgICAgICAgdGhpcy5uYXZTZWxlY3RvclN0YXRlLnZpZXdNb3JlKGxlYWYpO1xuICAgIH1cblxuICAgIHZpZXdMZXNzKGxlYWY6IEludGVybmFsTmF2U2VsZWN0b3JMZWFmKSB7XG4gICAgICAgIHRoaXMubmF2U2VsZWN0b3JTdGF0ZS52aWV3TGVzcyhsZWFmKTtcbiAgICB9XG59XG4iXX0=
@@ -0,0 +1,123 @@
1
+ import { IconButtonComponent } from '@agorapulse/ui-components/icon-button';
2
+ import { InputSearchComponent } from '@agorapulse/ui-components/input-search';
3
+ import { apArrowExpand, apArrowReduce, apSearch, SymbolComponent, withSymbols } from '@agorapulse/ui-symbol';
4
+ import { animate, keyframes, state, style, transition, trigger } from '@angular/animations';
5
+ import { NgTemplateOutlet } from '@angular/common';
6
+ import { afterNextRender, ChangeDetectionStrategy, Component, computed, contentChild, contentChildren, effect, input, model, TemplateRef, } from '@angular/core';
7
+ import { FormsModule } from '@angular/forms';
8
+ import { EventPluginsModule } from '@tinkoff/ng-event-plugins';
9
+ import { NavSelectorCategoryComponent } from './nav-selector-category/nav-selector-category.component';
10
+ import { NavSelectorGroupComponent } from './nav-selector-group/nav-selector-group.component';
11
+ import { NavSelectorLeafComponent } from './nav-selector-leaf/nav-selector-leaf.component';
12
+ import { NavSelectorState } from './nav-selector.state';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "./nav-selector.state";
15
+ import * as i2 from "@angular/forms";
16
+ let nextUniqueId = 0;
17
+ export class NavSelectorComponent {
18
+ navSelectorState;
19
+ el;
20
+ navSelectorEntries = input.required();
21
+ translatedTexts = input.required();
22
+ multipleModeEnabled = input(false);
23
+ detailsDisplayedLimit = input(Number.MAX_SAFE_INTEGER);
24
+ /**
25
+ * If true, the nav selector will be expanded by default. Even if the width of the screen is smaller than 1280px.
26
+ */
27
+ forceExpanded = input(false);
28
+ selectedEntryUids = model.required();
29
+ headerProjection = contentChild('header');
30
+ footerProjection = contentChild('footer');
31
+ displayFooter = computed(() => this.footerProjection() !== undefined);
32
+ leafActionProjection = contentChildren('leafAction');
33
+ expansionState = computed(() => (this.navSelectorState.expanded() ? 'expanded' : 'minified'));
34
+ componentUid = `nav-selector-${nextUniqueId++}`;
35
+ constructor(navSelectorState, el) {
36
+ this.navSelectorState = navSelectorState;
37
+ this.el = el;
38
+ navSelectorState.registerOnSelectedUidsChange(selectedUids => {
39
+ this.selectedEntryUids.set(selectedUids);
40
+ });
41
+ afterNextRender(() => {
42
+ if (!this.forceExpanded() && document.body.clientWidth < 1200) {
43
+ this.navSelectorState.toggleExpanded();
44
+ }
45
+ });
46
+ effect(() => {
47
+ this.navSelectorState.updateMultiModeEnabled(this.multipleModeEnabled());
48
+ if (!this.navSelectorState.filteredEntries().length) {
49
+ this.navSelectorState.updateEntries(this.navSelectorEntries(), this.selectedEntryUids(), this.detailsDisplayedLimit());
50
+ }
51
+ }, { allowSignalWrites: true });
52
+ effect(() => {
53
+ this.navSelectorState.updateDetailsDisplayedLimit(this.detailsDisplayedLimit());
54
+ }, { allowSignalWrites: true });
55
+ const leafActionProjectionEffect = effect(() => {
56
+ if (this.leafActionProjection()) {
57
+ this.navSelectorState.updateLeafAction(this.leafActionProjection());
58
+ leafActionProjectionEffect.destroy();
59
+ }
60
+ }, { manualCleanup: true, allowSignalWrites: true });
61
+ effect(() => {
62
+ this.navSelectorState.onSelectionChange(this.selectedEntryUids());
63
+ }, { allowSignalWrites: true });
64
+ effect(() => {
65
+ this.navSelectorState.updateTexts(this.translatedTexts());
66
+ }, { allowSignalWrites: true });
67
+ }
68
+ onArrowDown($event) {
69
+ if (this.el.nativeElement.querySelector('.nav-selector__content').contains(document.activeElement)) {
70
+ $event.stopImmediatePropagation();
71
+ this.navSelectorState.onArrowDown();
72
+ setTimeout(() => this.el.nativeElement.querySelector('[role="treeitem"][tabindex="0"]').focus());
73
+ }
74
+ }
75
+ onArrowUp($event) {
76
+ if (this.el.nativeElement.querySelector('.nav-selector__content').contains(document.activeElement)) {
77
+ $event.stopImmediatePropagation();
78
+ this.navSelectorState.onArrowUp();
79
+ setTimeout(() => this.el.nativeElement.querySelector('[role="treeitem"][tabindex="0"]').focus());
80
+ }
81
+ }
82
+ TemplateRef = TemplateRef;
83
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorComponent, deps: [{ token: i1.NavSelectorState }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
84
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.9", type: NavSelectorComponent, isStandalone: true, selector: "ap-nav-selector", inputs: { navSelectorEntries: { classPropertyName: "navSelectorEntries", publicName: "navSelectorEntries", isSignal: true, isRequired: true, transformFunction: null }, translatedTexts: { classPropertyName: "translatedTexts", publicName: "translatedTexts", isSignal: true, isRequired: true, transformFunction: null }, multipleModeEnabled: { classPropertyName: "multipleModeEnabled", publicName: "multipleModeEnabled", isSignal: true, isRequired: false, transformFunction: null }, detailsDisplayedLimit: { classPropertyName: "detailsDisplayedLimit", publicName: "detailsDisplayedLimit", isSignal: true, isRequired: false, transformFunction: null }, forceExpanded: { classPropertyName: "forceExpanded", publicName: "forceExpanded", isSignal: true, isRequired: false, transformFunction: null }, selectedEntryUids: { classPropertyName: "selectedEntryUids", publicName: "selectedEntryUids", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { selectedEntryUids: "selectedEntryUidsChange" }, host: { attributes: { "role": "tree" }, listeners: { "keydown.arrowDown": "onArrowDown($event)", "keydown.arrowUp": "onArrowUp($event)" }, properties: { "class.minified": "!navSelectorState.expanded()" } }, providers: [NavSelectorState, withSymbols(apArrowExpand, apArrowReduce, apSearch)], queries: [{ propertyName: "headerProjection", first: true, predicate: ["header"], descendants: true, isSignal: true }, { propertyName: "footerProjection", first: true, predicate: ["footer"], descendants: true, isSignal: true }, { propertyName: "leafActionProjection", predicate: ["leafAction"], isSignal: true }], ngImport: i0, template: "<nav [@expand]=\"expansionState()\">\n @let headerProjectionNotNull = headerProjection();\n @let footerProjectionNotNull = footerProjection();\n\n @if (navSelectorState.expandedAfterDelay()) {\n <div class=\"nav-selector__header\">\n <span class=\"h3\">{{ translatedTexts().title }}</span>\n\n <button\n type=\"button\"\n class=\"expand-button expanded\"\n (click)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol\n size=\"sm\"\n symbolId=\"arrow-reduce\" />\n </button>\n\n\n @if (headerProjectionNotNull) {\n <ng-container\n [ngTemplateOutlet]=\"headerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n }\n\n <ap-input-search\n [id]=\"componentUid + '_search'\"\n class=\"nav-selector-search\"\n [placeholder]=\"translatedTexts().searchPlaceholder\"\n [ngModel]=\"navSelectorState.search()\"\n (ngModelChange)=\"navSelectorState.search.set($event ?? '')\" />\n </div>\n\n <div\n class=\"nav-selector__content\"\n role=\"tree\">\n @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n @if (!entry.hidden) {\n <div class=\"entry\">\n @if (entry.type === 'LEAF') {\n <ap-nav-selector-leaf [leaf]=\"entry\" />\n } @else if (entry.type === 'GROUP') {\n <ap-nav-selector-group [group]=\"entry\" />\n } @else if (entry.type === 'CATEGORY') {\n <ap-nav-selector-category [category]=\"$any(entry)\" />\n }\n </div>\n }\n }\n @if (navSelectorState.noResults()) {\n <div class=\"no-result\">{{ translatedTexts().noResults }}</div>\n }\n </div>\n\n @if (displayFooter() && footerProjectionNotNull) {\n <div class=\"nav-selector__footer\">\n <ng-container\n [ngTemplateOutlet]=\"footerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n </div>\n }\n } @else {\n <div class=\"nav-selector__header\">\n <button\n type=\"button\"\n class=\"expand-button\"\n (click)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol\n size=\"sm\"\n symbolId=\"arrow-expand\" />\n </button>\n @if (headerProjectionNotNull) {\n <ng-container\n [ngTemplateOutlet]=\"headerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n }\n\n <ap-icon-button (onClick)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol symbolId=\"search\" />\n </ap-icon-button>\n </div>\n\n <div class=\"nav-selector__content\">\n @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n @if (!entry.hidden) {\n <div class=\"entry\">\n @if (entry.type === 'LEAF') {\n <ap-nav-selector-leaf [leaf]=\"entry\" />\n } @else if (entry.type === 'GROUP') {\n <ap-nav-selector-group [group]=\"entry\" />\n } @else if (entry.type === 'CATEGORY') {\n <ap-nav-selector-category [category]=\"entry\" />\n }\n </div>\n }\n }\n </div>\n\n @if (displayFooter() && footerProjectionNotNull) {\n <div class=\"nav-selector__footer\">\n <ng-container\n [ngTemplateOutlet]=\"footerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n </div>\n }\n }\n</nav>\n", styles: [":host{display:flex;height:100%;flex-direction:column;align-items:flex-start;flex-shrink:0;width:224px;max-width:224px}@media only screen and (min-device-width: 1400px){:host{width:100%;max-width:250px}}:host :hover .expand-button.expanded{animation-name:translateExpandButton}:host nav{display:flex;height:100%;width:100%;flex-direction:column;align-items:flex-start;flex-shrink:0;overflow:hidden;border-right:1px solid var(--ref-color-grey-10);background:var(--ref-color-white)}:host .nav-selector__header{position:relative;display:flex;padding:var(--ref-spacing-xs) var(--ref-spacing-sm);flex-direction:column;align-items:flex-start;gap:8px;align-self:stretch;border-bottom:1px solid var(--ref-color-grey-10);background:var(--ref-color-white)}:host .nav-selector__header .h3{color:var(--ref-color-grey-100);font-family:Averta;font-size:var(--sys-text-h3-size);font-style:normal;font-weight:var(--ref-font-weight-bold);line-height:var(--ref-font-line-height-lg)}:host .nav-selector__header .expand-button-container{position:absolute;right:0;top:var(--ref-spacing-xs)}:host .nav-selector__header .expand-button{animation-duration:70ms;animation-timing-function:cubic-bezier(0,0,.2,1);animation-fill-mode:forwards;display:flex;width:24px;height:24px;justify-content:center;align-items:center;background:none;border:1px solid var(--ref-color-grey-20);border-top-left-radius:var(--ref-border-radius-sm);border-bottom-left-radius:var(--ref-border-radius-sm);cursor:pointer}@keyframes translateExpandButton{0%{transform:translate(100%)}to{transform:translate(0)}}:host .nav-selector__header .expand-button.expanded{position:absolute;right:0;top:12px;transform:translate(100%)}:host .nav-selector__header .expand-button ap-symbol[symbol-id=arrow-reduce],:host .nav-selector__header .expand-button ap-symbol[symbol-id=arrow-expand]{color:var(--ref-color-grey-80)}:host .nav-selector__header .expand-button:focus{border-radius:var(--ref-border-radius-sm) 0px 0px var(--ref-border-radius-sm);border-top:1px solid var(--ref-color-grey-20);border-bottom:1px solid var(--ref-color-grey-20);border-left:1px solid var(--ref-color-grey-20);background:var(--ref-color-grey-10);box-shadow:0 0 0 1px #fff,0 0 0 3px #178dfe}:host .nav-selector__header .expand-button:focus.expanded{animation-name:translateExpandButton}:host .nav-selector__header .expand-button:hover{background-color:var(--ref-color-grey-10)}:host .nav-selector__header .expand-button:active{background-color:var(--ref-color-grey-20)}:host .nav-selector__header .nav-selector-search{width:100%}:host .nav-selector__content{display:flex;flex-direction:column;align-items:flex-start;flex:1 0 0;align-self:stretch;overflow:auto}:host .nav-selector__content .entry{padding:var(--ref-spacing-xxs);align-self:stretch;display:flex;flex-direction:column}:host .nav-selector__content .entry+.entry{border-top:1px solid var(--sys-border-color-default)}:host .nav-selector__content .no-result{display:flex;padding:var(--ref-spacing-sm);flex-direction:column;align-items:flex-start;flex:1 0 0;align-self:stretch;color:var(--ref-color-grey-80);font-family:Averta;font-size:var(--ref-font-size-sm);font-style:italic;font-weight:400;line-height:var(--ref-font-line-height-sm)}:host .nav-selector__footer{display:flex;padding:var(--ref-spacing-xs);flex-direction:column;align-items:flex-start;gap:var(--ref-spacing-xxs);align-self:stretch;border-top:1px solid var(--sys-border-color-default);background:var(--ref-color-white)}:host.minified nav{width:64px}:host.minified nav .nav-selector__header{align-items:center}:host.minified nav .nav-selector__content .entry{padding:var(--ref-spacing-xxxs)}:host.minified nav .nav-selector__footer{padding:var(--ref-spacing-xs) var(--ref-spacing-xxxs) var(--ref-spacing-xs) var(--ref-spacing-xxxs)}\n"], dependencies: [{ kind: "component", type: NavSelectorLeafComponent, selector: "ap-nav-selector-leaf", inputs: ["leaf"] }, { kind: "component", type: NavSelectorGroupComponent, selector: "ap-nav-selector-group", inputs: ["group"] }, { kind: "component", type: NavSelectorCategoryComponent, selector: "ap-nav-selector-category", inputs: ["category"] }, { kind: "component", type: InputSearchComponent, selector: "ap-input-search", inputs: ["id", "placeholder", "clearable"], outputs: ["focus", "blur", "keyup"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: SymbolComponent, selector: "ap-symbol", inputs: ["symbolId", "color", "size"], outputs: ["sizeChange"] }, { kind: "component", type: IconButtonComponent, selector: "ap-icon-button", inputs: ["ariaLabel", "name", "color", "disabled", "menuTrigger", "locked", "loading", "type"], outputs: ["onClick", "onFocus", "onBlur", "menuOpened", "menuClosed"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: EventPluginsModule }], animations: [
85
+ trigger('expand', [
86
+ state('expanded', style({ width: '100%' })),
87
+ state('minified', style({ with: '64px' })),
88
+ transition('expanded => minified', animate('250ms cubic-bezier(.4, 0, .3, 1)')),
89
+ transition('minified => expanded', animate('250ms cubic-bezier(.4, 0, .3, 1)',
90
+ // Force animation to do not be with 100% instantly
91
+ keyframes([style({ width: '64px' }), style({ width: '50%' }), style({ width: '100%' })]))),
92
+ ]),
93
+ ], changeDetection: i0.ChangeDetectionStrategy.OnPush });
94
+ }
95
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorComponent, decorators: [{
96
+ type: Component,
97
+ args: [{ selector: 'ap-nav-selector', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [
98
+ NavSelectorLeafComponent,
99
+ NavSelectorGroupComponent,
100
+ NavSelectorCategoryComponent,
101
+ InputSearchComponent,
102
+ FormsModule,
103
+ SymbolComponent,
104
+ IconButtonComponent,
105
+ NgTemplateOutlet,
106
+ EventPluginsModule,
107
+ ], providers: [NavSelectorState, withSymbols(apArrowExpand, apArrowReduce, apSearch)], host: {
108
+ '[class.minified]': '!navSelectorState.expanded()',
109
+ role: 'tree',
110
+ '(keydown.arrowDown)': 'onArrowDown($event)',
111
+ '(keydown.arrowUp)': 'onArrowUp($event)',
112
+ }, animations: [
113
+ trigger('expand', [
114
+ state('expanded', style({ width: '100%' })),
115
+ state('minified', style({ with: '64px' })),
116
+ transition('expanded => minified', animate('250ms cubic-bezier(.4, 0, .3, 1)')),
117
+ transition('minified => expanded', animate('250ms cubic-bezier(.4, 0, .3, 1)',
118
+ // Force animation to do not be with 100% instantly
119
+ keyframes([style({ width: '64px' }), style({ width: '50%' }), style({ width: '100%' })]))),
120
+ ]),
121
+ ], template: "<nav [@expand]=\"expansionState()\">\n @let headerProjectionNotNull = headerProjection();\n @let footerProjectionNotNull = footerProjection();\n\n @if (navSelectorState.expandedAfterDelay()) {\n <div class=\"nav-selector__header\">\n <span class=\"h3\">{{ translatedTexts().title }}</span>\n\n <button\n type=\"button\"\n class=\"expand-button expanded\"\n (click)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol\n size=\"sm\"\n symbolId=\"arrow-reduce\" />\n </button>\n\n\n @if (headerProjectionNotNull) {\n <ng-container\n [ngTemplateOutlet]=\"headerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n }\n\n <ap-input-search\n [id]=\"componentUid + '_search'\"\n class=\"nav-selector-search\"\n [placeholder]=\"translatedTexts().searchPlaceholder\"\n [ngModel]=\"navSelectorState.search()\"\n (ngModelChange)=\"navSelectorState.search.set($event ?? '')\" />\n </div>\n\n <div\n class=\"nav-selector__content\"\n role=\"tree\">\n @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n @if (!entry.hidden) {\n <div class=\"entry\">\n @if (entry.type === 'LEAF') {\n <ap-nav-selector-leaf [leaf]=\"entry\" />\n } @else if (entry.type === 'GROUP') {\n <ap-nav-selector-group [group]=\"entry\" />\n } @else if (entry.type === 'CATEGORY') {\n <ap-nav-selector-category [category]=\"$any(entry)\" />\n }\n </div>\n }\n }\n @if (navSelectorState.noResults()) {\n <div class=\"no-result\">{{ translatedTexts().noResults }}</div>\n }\n </div>\n\n @if (displayFooter() && footerProjectionNotNull) {\n <div class=\"nav-selector__footer\">\n <ng-container\n [ngTemplateOutlet]=\"footerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n </div>\n }\n } @else {\n <div class=\"nav-selector__header\">\n <button\n type=\"button\"\n class=\"expand-button\"\n (click)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol\n size=\"sm\"\n symbolId=\"arrow-expand\" />\n </button>\n @if (headerProjectionNotNull) {\n <ng-container\n [ngTemplateOutlet]=\"headerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n }\n\n <ap-icon-button (onClick)=\"navSelectorState.toggleExpanded()\">\n <ap-symbol symbolId=\"search\" />\n </ap-icon-button>\n </div>\n\n <div class=\"nav-selector__content\">\n @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n @if (!entry.hidden) {\n <div class=\"entry\">\n @if (entry.type === 'LEAF') {\n <ap-nav-selector-leaf [leaf]=\"entry\" />\n } @else if (entry.type === 'GROUP') {\n <ap-nav-selector-group [group]=\"entry\" />\n } @else if (entry.type === 'CATEGORY') {\n <ap-nav-selector-category [category]=\"entry\" />\n }\n </div>\n }\n }\n </div>\n\n @if (displayFooter() && footerProjectionNotNull) {\n <div class=\"nav-selector__footer\">\n <ng-container\n [ngTemplateOutlet]=\"footerProjectionNotNull\"\n [ngTemplateOutletContext]=\"{\n expanded: navSelectorState.expanded()\n }\" />\n </div>\n }\n }\n</nav>\n", styles: [":host{display:flex;height:100%;flex-direction:column;align-items:flex-start;flex-shrink:0;width:224px;max-width:224px}@media only screen and (min-device-width: 1400px){:host{width:100%;max-width:250px}}:host :hover .expand-button.expanded{animation-name:translateExpandButton}:host nav{display:flex;height:100%;width:100%;flex-direction:column;align-items:flex-start;flex-shrink:0;overflow:hidden;border-right:1px solid var(--ref-color-grey-10);background:var(--ref-color-white)}:host .nav-selector__header{position:relative;display:flex;padding:var(--ref-spacing-xs) var(--ref-spacing-sm);flex-direction:column;align-items:flex-start;gap:8px;align-self:stretch;border-bottom:1px solid var(--ref-color-grey-10);background:var(--ref-color-white)}:host .nav-selector__header .h3{color:var(--ref-color-grey-100);font-family:Averta;font-size:var(--sys-text-h3-size);font-style:normal;font-weight:var(--ref-font-weight-bold);line-height:var(--ref-font-line-height-lg)}:host .nav-selector__header .expand-button-container{position:absolute;right:0;top:var(--ref-spacing-xs)}:host .nav-selector__header .expand-button{animation-duration:70ms;animation-timing-function:cubic-bezier(0,0,.2,1);animation-fill-mode:forwards;display:flex;width:24px;height:24px;justify-content:center;align-items:center;background:none;border:1px solid var(--ref-color-grey-20);border-top-left-radius:var(--ref-border-radius-sm);border-bottom-left-radius:var(--ref-border-radius-sm);cursor:pointer}@keyframes translateExpandButton{0%{transform:translate(100%)}to{transform:translate(0)}}:host .nav-selector__header .expand-button.expanded{position:absolute;right:0;top:12px;transform:translate(100%)}:host .nav-selector__header .expand-button ap-symbol[symbol-id=arrow-reduce],:host .nav-selector__header .expand-button ap-symbol[symbol-id=arrow-expand]{color:var(--ref-color-grey-80)}:host .nav-selector__header .expand-button:focus{border-radius:var(--ref-border-radius-sm) 0px 0px var(--ref-border-radius-sm);border-top:1px solid var(--ref-color-grey-20);border-bottom:1px solid var(--ref-color-grey-20);border-left:1px solid var(--ref-color-grey-20);background:var(--ref-color-grey-10);box-shadow:0 0 0 1px #fff,0 0 0 3px #178dfe}:host .nav-selector__header .expand-button:focus.expanded{animation-name:translateExpandButton}:host .nav-selector__header .expand-button:hover{background-color:var(--ref-color-grey-10)}:host .nav-selector__header .expand-button:active{background-color:var(--ref-color-grey-20)}:host .nav-selector__header .nav-selector-search{width:100%}:host .nav-selector__content{display:flex;flex-direction:column;align-items:flex-start;flex:1 0 0;align-self:stretch;overflow:auto}:host .nav-selector__content .entry{padding:var(--ref-spacing-xxs);align-self:stretch;display:flex;flex-direction:column}:host .nav-selector__content .entry+.entry{border-top:1px solid var(--sys-border-color-default)}:host .nav-selector__content .no-result{display:flex;padding:var(--ref-spacing-sm);flex-direction:column;align-items:flex-start;flex:1 0 0;align-self:stretch;color:var(--ref-color-grey-80);font-family:Averta;font-size:var(--ref-font-size-sm);font-style:italic;font-weight:400;line-height:var(--ref-font-line-height-sm)}:host .nav-selector__footer{display:flex;padding:var(--ref-spacing-xs);flex-direction:column;align-items:flex-start;gap:var(--ref-spacing-xxs);align-self:stretch;border-top:1px solid var(--sys-border-color-default);background:var(--ref-color-white)}:host.minified nav{width:64px}:host.minified nav .nav-selector__header{align-items:center}:host.minified nav .nav-selector__content .entry{padding:var(--ref-spacing-xxxs)}:host.minified nav .nav-selector__footer{padding:var(--ref-spacing-xs) var(--ref-spacing-xxxs) var(--ref-spacing-xs) var(--ref-spacing-xxxs)}\n"] }]
122
+ }], ctorParameters: () => [{ type: i1.NavSelectorState }, { type: i0.ElementRef }] });
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nav-selector.component.js","sourceRoot":"","sources":["../../../../libs/ui-components/nav-selector/src/nav-selector.component.ts","../../../../libs/ui-components/nav-selector/src/nav-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import { IconButtonComponent } from '@agorapulse/ui-components/icon-button';\nimport { InputSearchComponent } from '@agorapulse/ui-components/input-search';\nimport { apArrowExpand, apArrowReduce, apSearch, SymbolComponent, withSymbols } from '@agorapulse/ui-symbol';\nimport { animate, keyframes, state, style, transition, trigger } from '@angular/animations';\nimport { NgTemplateOutlet } from '@angular/common';\nimport {\n    afterNextRender,\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    contentChild,\n    contentChildren,\n    effect,\n    ElementRef,\n    input,\n    model,\n    TemplateRef,\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { EventPluginsModule } from '@tinkoff/ng-event-plugins';\nimport { NavSelectorEntry, NavSelectorTranslatedTexts } from './nav-selector';\nimport { NavSelectorCategoryComponent } from './nav-selector-category/nav-selector-category.component';\nimport { NavSelectorGroupComponent } from './nav-selector-group/nav-selector-group.component';\nimport { NavSelectorLeafComponent } from './nav-selector-leaf/nav-selector-leaf.component';\nimport { NavSelectorState } from './nav-selector.state';\n\nlet nextUniqueId = 0;\n\n@Component({\n    selector: 'ap-nav-selector',\n    templateUrl: './nav-selector.component.html',\n    styleUrls: ['./nav-selector.component.scss'],\n    standalone: true,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    imports: [\n        NavSelectorLeafComponent,\n        NavSelectorGroupComponent,\n        NavSelectorCategoryComponent,\n        InputSearchComponent,\n        FormsModule,\n        SymbolComponent,\n        IconButtonComponent,\n        NgTemplateOutlet,\n        EventPluginsModule,\n    ],\n    providers: [NavSelectorState, withSymbols(apArrowExpand, apArrowReduce, apSearch)],\n    host: {\n        '[class.minified]': '!navSelectorState.expanded()',\n        role: 'tree',\n        '(keydown.arrowDown)': 'onArrowDown($event)',\n        '(keydown.arrowUp)': 'onArrowUp($event)',\n    },\n    animations: [\n        trigger('expand', [\n            state('expanded', style({ width: '100%' })),\n            state('minified', style({ with: '64px' })),\n            transition('expanded => minified', animate('250ms cubic-bezier(.4, 0, .3, 1)')),\n            transition(\n                'minified => expanded',\n                animate(\n                    '250ms cubic-bezier(.4, 0, .3, 1)',\n                    // Force animation to do not be with 100% instantly\n                    keyframes([style({ width: '64px' }), style({ width: '50%' }), style({ width: '100%' })])\n                )\n            ),\n        ]),\n    ],\n})\nexport class NavSelectorComponent {\n    navSelectorEntries = input.required<NavSelectorEntry[]>();\n    translatedTexts = input.required<NavSelectorTranslatedTexts>();\n    multipleModeEnabled = input<boolean>(false);\n\n    detailsDisplayedLimit = input<number>(Number.MAX_SAFE_INTEGER);\n\n    /**\n     * If true, the nav selector will be expanded by default. Even if the width of the screen is smaller than 1280px.\n     */\n    forceExpanded = input<boolean>(false);\n\n    selectedEntryUids = model.required<string[]>();\n\n    headerProjection = contentChild<TemplateRef<unknown>>('header');\n    footerProjection = contentChild<TemplateRef<unknown>>('footer');\n\n    displayFooter = computed(() => this.footerProjection() !== undefined);\n\n    leafActionProjection = contentChildren<TemplateRef<unknown>>('leafAction');\n\n    expansionState = computed(() => (this.navSelectorState.expanded() ? 'expanded' : 'minified'));\n\n    componentUid = `nav-selector-${nextUniqueId++}`;\n\n    constructor(\n        public navSelectorState: NavSelectorState,\n        private el: ElementRef\n    ) {\n        navSelectorState.registerOnSelectedUidsChange(selectedUids => {\n            this.selectedEntryUids.set(selectedUids);\n        });\n\n        afterNextRender(() => {\n            if (!this.forceExpanded() && document.body.clientWidth < 1200) {\n                this.navSelectorState.toggleExpanded();\n            }\n        });\n\n        effect(\n            () => {\n                this.navSelectorState.updateMultiModeEnabled(this.multipleModeEnabled());\n                if (!this.navSelectorState.filteredEntries().length) {\n                    this.navSelectorState.updateEntries(this.navSelectorEntries(), this.selectedEntryUids(), this.detailsDisplayedLimit());\n                }\n            },\n            { allowSignalWrites: true }\n        );\n\n        effect(\n            () => {\n                this.navSelectorState.updateDetailsDisplayedLimit(this.detailsDisplayedLimit());\n            },\n            { allowSignalWrites: true }\n        );\n\n        const leafActionProjectionEffect = effect(\n            () => {\n                if (this.leafActionProjection()) {\n                    this.navSelectorState.updateLeafAction(this.leafActionProjection());\n                    leafActionProjectionEffect.destroy();\n                }\n            },\n            { manualCleanup: true, allowSignalWrites: true }\n        );\n\n        effect(\n            () => {\n                this.navSelectorState.onSelectionChange(this.selectedEntryUids());\n            },\n            { allowSignalWrites: true }\n        );\n        effect(\n            () => {\n                this.navSelectorState.updateTexts(this.translatedTexts());\n            },\n            { allowSignalWrites: true }\n        );\n    }\n\n    onArrowDown($event: Event) {\n        if (this.el.nativeElement.querySelector('.nav-selector__content').contains(document.activeElement)) {\n            $event.stopImmediatePropagation();\n            this.navSelectorState.onArrowDown();\n            setTimeout(() => this.el.nativeElement.querySelector('[role=\"treeitem\"][tabindex=\"0\"]').focus());\n        }\n    }\n\n    onArrowUp($event: Event) {\n        if (this.el.nativeElement.querySelector('.nav-selector__content').contains(document.activeElement)) {\n            $event.stopImmediatePropagation();\n            this.navSelectorState.onArrowUp();\n            setTimeout(() => this.el.nativeElement.querySelector('[role=\"treeitem\"][tabindex=\"0\"]').focus());\n        }\n    }\n\n    protected readonly TemplateRef = TemplateRef;\n}\n","<nav [@expand]=\"expansionState()\">\n    @let headerProjectionNotNull = headerProjection();\n    @let footerProjectionNotNull = footerProjection();\n\n    @if (navSelectorState.expandedAfterDelay()) {\n        <div class=\"nav-selector__header\">\n            <span class=\"h3\">{{ translatedTexts().title }}</span>\n\n            <button\n                type=\"button\"\n                class=\"expand-button expanded\"\n                (click)=\"navSelectorState.toggleExpanded()\">\n                <ap-symbol\n                    size=\"sm\"\n                    symbolId=\"arrow-reduce\" />\n            </button>\n\n\n            @if (headerProjectionNotNull) {\n                <ng-container\n                    [ngTemplateOutlet]=\"headerProjectionNotNull\"\n                    [ngTemplateOutletContext]=\"{\n                        expanded: navSelectorState.expanded()\n                    }\" />\n            }\n\n            <ap-input-search\n                [id]=\"componentUid + '_search'\"\n                class=\"nav-selector-search\"\n                [placeholder]=\"translatedTexts().searchPlaceholder\"\n                [ngModel]=\"navSelectorState.search()\"\n                (ngModelChange)=\"navSelectorState.search.set($event ?? '')\" />\n        </div>\n\n        <div\n            class=\"nav-selector__content\"\n            role=\"tree\">\n            @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n                @if (!entry.hidden) {\n                    <div class=\"entry\">\n                        @if (entry.type === 'LEAF') {\n                            <ap-nav-selector-leaf [leaf]=\"entry\" />\n                        } @else if (entry.type === 'GROUP') {\n                            <ap-nav-selector-group [group]=\"entry\" />\n                        } @else if (entry.type === 'CATEGORY') {\n                            <ap-nav-selector-category [category]=\"$any(entry)\" />\n                        }\n                    </div>\n                }\n            }\n            @if (navSelectorState.noResults()) {\n                <div class=\"no-result\">{{ translatedTexts().noResults }}</div>\n            }\n        </div>\n\n        @if (displayFooter() && footerProjectionNotNull) {\n            <div class=\"nav-selector__footer\">\n                <ng-container\n                    [ngTemplateOutlet]=\"footerProjectionNotNull\"\n                    [ngTemplateOutletContext]=\"{\n                        expanded: navSelectorState.expanded()\n                    }\" />\n            </div>\n        }\n    } @else {\n        <div class=\"nav-selector__header\">\n            <button\n                type=\"button\"\n                class=\"expand-button\"\n                (click)=\"navSelectorState.toggleExpanded()\">\n                <ap-symbol\n                    size=\"sm\"\n                    symbolId=\"arrow-expand\" />\n            </button>\n            @if (headerProjectionNotNull) {\n                <ng-container\n                    [ngTemplateOutlet]=\"headerProjectionNotNull\"\n                    [ngTemplateOutletContext]=\"{\n                        expanded: navSelectorState.expanded()\n                    }\" />\n            }\n\n            <ap-icon-button (onClick)=\"navSelectorState.toggleExpanded()\">\n                <ap-symbol symbolId=\"search\" />\n            </ap-icon-button>\n        </div>\n\n        <div class=\"nav-selector__content\">\n            @for (entry of navSelectorState.filteredEntries(); track entry.uid) {\n                @if (!entry.hidden) {\n                    <div class=\"entry\">\n                        @if (entry.type === 'LEAF') {\n                            <ap-nav-selector-leaf [leaf]=\"entry\" />\n                        } @else if (entry.type === 'GROUP') {\n                            <ap-nav-selector-group [group]=\"entry\" />\n                        } @else if (entry.type === 'CATEGORY') {\n                            <ap-nav-selector-category [category]=\"entry\" />\n                        }\n                    </div>\n                }\n            }\n        </div>\n\n        @if (displayFooter() && footerProjectionNotNull) {\n            <div class=\"nav-selector__footer\">\n                <ng-container\n                    [ngTemplateOutlet]=\"footerProjectionNotNull\"\n                    [ngTemplateOutletContext]=\"{\n                        expanded: navSelectorState.expanded()\n                    }\" />\n            </div>\n        }\n    }\n</nav>\n"]}
@@ -0,0 +1,9 @@
1
+ export const isNavSelectorEntryALeaf = (entry) => entry.type === 'LEAF';
2
+ export const isNavSelectorEntryAGroup = (entry) => entry.type === 'GROUP';
3
+ export const isNavSelectorEntryACategory = (entry) => entry.type === 'CATEGORY';
4
+ export const isInternalNavSelectorEntryALeaf = (entry) => entry.type === 'LEAF';
5
+ export const isInternalNavSelectorEntryANode = (entry) => `children` in entry;
6
+ export const isInternalNavSelectorEntryACategory = (entry) => entry.type === 'CATEGORY';
7
+ export const isInternalNavSelectorEntryAGroup = (entry) => entry.type === 'GROUP';
8
+ export const isInternalNavSelectorEntryALeafDetails = (entry) => entry.type === 'LEAF_DETAILS';
9
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nav-selector.js","sourceRoot":"","sources":["../../../../libs/ui-components/nav-selector/src/nav-selector.ts"],"names":[],"mappings":"AA8FA,MAAM,CAAC,MAAM,uBAAuB,GAAG,CAAC,KAAuB,EAA4B,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,MAAM,CAAC;AAEpH,MAAM,CAAC,MAAM,wBAAwB,GAAG,CAAC,KAAuB,EAA6B,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,OAAO,CAAC;AAEvH,MAAM,CAAC,MAAM,2BAA2B,GAAG,CAAC,KAAuB,EAAgC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,UAAU,CAAC;AAwIhI,MAAM,CAAC,MAAM,+BAA+B,GAAG,CAAC,KAA+B,EAAoC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,MAAM,CAAC;AAE5I,MAAM,CAAC,MAAM,+BAA+B,GAAG,CAC3C,KAA+B,EACgC,EAAE,CAAC,UAAU,IAAI,KAAK,CAAC;AAE1F,MAAM,CAAC,MAAM,mCAAmC,GAAG,CAAC,KAA+B,EAAwC,EAAE,CACzH,KAAK,CAAC,IAAI,KAAK,UAAU,CAAC;AAE9B,MAAM,CAAC,MAAM,gCAAgC,GAAG,CAAC,KAA+B,EAAqC,EAAE,CACnH,KAAK,CAAC,IAAI,KAAK,OAAO,CAAC;AAE3B,MAAM,CAAC,MAAM,sCAAsC,GAAG,CAAC,KAA+B,EAA2C,EAAE,CAC/H,KAAK,CAAC,IAAI,KAAK,cAAc,CAAC","sourcesContent":["import { AvatarNetwork } from '@agorapulse/ui-components/avatar';\n\n/**\n * As translation is not available in the DS, this all translations needed\n */\nexport interface NavSelectorTranslatedTexts {\n    /** Title of the nav selector */\n    title: string;\n    /** Only text in multi select mode */\n    only: string;\n    /** Toolip displaying that token is invalid */\n    tokenInvalid: string;\n    /** Toolip displaying that feature is locked */\n    featureLocked: string;\n    /** placeholder for the search input */\n    searchPlaceholder: string;\n    /** Text displayed when the search return no results */\n    noResults: string;\n    /** Text of the button to show more details, it’s used when we limit the number of details displayed  */\n    viewMore: string;\n    /** Text of the button to show less details, it’s used when we limit the number of details displayed  */\n    viewLess: string;\n}\n\n/**\n * Common interface for the NavSelector\n */\nexport type NavSelectorBaseNode = {\n    /** Unique identifier of the node */\n    uid: string;\n    /** Displayed name of the node */\n    alias: string;\n};\n\n/**\n * Represent final entry of the nav selector\n */\nexport type NavSelectorLeaf = {\n    pictureUrl: string | null;\n    /** Counter to display on the entry, to ignore if the leaf has details */\n    counter: number | null;\n    /** When true, the leaf will be hidden */\n    missingPermission: boolean;\n    /** For social profiles or ad accounts the corresponding network */\n    network: AvatarNetwork | null;\n    /** Translated text when the entry is disabled */\n    disableReason: string | null;\n    /** Indicates if entry has its token invalid */\n    tokenInvalid: boolean;\n    /** Indicates if entry is locked */\n    featureLocked: boolean;\n    /** Sub elements of the leaf, they are optional */\n    details: NavSelectorLeafDetails[];\n    type: 'LEAF';\n} & NavSelectorBaseNode;\n\n/**\n * Represent a group of entries which has been created by the user (ex: menuGroup)\n */\nexport type NavSelectorGroup = {\n    /** A group can only have leaves as children */\n    children: NavSelectorLeaf[];\n    /** Indicates if the group should be folded or expanded when generating the nav selector */\n    folded: boolean;\n    type: 'GROUP';\n} & NavSelectorBaseNode;\n\n/**\n * Represent a category of entries which has been created by the product to separate different kind of entries (ex: social profiles, reports, ad accounts…)\n */\nexport type NavSelectorCategory = {\n    /** A Category can have leaves or groups as children, you can have leaves alongside with groups */\n    children: (NavSelectorLeaf | NavSelectorGroup)[];\n    /** Indicates if the group should be folded or expanded when generating the nav selector */\n    folded: boolean;\n    type: 'CATEGORY';\n} & NavSelectorBaseNode;\n\nexport type NavSelectorEntry = NavSelectorLeaf | NavSelectorGroup | NavSelectorCategory;\n\n/**\n * Represent details of a leaf, they are used to allow to display more specific information about the leaf\n */\nexport interface NavSelectorLeafDetails {\n    /** Unique identifier of the node */\n    uid: string;\n    /** Counter to display on the entry */\n    counter: number | null;\n    /** Displayed name of the node */\n    name: string;\n    /** Translation of the error on the leaf */\n    errorReason: string | null;\n}\n\nexport const isNavSelectorEntryALeaf = (entry: NavSelectorEntry): entry is NavSelectorLeaf => entry.type === 'LEAF';\n\nexport const isNavSelectorEntryAGroup = (entry: NavSelectorEntry): entry is NavSelectorGroup => entry.type === 'GROUP';\n\nexport const isNavSelectorEntryACategory = (entry: NavSelectorEntry): entry is NavSelectorCategory => entry.type === 'CATEGORY';\n\n/**\n * Represent the accessibility on a node\n */\nexport type NodeAccessibility = {\n    /** Number of sibling elements, itself included. Ex: there are 7 social profiles display in the group */\n    ariaSetsize: number;\n    /** Index starting from 1 of the element alongside its sibling */\n    ariaPosinset: number;\n    /** 0 to be focusable by Tab key and -1 to be focusable by keyboard with JS */\n    tabIndex: number;\n    /** In a tree, the deep level, starting from 1 */\n    ariaLevel: number;\n};\n\n/**\n * Represent a node\n */\nexport type InternalNavSelectorBaseNode = {\n    /** Unique identifier of the entry */\n    uid: string;\n    /** Displayed name of the entry */\n    alias: string;\n    accessibility: NodeAccessibility;\n    /** When hidden, the entry is not displayed, used when searching an element */\n    hidden: boolean;\n    /** When true, you can navigate with arrow keys on this entry */\n    focusable: boolean;\n};\n\n/**\n * Represent a group of entries which has been created by the user (ex: menuGroup)\n */\nexport type InternalNavSelectorGroup = {\n    /** A group can only have leaves as children */\n    children: InternalNavSelectorLeaf[];\n    /** Picture url of the group, used when the nav selector is minified */\n    childrenPictureUrlSample: { url: string | null; initial: string }[];\n    /** Indicates whenever the group is folded or not */\n    folded: boolean;\n    /** Indicates the counter value to display when the group is folded */\n    counter: number;\n    /** Indicates if the counter should be display when the group is folded */\n    displayCounter: boolean;\n    /** Indicates if the token invalid icon should be display when the group is folded */\n    displayTokenInvalid: boolean;\n    /** Indicates in multi select mode if the group has some children selected */\n    undeterminedSelection: boolean;\n    /** Indicates if the group is selected */\n    selected: boolean;\n    /** Indicates if the group is selectable */\n    selectable: boolean;\n    type: 'GROUP';\n} & InternalNavSelectorBaseNode;\n\n/**\n * Represent a category of entries which has been created by the product to separate different kind of entries (ex: social profiles, reports, ad accounts…)\n */\nexport type InternalNavSelectorCategory = {\n    /** A Category can have leaves or groups as children, you can have leaves alongside with groups */\n    children: (InternalNavSelectorGroup | InternalNavSelectorLeaf)[];\n    /** Indicates whenever the category is folded or not */\n    folded: boolean;\n    type: 'CATEGORY';\n} & InternalNavSelectorBaseNode;\n\n/**\n * Represent final entry of the nav selector\n */\nexport type InternalNavSelectorLeaf = {\n    pictureUrl: string | null;\n    /** Indicates if the counter could be display. The leaf should not be disabled, feature locked or has token invalid */\n    counterDisplayable: boolean;\n    /** Indicates if the token invalid icon should be display. So when displayable when no details or folded */\n    displayCounter: boolean;\n    /** Counter value to display */\n    counter: number;\n    /** Indicates if the token invalid icon should be display */\n    displayTokenInvalid: boolean;\n    /** Indicates if the feature is locked */\n    displayFeatureLocked: boolean;\n    /** Indicates if the leaf is disabled */\n    disabled: boolean;\n    /** The disabled reason to display in tooltip */\n    disableReason: string | null;\n    /** Indicates if the leaf is selected */\n    selected: boolean;\n    /** Indicates if the leaf is selectable */\n    selectable: boolean;\n    /** For social profiles or ad accounts, the corresponding network */\n    network: AvatarNetwork | null;\n    /** Indicates if the leaf foldable. So when it has details and no error status */\n    foldable: boolean;\n    /** Indicates if the leaf is folded */\n    folded: boolean;\n    /** Indicates if the leaf has details and they are displayable. So no error status */\n    detailsDisplayable: boolean;\n    /** Details of the leaf */\n    details: InternalNavSelectorLeafDetails[];\n    type: 'LEAF';\n    /** Indicates if the view more section should be display for details */\n    viewMoreDisplayable: boolean;\n    /** Indicates if the view more or view less button is displayed */\n    viewMoreDisplayed: boolean;\n    /** Number of details to display when the view more is not clicked */\n    viewMoreDetailsDisplayedLimit: number;\n} & InternalNavSelectorBaseNode;\n\n/**\n * Represent details of a leaf, they are used to allow to display more specific information about the leaf\n */\nexport type InternalNavSelectorLeafDetails = {\n    /** Unique identifier of the node */\n    uid: string;\n    /** Counter to display on the detail */\n    counter: number;\n    /** Indicates if the counter should be display. So no error on the detail */\n    displayCounter: boolean;\n    /** Displayed name of the node */\n    alias: string;\n    /** Indicates if the detail is selected */\n    selected: boolean;\n    /** Indicates if we should display an error icon */\n    displayError: boolean;\n    /** Translated text to display in the tooltip */\n    errorReason: string | null;\n    type: 'LEAF_DETAILS';\n} & InternalNavSelectorBaseNode;\n\nexport type InternalNavSelectorEntry =\n    | InternalNavSelectorLeaf\n    | InternalNavSelectorGroup\n    | InternalNavSelectorCategory\n    | InternalNavSelectorLeafDetails;\n\nexport const isInternalNavSelectorEntryALeaf = (entry: InternalNavSelectorEntry): entry is InternalNavSelectorLeaf => entry.type === 'LEAF';\n\nexport const isInternalNavSelectorEntryANode = (\n    entry: InternalNavSelectorEntry\n): entry is InternalNavSelectorCategory | InternalNavSelectorGroup => `children` in entry;\n\nexport const isInternalNavSelectorEntryACategory = (entry: InternalNavSelectorEntry): entry is InternalNavSelectorCategory =>\n    entry.type === 'CATEGORY';\n\nexport const isInternalNavSelectorEntryAGroup = (entry: InternalNavSelectorEntry): entry is InternalNavSelectorGroup =>\n    entry.type === 'GROUP';\n\nexport const isInternalNavSelectorEntryALeafDetails = (entry: InternalNavSelectorEntry): entry is InternalNavSelectorLeafDetails =>\n    entry.type === 'LEAF_DETAILS';\n"]}