angular-select-dropdown
Version:
A highly customizable and flexible dropdown select component for Angular applications. Ideal for creating dynamic select menus with search, filtering.
184 lines • 34.9 kB
JavaScript
import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, HostBinding, Input, Output, ViewEncapsulation } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "../directives/click-outside.directive";
export class AngularSelectDropdownComponent {
/**
* List of items to populate the dropdown.
* This setter updates the internal store of items and triggers an update.
* @param items The array of items to be displayed in the dropdown.
*/
set items(items) {
this._items = items;
this.updateItems(items);
}
get items() {
return this._items;
}
constructor(cdRef) {
this.cdRef = cdRef;
/**
* Handler function to extract the label for each item.
* @param item The item from the list.
* @returns The label (name or value) of the item.
*/
this.labelHandler = (item) => item['name'] || item;
/**
* Defines the key used to sort the dropdown items.
* @param sortKeyName The key used to sort the items, default is 'value'.
* @default 'value'
*/
this.sortKeyName = 'value';
/**
* Determines whether the dropdown allows clearing the selection.
* @param isClearable If true, the dropdown will show a clear button.
* @default false
*/
this.isClearable = false;
/**
* Determines whether the dropdown is disabled.
* @param disabled If true, the dropdown is disabled and cannot be interacted with.
* @default false
*/
this.disabled = false;
/**
* Determines whether the dropdown menu should close when the selected item is cleared.
* @param closeMenuOnClear If true, the dropdown will close when the selection is cleared.
* @default true
*/
this.closeMenuOnClear = true;
/**
* Placeholder text displayed when no item is selected.
* @param placeholder The placeholder text.
* @default ''
*/
this.placeholder = '';
/**
* Message displayed when no items match the search query.
* @param notFoundMessage The message to display when no items are found.
* @default 'Nothing found'
*/
this.notFoundMessage = 'Nothing found';
/**
* Event emitted when an item is selected.
* @param selectItem The selected item is emitted as an event.
*/
this.selectItem = new EventEmitter();
this.hostClasses = 'angular-select-dropdown';
this._items = [];
this.data = '';
this.isDropdownOpen = false;
this.itemsStore = [];
this.selectedItem = {
value: null,
name: ''
};
this.onTouched = () => { };
this.onChange = () => { };
}
ngOnDestroy() {
this.isDropdownOpen = false;
}
registerOnChange(onChange) {
this.onChange = onChange;
}
registerOnTouched(onTouched) {
this.onTouched = onTouched;
}
writeValue(obj) {
this.data = obj;
this.updateItems(this.items);
}
updateData(data) {
this.onChange(data);
this.onTouched();
}
updateItems(items) {
if (items && items.length) {
this.itemsStore = items.map((item) => ({
value: item[this.sortKeyName] ?? this.labelHandler(item),
name: this.labelHandler(item),
disabled: !!item?.['disabled']
}));
}
else {
this.itemsStore = [];
}
this.updateSelectedItem();
this.cdRef.markForCheck();
}
updateSelectedItem(item) {
const selectedItem = this.itemsStore.find(item => item?.value == this.data);
this.selectedItem = {
value: item?.value ?? (selectedItem?.value ?? null),
name: item?.name ?? (selectedItem?.name ?? '')
};
}
trackByValue(index, item) {
return item.value;
}
closeDropdownMenu() {
this.isDropdownOpen = false;
}
onClickRemove(event) {
event.stopImmediatePropagation();
if (this.selectedItem.value && this.selectedItem.name) {
this.updateSelectedItem();
this.updateData(null);
this.selectItem.emit();
if (this.closeMenuOnClear) {
this.closeDropdownMenu();
}
}
}
onSelectItem(item) {
this.updateData(item.value);
this.updateSelectedItem(item);
this.closeDropdownMenu();
this.selectItem.emit(item.value);
}
onSelectClick() {
this.isDropdownOpen = !this.isDropdownOpen;
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: AngularSelectDropdownComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: AngularSelectDropdownComponent, selector: "angular-select-dropdown", inputs: { labelHandler: "labelHandler", sortKeyName: "sortKeyName", items: "items", isClearable: "isClearable", disabled: "disabled", closeMenuOnClear: "closeMenuOnClear", placeholder: "placeholder", notFoundMessage: "notFoundMessage" }, outputs: { selectItem: "selectItem" }, host: { properties: { "class": "this.hostClasses" } }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => AngularSelectDropdownComponent),
multi: true
}
], ngImport: i0, template: "<div\n class=\"angular-select-dropdown-wrapper\"\n [excludeElements]=\"['.angular-select-dropdown-menu']\"\n [class.angular-select-dropdown-wrapper_disabled]=\"disabled\"\n (clickOutside)=\"closeDropdownMenu()\"\n (click)=\"onSelectClick()\"\n>\n <div\n class=\"angular-select-dropdown__text\"\n [class.angular-select-dropdown__text_placeholder]=\"!selectedItem.name && placeholder\"\n [title]=\"selectedItem.name\"\n >\n {{selectedItem.name || placeholder}}\n </div>\n\n <div class=\"angular-select-dropdown-actions\">\n <div\n *ngIf=\"isClearable\"\n class=\"angular-select-dropdown-actions__close\"\n (click)=\"onClickRemove($event)\"\n ></div>\n <div\n class=\"angular-select-dropdown-actions__arrow\"\n [class.angular-select-dropdown-actions__arrow_rotate]=\"isDropdownOpen\"\n ></div>\n </div>\n</div>\n\n<div\n *ngIf=\"isDropdownOpen\"\n class=\"angular-select-dropdown-menu\"\n>\n <div class=\"angular-select-dropdown-menu__wrapper\">\n <div\n *ngIf=\"itemsStore.length; else itemsListEmpty\"\n class=\"angular-select-dropdown-menu__items\"\n >\n <div\n *ngFor=\"let item of itemsStore; let i = index; trackBy: trackByValue\"\n class=\"angular-select-dropdown-menu__item\"\n [class.angular-select-dropdown-menu__item_active]=\"item.value === selectedItem.value\"\n [title]=\"item.name\"\n (click)=\"onSelectItem(item)\"\n >\n {{item.name}}\n </div>\n </div>\n\n <ng-template #itemsListEmpty>\n <div\n class=\"angular-select-dropdown-menu__not-found\"\n >\n {{notFoundMessage}}\n </div>\n </ng-template>\n </div>\n</div>\n", styles: [":host,.angular-select-dropdown{display:block;position:relative;width:100%}.angular-select-dropdown-wrapper{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px 10px 10px 15px;border:1px solid rgba(0,0,0,.1019607843);border-radius:.625rem}.angular-select-dropdown-wrapper:hover .angular-select-dropdown-actions__arrow{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M7 10L12 15%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M12 15L17 10%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-wrapper_disabled{pointer-events:none;opacity:.5}.angular-select-dropdown__text{text-overflow:ellipsis;white-space:nowrap;overflow:hidden;color:#1b1f3b;font-family:inherit;width:100%}.angular-select-dropdown__text_placeholder{color:#1b1f3ba6}.angular-select-dropdown-actions{display:flex;align-items:center}.angular-select-dropdown-actions__close,.angular-select-dropdown-actions__arrow{width:20px;height:20px;background-repeat:no-repeat}.angular-select-dropdown-actions__close{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M8 8L16 16%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M16 8L8 16%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__close:hover{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M8 8L16 16%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M16 8L8 16%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__arrow{transition:.4s;background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M7 10L12 15%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M12 15L17 10%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__arrow_rotate{transform:rotate(180deg)}.angular-select-dropdown-menu{position:absolute;width:100%;left:0;top:45px;border:1px solid rgba(0,0,0,.1019607843);border-radius:.625rem}.angular-select-dropdown-menu__wrapper{padding:4px}.angular-select-dropdown-menu__items{padding:1px;max-height:200px;overflow-y:auto;overflow-x:hidden;box-sizing:border-box}.angular-select-dropdown-menu__items::-webkit-scrollbar{width:4px}.angular-select-dropdown-menu__items::-webkit-scrollbar-thumb{background:#888;border-radius:.625rem;cursor:pointer}.angular-select-dropdown-menu__items::-webkit-scrollbar-thumb:hover{background:#555}.angular-select-dropdown-menu__item{padding:10px;cursor:pointer;border-radius:.625rem;margin-bottom:2px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden;font-family:inherit;color:#1b1f3b}.angular-select-dropdown-menu__item_active,.angular-select-dropdown-menu__item:hover{background-color:#0000000a}.angular-select-dropdown-menu__item:last-child{margin-bottom:0}.angular-select-dropdown-menu__not-found{padding:10px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden;font-family:inherit;color:#1b1f3b66}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.ClickOutsideDirective, selector: "[clickOutside]", inputs: ["excludeElements"], outputs: ["clickOutside"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: AngularSelectDropdownComponent, decorators: [{
type: Component,
args: [{ selector: 'angular-select-dropdown', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => AngularSelectDropdownComponent),
multi: true
}
], template: "<div\n class=\"angular-select-dropdown-wrapper\"\n [excludeElements]=\"['.angular-select-dropdown-menu']\"\n [class.angular-select-dropdown-wrapper_disabled]=\"disabled\"\n (clickOutside)=\"closeDropdownMenu()\"\n (click)=\"onSelectClick()\"\n>\n <div\n class=\"angular-select-dropdown__text\"\n [class.angular-select-dropdown__text_placeholder]=\"!selectedItem.name && placeholder\"\n [title]=\"selectedItem.name\"\n >\n {{selectedItem.name || placeholder}}\n </div>\n\n <div class=\"angular-select-dropdown-actions\">\n <div\n *ngIf=\"isClearable\"\n class=\"angular-select-dropdown-actions__close\"\n (click)=\"onClickRemove($event)\"\n ></div>\n <div\n class=\"angular-select-dropdown-actions__arrow\"\n [class.angular-select-dropdown-actions__arrow_rotate]=\"isDropdownOpen\"\n ></div>\n </div>\n</div>\n\n<div\n *ngIf=\"isDropdownOpen\"\n class=\"angular-select-dropdown-menu\"\n>\n <div class=\"angular-select-dropdown-menu__wrapper\">\n <div\n *ngIf=\"itemsStore.length; else itemsListEmpty\"\n class=\"angular-select-dropdown-menu__items\"\n >\n <div\n *ngFor=\"let item of itemsStore; let i = index; trackBy: trackByValue\"\n class=\"angular-select-dropdown-menu__item\"\n [class.angular-select-dropdown-menu__item_active]=\"item.value === selectedItem.value\"\n [title]=\"item.name\"\n (click)=\"onSelectItem(item)\"\n >\n {{item.name}}\n </div>\n </div>\n\n <ng-template #itemsListEmpty>\n <div\n class=\"angular-select-dropdown-menu__not-found\"\n >\n {{notFoundMessage}}\n </div>\n </ng-template>\n </div>\n</div>\n", styles: [":host,.angular-select-dropdown{display:block;position:relative;width:100%}.angular-select-dropdown-wrapper{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px 10px 10px 15px;border:1px solid rgba(0,0,0,.1019607843);border-radius:.625rem}.angular-select-dropdown-wrapper:hover .angular-select-dropdown-actions__arrow{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M7 10L12 15%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M12 15L17 10%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-wrapper_disabled{pointer-events:none;opacity:.5}.angular-select-dropdown__text{text-overflow:ellipsis;white-space:nowrap;overflow:hidden;color:#1b1f3b;font-family:inherit;width:100%}.angular-select-dropdown__text_placeholder{color:#1b1f3ba6}.angular-select-dropdown-actions{display:flex;align-items:center}.angular-select-dropdown-actions__close,.angular-select-dropdown-actions__arrow{width:20px;height:20px;background-repeat:no-repeat}.angular-select-dropdown-actions__close{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M8 8L16 16%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M16 8L8 16%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__close:hover{background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M8 8L16 16%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M16 8L8 16%22 stroke%3D%22%23333%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__arrow{transition:.4s;background-image:url(\"data:image/svg+xml,%3Csvg viewBox%3D%220 0 24 24%22 fill%3D%22none%22 xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E %3Cg%3E %3Cpath d%3D%22M7 10L12 15%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3Cpath d%3D%22M12 15L17 10%22 stroke%3D%22%23808080%22 stroke-width%3D%222%22 stroke-linecap%3D%22round%22 stroke-linejoin%3D%22round%22 %3E%3C%2Fpath%3E %3C%2Fg%3E%3C%2Fsvg%3E\")}.angular-select-dropdown-actions__arrow_rotate{transform:rotate(180deg)}.angular-select-dropdown-menu{position:absolute;width:100%;left:0;top:45px;border:1px solid rgba(0,0,0,.1019607843);border-radius:.625rem}.angular-select-dropdown-menu__wrapper{padding:4px}.angular-select-dropdown-menu__items{padding:1px;max-height:200px;overflow-y:auto;overflow-x:hidden;box-sizing:border-box}.angular-select-dropdown-menu__items::-webkit-scrollbar{width:4px}.angular-select-dropdown-menu__items::-webkit-scrollbar-thumb{background:#888;border-radius:.625rem;cursor:pointer}.angular-select-dropdown-menu__items::-webkit-scrollbar-thumb:hover{background:#555}.angular-select-dropdown-menu__item{padding:10px;cursor:pointer;border-radius:.625rem;margin-bottom:2px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden;font-family:inherit;color:#1b1f3b}.angular-select-dropdown-menu__item_active,.angular-select-dropdown-menu__item:hover{background-color:#0000000a}.angular-select-dropdown-menu__item:last-child{margin-bottom:0}.angular-select-dropdown-menu__not-found{padding:10px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden;font-family:inherit;color:#1b1f3b66}\n"] }]
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { labelHandler: [{
type: Input
}], sortKeyName: [{
type: Input
}], items: [{
type: Input
}], isClearable: [{
type: Input
}], disabled: [{
type: Input
}], closeMenuOnClear: [{
type: Input
}], placeholder: [{
type: Input
}], notFoundMessage: [{
type: Input
}], selectItem: [{
type: Output
}], hostClasses: [{
type: HostBinding,
args: ['class']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-select-dropdown.component.js","sourceRoot":"","sources":["../../../../../projects/angular-select-dropdown/src/lib/components/angular-select-dropdown.component.ts","../../../../../projects/angular-select-dropdown/src/lib/components/angular-select-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  Input,\n  OnDestroy,\n  Output,\n  ViewEncapsulation\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\n\nimport { AngularSelectDropdown } from '../models';\n\ninterface SelectedItem {\n  value: any;\n  name: string;\n}\n\n@Component({\n  selector: 'angular-select-dropdown',\n  templateUrl: 'angular-select-dropdown.component.html',\n  styleUrls: [ 'angular-select-dropdown.component.scss' ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => AngularSelectDropdownComponent),\n      multi: true\n    }\n  ]\n})\nexport class AngularSelectDropdownComponent<T> implements ControlValueAccessor, OnDestroy {\n  /**\n   * Handler function to extract the label for each item.\n   * @param item The item from the list.\n   * @returns The label (name or value) of the item.\n   */\n  @Input() labelHandler: (item: T) => any = (item) => (item as AngularSelectDropdown)['name'] || item;\n\n  /**\n   * Defines the key used to sort the dropdown items.\n   * @param sortKeyName The key used to sort the items, default is 'value'.\n   * @default 'value'\n   */\n  @Input() sortKeyName: string = 'value';\n\n  /**\n   * List of items to populate the dropdown.\n   * This setter updates the internal store of items and triggers an update.\n   * @param items The array of items to be displayed in the dropdown.\n   */\n  @Input() set items(items: T[]) {\n    this._items = items;\n    this.updateItems(items);\n  }\n\n  /**\n   * Determines whether the dropdown allows clearing the selection.\n   * @param isClearable If true, the dropdown will show a clear button.\n   * @default false\n   */\n  @Input() isClearable: boolean = false;\n\n  /**\n   * Determines whether the dropdown is disabled.\n   * @param disabled If true, the dropdown is disabled and cannot be interacted with.\n   * @default false\n   */\n  @Input() disabled: boolean = false;\n\n  /**\n   * Determines whether the dropdown menu should close when the selected item is cleared.\n   * @param closeMenuOnClear If true, the dropdown will close when the selection is cleared.\n   * @default true\n   */\n  @Input() closeMenuOnClear: boolean = true;\n\n  /**\n   * Placeholder text displayed when no item is selected.\n   * @param placeholder The placeholder text.\n   * @default ''\n   */\n  @Input() placeholder: string = '';\n\n  /**\n   * Message displayed when no items match the search query.\n   * @param notFoundMessage The message to display when no items are found.\n   * @default 'Nothing found'\n   */\n  @Input() notFoundMessage: string = 'Nothing found';\n\n  /**\n   * Event emitted when an item is selected.\n   * @param selectItem The selected item is emitted as an event.\n   */\n  @Output() selectItem: EventEmitter<T> = new EventEmitter();\n\n  @HostBinding('class') hostClasses = 'angular-select-dropdown';\n\n  private _items: T[] = [];\n\n  get items(): T[] {\n    return this._items;\n  }\n\n  data: (number | string) | (number | string)[] = '';\n  isDropdownOpen: boolean = false;\n  itemsStore: AngularSelectDropdown[] = [];\n  selectedItem: SelectedItem = {\n    value: null,\n    name: ''\n  };\n\n  constructor(\n    private cdRef: ChangeDetectorRef\n  ) {}\n\n  ngOnDestroy() {\n    this.isDropdownOpen = false;\n  }\n\n  private onTouched = () => {};\n\n  private onChange: (value: number | string | null) => void = () => {};\n\n  registerOnChange(onChange: (value: number | string | null) => void) {\n    this.onChange = onChange;\n  }\n\n  registerOnTouched(onTouched: () => void) {\n    this.onTouched = onTouched;\n  }\n\n  writeValue(obj: number | string) {\n    this.data = obj;\n    this.updateItems(this.items);\n  }\n\n  private updateData(data: number | string | null) {\n    this.onChange(data);\n    this.onTouched();\n  }\n\n  private updateItems(items: T[]) {\n    if (items && items.length) {\n      this.itemsStore = items.map((item: any) => ({\n        value: item[this.sortKeyName as keyof T] ?? this.labelHandler(item),\n        name: this.labelHandler(item),\n        disabled: !!item?.['disabled']\n      })) as AngularSelectDropdown[];\n    } else {\n      this.itemsStore = [];\n    }\n\n    this.updateSelectedItem();\n    this.cdRef.markForCheck();\n  }\n\n  private updateSelectedItem(item?: AngularSelectDropdown) {\n    const selectedItem = this.itemsStore.find(item => item?.value == this.data);\n\n    this.selectedItem = {\n      value: item?.value ?? (selectedItem?.value ?? null),\n      name: item?.name ?? (selectedItem?.name ?? '')\n    };\n  }\n\n  trackByValue(index: number, item: AngularSelectDropdown) {\n    return item.value;\n  }\n\n  closeDropdownMenu() {\n    this.isDropdownOpen = false;\n  }\n\n  onClickRemove(event: MouseEvent) {\n    event.stopImmediatePropagation();\n\n    if (this.selectedItem.value && this.selectedItem.name) {\n      this.updateSelectedItem();\n      this.updateData(null);\n      this.selectItem.emit();\n\n      if (this.closeMenuOnClear) {\n        this.closeDropdownMenu();\n      }\n    }\n  }\n\n  onSelectItem(item: AngularSelectDropdown) {\n    this.updateData(item.value);\n    this.updateSelectedItem(item);\n    this.closeDropdownMenu();\n\n    this.selectItem.emit(item.value);\n  }\n\n  onSelectClick() {\n    this.isDropdownOpen = !this.isDropdownOpen;\n  }\n}\n","<div\n  class=\"angular-select-dropdown-wrapper\"\n  [excludeElements]=\"['.angular-select-dropdown-menu']\"\n  [class.angular-select-dropdown-wrapper_disabled]=\"disabled\"\n  (clickOutside)=\"closeDropdownMenu()\"\n  (click)=\"onSelectClick()\"\n>\n  <div\n    class=\"angular-select-dropdown__text\"\n    [class.angular-select-dropdown__text_placeholder]=\"!selectedItem.name && placeholder\"\n    [title]=\"selectedItem.name\"\n  >\n    {{selectedItem.name || placeholder}}\n  </div>\n\n  <div class=\"angular-select-dropdown-actions\">\n    <div\n      *ngIf=\"isClearable\"\n      class=\"angular-select-dropdown-actions__close\"\n      (click)=\"onClickRemove($event)\"\n    ></div>\n    <div\n      class=\"angular-select-dropdown-actions__arrow\"\n      [class.angular-select-dropdown-actions__arrow_rotate]=\"isDropdownOpen\"\n    ></div>\n  </div>\n</div>\n\n<div\n  *ngIf=\"isDropdownOpen\"\n  class=\"angular-select-dropdown-menu\"\n>\n  <div class=\"angular-select-dropdown-menu__wrapper\">\n    <div\n      *ngIf=\"itemsStore.length; else itemsListEmpty\"\n      class=\"angular-select-dropdown-menu__items\"\n    >\n      <div\n        *ngFor=\"let item of itemsStore; let i = index; trackBy: trackByValue\"\n        class=\"angular-select-dropdown-menu__item\"\n        [class.angular-select-dropdown-menu__item_active]=\"item.value === selectedItem.value\"\n        [title]=\"item.name\"\n        (click)=\"onSelectItem(item)\"\n      >\n        {{item.name}}\n      </div>\n    </div>\n\n    <ng-template #itemsListEmpty>\n      <div\n        class=\"angular-select-dropdown-menu__not-found\"\n      >\n        {{notFoundMessage}}\n      </div>\n    </ng-template>\n  </div>\n</div>\n"]}