@nova-ui/bits
Version:
SolarWinds Nova Framework
237 lines • 41.5 kB
JavaScript
import { Component, ElementRef, EventEmitter, Input, Output, Renderer2, ViewChild, ViewEncapsulation, } from "@angular/core";
import _assign from "lodash/assign";
import _isEqual from "lodash/isEqual";
import _values from "lodash/values";
import { Subject } from "rxjs";
import { takeUntil } from "rxjs/operators";
import { SorterDirection } from "./public-api";
import { SorterKeyboardService } from "./sorter-keyboard.service";
import { LoggerService } from "../../services/log-service";
import { MenuPopupComponent } from "../menu";
import { OVERLAY_WITH_POPUP_STYLES_CLASS } from "../overlay/constants";
import { OverlayComponent } from "../overlay/overlay-component/overlay.component";
import * as i0 from "@angular/core";
import * as i1 from "../../services/log-service";
import * as i2 from "./sorter-keyboard.service";
import * as i3 from "@angular/common";
import * as i4 from "../menu/menu-popup/menu-popup.component";
import * as i5 from "../button/button.component";
import * as i6 from "../overlay/overlay-component/overlay.component";
// <example-url>./../examples/index.html#/sorter</example-url>
export class SorterComponent {
constructor(logger, sorterKeyboardService, elRef, renderer) {
this.logger = logger;
this.sorterKeyboardService = sorterKeyboardService;
this.elRef = elRef;
this.renderer = renderer;
this.appendToBody = false;
this.sorterAction = new EventEmitter();
// mark this filter to be monitored by our datasource for any changes in order reset other filters(eg: pagination)
// before any new search is performed
this.detectFilterChanges = true;
this.onDestroy$ = new Subject();
this.items = [
{
itemsSource: [],
},
];
this.overlayConfig = {
panelClass: [OVERLAY_WITH_POPUP_STYLES_CLASS],
};
this.sortIcons = {
[SorterDirection.ascending]: "arrow-up",
[SorterDirection.descending]: "arrow-down",
};
this.menuKeyControlListeners = [];
}
ngOnInit() {
this.onAppendToBodyChange(this.appendToBody);
}
ngOnChanges(changes) {
if (changes.itemsSource &&
!_isEqual(changes.itemsSource.previousValue, changes.itemsSource.currentValue)) {
if (this.itemsSource?.length > 0 &&
typeof this.itemsSource[0] === "string") {
this.logger
.warn(`The 'string[]' type for the sorter's itemsSource input is deprecated as of Nova v9. \
Instead, use type 'IMenuItem[]' which is internationalizable with title and value properties for each item.`);
}
this.initSelectedItem();
this.initPopupItems();
}
if (changes.selectedItem &&
this.sortConfig?.sortBy !== changes.selectedItem.currentValue &&
!changes.selectedItem.firstChange) {
const oldValue = _assign({}, this.sortConfig, {
sortBy: changes.selectedItem.previousValue,
});
this.selectedItem = changes.selectedItem.currentValue;
this.sortConfig = _assign({}, this.sortConfig, {
sortBy: changes.selectedItem.currentValue,
});
this.triggerSorterAction(oldValue);
this.setPopupSelection();
}
if (changes.sortDirection &&
!changes.sortDirection.firstChange &&
this.sortConfig?.direction !== changes.sortDirection.currentValue) {
const oldValue = this.sortConfig;
this.sortDirection = changes.sortDirection.currentValue;
this.sortConfig = _assign({}, this.sortConfig, {
direction: this.sortDirection,
});
this.triggerSorterAction(oldValue);
}
if (changes.appendToBody) {
this.onAppendToBodyChange(changes.appendToBody.currentValue);
}
}
ngAfterViewInit() {
this.initSelectedItem();
this.initSortDirection();
this.sortConfig = {
sortBy: this.selectedItem,
direction: this.sortDirection,
};
this.overlay.clickOutside
.pipe(takeUntil(this.onDestroy$))
.subscribe((_) => this.overlay.hide());
this.updateOverlayWidth();
this.initKeyboardService();
this.menuKeyControlListeners.push(this.renderer.listen(this.elRef.nativeElement, "keydown", (event) => {
this.sorterKeyboardService.handleKeydown(event);
}));
}
select(item) {
// perform update only if the new value actually changes
if (this.selectedItem !== item.value) {
const oldValue = this.sortConfig;
this.selectedItem = item.value;
this.sortConfig = _assign({}, this.sortConfig, {
sortBy: item.value,
});
this.updateOverlayWidth();
this.triggerSorterAction(oldValue);
this.setPopupSelection();
this.overlay.hide();
}
}
toggleSortDirection() {
const oldValue = this.sortConfig;
this.sortDirection =
this.sortDirection === SorterDirection.ascending
? SorterDirection.descending
: SorterDirection.ascending;
this.sortConfig = _assign({}, this.sortConfig, {
direction: this.sortDirection,
});
this.sorterAction.emit({ newValue: this.sortConfig, oldValue });
}
getSelectedItem() {
return this.selectedItem;
}
getSelectedItemTitle() {
return this.items[0].itemsSource.find((item) => item.isSelected)?.title;
}
getSortIcon() {
return this.sortIcons[this.sortDirection];
}
getFilters() {
return {
type: "sorter",
value: { ...this.sortConfig },
};
}
updateOverlayWidth() {
this.overlayConfig.minWidth = this.toggleRef.nativeElement.offsetWidth;
}
getAriaLabelForSortingButton() {
return this.sortDirection === SorterDirection.descending
? `${this.getSelectedItemTitle()}. Sorter direction - descending`
: `${this.getSelectedItemTitle()}. Sorter direction - ascending`;
}
ngOnDestroy() {
this.menuKeyControlListeners.forEach((listener) => listener());
this.onDestroy$.next();
this.onDestroy$.complete();
}
toggleSorterMenu() {
this.overlay.toggle();
this.sorterKeyboardService.announceDropdown();
}
initSelectedItem() {
// skip initialization in case we already have a value
// or if the itemsSource are lazy loaded
if (this.selectedItem || !this.itemsSource?.length) {
return;
}
const firstItem = this.itemsSource[0];
this.selectedItem =
typeof firstItem === "string"
? firstItem
: firstItem.value;
}
initPopupItems() {
this.items[0].itemsSource = this.itemsSource.map((item) => {
const menuItem = typeof item === "string"
? { title: item, value: item }
: item;
menuItem.isSelected = this.selectedItem === menuItem.value;
return menuItem;
});
}
initSortDirection() {
if (_values(SorterDirection).indexOf(this.sortDirection) === -1) {
this.sortDirection = SorterDirection.ascending;
}
}
setPopupSelection() {
this.items[0].itemsSource.forEach((popupItem) => {
popupItem.isSelected = popupItem.value === this.selectedItem;
});
}
triggerSorterAction(oldValue) {
this.sorterAction.emit({ newValue: this.sortConfig, oldValue });
}
onAppendToBodyChange(appendToBody) {
this.customContainer = appendToBody ? undefined : this.popupArea;
}
initKeyboardService() {
this.sorterKeyboardService.menuItems = this.menuPopup?.menuItems;
this.sorterKeyboardService.overlay = this.overlay;
this.sorterKeyboardService.initKeyboardManager();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SorterComponent, deps: [{ token: i1.LoggerService }, { token: i2.SorterKeyboardService }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SorterComponent, selector: "nui-sorter", inputs: { appendToBody: "appendToBody", caption: "caption", itemsSource: "itemsSource", selectedItem: "selectedItem", sortDirection: "sortDirection" }, outputs: { sorterAction: "sorterAction" }, host: { classAttribute: "nui-sorter" }, providers: [SorterKeyboardService], viewQueries: [{ propertyName: "popupArea", first: true, predicate: ["popupArea"], descendants: true, static: true }, { propertyName: "overlay", first: true, predicate: OverlayComponent, descendants: true }, { propertyName: "menuPopup", first: true, predicate: ["popup"], descendants: true }, { propertyName: "toggleRef", first: true, predicate: ["toggleRef"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div>\n <label *ngIf=\"caption\" class=\"nui-sorter__label\">{{ caption }}</label>\n <div class=\"btn-group nui-sorter__popup\" #toggleRef>\n <button\n nui-button\n type=\"button\"\n class=\"nui-sorter__toggle-button\"\n [icon]=\"getSortIcon()\"\n [ariaLabel]=\"getAriaLabelForSortingButton()\"\n (click)=\"toggleSortDirection()\"\n >\n <span class=\"nui-sorter__display-value\">\n {{ getSelectedItemTitle() }}\n </span>\n </button>\n <button\n nui-button\n type=\"button\"\n class=\"nui-selector__toggle\"\n (click)=\"toggleSorterMenu()\"\n ariaLabel=\"Open Sorter Menu\"\n icon=\"caret-down\"\n ></button>\n </div>\n\n <div #popupArea></div>\n\n <nui-overlay\n #overlay\n [toggleReference]=\"toggleRef\"\n [overlayConfig]=\"overlayConfig\"\n [customContainer]=\"customContainer\"\n >\n <nui-menu-popup\n #popup\n [itemsSource]=\"items\"\n (menuItemClicked)=\"select($event)\"\n ></nui-menu-popup>\n </nui-overlay>\n</div>\n", styles: [".nui .nui-sorter{max-height:30px}.nui .nui-sorter .list-group{margin:5px 0}.nui .nui-sorter__label{display:block}.nui .nui-sorter__popup{display:flex}.nui .nui-sorter__toggle-button.nui-button{min-width:auto}.nui .nui-sorter .nui-popup .nui-popup__area{min-width:100%;border:none}.nui .nui-sorter .nui-popup .nui-popup__area [popupAreaContent]{padding:0}.nui .nui-sorter .nui-popup .nui-popup__area .nui-list .nui-list-item{border:0}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.MenuPopupComponent, selector: "nui-menu-popup", inputs: ["itemsSource", "size"], outputs: ["menuItemClicked"] }, { kind: "component", type: i5.ButtonComponent, selector: "[nui-button]", inputs: ["displayStyle", "icon", "iconColor", "iconRight", "isBusy", "isEmpty", "ariaLabel", "isRepeat", "size"] }, { kind: "component", type: i6.OverlayComponent, selector: "nui-overlay", inputs: ["overlayConfig", "toggleReference", "viewportMargin", "customContainer", "roleAttr"], outputs: ["clickOutside"] }], encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SorterComponent, decorators: [{
type: Component,
args: [{ selector: "nui-sorter", host: {
class: "nui-sorter",
}, encapsulation: ViewEncapsulation.None, providers: [SorterKeyboardService], template: "<div>\n <label *ngIf=\"caption\" class=\"nui-sorter__label\">{{ caption }}</label>\n <div class=\"btn-group nui-sorter__popup\" #toggleRef>\n <button\n nui-button\n type=\"button\"\n class=\"nui-sorter__toggle-button\"\n [icon]=\"getSortIcon()\"\n [ariaLabel]=\"getAriaLabelForSortingButton()\"\n (click)=\"toggleSortDirection()\"\n >\n <span class=\"nui-sorter__display-value\">\n {{ getSelectedItemTitle() }}\n </span>\n </button>\n <button\n nui-button\n type=\"button\"\n class=\"nui-selector__toggle\"\n (click)=\"toggleSorterMenu()\"\n ariaLabel=\"Open Sorter Menu\"\n icon=\"caret-down\"\n ></button>\n </div>\n\n <div #popupArea></div>\n\n <nui-overlay\n #overlay\n [toggleReference]=\"toggleRef\"\n [overlayConfig]=\"overlayConfig\"\n [customContainer]=\"customContainer\"\n >\n <nui-menu-popup\n #popup\n [itemsSource]=\"items\"\n (menuItemClicked)=\"select($event)\"\n ></nui-menu-popup>\n </nui-overlay>\n</div>\n", styles: [".nui .nui-sorter{max-height:30px}.nui .nui-sorter .list-group{margin:5px 0}.nui .nui-sorter__label{display:block}.nui .nui-sorter__popup{display:flex}.nui .nui-sorter__toggle-button.nui-button{min-width:auto}.nui .nui-sorter .nui-popup .nui-popup__area{min-width:100%;border:none}.nui .nui-sorter .nui-popup .nui-popup__area [popupAreaContent]{padding:0}.nui .nui-sorter .nui-popup .nui-popup__area .nui-list .nui-list-item{border:0}\n"] }]
}], ctorParameters: () => [{ type: i1.LoggerService }, { type: i2.SorterKeyboardService }, { type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { appendToBody: [{
type: Input
}], caption: [{
type: Input
}], itemsSource: [{
type: Input
}], selectedItem: [{
type: Input
}], sortDirection: [{
type: Input
}], sorterAction: [{
type: Output
}], popupArea: [{
type: ViewChild,
args: ["popupArea", { static: true }]
}], overlay: [{
type: ViewChild,
args: [OverlayComponent]
}], menuPopup: [{
type: ViewChild,
args: ["popup"]
}], toggleRef: [{
type: ViewChild,
args: ["toggleRef", { static: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sorter.component.js","sourceRoot":"","sources":["../../../../src/lib/sorter/sorter.component.ts","../../../../src/lib/sorter/sorter.component.html"],"names":[],"mappings":"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","sourcesContent":["// © 2022 SolarWinds Worldwide, LLC. All rights reserved.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n//  of this software and associated documentation files (the \"Software\"), to\n//  deal in the Software without restriction, including without limitation the\n//  rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n//  sell copies of the Software, and to permit persons to whom the Software is\n//  furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n//  all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n//  THE SOFTWARE.\n\nimport { OverlayConfig } from \"@angular/cdk/overlay\";\nimport {\n    AfterViewInit,\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    Renderer2,\n    SimpleChanges,\n    ViewChild,\n    ViewEncapsulation,\n} from \"@angular/core\";\nimport _assign from \"lodash/assign\";\nimport _isEqual from \"lodash/isEqual\";\nimport _values from \"lodash/values\";\nimport { Subject } from \"rxjs\";\nimport { takeUntil } from \"rxjs/operators\";\n\nimport { ISortedItem, ISorterChanges, SorterDirection } from \"./public-api\";\nimport { SorterKeyboardService } from \"./sorter-keyboard.service\";\nimport {\n    IFilter,\n    IFilterPub,\n    ISorterFilter,\n} from \"../../services/data-source/public-api\";\nimport { LoggerService } from \"../../services/log-service\";\nimport { MenuPopupComponent } from \"../menu\";\nimport { IMenuGroup, IMenuItem } from \"../menu/public-api\";\nimport { OVERLAY_WITH_POPUP_STYLES_CLASS } from \"../overlay/constants\";\nimport { OverlayComponent } from \"../overlay/overlay-component/overlay.component\";\n\n// <example-url>./../examples/index.html#/sorter</example-url>\n\n@Component({\n    selector: \"nui-sorter\",\n    host: {\n        class: \"nui-sorter\",\n    },\n    templateUrl: \"./sorter.component.html\",\n    styleUrls: [\"./sorter.component.less\"],\n    encapsulation: ViewEncapsulation.None,\n    providers: [SorterKeyboardService],\n})\nexport class SorterComponent\n    implements OnChanges, OnInit, OnDestroy, AfterViewInit, IFilterPub\n{\n    @Input() appendToBody: boolean = false;\n    @Input() caption: string;\n\n    /**\n     * The string[] type for itemsSource is the legacy non-i18n-friendly type\n     * and it should be removed as an option in scope of NUI-5801\n     */\n    @Input() itemsSource: string[] | IMenuItem[];\n\n    @Input() selectedItem: string;\n    @Input() sortDirection: any;\n\n    @Output() sorterAction = new EventEmitter<ISorterChanges>();\n\n    @ViewChild(\"popupArea\", { static: true }) popupArea: ElementRef;\n    @ViewChild(OverlayComponent) public overlay: OverlayComponent;\n    @ViewChild(\"popup\") public menuPopup: MenuPopupComponent;\n\n    // mark this filter to be monitored by our datasource for any changes in order reset other filters(eg: pagination)\n    // before any new search is performed\n    public detectFilterChanges = true;\n\n    public customContainer: ElementRef | undefined;\n    public onDestroy$ = new Subject<void>();\n    public items: IMenuGroup[] = [\n        {\n            itemsSource: [],\n        },\n    ];\n    public overlayConfig: OverlayConfig = {\n        panelClass: [OVERLAY_WITH_POPUP_STYLES_CLASS],\n    };\n\n    @ViewChild(\"toggleRef\", { static: true }) private toggleRef: ElementRef;\n    private sortConfig: ISortedItem;\n    private sortIcons: { [key: string]: string } = {\n        [SorterDirection.ascending]: \"arrow-up\",\n        [SorterDirection.descending]: \"arrow-down\",\n    };\n    private menuKeyControlListeners: Function[] = [];\n\n    constructor(\n        private logger: LoggerService,\n        private sorterKeyboardService: SorterKeyboardService,\n        private elRef: ElementRef,\n        private renderer: Renderer2\n    ) {}\n\n    public ngOnInit(): void {\n        this.onAppendToBodyChange(this.appendToBody);\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (\n            changes.itemsSource &&\n            !_isEqual(\n                changes.itemsSource.previousValue,\n                changes.itemsSource.currentValue\n            )\n        ) {\n            if (\n                this.itemsSource?.length > 0 &&\n                typeof this.itemsSource[0] === \"string\"\n            ) {\n                this.logger\n                    .warn(`The 'string[]' type for the sorter's itemsSource input is deprecated as of Nova v9. \\\n                                  Instead, use type 'IMenuItem[]' which is internationalizable with title and value properties for each item.`);\n            }\n            this.initSelectedItem();\n            this.initPopupItems();\n        }\n\n        if (\n            changes.selectedItem &&\n            this.sortConfig?.sortBy !== changes.selectedItem.currentValue &&\n            !changes.selectedItem.firstChange\n        ) {\n            const oldValue = _assign({}, this.sortConfig, {\n                sortBy: changes.selectedItem.previousValue,\n            });\n\n            this.selectedItem = changes.selectedItem.currentValue;\n            this.sortConfig = _assign({}, this.sortConfig, {\n                sortBy: changes.selectedItem.currentValue,\n            });\n\n            this.triggerSorterAction(oldValue);\n\n            this.setPopupSelection();\n        }\n\n        if (\n            changes.sortDirection &&\n            !changes.sortDirection.firstChange &&\n            this.sortConfig?.direction !== changes.sortDirection.currentValue\n        ) {\n            const oldValue = this.sortConfig;\n            this.sortDirection = changes.sortDirection.currentValue;\n            this.sortConfig = _assign({}, this.sortConfig, {\n                direction: this.sortDirection,\n            });\n\n            this.triggerSorterAction(oldValue);\n        }\n\n        if (changes.appendToBody) {\n            this.onAppendToBodyChange(changes.appendToBody.currentValue);\n        }\n    }\n\n    public ngAfterViewInit(): void {\n        this.initSelectedItem();\n        this.initSortDirection();\n\n        this.sortConfig = {\n            sortBy: this.selectedItem,\n            direction: this.sortDirection,\n        };\n        this.overlay.clickOutside\n            .pipe(takeUntil(this.onDestroy$))\n            .subscribe((_) => this.overlay.hide());\n\n        this.updateOverlayWidth();\n        this.initKeyboardService();\n        this.menuKeyControlListeners.push(\n            this.renderer.listen(\n                this.elRef.nativeElement,\n                \"keydown\",\n                (event: KeyboardEvent) => {\n                    this.sorterKeyboardService.handleKeydown(event);\n                }\n            )\n        );\n    }\n\n    public select(item: IMenuItem): void {\n        // perform update only if the new value actually changes\n        if (this.selectedItem !== item.value) {\n            const oldValue = this.sortConfig;\n\n            this.selectedItem = item.value;\n            this.sortConfig = _assign({}, this.sortConfig, {\n                sortBy: item.value,\n            });\n\n            this.updateOverlayWidth();\n            this.triggerSorterAction(oldValue);\n            this.setPopupSelection();\n            this.overlay.hide();\n        }\n    }\n\n    public toggleSortDirection(): void {\n        const oldValue = this.sortConfig;\n\n        this.sortDirection =\n            this.sortDirection === SorterDirection.ascending\n                ? SorterDirection.descending\n                : SorterDirection.ascending;\n        this.sortConfig = _assign({}, this.sortConfig, {\n            direction: this.sortDirection,\n        });\n        this.sorterAction.emit({ newValue: this.sortConfig, oldValue });\n    }\n\n    public getSelectedItem(): any {\n        return this.selectedItem;\n    }\n\n    public getSelectedItemTitle(): string {\n        return this.items[0].itemsSource.find(\n            (item: IMenuItem) => item.isSelected\n        )?.title;\n    }\n\n    public getSortIcon(): string {\n        return this.sortIcons[this.sortDirection];\n    }\n\n    public getFilters(): IFilter<ISorterFilter> {\n        return {\n            type: \"sorter\",\n            value: { ...this.sortConfig },\n        };\n    }\n\n    public updateOverlayWidth(): void {\n        this.overlayConfig.minWidth = (\n            this.toggleRef.nativeElement as HTMLElement\n        ).offsetWidth;\n    }\n\n    public getAriaLabelForSortingButton(): string {\n        return this.sortDirection === SorterDirection.descending\n            ? `${this.getSelectedItemTitle()}. Sorter direction - descending`\n            : `${this.getSelectedItemTitle()}. Sorter direction - ascending`;\n    }\n\n    public ngOnDestroy(): void {\n        this.menuKeyControlListeners.forEach((listener) => listener());\n        this.onDestroy$.next();\n        this.onDestroy$.complete();\n    }\n\n    public toggleSorterMenu(): void {\n        this.overlay.toggle();\n        this.sorterKeyboardService.announceDropdown();\n    }\n\n    private initSelectedItem() {\n        // skip initialization in case we already have a value\n        // or if the itemsSource are lazy loaded\n        if (this.selectedItem || !this.itemsSource?.length) {\n            return;\n        }\n\n        const firstItem = this.itemsSource[0];\n        this.selectedItem =\n            typeof firstItem === \"string\"\n                ? firstItem\n                : (firstItem as IMenuItem).value;\n    }\n\n    private initPopupItems(): void {\n        this.items[0].itemsSource = (this.itemsSource as any[]).map(\n            (item: string | IMenuItem) => {\n                const menuItem: IMenuItem =\n                    typeof item === \"string\"\n                        ? { title: item, value: item }\n                        : (item as IMenuItem);\n                menuItem.isSelected = this.selectedItem === menuItem.value;\n                return menuItem;\n            }\n        );\n    }\n\n    private initSortDirection(): void {\n        if (_values(SorterDirection).indexOf(this.sortDirection) === -1) {\n            this.sortDirection = SorterDirection.ascending;\n        }\n    }\n\n    private setPopupSelection(): void {\n        this.items[0].itemsSource.forEach((popupItem: IMenuItem) => {\n            popupItem.isSelected = popupItem.value === this.selectedItem;\n        });\n    }\n\n    private triggerSorterAction(oldValue: ISortedItem): void {\n        this.sorterAction.emit({ newValue: this.sortConfig, oldValue });\n    }\n\n    private onAppendToBodyChange(appendToBody: boolean): void {\n        this.customContainer = appendToBody ? undefined : this.popupArea;\n    }\n\n    private initKeyboardService(): void {\n        this.sorterKeyboardService.menuItems = this.menuPopup?.menuItems;\n        this.sorterKeyboardService.overlay = this.overlay;\n        this.sorterKeyboardService.initKeyboardManager();\n    }\n}\n","<div>\n    <label *ngIf=\"caption\" class=\"nui-sorter__label\">{{ caption }}</label>\n    <div class=\"btn-group nui-sorter__popup\" #toggleRef>\n        <button\n            nui-button\n            type=\"button\"\n            class=\"nui-sorter__toggle-button\"\n            [icon]=\"getSortIcon()\"\n            [ariaLabel]=\"getAriaLabelForSortingButton()\"\n            (click)=\"toggleSortDirection()\"\n        >\n            <span class=\"nui-sorter__display-value\">\n                {{ getSelectedItemTitle() }}\n            </span>\n        </button>\n        <button\n            nui-button\n            type=\"button\"\n            class=\"nui-selector__toggle\"\n            (click)=\"toggleSorterMenu()\"\n            ariaLabel=\"Open Sorter Menu\"\n            icon=\"caret-down\"\n        ></button>\n    </div>\n\n    <div #popupArea></div>\n\n    <nui-overlay\n        #overlay\n        [toggleReference]=\"toggleRef\"\n        [overlayConfig]=\"overlayConfig\"\n        [customContainer]=\"customContainer\"\n    >\n        <nui-menu-popup\n            #popup\n            [itemsSource]=\"items\"\n            (menuItemClicked)=\"select($event)\"\n        ></nui-menu-popup>\n    </nui-overlay>\n</div>\n"]}