@hestia-earth/ui-components 0.29.2 → 0.29.3

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 (52) hide show
  1. package/common/navigation-menu/navigation-menu.component.d.ts +1 -0
  2. package/common/semver-utils.d.ts +0 -1
  3. package/engine/engine-models-version-link/engine-models-version-link.component.d.ts +1 -1
  4. package/esm2022/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.mjs +3 -3
  5. package/esm2022/chart/chart.component.mjs +3 -3
  6. package/esm2022/common/blank-node-state/blank-node-state.component.mjs +3 -3
  7. package/esm2022/common/clipboard/clipboard.component.mjs +3 -3
  8. package/esm2022/common/collapsible-box/collapsible-box.component.mjs +3 -3
  9. package/esm2022/common/data-table/data-table.component.mjs +3 -3
  10. package/esm2022/common/drawer-container/drawer-container.component.mjs +3 -3
  11. package/esm2022/common/link-key-value/link-key-value.component.mjs +3 -3
  12. package/esm2022/common/maps-drawing-confirm/maps-drawing-confirm.component.mjs +3 -3
  13. package/esm2022/common/mobile-shell/mobile-shell.component.mjs +3 -3
  14. package/esm2022/common/navigation-menu/navigation-menu.component.mjs +7 -3
  15. package/esm2022/common/schema-version-link/schema-version-link.component.mjs +3 -3
  16. package/esm2022/common/search-extend/search-extend.component.mjs +3 -3
  17. package/esm2022/common/semver-utils.mjs +2 -3
  18. package/esm2022/common/shelf-dialog/shelf-dialog.component.mjs +3 -3
  19. package/esm2022/common/shell/shell.component.mjs +3 -3
  20. package/esm2022/common/sort-select/sort-select.component.mjs +3 -3
  21. package/esm2022/cycles/cycles-completeness/cycles-completeness.component.mjs +3 -3
  22. package/esm2022/cycles/cycles-nodes/cycles-nodes.component.mjs +3 -3
  23. package/esm2022/cycles/cycles-nodes-timeline/cycles-nodes-timeline.component.mjs +3 -3
  24. package/esm2022/engine/engine-models-stage-deep/engine-models-stage-deep.component.mjs +3 -3
  25. package/esm2022/engine/engine-models-version-link/engine-models-version-link.component.mjs +3 -3
  26. package/esm2022/engine/engine-requirements-form/engine-requirements-form.component.mjs +3 -3
  27. package/esm2022/files/files-error-summary/files-error-summary-item/files-error-summary-item.component.mjs +3 -3
  28. package/esm2022/files/files-upload-errors/files-upload-errors.component.mjs +3 -3
  29. package/esm2022/impact-assessments/impact-assessments-graph/impact-assessments-graph.component.mjs +3 -3
  30. package/esm2022/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.mjs +3 -3
  31. package/esm2022/impact-assessments/impact-assessments-products/impact-assessments-products.component.mjs +3 -3
  32. package/esm2022/node/node-csv-export-confirm/node-csv-export-confirm.component.mjs +3 -3
  33. package/esm2022/node/node-csv-select-headers/node-csv-select-headers.component.mjs +3 -3
  34. package/esm2022/node/node-diffs/node-diffs.component.mjs +3 -3
  35. package/esm2022/node/node-link/node-link.component.mjs +3 -3
  36. package/esm2022/node/node-logs-file/node-logs-file.component.mjs +3 -3
  37. package/esm2022/node/node-missing-lookup-factors/node-missing-lookup-factors.component.mjs +3 -3
  38. package/esm2022/node/node-recommendations/node-recommendations.component.mjs +3 -3
  39. package/esm2022/node/node-value-details/node-value-details.component.mjs +5 -3
  40. package/esm2022/schema/schema-info/schema-info.component.mjs +3 -3
  41. package/esm2022/select/select-option/select-option.component.mjs +3 -3
  42. package/esm2022/select/select-option-group/select-option-group.component.mjs +3 -3
  43. package/esm2022/select/select.component.mjs +3 -3
  44. package/esm2022/sites/sites-nodes/sites-nodes.component.mjs +3 -3
  45. package/esm2022/terms/terms-property-content/terms-property-content.component.mjs +3 -3
  46. package/esm2022/terms/terms-sub-class-of-content/terms-sub-class-of-content.component.mjs +3 -3
  47. package/esm2022/terms/terms-units-description/terms-units-description.component.mjs +3 -3
  48. package/fesm2022/hestia-earth-ui-components.mjs +93 -88
  49. package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
  50. package/node/node-logs-models/node-logs-models.component.d.ts +1 -1
  51. package/package.json +1 -1
  52. package/styles.scss +0 -34
@@ -19,10 +19,10 @@ export class ShelfDialogComponent {
19
19
  this.showContent.set(false);
20
20
  }
21
21
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: ShelfDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
22
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.11", type: ShelfDialogComponent, isStandalone: true, selector: "he-shelf-dialog", inputs: { headerTemplate: { classPropertyName: "headerTemplate", publicName: "headerTemplate", isSignal: true, isRequired: false, transformFunction: null }, title: { classPropertyName: "title", publicName: "title", isSignal: true, isRequired: false, transformFunction: null }, bottom: { classPropertyName: "bottom", publicName: "bottom", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed" }, ngImport: i0, template: "<div [style.bottom]=\"bottom()\" class=\"shelf-dialog | is-fixed\">\n <header class=\"shelf-dialog__header | px-4 py-2 is-flex is-rounded has-text-white is-align-items-center\">\n <div class=\"is-flex-grow-1\">\n <ng-container [ngTemplateOutlet]=\"headerTemplate() ?? defaultTemplate\" />\n </div>\n <button (click)=\"showContent.set(!showContent())\" class=\"ml-3 dialog-btn has-text-white\">\n <svg-icon class=\"is-size-4\" [name]=\"toggleIcon()\" />\n </button>\n <button (click)=\"closed.emit()\" class=\"ml-3 dialog-btn has-text-white\">\n <svg-icon class=\"is-size-4\" name=\"far-close\" />\n </button>\n </header>\n <div [class.max-h-0]=\"!showContent()\" class=\"shelf-dialog__content | has-background-white\">\n <div>\n <div class=\"p-4\">\n <ng-content />\n </div>\n </div>\n </div>\n</div>\n\n<ng-template #defaultTemplate>\n {{ title() }}\n</ng-template>\n", styles: [".shelf-dialog{min-width:360px;right:10%;border-radius:.5em .5em 0 0;overflow:hidden;box-shadow:0 8px 8px #00000026;z-index:99}.shelf-dialog__header{background:#4c7194}.shelf-dialog__content{overflow:hidden;display:grid;grid-template-rows:1fr;transition:grid-template-rows 125ms}.shelf-dialog__content>*{overflow:hidden}.shelf-dialog .dialog-btn{background:transparent;border:none;cursor:pointer;width:1.5rem;height:1.5rem}.shelf-dialog .dialog-btn fa-icon{font-size:1.5rem}.max-h-0{grid-template-rows:0fr}@media screen and (max-width: 767px){.shelf-dialog{width:100dvw;right:0;border-radius:0}}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: SvgIconComponent, selector: "svg-icon", inputs: ["src", "name", "stretch", "applyClass", "svgClass", "class", "viewBox", "svgAriaLabel", "svgStyle"] }, { kind: "ngmodule", type: HeSvgIconsModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
22
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.11", type: ShelfDialogComponent, isStandalone: true, selector: "he-shelf-dialog", inputs: { headerTemplate: { classPropertyName: "headerTemplate", publicName: "headerTemplate", isSignal: true, isRequired: false, transformFunction: null }, title: { classPropertyName: "title", publicName: "title", isSignal: true, isRequired: false, transformFunction: null }, bottom: { classPropertyName: "bottom", publicName: "bottom", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed" }, ngImport: i0, template: "<div [style.bottom]=\"bottom()\" class=\"shelf-dialog | is-fixed is-overflow-hidden\">\n <header class=\"shelf-dialog__header | px-4 py-2 is-flex is-rounded has-text-white is-align-items-center\">\n <div class=\"is-flex-grow-1\">\n <ng-container [ngTemplateOutlet]=\"headerTemplate() ?? defaultTemplate\" />\n </div>\n <button (click)=\"showContent.set(!showContent())\" class=\"ml-3 has-text-white | dialog-btn\">\n <svg-icon class=\"is-size-4\" [name]=\"toggleIcon()\" />\n </button>\n <button (click)=\"closed.emit()\" class=\"ml-3 has-text-white | dialog-btn\">\n <svg-icon class=\"is-size-4\" name=\"far-close\" />\n </button>\n </header>\n <div [class.max-h-0]=\"!showContent()\" class=\"shelf-dialog__content | has-background-white is-overflow-hidden\">\n <div>\n <div class=\"p-4\">\n <ng-content />\n </div>\n </div>\n </div>\n</div>\n\n<ng-template #defaultTemplate>\n {{ title() }}\n</ng-template>\n", styles: [".shelf-dialog{min-width:360px;right:10%;border-radius:.5em .5em 0 0;box-shadow:0 8px 8px #00000026;z-index:99}.shelf-dialog__header{background:#4c7194}.shelf-dialog__content{display:grid;grid-template-rows:1fr;transition:grid-template-rows 125ms}.shelf-dialog__content>*{overflow:hidden}.shelf-dialog .dialog-btn{background:transparent;border:none;width:1.5rem;height:1.5rem}.shelf-dialog .dialog-btn fa-icon{font-size:1.5rem}.max-h-0{grid-template-rows:0fr}@media screen and (max-width: 767px){.shelf-dialog{width:100dvw;right:0;border-radius:0}}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: SvgIconComponent, selector: "svg-icon", inputs: ["src", "name", "stretch", "applyClass", "svgClass", "class", "viewBox", "svgAriaLabel", "svgStyle"] }, { kind: "ngmodule", type: HeSvgIconsModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
23
23
  }
24
24
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: ShelfDialogComponent, decorators: [{
25
25
  type: Component,
26
- args: [{ selector: 'he-shelf-dialog', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgTemplateOutlet, SvgIconComponent, HeSvgIconsModule], template: "<div [style.bottom]=\"bottom()\" class=\"shelf-dialog | is-fixed\">\n <header class=\"shelf-dialog__header | px-4 py-2 is-flex is-rounded has-text-white is-align-items-center\">\n <div class=\"is-flex-grow-1\">\n <ng-container [ngTemplateOutlet]=\"headerTemplate() ?? defaultTemplate\" />\n </div>\n <button (click)=\"showContent.set(!showContent())\" class=\"ml-3 dialog-btn has-text-white\">\n <svg-icon class=\"is-size-4\" [name]=\"toggleIcon()\" />\n </button>\n <button (click)=\"closed.emit()\" class=\"ml-3 dialog-btn has-text-white\">\n <svg-icon class=\"is-size-4\" name=\"far-close\" />\n </button>\n </header>\n <div [class.max-h-0]=\"!showContent()\" class=\"shelf-dialog__content | has-background-white\">\n <div>\n <div class=\"p-4\">\n <ng-content />\n </div>\n </div>\n </div>\n</div>\n\n<ng-template #defaultTemplate>\n {{ title() }}\n</ng-template>\n", styles: [".shelf-dialog{min-width:360px;right:10%;border-radius:.5em .5em 0 0;overflow:hidden;box-shadow:0 8px 8px #00000026;z-index:99}.shelf-dialog__header{background:#4c7194}.shelf-dialog__content{overflow:hidden;display:grid;grid-template-rows:1fr;transition:grid-template-rows 125ms}.shelf-dialog__content>*{overflow:hidden}.shelf-dialog .dialog-btn{background:transparent;border:none;cursor:pointer;width:1.5rem;height:1.5rem}.shelf-dialog .dialog-btn fa-icon{font-size:1.5rem}.max-h-0{grid-template-rows:0fr}@media screen and (max-width: 767px){.shelf-dialog{width:100dvw;right:0;border-radius:0}}\n"] }]
26
+ args: [{ selector: 'he-shelf-dialog', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgTemplateOutlet, SvgIconComponent, HeSvgIconsModule], template: "<div [style.bottom]=\"bottom()\" class=\"shelf-dialog | is-fixed is-overflow-hidden\">\n <header class=\"shelf-dialog__header | px-4 py-2 is-flex is-rounded has-text-white is-align-items-center\">\n <div class=\"is-flex-grow-1\">\n <ng-container [ngTemplateOutlet]=\"headerTemplate() ?? defaultTemplate\" />\n </div>\n <button (click)=\"showContent.set(!showContent())\" class=\"ml-3 has-text-white | dialog-btn\">\n <svg-icon class=\"is-size-4\" [name]=\"toggleIcon()\" />\n </button>\n <button (click)=\"closed.emit()\" class=\"ml-3 has-text-white | dialog-btn\">\n <svg-icon class=\"is-size-4\" name=\"far-close\" />\n </button>\n </header>\n <div [class.max-h-0]=\"!showContent()\" class=\"shelf-dialog__content | has-background-white is-overflow-hidden\">\n <div>\n <div class=\"p-4\">\n <ng-content />\n </div>\n </div>\n </div>\n</div>\n\n<ng-template #defaultTemplate>\n {{ title() }}\n</ng-template>\n", styles: [".shelf-dialog{min-width:360px;right:10%;border-radius:.5em .5em 0 0;box-shadow:0 8px 8px #00000026;z-index:99}.shelf-dialog__header{background:#4c7194}.shelf-dialog__content{display:grid;grid-template-rows:1fr;transition:grid-template-rows 125ms}.shelf-dialog__content>*{overflow:hidden}.shelf-dialog .dialog-btn{background:transparent;border:none;width:1.5rem;height:1.5rem}.shelf-dialog .dialog-btn fa-icon{font-size:1.5rem}.max-h-0{grid-template-rows:0fr}@media screen and (max-width: 767px){.shelf-dialog{width:100dvw;right:0;border-radius:0}}\n"] }]
27
27
  }] });
28
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2hlbGYtZGlhbG9nLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb21tb24vc2hlbGYtZGlhbG9nL3NoZWxmLWRpYWxvZy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi9zcmMvY29tbW9uL3NoZWxmLWRpYWxvZy9zaGVsZi1kaWFsb2cuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQWUsTUFBTSxlQUFlLENBQUM7QUFDakgsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDbkQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFDcEQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBVW5ELE1BQU0sT0FBTyxvQkFBb0I7SUFSakM7UUFTcUIsbUJBQWMsR0FBRyxLQUFLLEVBQW9CLENBQUM7UUFFM0MsVUFBSyxHQUFHLEtBQUssRUFBVSxDQUFDO1FBRXhCLFdBQU0sR0FBRyxLQUFLLENBQWtCLEdBQUcsQ0FBQyxDQUFDO1FBRXJDLFdBQU0sR0FBRyxNQUFNLEVBQVEsQ0FBQztRQUV4QixnQkFBVyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUUzQixlQUFVLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDO0tBUzlHO0lBUFEsTUFBTTtRQUNYLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdCLENBQUM7SUFFTSxRQUFRO1FBQ2IsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDOUIsQ0FBQzsrR0FuQlUsb0JBQW9CO21HQUFwQixvQkFBb0IsNGZDYmpDLG02QkF3QkEsOG9CRGJZLGdCQUFnQixvSkFBRSxnQkFBZ0Isa0tBQUUsZ0JBQWdCOzs0RkFFbkQsb0JBQW9CO2tCQVJoQyxTQUFTOytCQUNFLGlCQUFpQixtQkFHVix1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgY29tcHV0ZWQsIGlucHV0LCBvdXRwdXQsIHNpZ25hbCwgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5nVGVtcGxhdGVPdXRsZXQgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgU3ZnSWNvbkNvbXBvbmVudCB9IGZyb20gJ2FuZ3VsYXItc3ZnLWljb24nO1xuaW1wb3J0IHsgSGVTdmdJY29uc01vZHVsZSB9IGZyb20gJy4uLy4uL3N2Zy1pY29ucyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2hlLXNoZWxmLWRpYWxvZycsXG4gIHRlbXBsYXRlVXJsOiAnLi9zaGVsZi1kaWFsb2cuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9zaGVsZi1kaWFsb2cuY29tcG9uZW50LnNjc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtOZ1RlbXBsYXRlT3V0bGV0LCBTdmdJY29uQ29tcG9uZW50LCBIZVN2Z0ljb25zTW9kdWxlXVxufSlcbmV4cG9ydCBjbGFzcyBTaGVsZkRpYWxvZ0NvbXBvbmVudCB7XG4gIHByb3RlY3RlZCByZWFkb25seSBoZWFkZXJUZW1wbGF0ZSA9IGlucHV0PFRlbXBsYXRlUmVmPGFueT4+KCk7XG5cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IHRpdGxlID0gaW5wdXQ8c3RyaW5nPigpO1xuXG4gIHByb3RlY3RlZCByZWFkb25seSBib3R0b20gPSBpbnB1dDxzdHJpbmcgfCBudW1iZXI+KCcwJyk7XG5cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGNsb3NlZCA9IG91dHB1dDx2b2lkPigpO1xuXG4gIHByb3RlY3RlZCByZWFkb25seSBzaG93Q29udGVudCA9IHNpZ25hbCh0cnVlKTtcblxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgdG9nZ2xlSWNvbiA9IGNvbXB1dGVkKCgpID0+ICh0aGlzLnNob3dDb250ZW50KCkgPyAnZmFyLWNoZXZyb24tZG93bicgOiAnZmFyLWNoZXZyb24tdXAnKSk7XG5cbiAgcHVibGljIGV4cGFuZCgpIHtcbiAgICB0aGlzLnNob3dDb250ZW50LnNldCh0cnVlKTtcbiAgfVxuXG4gIHB1YmxpYyBjb2xsYXBzZSgpIHtcbiAgICB0aGlzLnNob3dDb250ZW50LnNldChmYWxzZSk7XG4gIH1cbn1cbiIsIjxkaXYgW3N0eWxlLmJvdHRvbV09XCJib3R0b20oKVwiIGNsYXNzPVwic2hlbGYtZGlhbG9nIHwgaXMtZml4ZWRcIj5cbiAgPGhlYWRlciBjbGFzcz1cInNoZWxmLWRpYWxvZ19faGVhZGVyIHwgcHgtNCBweS0yIGlzLWZsZXggaXMtcm91bmRlZCBoYXMtdGV4dC13aGl0ZSBpcy1hbGlnbi1pdGVtcy1jZW50ZXJcIj5cbiAgICA8ZGl2IGNsYXNzPVwiaXMtZmxleC1ncm93LTFcIj5cbiAgICAgIDxuZy1jb250YWluZXIgW25nVGVtcGxhdGVPdXRsZXRdPVwiaGVhZGVyVGVtcGxhdGUoKSA/PyBkZWZhdWx0VGVtcGxhdGVcIiAvPlxuICAgIDwvZGl2PlxuICAgIDxidXR0b24gKGNsaWNrKT1cInNob3dDb250ZW50LnNldCghc2hvd0NvbnRlbnQoKSlcIiBjbGFzcz1cIm1sLTMgZGlhbG9nLWJ0biBoYXMtdGV4dC13aGl0ZVwiPlxuICAgICAgPHN2Zy1pY29uIGNsYXNzPVwiaXMtc2l6ZS00XCIgW25hbWVdPVwidG9nZ2xlSWNvbigpXCIgLz5cbiAgICA8L2J1dHRvbj5cbiAgICA8YnV0dG9uIChjbGljayk9XCJjbG9zZWQuZW1pdCgpXCIgY2xhc3M9XCJtbC0zIGRpYWxvZy1idG4gaGFzLXRleHQtd2hpdGVcIj5cbiAgICAgIDxzdmctaWNvbiBjbGFzcz1cImlzLXNpemUtNFwiIG5hbWU9XCJmYXItY2xvc2VcIiAvPlxuICAgIDwvYnV0dG9uPlxuICA8L2hlYWRlcj5cbiAgPGRpdiBbY2xhc3MubWF4LWgtMF09XCIhc2hvd0NvbnRlbnQoKVwiIGNsYXNzPVwic2hlbGYtZGlhbG9nX19jb250ZW50IHwgaGFzLWJhY2tncm91bmQtd2hpdGVcIj5cbiAgICA8ZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInAtNFwiPlxuICAgICAgICA8bmctY29udGVudCAvPlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuPC9kaXY+XG5cbjxuZy10ZW1wbGF0ZSAjZGVmYXVsdFRlbXBsYXRlPlxuICB7eyB0aXRsZSgpIH19XG48L25nLXRlbXBsYXRlPlxuIl19
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2hlbGYtZGlhbG9nLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb21tb24vc2hlbGYtZGlhbG9nL3NoZWxmLWRpYWxvZy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi9zcmMvY29tbW9uL3NoZWxmLWRpYWxvZy9zaGVsZi1kaWFsb2cuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQWUsTUFBTSxlQUFlLENBQUM7QUFDakgsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDbkQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFDcEQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBVW5ELE1BQU0sT0FBTyxvQkFBb0I7SUFSakM7UUFTcUIsbUJBQWMsR0FBRyxLQUFLLEVBQW9CLENBQUM7UUFFM0MsVUFBSyxHQUFHLEtBQUssRUFBVSxDQUFDO1FBRXhCLFdBQU0sR0FBRyxLQUFLLENBQWtCLEdBQUcsQ0FBQyxDQUFDO1FBRXJDLFdBQU0sR0FBRyxNQUFNLEVBQVEsQ0FBQztRQUV4QixnQkFBVyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUUzQixlQUFVLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDO0tBUzlHO0lBUFEsTUFBTTtRQUNYLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdCLENBQUM7SUFFTSxRQUFRO1FBQ2IsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDOUIsQ0FBQzsrR0FuQlUsb0JBQW9CO21HQUFwQixvQkFBb0IsNGZDYmpDLDY4QkF3QkEsK2xCRGJZLGdCQUFnQixvSkFBRSxnQkFBZ0Isa0tBQUUsZ0JBQWdCOzs0RkFFbkQsb0JBQW9CO2tCQVJoQyxTQUFTOytCQUNFLGlCQUFpQixtQkFHVix1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgY29tcHV0ZWQsIGlucHV0LCBvdXRwdXQsIHNpZ25hbCwgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5nVGVtcGxhdGVPdXRsZXQgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgU3ZnSWNvbkNvbXBvbmVudCB9IGZyb20gJ2FuZ3VsYXItc3ZnLWljb24nO1xuaW1wb3J0IHsgSGVTdmdJY29uc01vZHVsZSB9IGZyb20gJy4uLy4uL3N2Zy1pY29ucyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2hlLXNoZWxmLWRpYWxvZycsXG4gIHRlbXBsYXRlVXJsOiAnLi9zaGVsZi1kaWFsb2cuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9zaGVsZi1kaWFsb2cuY29tcG9uZW50LnNjc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtOZ1RlbXBsYXRlT3V0bGV0LCBTdmdJY29uQ29tcG9uZW50LCBIZVN2Z0ljb25zTW9kdWxlXVxufSlcbmV4cG9ydCBjbGFzcyBTaGVsZkRpYWxvZ0NvbXBvbmVudCB7XG4gIHByb3RlY3RlZCByZWFkb25seSBoZWFkZXJUZW1wbGF0ZSA9IGlucHV0PFRlbXBsYXRlUmVmPGFueT4+KCk7XG5cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IHRpdGxlID0gaW5wdXQ8c3RyaW5nPigpO1xuXG4gIHByb3RlY3RlZCByZWFkb25seSBib3R0b20gPSBpbnB1dDxzdHJpbmcgfCBudW1iZXI+KCcwJyk7XG5cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGNsb3NlZCA9IG91dHB1dDx2b2lkPigpO1xuXG4gIHByb3RlY3RlZCByZWFkb25seSBzaG93Q29udGVudCA9IHNpZ25hbCh0cnVlKTtcblxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgdG9nZ2xlSWNvbiA9IGNvbXB1dGVkKCgpID0+ICh0aGlzLnNob3dDb250ZW50KCkgPyAnZmFyLWNoZXZyb24tZG93bicgOiAnZmFyLWNoZXZyb24tdXAnKSk7XG5cbiAgcHVibGljIGV4cGFuZCgpIHtcbiAgICB0aGlzLnNob3dDb250ZW50LnNldCh0cnVlKTtcbiAgfVxuXG4gIHB1YmxpYyBjb2xsYXBzZSgpIHtcbiAgICB0aGlzLnNob3dDb250ZW50LnNldChmYWxzZSk7XG4gIH1cbn1cbiIsIjxkaXYgW3N0eWxlLmJvdHRvbV09XCJib3R0b20oKVwiIGNsYXNzPVwic2hlbGYtZGlhbG9nIHwgaXMtZml4ZWQgaXMtb3ZlcmZsb3ctaGlkZGVuXCI+XG4gIDxoZWFkZXIgY2xhc3M9XCJzaGVsZi1kaWFsb2dfX2hlYWRlciB8IHB4LTQgcHktMiBpcy1mbGV4IGlzLXJvdW5kZWQgaGFzLXRleHQtd2hpdGUgaXMtYWxpZ24taXRlbXMtY2VudGVyXCI+XG4gICAgPGRpdiBjbGFzcz1cImlzLWZsZXgtZ3Jvdy0xXCI+XG4gICAgICA8bmctY29udGFpbmVyIFtuZ1RlbXBsYXRlT3V0bGV0XT1cImhlYWRlclRlbXBsYXRlKCkgPz8gZGVmYXVsdFRlbXBsYXRlXCIgLz5cbiAgICA8L2Rpdj5cbiAgICA8YnV0dG9uIChjbGljayk9XCJzaG93Q29udGVudC5zZXQoIXNob3dDb250ZW50KCkpXCIgY2xhc3M9XCJtbC0zIGhhcy10ZXh0LXdoaXRlIHwgZGlhbG9nLWJ0blwiPlxuICAgICAgPHN2Zy1pY29uIGNsYXNzPVwiaXMtc2l6ZS00XCIgW25hbWVdPVwidG9nZ2xlSWNvbigpXCIgLz5cbiAgICA8L2J1dHRvbj5cbiAgICA8YnV0dG9uIChjbGljayk9XCJjbG9zZWQuZW1pdCgpXCIgY2xhc3M9XCJtbC0zIGhhcy10ZXh0LXdoaXRlIHwgZGlhbG9nLWJ0blwiPlxuICAgICAgPHN2Zy1pY29uIGNsYXNzPVwiaXMtc2l6ZS00XCIgbmFtZT1cImZhci1jbG9zZVwiIC8+XG4gICAgPC9idXR0b24+XG4gIDwvaGVhZGVyPlxuICA8ZGl2IFtjbGFzcy5tYXgtaC0wXT1cIiFzaG93Q29udGVudCgpXCIgY2xhc3M9XCJzaGVsZi1kaWFsb2dfX2NvbnRlbnQgfCBoYXMtYmFja2dyb3VuZC13aGl0ZSBpcy1vdmVyZmxvdy1oaWRkZW5cIj5cbiAgICA8ZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInAtNFwiPlxuICAgICAgICA8bmctY29udGVudCAvPlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuPC9kaXY+XG5cbjxuZy10ZW1wbGF0ZSAjZGVmYXVsdFRlbXBsYXRlPlxuICB7eyB0aXRsZSgpIH19XG48L25nLXRlbXBsYXRlPlxuIl19
@@ -77,11 +77,11 @@ export class ShellComponent {
77
77
  button.onClick?.();
78
78
  }
79
79
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: ShellComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
80
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: ShellComponent, isStandalone: true, selector: "he-shell", inputs: { shelfButtons: "shelfButtons", menuOverlap: "menuOverlap", closeMenuOnOutsideClick: "closeMenuOnOutsideClick", defaultActiveButtonId: "defaultActiveButtonId" }, outputs: { menuShown: "menuShown" }, queries: [{ propertyName: "iconTemplate", first: true, predicate: ["iconTemplate"], descendants: true }], ngImport: i0, template: "<div class=\"shell | h-100\">\n <aside>\n <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of topMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n </div>\n <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n </div>\n </div>\n </aside>\n\n <div class=\"shell__content | h-100 w-100 is-flex is-justify-content-flex-end\">\n <div\n class=\"shell__content--body | w-100\"\n [class.menu-expanded]=\"!menuOverlap && expanded()\"\n (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n <ng-content></ng-content>\n </div>\n </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n <button\n (click)=\"onMenuButtonClick(button)\"\n class=\"menu-button button is-secondary\"\n [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n [class.is-active]=\"button.isActive\">\n <ng-container\n *ngTemplateOutlet=\"\n iconTemplate ?? defaultIconTemplate;\n context: { $implicit: button, isActive: button !== clickedButton }\n \"></ng-container>\n </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n <fa-icon [icon]=\"button.icon\"></fa-icon>\n</ng-template>\n", styles: ["::ng-deep :root{--menu-width: 200px;--menu-transition-time: .15s}.shell{display:grid;grid-template-columns:auto 1fr;grid-template-areas:\"menu content\";overflow:hidden}.shell>aside{position:relative;grid-area:menu}.shell>aside .shell__menu{background:#193957;padding:1.5em 0;z-index:10;min-width:2.875em}.shell>aside .shell__menuContent{background-color:#fefaef;width:var(--menu-width);top:0;right:1px;transform:translate(0);transition:transform var(--menu-transition-time) ease-out}.shell>aside .shell__menuContent.expanded{transform:translate(100%)}.shell__content{grid-area:content;overflow:hidden}.shell__content--body{transition:width var(--menu-transition-time) ease-out}.shell__content .menu-expanded{width:calc(100% - var(--menu-width))}.menu-button{border-radius:0;padding:1.25em .6em;box-shadow:none!important;border-width:0}.menu-button.is-active{background-color:#fff;color:#0a0a0a}.menu-button:hover:not(.is-active){background-color:initial}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }] }); }
80
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: ShellComponent, isStandalone: true, selector: "he-shell", inputs: { shelfButtons: "shelfButtons", menuOverlap: "menuOverlap", closeMenuOnOutsideClick: "closeMenuOnOutsideClick", defaultActiveButtonId: "defaultActiveButtonId" }, outputs: { menuShown: "menuShown" }, queries: [{ propertyName: "iconTemplate", first: true, predicate: ["iconTemplate"], descendants: true }], ngImport: i0, template: "<div class=\"shell | h-100 is-overflow-hidden\">\n <aside>\n <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of topMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n </div>\n <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n </div>\n </div>\n </aside>\n\n <div class=\"shell__content | h-100 w-100 is-overflow-hidden is-flex is-justify-content-flex-end\">\n <div\n class=\"shell__content--body | w-100\"\n [class.menu-expanded]=\"!menuOverlap && expanded()\"\n (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n <ng-content></ng-content>\n </div>\n </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n <button\n (click)=\"onMenuButtonClick(button)\"\n class=\"menu-button button is-secondary\"\n [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n [class.is-active]=\"button.isActive\">\n <ng-container\n *ngTemplateOutlet=\"\n iconTemplate ?? defaultIconTemplate;\n context: { $implicit: button, isActive: button !== clickedButton }\n \"></ng-container>\n </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n <fa-icon [icon]=\"button.icon\" />\n</ng-template>\n", styles: ["::ng-deep :root{--menu-width: 200px;--menu-transition-time: .15s}.shell{display:grid;grid-template-columns:auto 1fr;grid-template-areas:\"menu content\"}.shell>aside{position:relative;grid-area:menu}.shell>aside .shell__menu{background:#193957;padding:1.5em 0;z-index:10;min-width:2.875em}.shell>aside .shell__menuContent{background-color:#fefaef;width:var(--menu-width);top:0;right:1px;transform:translate(0);transition:transform var(--menu-transition-time) ease-out}.shell>aside .shell__menuContent.expanded{transform:translate(100%)}.shell__content{grid-area:content}.shell__content--body{transition:width var(--menu-transition-time) ease-out}.shell__content .menu-expanded{width:calc(100% - var(--menu-width))}.menu-button{border-radius:0;padding:1.25em .6em;box-shadow:none!important;border-width:0}.menu-button.is-active{background-color:#fff;color:#0a0a0a}.menu-button:hover:not(.is-active){background-color:initial}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }] }); }
81
81
  }
82
82
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: ShellComponent, decorators: [{
83
83
  type: Component,
84
- args: [{ selector: 'he-shell', standalone: true, imports: [NgTemplateOutlet, NgClass, FaIconComponent], template: "<div class=\"shell | h-100\">\n <aside>\n <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of topMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n </div>\n <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n </div>\n </div>\n </aside>\n\n <div class=\"shell__content | h-100 w-100 is-flex is-justify-content-flex-end\">\n <div\n class=\"shell__content--body | w-100\"\n [class.menu-expanded]=\"!menuOverlap && expanded()\"\n (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n <ng-content></ng-content>\n </div>\n </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n <button\n (click)=\"onMenuButtonClick(button)\"\n class=\"menu-button button is-secondary\"\n [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n [class.is-active]=\"button.isActive\">\n <ng-container\n *ngTemplateOutlet=\"\n iconTemplate ?? defaultIconTemplate;\n context: { $implicit: button, isActive: button !== clickedButton }\n \"></ng-container>\n </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n <fa-icon [icon]=\"button.icon\"></fa-icon>\n</ng-template>\n", styles: ["::ng-deep :root{--menu-width: 200px;--menu-transition-time: .15s}.shell{display:grid;grid-template-columns:auto 1fr;grid-template-areas:\"menu content\";overflow:hidden}.shell>aside{position:relative;grid-area:menu}.shell>aside .shell__menu{background:#193957;padding:1.5em 0;z-index:10;min-width:2.875em}.shell>aside .shell__menuContent{background-color:#fefaef;width:var(--menu-width);top:0;right:1px;transform:translate(0);transition:transform var(--menu-transition-time) ease-out}.shell>aside .shell__menuContent.expanded{transform:translate(100%)}.shell__content{grid-area:content;overflow:hidden}.shell__content--body{transition:width var(--menu-transition-time) ease-out}.shell__content .menu-expanded{width:calc(100% - var(--menu-width))}.menu-button{border-radius:0;padding:1.25em .6em;box-shadow:none!important;border-width:0}.menu-button.is-active{background-color:#fff;color:#0a0a0a}.menu-button:hover:not(.is-active){background-color:initial}\n"] }]
84
+ args: [{ selector: 'he-shell', standalone: true, imports: [NgTemplateOutlet, NgClass, FaIconComponent], template: "<div class=\"shell | h-100 is-overflow-hidden\">\n <aside>\n <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of topMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n\n <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n <ng-container\n [ngTemplateOutlet]=\"buttonTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n }\n </div>\n </div>\n <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n </div>\n </div>\n </aside>\n\n <div class=\"shell__content | h-100 w-100 is-overflow-hidden is-flex is-justify-content-flex-end\">\n <div\n class=\"shell__content--body | w-100\"\n [class.menu-expanded]=\"!menuOverlap && expanded()\"\n (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n <ng-content></ng-content>\n </div>\n </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n <button\n (click)=\"onMenuButtonClick(button)\"\n class=\"menu-button button is-secondary\"\n [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n [class.is-active]=\"button.isActive\">\n <ng-container\n *ngTemplateOutlet=\"\n iconTemplate ?? defaultIconTemplate;\n context: { $implicit: button, isActive: button !== clickedButton }\n \"></ng-container>\n </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n <fa-icon [icon]=\"button.icon\" />\n</ng-template>\n", styles: ["::ng-deep :root{--menu-width: 200px;--menu-transition-time: .15s}.shell{display:grid;grid-template-columns:auto 1fr;grid-template-areas:\"menu content\"}.shell>aside{position:relative;grid-area:menu}.shell>aside .shell__menu{background:#193957;padding:1.5em 0;z-index:10;min-width:2.875em}.shell>aside .shell__menuContent{background-color:#fefaef;width:var(--menu-width);top:0;right:1px;transform:translate(0);transition:transform var(--menu-transition-time) ease-out}.shell>aside .shell__menuContent.expanded{transform:translate(100%)}.shell__content{grid-area:content}.shell__content--body{transition:width var(--menu-transition-time) ease-out}.shell__content .menu-expanded{width:calc(100% - var(--menu-width))}.menu-button{border-radius:0;padding:1.25em .6em;box-shadow:none!important;border-width:0}.menu-button.is-active{background-color:#fff;color:#0a0a0a}.menu-button:hover:not(.is-active){background-color:initial}\n"] }]
85
85
  }], propDecorators: { iconTemplate: [{
86
86
  type: ContentChild,
87
87
  args: ['iconTemplate']
@@ -96,4 +96,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
96
96
  }], menuShown: [{
97
97
  type: Output
98
98
  }] } });
99
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.component.js","sourceRoot":"","sources":["../../../../src/common/shell/shell.component.ts","../../../../src/common/shell/shell.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  computed,\n  ContentChild,\n  EventEmitter,\n  Input,\n  Output,\n  signal,\n  TemplateRef,\n  WritableSignal\n} from '@angular/core';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgTemplateOutlet, NgClass } from '@angular/common';\n\nexport interface IShellMenuButton {\n  icon?: any;\n  iconPath?: string;\n  onClick: () => void;\n  menuRef?: TemplateRef<any>;\n  position?: 'top' | 'bottom';\n  activeDisabled?: boolean;\n  id?: string;\n  className?: string;\n  activeClassName?: string;\n}\n\n@Component({\n  selector: 'he-shell',\n  templateUrl: './shell.component.html',\n  styleUrls: ['./shell.component.scss'],\n  standalone: true,\n  imports: [NgTemplateOutlet, NgClass, FaIconComponent]\n})\nexport class ShellComponent {\n  @ContentChild('iconTemplate') iconTemplate: TemplateRef<any> | null = null;\n\n  private _shelfButtons: WritableSignal<IShellMenuButton[]> = signal([]);\n  private _defaultActiveButtonId: WritableSignal<string | null> = signal(null);\n\n  @Input() set shelfButtons(value: IShellMenuButton[]) {\n    this._shelfButtons.set(value);\n  }\n\n  @Input() menuOverlap = false;\n\n  @Input() closeMenuOnOutsideClick = false;\n\n  @Input() set defaultActiveButtonId(value: string | null) {\n    this._defaultActiveButtonId.set(value);\n  }\n\n  @Output() menuShown = new EventEmitter();\n\n  buttons = computed(() =>\n    this._shelfButtons()\n      .map((button, index) => ({\n        ...button,\n        id: button.id || index.toString()\n      }))\n      .map(button => {\n        const activeMenuId = this.activeMenuId();\n        return { ...button, isActive: button.id === (activeMenuId || this.activeButtonId()) };\n      })\n  );\n\n  protected clickedButton: WritableSignal<string | null> = signal(null);\n  protected expanded = signal(false);\n\n  protected menuRef = signal(null as TemplateRef<any> | null);\n\n  activeButtonId = computed(() => {\n    return this.clickedButton() || this._defaultActiveButtonId();\n  });\n\n  activeMenuId = computed(() => {\n    if (this.expanded() && this.menuRef()) {\n      return this._shelfButtons()\n        .findIndex(button => button.menuRef === this.menuRef())\n        ?.toString();\n    }\n  });\n\n  protected topMenuButtons = computed(() => {\n    return this.buttons().filter(button => !button.position || button.position === 'top');\n  });\n\n  protected trackByFn(index: number, item: IShellMenuButton) {\n    return item.id;\n  }\n\n  protected bottomMenuButtons = computed(() => {\n    return this.buttons().filter(button => button.position === 'bottom');\n  });\n\n  hideMenu() {\n    this.expanded.set(false);\n    this.clickedButton.set(null);\n  }\n\n  showMenu(templateRef?: TemplateRef<any>) {\n    if (templateRef) {\n      this.menuRef.set(templateRef);\n    }\n    this.expanded.set(true);\n    this.menuShown.emit();\n  }\n\n  protected onMenuButtonClick(button: IShellMenuButton) {\n    if (button.id === this.clickedButton()) {\n      this.hideMenu();\n      return;\n    }\n\n    if (button.menuRef) {\n      this.menuRef.set(button.menuRef);\n      this.showMenu();\n    } else {\n      this.hideMenu();\n    }\n\n    if (!button.activeDisabled) {\n      this.clickedButton.set(button.id);\n    }\n\n    button.onClick?.();\n  }\n}\n","<div class=\"shell | h-100\">\n  <aside>\n    <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n      <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n        @for (button of topMenuButtons(); track trackByFn($index, button)) {\n          <ng-container\n            [ngTemplateOutlet]=\"buttonTemplate\"\n            [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n        }\n      </div>\n\n      <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n        @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n          <ng-container\n            [ngTemplateOutlet]=\"buttonTemplate\"\n            [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n        }\n      </div>\n    </div>\n    <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n      <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n        <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n      </div>\n    </div>\n  </aside>\n\n  <div class=\"shell__content | h-100 w-100 is-flex is-justify-content-flex-end\">\n    <div\n      class=\"shell__content--body | w-100\"\n      [class.menu-expanded]=\"!menuOverlap && expanded()\"\n      (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n      <ng-content></ng-content>\n    </div>\n  </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n  <button\n    (click)=\"onMenuButtonClick(button)\"\n    class=\"menu-button button is-secondary\"\n    [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n    [class.is-active]=\"button.isActive\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        iconTemplate ?? defaultIconTemplate;\n        context: { $implicit: button, isActive: button !== clickedButton }\n      \"></ng-container>\n  </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n  <fa-icon [icon]=\"button.icon\"></fa-icon>\n</ng-template>\n"]}
99
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.component.js","sourceRoot":"","sources":["../../../../src/common/shell/shell.component.ts","../../../../src/common/shell/shell.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  computed,\n  ContentChild,\n  EventEmitter,\n  Input,\n  Output,\n  signal,\n  TemplateRef,\n  WritableSignal\n} from '@angular/core';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgTemplateOutlet, NgClass } from '@angular/common';\n\nexport interface IShellMenuButton {\n  icon?: any;\n  iconPath?: string;\n  onClick: () => void;\n  menuRef?: TemplateRef<any>;\n  position?: 'top' | 'bottom';\n  activeDisabled?: boolean;\n  id?: string;\n  className?: string;\n  activeClassName?: string;\n}\n\n@Component({\n  selector: 'he-shell',\n  templateUrl: './shell.component.html',\n  styleUrls: ['./shell.component.scss'],\n  standalone: true,\n  imports: [NgTemplateOutlet, NgClass, FaIconComponent]\n})\nexport class ShellComponent {\n  @ContentChild('iconTemplate') iconTemplate: TemplateRef<any> | null = null;\n\n  private _shelfButtons: WritableSignal<IShellMenuButton[]> = signal([]);\n  private _defaultActiveButtonId: WritableSignal<string | null> = signal(null);\n\n  @Input() set shelfButtons(value: IShellMenuButton[]) {\n    this._shelfButtons.set(value);\n  }\n\n  @Input() menuOverlap = false;\n\n  @Input() closeMenuOnOutsideClick = false;\n\n  @Input() set defaultActiveButtonId(value: string | null) {\n    this._defaultActiveButtonId.set(value);\n  }\n\n  @Output() menuShown = new EventEmitter();\n\n  buttons = computed(() =>\n    this._shelfButtons()\n      .map((button, index) => ({\n        ...button,\n        id: button.id || index.toString()\n      }))\n      .map(button => {\n        const activeMenuId = this.activeMenuId();\n        return { ...button, isActive: button.id === (activeMenuId || this.activeButtonId()) };\n      })\n  );\n\n  protected clickedButton: WritableSignal<string | null> = signal(null);\n  protected expanded = signal(false);\n\n  protected menuRef = signal(null as TemplateRef<any> | null);\n\n  activeButtonId = computed(() => {\n    return this.clickedButton() || this._defaultActiveButtonId();\n  });\n\n  activeMenuId = computed(() => {\n    if (this.expanded() && this.menuRef()) {\n      return this._shelfButtons()\n        .findIndex(button => button.menuRef === this.menuRef())\n        ?.toString();\n    }\n  });\n\n  protected topMenuButtons = computed(() => {\n    return this.buttons().filter(button => !button.position || button.position === 'top');\n  });\n\n  protected trackByFn(index: number, item: IShellMenuButton) {\n    return item.id;\n  }\n\n  protected bottomMenuButtons = computed(() => {\n    return this.buttons().filter(button => button.position === 'bottom');\n  });\n\n  hideMenu() {\n    this.expanded.set(false);\n    this.clickedButton.set(null);\n  }\n\n  showMenu(templateRef?: TemplateRef<any>) {\n    if (templateRef) {\n      this.menuRef.set(templateRef);\n    }\n    this.expanded.set(true);\n    this.menuShown.emit();\n  }\n\n  protected onMenuButtonClick(button: IShellMenuButton) {\n    if (button.id === this.clickedButton()) {\n      this.hideMenu();\n      return;\n    }\n\n    if (button.menuRef) {\n      this.menuRef.set(button.menuRef);\n      this.showMenu();\n    } else {\n      this.hideMenu();\n    }\n\n    if (!button.activeDisabled) {\n      this.clickedButton.set(button.id);\n    }\n\n    button.onClick?.();\n  }\n}\n","<div class=\"shell | h-100 is-overflow-hidden\">\n  <aside>\n    <div class=\"shell__menu | is-relative h-100 is-flex is-flex-direction-column is-justify-content-space-between\">\n      <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n        @for (button of topMenuButtons(); track trackByFn($index, button)) {\n          <ng-container\n            [ngTemplateOutlet]=\"buttonTemplate\"\n            [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n        }\n      </div>\n\n      <div class=\"shell__menu--section | is-flex is-flex-direction-column\">\n        @for (button of bottomMenuButtons(); track trackByFn($index, button)) {\n          <ng-container\n            [ngTemplateOutlet]=\"buttonTemplate\"\n            [ngTemplateOutletContext]=\"{ $implicit: button }\"></ng-container>\n        }\n      </div>\n    </div>\n    <div class=\"shell__menuContent | h-100 is-absolute\" [class.expanded]=\"expanded()\">\n      <div class=\"h-100\" [class.is-hidden]=\"!expanded()\">\n        <ng-template [ngTemplateOutlet]=\"menuRef()\"></ng-template>\n      </div>\n    </div>\n  </aside>\n\n  <div class=\"shell__content | h-100 w-100 is-overflow-hidden is-flex is-justify-content-flex-end\">\n    <div\n      class=\"shell__content--body | w-100\"\n      [class.menu-expanded]=\"!menuOverlap && expanded()\"\n      (click)=\"closeMenuOnOutsideClick ? hideMenu() : null\">\n      <ng-content></ng-content>\n    </div>\n  </div>\n</div>\n\n<ng-template #buttonTemplate let-button>\n  <button\n    (click)=\"onMenuButtonClick(button)\"\n    class=\"menu-button button is-secondary\"\n    [ngClass]=\"button.isActive ? button.activeClassName : ''\"\n    [class.is-active]=\"button.isActive\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        iconTemplate ?? defaultIconTemplate;\n        context: { $implicit: button, isActive: button !== clickedButton }\n      \"></ng-container>\n  </button>\n</ng-template>\n\n<ng-template #defaultIconTemplate let-button>\n  <fa-icon [icon]=\"button.icon\" />\n</ng-template>\n"]}
@@ -41,11 +41,11 @@ export class SortSelectComponent {
41
41
  });
42
42
  }
43
43
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: SortSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
44
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: SortSelectComponent, isStandalone: true, selector: "he-sort-select", inputs: { sortOptions: "sortOptions", alignment: "alignment", sortBy: "sortBy", sortOrder: "sortOrder" }, outputs: { sortChange: "sortChange" }, ngImport: i0, template: "<div (clickOutside)=\"showSortBy = false\" [class.is-active]=\"showSortBy\" [ngClass]=\"'is-' + alignment\" class=\"dropdown\">\n <div (click)=\"showSortBy = !showSortBy\" class=\"dropdown-trigger\">\n <button\n [class.is-active]=\"showSortBy\"\n [disableTooltip]=\"showSortBy\"\n aria-controls=\"sort-menu\"\n aria-haspopup=\"true\"\n class=\"button is-ghost\"\n container=\"body\"\n ngbTooltip=\"Sort options\"\n placement=\"bottom\">\n <fa-icon aria-hidden=\"true\" [icon]=\"faSortAmountUpAlt\"></fa-icon>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"sort-menu\" role=\"menu\">\n <div (click)=\"showSortBy = false\" class=\"dropdown-content\">\n @for (option of sortOptions; track option) {\n <a (click)=\"sort(option)\" [class.is-selected]=\"sortBySignal() === option.id\" class=\"dropdown-item\">\n <fa-icon\n [icon]=\"arrowIcon()\"\n [style.visibility]=\"sortBySignal() === option.id ? 'visible' : 'hidden'\"\n class=\"is-pr-2\"></fa-icon>\n <span [style.border-color]=\"option.color ?? 'transparent'\" class=\"is-pl-2 is-inline-block is-capitalized\">\n {{ option.label }}\n </span>\n </a>\n }\n </div>\n </div>\n</div>\n", styles: [".dropdown-item>span{border-left:2px solid transparent}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: ClickOutsideDirective, selector: "[clickOutside]", inputs: ["clickOutsideListenAfter"], outputs: ["clickOutside"] }, { kind: "directive", type: NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
44
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: SortSelectComponent, isStandalone: true, selector: "he-sort-select", inputs: { sortOptions: "sortOptions", alignment: "alignment", sortBy: "sortBy", sortOrder: "sortOrder" }, outputs: { sortChange: "sortChange" }, ngImport: i0, template: "<div (clickOutside)=\"showSortBy = false\" [class.is-active]=\"showSortBy\" [ngClass]=\"'is-' + alignment\" class=\"dropdown\">\n <div (click)=\"showSortBy = !showSortBy\" class=\"dropdown-trigger\">\n <button\n [class.is-active]=\"showSortBy\"\n [disableTooltip]=\"showSortBy\"\n aria-controls=\"sort-menu\"\n aria-haspopup=\"true\"\n class=\"button is-ghost\"\n container=\"body\"\n ngbTooltip=\"Sort options\"\n placement=\"bottom\">\n <fa-icon aria-hidden=\"true\" [icon]=\"faSortAmountUpAlt\" />\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"sort-menu\" role=\"menu\">\n <div (click)=\"showSortBy = false\" class=\"dropdown-content\">\n @for (option of sortOptions; track option) {\n <a (click)=\"sort(option)\" [class.is-selected]=\"sortBySignal() === option.id\" class=\"dropdown-item\">\n <fa-icon\n [icon]=\"arrowIcon()\"\n [style.visibility]=\"sortBySignal() === option.id ? 'visible' : 'hidden'\"\n class=\"is-pr-2\" />\n <span [style.border-color]=\"option.color ?? 'transparent'\" class=\"is-pl-2 is-inline-block is-capitalized\">\n {{ option.label }}\n </span>\n </a>\n }\n </div>\n </div>\n</div>\n", styles: [".dropdown-item>span{border-left:2px solid transparent}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: ClickOutsideDirective, selector: "[clickOutside]", inputs: ["clickOutsideListenAfter"], outputs: ["clickOutside"] }, { kind: "directive", type: NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "popperOptions", "triggers", "positionTarget", "container", "disableTooltip", "tooltipClass", "tooltipContext", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
45
45
  }
46
46
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: SortSelectComponent, decorators: [{
47
47
  type: Component,
48
- args: [{ selector: 'he-sort-select', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgClass, ClickOutsideDirective, NgbTooltip, FaIconComponent], template: "<div (clickOutside)=\"showSortBy = false\" [class.is-active]=\"showSortBy\" [ngClass]=\"'is-' + alignment\" class=\"dropdown\">\n <div (click)=\"showSortBy = !showSortBy\" class=\"dropdown-trigger\">\n <button\n [class.is-active]=\"showSortBy\"\n [disableTooltip]=\"showSortBy\"\n aria-controls=\"sort-menu\"\n aria-haspopup=\"true\"\n class=\"button is-ghost\"\n container=\"body\"\n ngbTooltip=\"Sort options\"\n placement=\"bottom\">\n <fa-icon aria-hidden=\"true\" [icon]=\"faSortAmountUpAlt\"></fa-icon>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"sort-menu\" role=\"menu\">\n <div (click)=\"showSortBy = false\" class=\"dropdown-content\">\n @for (option of sortOptions; track option) {\n <a (click)=\"sort(option)\" [class.is-selected]=\"sortBySignal() === option.id\" class=\"dropdown-item\">\n <fa-icon\n [icon]=\"arrowIcon()\"\n [style.visibility]=\"sortBySignal() === option.id ? 'visible' : 'hidden'\"\n class=\"is-pr-2\"></fa-icon>\n <span [style.border-color]=\"option.color ?? 'transparent'\" class=\"is-pl-2 is-inline-block is-capitalized\">\n {{ option.label }}\n </span>\n </a>\n }\n </div>\n </div>\n</div>\n", styles: [".dropdown-item>span{border-left:2px solid transparent}\n"] }]
48
+ args: [{ selector: 'he-sort-select', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgClass, ClickOutsideDirective, NgbTooltip, FaIconComponent], template: "<div (clickOutside)=\"showSortBy = false\" [class.is-active]=\"showSortBy\" [ngClass]=\"'is-' + alignment\" class=\"dropdown\">\n <div (click)=\"showSortBy = !showSortBy\" class=\"dropdown-trigger\">\n <button\n [class.is-active]=\"showSortBy\"\n [disableTooltip]=\"showSortBy\"\n aria-controls=\"sort-menu\"\n aria-haspopup=\"true\"\n class=\"button is-ghost\"\n container=\"body\"\n ngbTooltip=\"Sort options\"\n placement=\"bottom\">\n <fa-icon aria-hidden=\"true\" [icon]=\"faSortAmountUpAlt\" />\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"sort-menu\" role=\"menu\">\n <div (click)=\"showSortBy = false\" class=\"dropdown-content\">\n @for (option of sortOptions; track option) {\n <a (click)=\"sort(option)\" [class.is-selected]=\"sortBySignal() === option.id\" class=\"dropdown-item\">\n <fa-icon\n [icon]=\"arrowIcon()\"\n [style.visibility]=\"sortBySignal() === option.id ? 'visible' : 'hidden'\"\n class=\"is-pr-2\" />\n <span [style.border-color]=\"option.color ?? 'transparent'\" class=\"is-pl-2 is-inline-block is-capitalized\">\n {{ option.label }}\n </span>\n </a>\n }\n </div>\n </div>\n</div>\n", styles: [".dropdown-item>span{border-left:2px solid transparent}\n"] }]
49
49
  }], propDecorators: { sortOptions: [{
50
50
  type: Input
51
51
  }], alignment: [{
@@ -55,4 +55,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
55
55
  }], sortOrder: [{
56
56
  type: Input
57
57
  }] } });
58
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ydC1zZWxlY3QuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbW1vbi9zb3J0LXNlbGVjdC9zb3J0LXNlbGVjdC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi9zcmMvY29tbW9uL3NvcnQtc2VsZWN0L3NvcnQtc2VsZWN0LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXBHLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDeEQsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQzFDLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxnQkFBZ0IsRUFBRSxpQkFBaUIsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBRTVHLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDRCQUE0QixDQUFDOztBQXVCbkUsTUFBTSxPQUFPLG1CQUFtQjtJQVJoQztRQVNxQixzQkFBaUIsR0FBRyxpQkFBaUIsQ0FBQztRQUtoRCxjQUFTLEdBQXFCLE9BQU8sQ0FBQztRQUVyQyxlQUFVLEdBQUcsTUFBTSxFQUFtQixDQUFDO1FBRXZDLGlCQUFZLEdBQUcsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRTFCLG9CQUFlLEdBQUcsTUFBTSxDQUFrQixNQUFNLENBQUMsQ0FBQztRQUVsRCxlQUFVLEdBQUcsS0FBSyxDQUFDO1FBVW5CLGNBQVMsR0FBRyxRQUFRLENBQzVCLEdBQUcsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLGVBQWUsRUFBRSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsa0JBQWtCLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixDQUFhLENBQzlGLENBQUM7S0FzQkg7SUFoQ0MsSUFBYSxNQUFNLENBQUMsS0FBYTtRQUMvQixJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBRUQsSUFBYSxTQUFTLENBQUMsS0FBc0I7UUFDM0MsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQU1TLElBQUksQ0FBQyxNQUFrQjtRQUMvQixJQUFJLElBQUksQ0FBQyxZQUFZLEVBQUUsS0FBSyxNQUFNLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDdEMsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1FBQ3ZCLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ2pDLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ25DLENBQUM7UUFDRCxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDekIsQ0FBQztJQUVPLGFBQWE7UUFDbkIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztJQUMzRSxDQUFDO0lBRU8sZUFBZTtRQUNyQixJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQztZQUNuQixNQUFNLEVBQUUsSUFBSSxDQUFDLFlBQVksRUFBRTtZQUMzQixTQUFTLEVBQUUsSUFBSSxDQUFDLGVBQWUsRUFBRTtTQUNsQyxDQUFDLENBQUM7SUFDTCxDQUFDOytHQS9DVSxtQkFBbUI7bUdBQW5CLG1CQUFtQiwyTkM5QmhDLGt4Q0E4QkEsa0hERlksT0FBTyxvRkFBRSxxQkFBcUIsMkhBQUUsVUFBVSwyVEFBRSxlQUFlOzs0RkFFMUQsbUJBQW1CO2tCQVIvQixTQUFTOytCQUNFLGdCQUFnQixtQkFHVCx1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxVQUFVLEVBQUUsZUFBZSxDQUFDOzhCQU0vRCxXQUFXO3NCQURqQixLQUFLO2dCQUdHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBVU8sTUFBTTtzQkFBbEIsS0FBSztnQkFJTyxTQUFTO3NCQUFyQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSW5wdXQsIGNvbXB1dGVkLCBvdXRwdXQsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSWNvblByb3AgfSBmcm9tICdAZm9ydGF3ZXNvbWUvZm9udGF3ZXNvbWUtc3ZnLWNvcmUnO1xuaW1wb3J0IHsgRmFJY29uQ29tcG9uZW50IH0gZnJvbSAnQGZvcnRhd2Vzb21lL2FuZ3VsYXItZm9udGF3ZXNvbWUnO1xuaW1wb3J0IHsgTmdiVG9vbHRpcCB9IGZyb20gJ0BuZy1ib290c3RyYXAvbmctYm9vdHN0cmFwJztcbmltcG9ydCB7IE5nQ2xhc3MgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgZmFMb25nQXJyb3dBbHREb3duLCBmYUxvbmdBcnJvd0FsdFVwLCBmYVNvcnRBbW91bnRVcEFsdCB9IGZyb20gJ0Bmb3J0YXdlc29tZS9mcmVlLXNvbGlkLXN2Zy1pY29ucyc7XG5cbmltcG9ydCB7IENsaWNrT3V0c2lkZURpcmVjdGl2ZSB9IGZyb20gJy4uL2NsaWNrLW91dHNpZGUuZGlyZWN0aXZlJztcblxuZXhwb3J0IGludGVyZmFjZSBTb3J0T3B0aW9uIHtcbiAgaWQ6IHN0cmluZztcbiAgbGFiZWw6IHN0cmluZztcbiAgY29sb3I/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU29ydFNlbGVjdEV2ZW50IHtcbiAgc29ydEJ5OiBzdHJpbmc7XG4gIHNvcnRPcmRlcjogU29ydFNlbGVjdE9yZGVyO1xufVxuXG5leHBvcnQgdHlwZSBTb3J0U2VsZWN0T3JkZXIgPSAnYXNjJyB8ICdkZXNjJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaGUtc29ydC1zZWxlY3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vc29ydC1zZWxlY3QuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9zb3J0LXNlbGVjdC5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW05nQ2xhc3MsIENsaWNrT3V0c2lkZURpcmVjdGl2ZSwgTmdiVG9vbHRpcCwgRmFJY29uQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBTb3J0U2VsZWN0Q29tcG9uZW50IHtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGZhU29ydEFtb3VudFVwQWx0ID0gZmFTb3J0QW1vdW50VXBBbHQ7XG5cbiAgQElucHV0KClcbiAgcHVibGljIHNvcnRPcHRpb25zOiBTb3J0T3B0aW9uW107XG5cbiAgQElucHV0KCkgYWxpZ25tZW50OiAnbGVmdCcgfCAncmlnaHQnID0gJ3JpZ2h0JztcblxuICBwcm90ZWN0ZWQgc29ydENoYW5nZSA9IG91dHB1dDxTb3J0U2VsZWN0RXZlbnQ+KCk7XG5cbiAgcHJvdGVjdGVkIHNvcnRCeVNpZ25hbCA9IHNpZ25hbCgnJyk7XG5cbiAgcHJvdGVjdGVkIHNvcnRPcmRlclNpZ25hbCA9IHNpZ25hbCg8U29ydFNlbGVjdE9yZGVyPidkZXNjJyk7XG5cbiAgcHJvdGVjdGVkIHNob3dTb3J0QnkgPSBmYWxzZTtcblxuICBASW5wdXQoKSBzZXQgc29ydEJ5KHZhbHVlOiBzdHJpbmcpIHtcbiAgICB0aGlzLnNvcnRCeVNpZ25hbC5zZXQodmFsdWUpO1xuICB9XG5cbiAgQElucHV0KCkgc2V0IHNvcnRPcmRlcih2YWx1ZTogU29ydFNlbGVjdE9yZGVyKSB7XG4gICAgdGhpcy5zb3J0T3JkZXJTaWduYWwuc2V0KHZhbHVlKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhcnJvd0ljb24gPSBjb21wdXRlZChcbiAgICAoKSA9PiAodGhpcy5zb3J0T3JkZXJTaWduYWwoKSA9PT0gJ2Rlc2MnID8gZmFMb25nQXJyb3dBbHREb3duIDogZmFMb25nQXJyb3dBbHRVcCkgYXMgSWNvblByb3BcbiAgKTtcblxuICBwcm90ZWN0ZWQgc29ydChvcHRpb246IFNvcnRPcHRpb24pIHtcbiAgICBpZiAodGhpcy5zb3J0QnlTaWduYWwoKSA9PT0gb3B0aW9uLmlkKSB7XG4gICAgICB0aGlzLl90b2dnbGVTb3J0QnkoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5zb3J0QnlTaWduYWwuc2V0KG9wdGlvbi5pZCk7XG4gICAgICB0aGlzLnNvcnRPcmRlclNpZ25hbC5zZXQoJ2Rlc2MnKTtcbiAgICB9XG4gICAgdGhpcy5fZW1pdFNvcnRDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3RvZ2dsZVNvcnRCeSgpIHtcbiAgICB0aGlzLnNvcnRPcmRlclNpZ25hbC51cGRhdGUodmFsdWUgPT4gKHZhbHVlID09PSAnYXNjJyA/ICdkZXNjJyA6ICdhc2MnKSk7XG4gIH1cblxuICBwcml2YXRlIF9lbWl0U29ydENoYW5nZSgpIHtcbiAgICB0aGlzLnNvcnRDaGFuZ2UuZW1pdCh7XG4gICAgICBzb3J0Qnk6IHRoaXMuc29ydEJ5U2lnbmFsKCksXG4gICAgICBzb3J0T3JkZXI6IHRoaXMuc29ydE9yZGVyU2lnbmFsKClcbiAgICB9KTtcbiAgfVxufVxuIiwiPGRpdiAoY2xpY2tPdXRzaWRlKT1cInNob3dTb3J0QnkgPSBmYWxzZVwiIFtjbGFzcy5pcy1hY3RpdmVdPVwic2hvd1NvcnRCeVwiIFtuZ0NsYXNzXT1cIidpcy0nICsgYWxpZ25tZW50XCIgY2xhc3M9XCJkcm9wZG93blwiPlxuICA8ZGl2IChjbGljayk9XCJzaG93U29ydEJ5ID0gIXNob3dTb3J0QnlcIiBjbGFzcz1cImRyb3Bkb3duLXRyaWdnZXJcIj5cbiAgICA8YnV0dG9uXG4gICAgICBbY2xhc3MuaXMtYWN0aXZlXT1cInNob3dTb3J0QnlcIlxuICAgICAgW2Rpc2FibGVUb29sdGlwXT1cInNob3dTb3J0QnlcIlxuICAgICAgYXJpYS1jb250cm9scz1cInNvcnQtbWVudVwiXG4gICAgICBhcmlhLWhhc3BvcHVwPVwidHJ1ZVwiXG4gICAgICBjbGFzcz1cImJ1dHRvbiBpcy1naG9zdFwiXG4gICAgICBjb250YWluZXI9XCJib2R5XCJcbiAgICAgIG5nYlRvb2x0aXA9XCJTb3J0IG9wdGlvbnNcIlxuICAgICAgcGxhY2VtZW50PVwiYm90dG9tXCI+XG4gICAgICA8ZmEtaWNvbiBhcmlhLWhpZGRlbj1cInRydWVcIiBbaWNvbl09XCJmYVNvcnRBbW91bnRVcEFsdFwiPjwvZmEtaWNvbj5cbiAgICA8L2J1dHRvbj5cbiAgPC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJkcm9wZG93bi1tZW51XCIgaWQ9XCJzb3J0LW1lbnVcIiByb2xlPVwibWVudVwiPlxuICAgIDxkaXYgKGNsaWNrKT1cInNob3dTb3J0QnkgPSBmYWxzZVwiIGNsYXNzPVwiZHJvcGRvd24tY29udGVudFwiPlxuICAgICAgQGZvciAob3B0aW9uIG9mIHNvcnRPcHRpb25zOyB0cmFjayBvcHRpb24pIHtcbiAgICAgICAgPGEgKGNsaWNrKT1cInNvcnQob3B0aW9uKVwiIFtjbGFzcy5pcy1zZWxlY3RlZF09XCJzb3J0QnlTaWduYWwoKSA9PT0gb3B0aW9uLmlkXCIgY2xhc3M9XCJkcm9wZG93bi1pdGVtXCI+XG4gICAgICAgICAgPGZhLWljb25cbiAgICAgICAgICAgIFtpY29uXT1cImFycm93SWNvbigpXCJcbiAgICAgICAgICAgIFtzdHlsZS52aXNpYmlsaXR5XT1cInNvcnRCeVNpZ25hbCgpID09PSBvcHRpb24uaWQgPyAndmlzaWJsZScgOiAnaGlkZGVuJ1wiXG4gICAgICAgICAgICBjbGFzcz1cImlzLXByLTJcIj48L2ZhLWljb24+XG4gICAgICAgICAgPHNwYW4gW3N0eWxlLmJvcmRlci1jb2xvcl09XCJvcHRpb24uY29sb3IgPz8gJ3RyYW5zcGFyZW50J1wiIGNsYXNzPVwiaXMtcGwtMiBpcy1pbmxpbmUtYmxvY2sgaXMtY2FwaXRhbGl6ZWRcIj5cbiAgICAgICAgICAgIHt7IG9wdGlvbi5sYWJlbCB9fVxuICAgICAgICAgIDwvc3Bhbj5cbiAgICAgICAgPC9hPlxuICAgICAgfVxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuIl19
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ydC1zZWxlY3QuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbW1vbi9zb3J0LXNlbGVjdC9zb3J0LXNlbGVjdC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi9zcmMvY29tbW9uL3NvcnQtc2VsZWN0L3NvcnQtc2VsZWN0LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXBHLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDeEQsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQzFDLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxnQkFBZ0IsRUFBRSxpQkFBaUIsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBRTVHLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDRCQUE0QixDQUFDOztBQXVCbkUsTUFBTSxPQUFPLG1CQUFtQjtJQVJoQztRQVNxQixzQkFBaUIsR0FBRyxpQkFBaUIsQ0FBQztRQUtoRCxjQUFTLEdBQXFCLE9BQU8sQ0FBQztRQUVyQyxlQUFVLEdBQUcsTUFBTSxFQUFtQixDQUFDO1FBRXZDLGlCQUFZLEdBQUcsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRTFCLG9CQUFlLEdBQUcsTUFBTSxDQUFrQixNQUFNLENBQUMsQ0FBQztRQUVsRCxlQUFVLEdBQUcsS0FBSyxDQUFDO1FBVW5CLGNBQVMsR0FBRyxRQUFRLENBQzVCLEdBQUcsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLGVBQWUsRUFBRSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsa0JBQWtCLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixDQUFhLENBQzlGLENBQUM7S0FzQkg7SUFoQ0MsSUFBYSxNQUFNLENBQUMsS0FBYTtRQUMvQixJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvQixDQUFDO0lBRUQsSUFBYSxTQUFTLENBQUMsS0FBc0I7UUFDM0MsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQU1TLElBQUksQ0FBQyxNQUFrQjtRQUMvQixJQUFJLElBQUksQ0FBQyxZQUFZLEVBQUUsS0FBSyxNQUFNLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDdEMsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1FBQ3ZCLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ2pDLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ25DLENBQUM7UUFDRCxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDekIsQ0FBQztJQUVPLGFBQWE7UUFDbkIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztJQUMzRSxDQUFDO0lBRU8sZUFBZTtRQUNyQixJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQztZQUNuQixNQUFNLEVBQUUsSUFBSSxDQUFDLFlBQVksRUFBRTtZQUMzQixTQUFTLEVBQUUsSUFBSSxDQUFDLGVBQWUsRUFBRTtTQUNsQyxDQUFDLENBQUM7SUFDTCxDQUFDOytHQS9DVSxtQkFBbUI7bUdBQW5CLG1CQUFtQiwyTkM5QmhDLGt3Q0E4QkEsa0hERlksT0FBTyxvRkFBRSxxQkFBcUIsMkhBQUUsVUFBVSwyVEFBRSxlQUFlOzs0RkFFMUQsbUJBQW1CO2tCQVIvQixTQUFTOytCQUNFLGdCQUFnQixtQkFHVCx1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxVQUFVLEVBQUUsZUFBZSxDQUFDOzhCQU0vRCxXQUFXO3NCQURqQixLQUFLO2dCQUdHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBVU8sTUFBTTtzQkFBbEIsS0FBSztnQkFJTyxTQUFTO3NCQUFyQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSW5wdXQsIGNvbXB1dGVkLCBvdXRwdXQsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSWNvblByb3AgfSBmcm9tICdAZm9ydGF3ZXNvbWUvZm9udGF3ZXNvbWUtc3ZnLWNvcmUnO1xuaW1wb3J0IHsgRmFJY29uQ29tcG9uZW50IH0gZnJvbSAnQGZvcnRhd2Vzb21lL2FuZ3VsYXItZm9udGF3ZXNvbWUnO1xuaW1wb3J0IHsgTmdiVG9vbHRpcCB9IGZyb20gJ0BuZy1ib290c3RyYXAvbmctYm9vdHN0cmFwJztcbmltcG9ydCB7IE5nQ2xhc3MgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgZmFMb25nQXJyb3dBbHREb3duLCBmYUxvbmdBcnJvd0FsdFVwLCBmYVNvcnRBbW91bnRVcEFsdCB9IGZyb20gJ0Bmb3J0YXdlc29tZS9mcmVlLXNvbGlkLXN2Zy1pY29ucyc7XG5cbmltcG9ydCB7IENsaWNrT3V0c2lkZURpcmVjdGl2ZSB9IGZyb20gJy4uL2NsaWNrLW91dHNpZGUuZGlyZWN0aXZlJztcblxuZXhwb3J0IGludGVyZmFjZSBTb3J0T3B0aW9uIHtcbiAgaWQ6IHN0cmluZztcbiAgbGFiZWw6IHN0cmluZztcbiAgY29sb3I/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU29ydFNlbGVjdEV2ZW50IHtcbiAgc29ydEJ5OiBzdHJpbmc7XG4gIHNvcnRPcmRlcjogU29ydFNlbGVjdE9yZGVyO1xufVxuXG5leHBvcnQgdHlwZSBTb3J0U2VsZWN0T3JkZXIgPSAnYXNjJyB8ICdkZXNjJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaGUtc29ydC1zZWxlY3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vc29ydC1zZWxlY3QuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9zb3J0LXNlbGVjdC5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW05nQ2xhc3MsIENsaWNrT3V0c2lkZURpcmVjdGl2ZSwgTmdiVG9vbHRpcCwgRmFJY29uQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBTb3J0U2VsZWN0Q29tcG9uZW50IHtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGZhU29ydEFtb3VudFVwQWx0ID0gZmFTb3J0QW1vdW50VXBBbHQ7XG5cbiAgQElucHV0KClcbiAgcHVibGljIHNvcnRPcHRpb25zOiBTb3J0T3B0aW9uW107XG5cbiAgQElucHV0KCkgYWxpZ25tZW50OiAnbGVmdCcgfCAncmlnaHQnID0gJ3JpZ2h0JztcblxuICBwcm90ZWN0ZWQgc29ydENoYW5nZSA9IG91dHB1dDxTb3J0U2VsZWN0RXZlbnQ+KCk7XG5cbiAgcHJvdGVjdGVkIHNvcnRCeVNpZ25hbCA9IHNpZ25hbCgnJyk7XG5cbiAgcHJvdGVjdGVkIHNvcnRPcmRlclNpZ25hbCA9IHNpZ25hbCg8U29ydFNlbGVjdE9yZGVyPidkZXNjJyk7XG5cbiAgcHJvdGVjdGVkIHNob3dTb3J0QnkgPSBmYWxzZTtcblxuICBASW5wdXQoKSBzZXQgc29ydEJ5KHZhbHVlOiBzdHJpbmcpIHtcbiAgICB0aGlzLnNvcnRCeVNpZ25hbC5zZXQodmFsdWUpO1xuICB9XG5cbiAgQElucHV0KCkgc2V0IHNvcnRPcmRlcih2YWx1ZTogU29ydFNlbGVjdE9yZGVyKSB7XG4gICAgdGhpcy5zb3J0T3JkZXJTaWduYWwuc2V0KHZhbHVlKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhcnJvd0ljb24gPSBjb21wdXRlZChcbiAgICAoKSA9PiAodGhpcy5zb3J0T3JkZXJTaWduYWwoKSA9PT0gJ2Rlc2MnID8gZmFMb25nQXJyb3dBbHREb3duIDogZmFMb25nQXJyb3dBbHRVcCkgYXMgSWNvblByb3BcbiAgKTtcblxuICBwcm90ZWN0ZWQgc29ydChvcHRpb246IFNvcnRPcHRpb24pIHtcbiAgICBpZiAodGhpcy5zb3J0QnlTaWduYWwoKSA9PT0gb3B0aW9uLmlkKSB7XG4gICAgICB0aGlzLl90b2dnbGVTb3J0QnkoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5zb3J0QnlTaWduYWwuc2V0KG9wdGlvbi5pZCk7XG4gICAgICB0aGlzLnNvcnRPcmRlclNpZ25hbC5zZXQoJ2Rlc2MnKTtcbiAgICB9XG4gICAgdGhpcy5fZW1pdFNvcnRDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3RvZ2dsZVNvcnRCeSgpIHtcbiAgICB0aGlzLnNvcnRPcmRlclNpZ25hbC51cGRhdGUodmFsdWUgPT4gKHZhbHVlID09PSAnYXNjJyA/ICdkZXNjJyA6ICdhc2MnKSk7XG4gIH1cblxuICBwcml2YXRlIF9lbWl0U29ydENoYW5nZSgpIHtcbiAgICB0aGlzLnNvcnRDaGFuZ2UuZW1pdCh7XG4gICAgICBzb3J0Qnk6IHRoaXMuc29ydEJ5U2lnbmFsKCksXG4gICAgICBzb3J0T3JkZXI6IHRoaXMuc29ydE9yZGVyU2lnbmFsKClcbiAgICB9KTtcbiAgfVxufVxuIiwiPGRpdiAoY2xpY2tPdXRzaWRlKT1cInNob3dTb3J0QnkgPSBmYWxzZVwiIFtjbGFzcy5pcy1hY3RpdmVdPVwic2hvd1NvcnRCeVwiIFtuZ0NsYXNzXT1cIidpcy0nICsgYWxpZ25tZW50XCIgY2xhc3M9XCJkcm9wZG93blwiPlxuICA8ZGl2IChjbGljayk9XCJzaG93U29ydEJ5ID0gIXNob3dTb3J0QnlcIiBjbGFzcz1cImRyb3Bkb3duLXRyaWdnZXJcIj5cbiAgICA8YnV0dG9uXG4gICAgICBbY2xhc3MuaXMtYWN0aXZlXT1cInNob3dTb3J0QnlcIlxuICAgICAgW2Rpc2FibGVUb29sdGlwXT1cInNob3dTb3J0QnlcIlxuICAgICAgYXJpYS1jb250cm9scz1cInNvcnQtbWVudVwiXG4gICAgICBhcmlhLWhhc3BvcHVwPVwidHJ1ZVwiXG4gICAgICBjbGFzcz1cImJ1dHRvbiBpcy1naG9zdFwiXG4gICAgICBjb250YWluZXI9XCJib2R5XCJcbiAgICAgIG5nYlRvb2x0aXA9XCJTb3J0IG9wdGlvbnNcIlxuICAgICAgcGxhY2VtZW50PVwiYm90dG9tXCI+XG4gICAgICA8ZmEtaWNvbiBhcmlhLWhpZGRlbj1cInRydWVcIiBbaWNvbl09XCJmYVNvcnRBbW91bnRVcEFsdFwiIC8+XG4gICAgPC9idXR0b24+XG4gIDwvZGl2PlxuICA8ZGl2IGNsYXNzPVwiZHJvcGRvd24tbWVudVwiIGlkPVwic29ydC1tZW51XCIgcm9sZT1cIm1lbnVcIj5cbiAgICA8ZGl2IChjbGljayk9XCJzaG93U29ydEJ5ID0gZmFsc2VcIiBjbGFzcz1cImRyb3Bkb3duLWNvbnRlbnRcIj5cbiAgICAgIEBmb3IgKG9wdGlvbiBvZiBzb3J0T3B0aW9uczsgdHJhY2sgb3B0aW9uKSB7XG4gICAgICAgIDxhIChjbGljayk9XCJzb3J0KG9wdGlvbilcIiBbY2xhc3MuaXMtc2VsZWN0ZWRdPVwic29ydEJ5U2lnbmFsKCkgPT09IG9wdGlvbi5pZFwiIGNsYXNzPVwiZHJvcGRvd24taXRlbVwiPlxuICAgICAgICAgIDxmYS1pY29uXG4gICAgICAgICAgICBbaWNvbl09XCJhcnJvd0ljb24oKVwiXG4gICAgICAgICAgICBbc3R5bGUudmlzaWJpbGl0eV09XCJzb3J0QnlTaWduYWwoKSA9PT0gb3B0aW9uLmlkID8gJ3Zpc2libGUnIDogJ2hpZGRlbidcIlxuICAgICAgICAgICAgY2xhc3M9XCJpcy1wci0yXCIgLz5cbiAgICAgICAgICA8c3BhbiBbc3R5bGUuYm9yZGVyLWNvbG9yXT1cIm9wdGlvbi5jb2xvciA/PyAndHJhbnNwYXJlbnQnXCIgY2xhc3M9XCJpcy1wbC0yIGlzLWlubGluZS1ibG9jayBpcy1jYXBpdGFsaXplZFwiPlxuICAgICAgICAgICAge3sgb3B0aW9uLmxhYmVsIH19XG4gICAgICAgICAgPC9zcGFuPlxuICAgICAgICA8L2E+XG4gICAgICB9XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuPC9kaXY+XG4iXX0=
@@ -92,7 +92,7 @@ export class CyclesCompletenessComponent {
92
92
  component.headerKeys.set(headerKeys);
93
93
  }
94
94
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesCompletenessComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
95
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesCompletenessComponent, isStandalone: true, selector: "he-cycles-completeness", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n", styles: [""], dependencies: [{ kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
95
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.11", type: CyclesCompletenessComponent, isStandalone: true, selector: "he-cycles-completeness", inputs: { dataState: { classPropertyName: "dataState", publicName: "dataState", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n", styles: [""], dependencies: [{ kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "mask", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["dataState", "nodeType", "dataKey", "key", "node", "state", "linkClass"] }, { kind: "component", type: BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "component", type: NodeLogsModelsComponent, selector: "he-node-logs-models", inputs: ["node", "nodeKey", "originalValues", "recalculatedValues", "terms", "filterTermTypes", "filterTermTypesLabel", "logsKey", "noDataMessage"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
96
96
  }
97
97
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: CyclesCompletenessComponent, decorators: [{
98
98
  type: Component,
@@ -105,6 +105,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
105
105
  BlankNodeStateNoticeComponent,
106
106
  FormsModule,
107
107
  NodeLogsModelsComponent
108
- ], template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n" }]
108
+ ], template: "<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n <div class=\"column\">\n @if (selectedView() === View.table) {\n <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n </button>\n }\n </div>\n @if (views()?.length > 1) {\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n @for (view of views(); track view) {\n <div class=\"control\">\n <button\n class=\"button is-small\"\n [class.is-selected]=\"selectedView() === view\"\n (click)=\"selectedView.set(view)\">\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n </span>\n <span class=\"is-hidden-mobile\">{{ view }}</span>\n </button>\n </div>\n }\n </div>\n </div>\n }\n</div>\n\n@switch (selectedView()) {\n @case (View.table) {\n <div class=\"px-3 pb-3\">\n @if (hasData()) {\n <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n <table class=\"table is-narrow is-striped\">\n <thead>\n <tr class=\"has-text-weight-semibold\">\n <th class=\"width-auto has-border-right\"></th>\n @for (completeness of completenessKeys(); track completeness) {\n <th [attr.title]=\"completeness\">\n <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n {{ keyToLabel(completeness) }}\n </a>\n </th>\n }\n </tr>\n </thead>\n <tbody>\n @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n <tr>\n <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n <he-node-link [node]=\"cycle\">\n <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n </he-node-link>\n </td>\n @for (key of completenessKeys(); track key) {\n <td class=\"is-nowrap\">\n <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n <he-blank-node-state\n class=\"ml-1\"\n [dataState]=\"dataState()\"\n [node]=\"getCompleteness(cycle)\"\n [key]=\"key\"></he-blank-node-state>\n </td>\n }\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n } @else {\n <div class=\"panel-block\">\n <span>No completeness data</span>\n </div>\n }\n </div>\n }\n @case (View.logs) {\n @if (cycles().length > 1) {\n <div class=\"field has-addons pt-2 px-3\">\n <div class=\"control\">\n <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-small is-fullwidth is-secondary\">\n <select (change)=\"selectIndex($event)\">\n @for (value of cycles(); track value; let i = $index) {\n <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n }\n </select>\n </div>\n </div>\n </div>\n }\n @if (selectedNode()) {\n <he-node-logs-models\n [node]=\"selectedNode()\"\n [nodeKey]=\"nodeKey\"\n [logsKey]=\"selectedLogsKey()\"\n [originalValues]=\"selectedOriginalValues()\"\n [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n }\n }\n}\n" }]
109
109
  }], ctorParameters: () => [] });
110
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-completeness.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-completeness/cycles-completeness.component.ts","../../../../src/cycles/cycles-completeness/cycles-completeness.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { IconDefinition, faCalculator, faDownload, faList } from '@fortawesome/free-solid-svg-icons';\nimport { NgbModal, NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, NodeType, NonBlankNodesKey } from '@hestia-earth/schema';\nimport { keyToLabel } from '@hestia-earth/utils';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { isValidKey } from '../../common/node-utils';\nimport { defaultLabel, schemaBaseUrl } from '../../common/utils';\nimport { logsKey } from '../cycles.model';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\nenum View {\n  table = 'Table view',\n  logs = 'Recalculations logs'\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.logs]: faCalculator,\n  [View.table]: faList\n};\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.completeness.'];\n\n// backward compatibility with schema version < 14\nconst getCompleteness = (cycle: any) => cycle?.completeness || cycle?.dataCompleteness || {};\n\n@Component({\n  selector: 'he-cycles-completeness',\n  templateUrl: './cycles-completeness.component.html',\n  styleUrls: ['./cycles-completeness.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    FormsModule,\n    NodeLogsModelsComponent\n  ]\n})\nexport class CyclesCompletenessComponent {\n  private readonly modalService = inject(NgbModal);\n  private readonly nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly defaultLabel = defaultLabel;\n  protected readonly keyToLabel = keyToLabel;\n  protected readonly getCompleteness = getCompleteness;\n  protected readonly nodeKey = NonBlankNodesKey.completeness;\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.logs]: !this.isOriginal(),\n        [View.table]: true\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(() => Object.values(View).filter(view => this.showView()[view]) ?? []);\n  protected readonly selectedView = signal(View.table);\n\n  private readonly originalCycles = toSignal(\n    this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original)\n  );\n  private readonly _allCycles = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  protected readonly cycles = computed(() => this._allCycles()?.map(data => data[this.dataState()]) || []);\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(() => getCompleteness(this.ogirinalSelectedCycle()));\n  protected readonly selectedRecalculatedValues = computed(() => getCompleteness(this.selectedCycle()));\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected readonly hasData = computed(() => this.cycles().length);\n  protected readonly isOriginal = computed(() => this.dataState() === DataState.original);\n\n  protected readonly completenessKeys = computed(() =>\n    Object.keys(getCompleteness(this.selectedCycle())).filter(isValidKey).sort()\n  );\n\n  constructor() {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView() === View.logs) {\n        this.selectedView.set(View.table);\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set('completeness.csv');\n    component.isUpload.set(false);\n    component.headerKeys.set(headerKeys);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    @if (selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\"></fa-icon>\n      </button>\n    }\n  </div>\n  @if (views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\"></fa-icon>\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@switch (selectedView()) {\n  @case (View.table) {\n    <div class=\"px-3 pb-3\">\n      @if (hasData()) {\n        <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n          <table class=\"table is-narrow is-striped\">\n            <thead>\n              <tr class=\"has-text-weight-semibold\">\n                <th class=\"width-auto has-border-right\"></th>\n                @for (completeness of completenessKeys(); track completeness) {\n                  <th [attr.title]=\"completeness\">\n                    <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n                      {{ keyToLabel(completeness) }}\n                    </a>\n                  </th>\n                }\n              </tr>\n            </thead>\n            <tbody>\n              @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n                <tr>\n                  <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                    <he-node-link [node]=\"cycle\">\n                      <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                    </he-node-link>\n                  </td>\n                  @for (key of completenessKeys(); track key) {\n                    <td class=\"is-nowrap\">\n                      <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n                      <he-blank-node-state\n                        class=\"ml-1\"\n                        [dataState]=\"dataState()\"\n                        [node]=\"getCompleteness(cycle)\"\n                        [key]=\"key\"></he-blank-node-state>\n                    </td>\n                  }\n                </tr>\n              }\n            </tbody>\n          </table>\n        </he-data-table>\n        <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n      } @else {\n        <div class=\"panel-block\">\n          <span>No completeness data</span>\n        </div>\n      }\n    </div>\n  }\n  @case (View.logs) {\n    @if (cycles().length > 1) {\n      <div class=\"field has-addons pt-2 px-3\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n        </div>\n        <div class=\"control is-expanded\">\n          <div class=\"select is-small is-fullwidth is-secondary\">\n            <select (change)=\"selectIndex($event)\">\n              @for (value of cycles(); track value; let i = $index) {\n                <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n              }\n            </select>\n          </div>\n        </div>\n      </div>\n    }\n    @if (selectedNode()) {\n      <he-node-logs-models\n        [node]=\"selectedNode()\"\n        [nodeKey]=\"nodeKey\"\n        [logsKey]=\"selectedLogsKey()\"\n        [originalValues]=\"selectedOriginalValues()\"\n        [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n    }\n  }\n}\n"]}
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cycles-completeness.component.js","sourceRoot":"","sources":["../../../../src/cycles/cycles-completeness/cycles-completeness.component.ts","../../../../src/cycles/cycles-completeness/cycles-completeness.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject, input, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { IconDefinition, faCalculator, faDownload, faList } from '@fortawesome/free-solid-svg-icons';\nimport { NgbModal, NgbTooltip } from '@ng-bootstrap/ng-bootstrap';\nimport { DataState } from '@hestia-earth/api';\nimport { ICycleJSONLD, NodeType, NonBlankNodesKey } from '@hestia-earth/schema';\nimport { keyToLabel } from '@hestia-earth/utils';\n\nimport { HeNodeStoreService } from '../../node/node-store.service';\nimport { isValidKey } from '../../common/node-utils';\nimport { defaultLabel, schemaBaseUrl } from '../../common/utils';\nimport { logsKey } from '../cycles.model';\nimport { NodeCsvExportConfirmComponent } from '../../node/node-csv-export-confirm/node-csv-export-confirm.component';\nimport { NodeLogsModelsComponent } from '../../node/node-logs-models/node-logs-models.component';\nimport { BlankNodeStateNoticeComponent } from '../../common/blank-node-state-notice/blank-node-state-notice.component';\nimport { BlankNodeStateComponent } from '../../common/blank-node-state/blank-node-state.component';\nimport { NodeLinkComponent } from '../../node/node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\nenum View {\n  table = 'Table view',\n  logs = 'Recalculations logs'\n}\n\nconst viewIcon: {\n  [view in View]: IconDefinition;\n} = {\n  [View.logs]: faCalculator,\n  [View.table]: faList\n};\n\nconst headerKeys = ['cycle.id', 'cycle.@id', 'cycle.completeness.'];\n\n// backward compatibility with schema version < 14\nconst getCompleteness = (cycle: any) => cycle?.completeness || cycle?.dataCompleteness || {};\n\n@Component({\n  selector: 'he-cycles-completeness',\n  templateUrl: './cycles-completeness.component.html',\n  styleUrls: ['./cycles-completeness.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgbTooltip,\n    FaIconComponent,\n    DataTableComponent,\n    NodeLinkComponent,\n    BlankNodeStateComponent,\n    BlankNodeStateNoticeComponent,\n    FormsModule,\n    NodeLogsModelsComponent\n  ]\n})\nexport class CyclesCompletenessComponent {\n  private readonly modalService = inject(NgbModal);\n  private readonly nodeStoreService = inject(HeNodeStoreService);\n\n  protected readonly faDownload = faDownload;\n\n  protected readonly dataState = input(DataState.original);\n\n  protected readonly schemaBaseUrl = schemaBaseUrl();\n  protected readonly defaultLabel = defaultLabel;\n  protected readonly keyToLabel = keyToLabel;\n  protected readonly getCompleteness = getCompleteness;\n  protected readonly nodeKey = NonBlankNodesKey.completeness;\n\n  protected readonly View = View;\n  protected readonly viewIcon = viewIcon;\n  private readonly showView = computed(\n    () =>\n      ({\n        [View.logs]: !this.isOriginal(),\n        [View.table]: true\n      }) as { [view in View]: boolean }\n  );\n  protected readonly views = computed(() => Object.values(View).filter(view => this.showView()[view]) ?? []);\n  protected readonly selectedView = signal(View.table);\n\n  private readonly originalCycles = toSignal(\n    this.nodeStoreService.findByState$<ICycleJSONLD>(NodeType.Cycle, DataState.original)\n  );\n  private readonly _allCycles = toSignal(this.nodeStoreService.find$<ICycleJSONLD>(NodeType.Cycle));\n  protected readonly cycles = computed(() => this._allCycles()?.map(data => data[this.dataState()]) || []);\n\n  private readonly selectedIndex = signal(0);\n  private readonly ogirinalSelectedCycle = computed(() => this.originalCycles()?.[this.selectedIndex()]);\n  private readonly selectedCycle = computed(() => this.cycles()?.[this.selectedIndex()]);\n  protected readonly selectedLogsKey = computed(() => logsKey(this.selectedCycle()));\n  protected readonly selectedOriginalValues = computed(() => getCompleteness(this.ogirinalSelectedCycle()));\n  protected readonly selectedRecalculatedValues = computed(() => getCompleteness(this.selectedCycle()));\n  protected readonly selectedNode = computed(() =>\n    this.selectedCycle()\n      ? {\n          ...this.selectedCycle(),\n          '@type': NodeType.Cycle,\n          type: NodeType.Cycle,\n          dataState: DataState.recalculated\n        }\n      : null\n  );\n\n  protected readonly hasData = computed(() => this.cycles().length);\n  protected readonly isOriginal = computed(() => this.dataState() === DataState.original);\n\n  protected readonly completenessKeys = computed(() =>\n    Object.keys(getCompleteness(this.selectedCycle())).filter(isValidKey).sort()\n  );\n\n  constructor() {\n    effect(() => {\n      // make sure logs does not remain displayed when switching back to original view\n      if (this.isOriginal() && this.selectedView() === View.logs) {\n        this.selectedView.set(View.table);\n      }\n    });\n  }\n\n  protected trackById(_index: number, item: ICycleJSONLD) {\n    return item['@id'];\n  }\n\n  protected selectIndex({ target: { value } }) {\n    this.selectedIndex.set(+value);\n  }\n\n  protected showDownload() {\n    const instance = this.modalService.open(NodeCsvExportConfirmComponent);\n    const component = instance.componentInstance as NodeCsvExportConfirmComponent;\n    component.nodes.set(this.cycles());\n    component.filename.set('completeness.csv');\n    component.isUpload.set(false);\n    component.headerKeys.set(headerKeys);\n  }\n}\n","<div class=\"columns is-variable is-align-items-center is-2 m-0\">\n  <div class=\"column\">\n    @if (selectedView() === View.table) {\n      <button class=\"button is-small is-ghost\" (click)=\"showDownload()\">\n        <fa-icon [icon]=\"faDownload\" size=\"lg\" />\n      </button>\n    }\n  </div>\n  @if (views()?.length > 1) {\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        @for (view of views(); track view) {\n          <div class=\"control\">\n            <button\n              class=\"button is-small\"\n              [class.is-selected]=\"selectedView() === view\"\n              (click)=\"selectedView.set(view)\">\n              <span class=\"icon is-small\">\n                <fa-icon [icon]=\"viewIcon[view]\" aria-hidden=\"true\" />\n              </span>\n              <span class=\"is-hidden-mobile\">{{ view }}</span>\n            </button>\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n@switch (selectedView()) {\n  @case (View.table) {\n    <div class=\"px-3 pb-3\">\n      @if (hasData()) {\n        <he-data-table class=\"mb-1 is-small\" [small]=\"true\" maxHeight=\"320\">\n          <table class=\"table is-narrow is-striped\">\n            <thead>\n              <tr class=\"has-text-weight-semibold\">\n                <th class=\"width-auto has-border-right\"></th>\n                @for (completeness of completenessKeys(); track completeness) {\n                  <th [attr.title]=\"completeness\">\n                    <a [href]=\"schemaBaseUrl + '/Completeness#' + completeness\" target=\"_blank\">\n                      {{ keyToLabel(completeness) }}\n                    </a>\n                  </th>\n                }\n              </tr>\n            </thead>\n            <tbody>\n              @for (cycle of cycles(); track trackById(i, cycle); let i = $index) {\n                <tr>\n                  <td class=\"width-auto has-border-right\" [attr.title]=\"defaultLabel(cycle)\">\n                    <he-node-link [node]=\"cycle\">\n                      <span>{{ i + 1 }}. {{ defaultLabel(cycle) }}</span>\n                    </he-node-link>\n                  </td>\n                  @for (key of completenessKeys(); track key) {\n                    <td class=\"is-nowrap\">\n                      <span>{{ getCompleteness(cycle)[key] ? 'Complete' : 'Incomplete' }}</span>\n                      <he-blank-node-state\n                        class=\"ml-1\"\n                        [dataState]=\"dataState()\"\n                        [node]=\"getCompleteness(cycle)\"\n                        [key]=\"key\"></he-blank-node-state>\n                    </td>\n                  }\n                </tr>\n              }\n            </tbody>\n          </table>\n        </he-data-table>\n        <he-blank-node-state-notice [dataState]=\"dataState()\"></he-blank-node-state-notice>\n      } @else {\n        <div class=\"panel-block\">\n          <span>No completeness data</span>\n        </div>\n      }\n    </div>\n  }\n  @case (View.logs) {\n    @if (cycles().length > 1) {\n      <div class=\"field has-addons pt-2 px-3\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static is-secondary\">Select a Cycle</span>\n        </div>\n        <div class=\"control is-expanded\">\n          <div class=\"select is-small is-fullwidth is-secondary\">\n            <select (change)=\"selectIndex($event)\">\n              @for (value of cycles(); track value; let i = $index) {\n                <option [value]=\"i\">{{ i + 1 }}. {{ defaultLabel(value) }}</option>\n              }\n            </select>\n          </div>\n        </div>\n      </div>\n    }\n    @if (selectedNode()) {\n      <he-node-logs-models\n        [node]=\"selectedNode()\"\n        [nodeKey]=\"nodeKey\"\n        [logsKey]=\"selectedLogsKey()\"\n        [originalValues]=\"selectedOriginalValues()\"\n        [recalculatedValues]=\"selectedRecalculatedValues()\"></he-node-logs-models>\n    }\n  }\n}\n"]}