UNPKG

ngx-slice-kit

Version:

[![npm version](https://badge.fury.io/js/ngx-slice-kit.svg)](https://badge.fury.io/js/ngx-slice-kit)

222 lines 41.5 kB
import { Component, EventEmitter, Inject, Input, Output, ViewChild, } from '@angular/core'; import { fromEvent } from 'rxjs'; import { take } from 'rxjs/operators'; import { DOCUMENT } from '@angular/common'; import * as i0 from "@angular/core"; import * as i1 from "./options.service"; import * as i2 from "@angular/common"; import * as i3 from "../buttons/icon/icon.component"; const DEFAULT_DROPDOWN_SIZE = 240; // I'm taking additional 16 pixels here, to prevent element sticking to bottom panel const DEFAULT_DROPDOWN_OFFSET = 256; export class DropdownComponent { constructor(document, elem, renderer, optionsService) { this.document = document; this.elem = elem; this.renderer = renderer; this.optionsService = optionsService; this.resultEvent = new EventEmitter(); this.rects = {}; } onOptionMouseEnter(o, index) { this.currentOption = o; this.highlightedIndex = index; } onOptionMouseLeave() { this.currentOption = undefined; this.highlightedIndex = undefined; } select(ev, option) { this.onResult(option); } onResult(res) { this.resultEvent.emit(res); this.resultEvent.complete(); this.sub?.unsubscribe(); } nextOption(direction) { if (!this.highlightedIndex && this.highlightedIndex !== 0) { const selected = this.optionsService.options.find(o => o.selected); if (!!selected) { this.highlightedIndex = this.optionsService.options.indexOf(selected); } else { this.highlightedIndex = direction === 'up' ? 0 : this.optionsService.options.length - 1; } } if (this.optionsService.options?.length === 1) { this.currentOption = this.optionsService.options[0]; this.highlightedIndex = 0; } else { const isFirstOption = this.highlightedIndex === 0; const optionsLength = this.optionsService.options.length; const isLastOption = this.highlightedIndex === optionsLength - 1; const index = direction === 'down' ? (isLastOption ? 0 : (this.highlightedIndex + 1)) : (isFirstOption ? (optionsLength - 1) : this.highlightedIndex - 1); this.currentOption = this.optionsService.options[index]; this.highlightedIndex = index; } setTimeout(() => this.autoScroll(direction)); } autoScroll(direction) { const dropdownPaddingTop = 8; /** * okay that padding is constant at the top and the bottom of the element, * so it should be used in calculations when we are counting top offset */ const { scrollHeight, scrollTop, offsetHeight } = this.dropdownElement.nativeElement; const currentItem = this.elem.nativeElement.querySelector(`.highlighted`); if (!currentItem) { return; } const itemHeight = currentItem.offsetHeight; switch (direction) { case 'up': if (currentItem.offsetTop - dropdownPaddingTop < scrollTop) { this.dropdownElement.nativeElement.scrollTop = scrollTop - offsetHeight; } if (currentItem.offsetTop + itemHeight > scrollTop + offsetHeight) { this.dropdownElement.nativeElement.scrollTop = scrollHeight - offsetHeight; } break; default: if (currentItem.offsetTop + itemHeight > offsetHeight + scrollTop) { this.dropdownElement.nativeElement.scrollTop = scrollTop + itemHeight; } else if (currentItem.offsetTop === dropdownPaddingTop) { this.dropdownElement.nativeElement.scrollTop = 0; } } } /** * click outside subscription if backdrop disabled. also dropdown must be attached during input */ initClickOutsideSub() { this.renderer.listen('window', 'click', event => { const isParent = this.config.parentElem?.contains(event.target); const isDropdown = this.elem.nativeElement.contains(event.target); if (!isParent && !isDropdown) { this.onResult(); } }); } /** * detect relative position to prevent dropdown being hidden over `overflow: none` */ getDropdownRects() { const windowHeight = window.innerHeight; const pixelsLeft = windowHeight - this.config.triggerRect.height - this.config.triggerRect.top; const rects = this.dropdownElement.nativeElement.getBoundingClientRect(); this.inverted = pixelsLeft <= DEFAULT_DROPDOWN_OFFSET; if (this.inverted) { this.rects.bottom = windowHeight - this.config.triggerRect.top; } else { this.rects.top = this.config.triggerRect.bottom; } if (this.config.fitWidth) { this.rects.width = this.config.triggerRect.width; } if (rects.width > this.config.triggerRect.width) { const rightOffset = window.innerWidth - this.config.triggerRect.width - this.config.triggerRect.left; if (rightOffset <= rects.width + 16) { this.rects.left = (this.config.triggerRect.left + this.config.triggerRect.width) - rects.width; } } } /** * - if `fitWidth` config options is true * there are width declared depending on its parent element * - if `rects.top` calculated there is enough place to drop it down, * if it hits `rects.bottom` – show it above the element */ setDropdownPosition() { if (this.rects.width) { this.renderer.setStyle(this.dropdownElement.nativeElement, `width`, `${this.rects.width}px`); } if (this.rects.top) { this.renderer.setStyle(this.dropdownElement.nativeElement, `top`, `${this.rects.top}px`); } else { this.renderer.setStyle(this.dropdownElement.nativeElement, `bottom`, `${this.rects.bottom}px`); } this.renderer.setStyle(this.dropdownElement.nativeElement, `left`, `${this.rects.left || this.config.triggerRect.left}px`); this.renderer.setStyle(this.dropdownElement.nativeElement, `opacity`, 1); } /** * detect window resize and scroll to prevent failed dropdown position */ initClosingSubscriptions() { this.sub = fromEvent(window, 'scroll').pipe(take(1)).subscribe(() => { this.onResult(); }); this.sub.add(fromEvent(window, 'resize').pipe(take(1)).subscribe(() => { this.onResult(); })); } /** * keyboard events */ initKeydownSubscription() { this.sub.add(fromEvent(this.document, 'keydown').subscribe((e) => { switch (e.key || e.code) { case 'ArrowDown': e.preventDefault(); e.stopPropagation(); this.nextOption('down'); break; case 'ArrowUp': e.preventDefault(); e.stopPropagation(); this.nextOption('up'); break; case 'Enter': e.preventDefault(); e.stopPropagation(); if (this.currentOption) { this.onResult(this.currentOption); } break; case 'Escape': e.preventDefault(); e.stopPropagation(); this.onResult(); break; } })); } ngOnInit() { this.initClosingSubscriptions(); this.initKeydownSubscription(); if (this.config && this.config.hideBackdrop) { this.initClickOutsideSub(); } } ngAfterViewInit() { this.getDropdownRects(); this.setDropdownPosition(); } ngOnDestroy() { this.sub?.unsubscribe(); this.currentOption = undefined; this.highlightedIndex = undefined; } } DropdownComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.3", ngImport: i0, type: DropdownComponent, deps: [{ token: DOCUMENT }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i1.OptionsService }], target: i0.ɵɵFactoryTarget.Component }); DropdownComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.3", type: DropdownComponent, selector: "sdk-dropdown-menu", inputs: { config: "config" }, outputs: { resultEvent: "resultEvent" }, viewQueries: [{ propertyName: "dropdownElement", first: true, predicate: ["dropdown"], descendants: true }], ngImport: i0, template: "<div class=\"sdk-dropdown-wrap\" [class.no-backdrop]=\"config?.hideBackdrop\">\n <div class=\"sdk-dropdown-backdrop\" *ngIf=\"!config?.hideBackdrop\" (click)=\"onResult()\"></div>\n <ul class=\"sdk-dropdown\" #dropdown [class.multi]=\"config?.multi\">\n <ng-container *ngIf=\"!optionsService.hasOptions\">\n <li class=\"sdk-dropdown-item\">\n <a>No options.</a>\n </li>\n </ng-container>\n <ng-container *ngIf=\"optionsService.hasOptions\">\n <ng-container *ngFor=\"let o of optionsService.optionsObservable | async; index as i\">\n <li class=\"sdk-dropdown-item\" (click)=\"select($event, o)\"\n [ngClass]=\"{\n highlighted: o.value === currentOption?.value,\n disabled: o.disabled,\n active: o.selected && !config?.multi\n }\"\n (mouseenter)=\"onOptionMouseEnter(o, i)\"\n (mouseleave)=\"onOptionMouseLeave()\">\n <span class=\"sdk-dropdown-item-checked\" *ngIf=\"config?.multi\">\n <ng-container *ngIf=\"o.selected\">\n <sdk-icon icon=\"check\" size=\"16\"></sdk-icon>\n </ng-container>\n </span>\n <ng-container *ngIf=\"o.image\">\n <img alt=\"{{o.label}}\" class=\"sdk-dropdown-item-image\" [src]=\"o.image\"/>\n </ng-container>\n <a>{{o.label}}</a>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n</div>\n\n\n", styles: [".sdk-dropdown-wrap{position:fixed;top:0;left:0;bottom:0;width:100vw;height:100vh;z-index:200;cursor:default}.sdk-dropdown-wrap.no-backdrop{width:auto;height:auto}.sdk-dropdown-backdrop{position:absolute;top:0;left:0;bottom:0;right:0;z-index:1}.sdk-dropdown{position:absolute;max-height:240px;min-width:64px;padding:8px 0;z-index:3;animation:sdk-dropdown .2s;transition:transform .2s,opacity .2s;background-color:var(--regular-a20);box-shadow:0 1px #1515150a,0 1px 2px #15151514;opacity:0;border-radius:4px;overflow-y:auto;overflow-x:hidden;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar;scroll-behavior:smooth}.sdk-dropdown.multi .sdk-dropdown-item{padding:8px 16px 8px 4px}.sdk-dropdown .sdk-dropdown-item{position:relative;display:flex;align-items:center;align-content:center;flex-direction:row;flex-wrap:nowrap;white-space:nowrap;width:100%;padding:8px 16px;transition:all .2s ease;cursor:pointer}.sdk-dropdown .sdk-dropdown-item.highlighted{background-color:var(--regular-a50)}.sdk-dropdown .sdk-dropdown-item.active{background-color:var(--regular-a80)}.sdk-dropdown .sdk-dropdown-item.disabled{cursor:default;pointer-events:none;opacity:.5}.sdk-dropdown .sdk-dropdown-item a{font-size:17px;line-height:24px}.sdk-dropdown .sdk-dropdown-item .sdk-dropdown-item-checked{display:flex;height:16px;width:16px;margin:0 4px}.sdk-dropdown .sdk-dropdown-item .sdk-dropdown-item-image{height:24px;width:24px;border-radius:100%}@keyframes sdk-dropdown{0%{opacity:0;transform:scale(.9)}}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.IconComponent, selector: "sdk-icon", inputs: ["icon", "image", "inline", "size", "color"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.3", ngImport: i0, type: DropdownComponent, decorators: [{ type: Component, args: [{ selector: 'sdk-dropdown-menu', template: "<div class=\"sdk-dropdown-wrap\" [class.no-backdrop]=\"config?.hideBackdrop\">\n <div class=\"sdk-dropdown-backdrop\" *ngIf=\"!config?.hideBackdrop\" (click)=\"onResult()\"></div>\n <ul class=\"sdk-dropdown\" #dropdown [class.multi]=\"config?.multi\">\n <ng-container *ngIf=\"!optionsService.hasOptions\">\n <li class=\"sdk-dropdown-item\">\n <a>No options.</a>\n </li>\n </ng-container>\n <ng-container *ngIf=\"optionsService.hasOptions\">\n <ng-container *ngFor=\"let o of optionsService.optionsObservable | async; index as i\">\n <li class=\"sdk-dropdown-item\" (click)=\"select($event, o)\"\n [ngClass]=\"{\n highlighted: o.value === currentOption?.value,\n disabled: o.disabled,\n active: o.selected && !config?.multi\n }\"\n (mouseenter)=\"onOptionMouseEnter(o, i)\"\n (mouseleave)=\"onOptionMouseLeave()\">\n <span class=\"sdk-dropdown-item-checked\" *ngIf=\"config?.multi\">\n <ng-container *ngIf=\"o.selected\">\n <sdk-icon icon=\"check\" size=\"16\"></sdk-icon>\n </ng-container>\n </span>\n <ng-container *ngIf=\"o.image\">\n <img alt=\"{{o.label}}\" class=\"sdk-dropdown-item-image\" [src]=\"o.image\"/>\n </ng-container>\n <a>{{o.label}}</a>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n</div>\n\n\n", styles: [".sdk-dropdown-wrap{position:fixed;top:0;left:0;bottom:0;width:100vw;height:100vh;z-index:200;cursor:default}.sdk-dropdown-wrap.no-backdrop{width:auto;height:auto}.sdk-dropdown-backdrop{position:absolute;top:0;left:0;bottom:0;right:0;z-index:1}.sdk-dropdown{position:absolute;max-height:240px;min-width:64px;padding:8px 0;z-index:3;animation:sdk-dropdown .2s;transition:transform .2s,opacity .2s;background-color:var(--regular-a20);box-shadow:0 1px #1515150a,0 1px 2px #15151514;opacity:0;border-radius:4px;overflow-y:auto;overflow-x:hidden;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar;scroll-behavior:smooth}.sdk-dropdown.multi .sdk-dropdown-item{padding:8px 16px 8px 4px}.sdk-dropdown .sdk-dropdown-item{position:relative;display:flex;align-items:center;align-content:center;flex-direction:row;flex-wrap:nowrap;white-space:nowrap;width:100%;padding:8px 16px;transition:all .2s ease;cursor:pointer}.sdk-dropdown .sdk-dropdown-item.highlighted{background-color:var(--regular-a50)}.sdk-dropdown .sdk-dropdown-item.active{background-color:var(--regular-a80)}.sdk-dropdown .sdk-dropdown-item.disabled{cursor:default;pointer-events:none;opacity:.5}.sdk-dropdown .sdk-dropdown-item a{font-size:17px;line-height:24px}.sdk-dropdown .sdk-dropdown-item .sdk-dropdown-item-checked{display:flex;height:16px;width:16px;margin:0 4px}.sdk-dropdown .sdk-dropdown-item .sdk-dropdown-item-image{height:24px;width:24px;border-radius:100%}@keyframes sdk-dropdown{0%{opacity:0;transform:scale(.9)}}\n"] }] }], ctorParameters: function () { return [{ type: undefined, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i1.OptionsService }]; }, propDecorators: { dropdownElement: [{ type: ViewChild, args: ['dropdown', { static: false }] }], resultEvent: [{ type: Output }], config: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.component.js","sourceRoot":"","sources":["../../../../../libs/ngx-slice-kit/src/lib/dropdowns/dropdown.component.ts","../../../../../libs/ngx-slice-kit/src/lib/dropdowns/dropdown.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEH,SAAS,EAET,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,MAAM,EAEN,SAAS,GACZ,MAAM,eAAe,CAAC;AAGvB,OAAO,EAAE,SAAS,EAAgB,MAAM,MAAM,CAAC;AAC/C,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;;;;;AAG3C,MAAM,qBAAqB,GAAG,GAAG,CAAC;AAClC,oFAAoF;AACpF,MAAM,uBAAuB,GAAG,GAAG,CAAC;AAOpC,MAAM,OAAO,iBAAiB;IAY1B,YAC8B,QAAa,EAC/B,IAAgB,EAChB,QAAmB,EACpB,cAA8B;QAHX,aAAQ,GAAR,QAAQ,CAAK;QAC/B,SAAI,GAAJ,IAAI,CAAY;QAChB,aAAQ,GAAR,QAAQ,CAAW;QACpB,mBAAc,GAAd,cAAc,CAAgB;QAbxB,gBAAW,GAAsB,IAAI,YAAY,EAAO,CAAC;QAMnE,UAAK,GAAqC,EAAE,CAAC;IASpD,CAAC;IAEM,kBAAkB,CAAC,CAAc,EAAE,KAAa;QACnD,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;QACvB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;IAClC,CAAC;IAEM,kBAAkB;QACrB,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;QAC/B,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;IACtC,CAAC;IAEM,MAAM,CAAC,EAAO,EAAE,MAAmB;QACtC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;IAC1B,CAAC;IAEM,QAAQ,CAAC,GAAiB;QAC7B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC3B,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;QAC5B,IAAI,CAAC,GAAG,EAAE,WAAW,EAAE,CAAC;IAC5B,CAAC;IAEM,UAAU,CAAC,SAAwB;QACtC,IAAI,CAAC,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,gBAAgB,KAAK,CAAC,EAAE;YACvD,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YACnE,IAAI,CAAC,CAAC,QAAQ,EAAE;gBACZ,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;aACzE;iBAAM;gBACH,IAAI,CAAC,gBAAgB,GAAG,SAAS,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;aAC3F;SACJ;QAED,IAAI,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,MAAM,KAAK,CAAC,EAAE;YAC3C,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YACpD,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;SAC7B;aAAM;YACH,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,KAAK,CAAC,CAAC;YAClD,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,MAAM,CAAC;YACzD,MAAM,YAAY,GAAG,IAAI,CAAC,gBAAgB,KAAK,aAAa,GAAG,CAAC,CAAC;YACjE,MAAM,KAAK,GAAG,SAAS,KAAK,MAAM,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;gBACnF,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC,CAAC;YAEtE,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YACxD,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;SACjC;QAED,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC;IACjD,CAAC;IAEM,UAAU,CAAC,SAAwB;QACtC,MAAM,kBAAkB,GAAG,CAAC,CAAC;QAC7B;;;WAGG;QACH,MAAM,EAAC,YAAY,EAAE,SAAS,EAAE,YAAY,EAAC,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC;QACnF,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC;QAC1E,IAAI,CAAC,WAAW,EAAE;YACd,OAAO;SACV;QACD,MAAM,UAAU,GAAG,WAAW,CAAC,YAAY,CAAC;QAE5C,QAAQ,SAAS,EAAE;YACf,KAAK,IAAI;gBACL,IAAI,WAAW,CAAC,SAAS,GAAG,kBAAkB,GAAG,SAAS,EAAE;oBACxD,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,SAAS,GAAG,SAAS,GAAG,YAAY,CAAC;iBAC3E;gBACD,IAAI,WAAW,CAAC,SAAS,GAAG,UAAU,GAAG,SAAS,GAAG,YAAY,EAAE;oBAC/D,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,SAAS,GAAG,YAAY,GAAG,YAAY,CAAC;iBAC9E;gBACD,MAAM;YACV;gBACI,IAAI,WAAW,CAAC,SAAS,GAAG,UAAU,GAAG,YAAY,GAAG,SAAS,EAAE;oBAC/D,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,SAAS,GAAG,SAAS,GAAG,UAAU,CAAC;iBACzE;qBAAM,IAAI,WAAW,CAAC,SAAS,KAAK,kBAAkB,EAAE;oBACrD,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,SAAS,GAAG,CAAC,CAAC;iBACpD;SACR;IACL,CAAC;IAED;;OAEG;IACI,mBAAmB;QACtB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,EAAE,KAAK,CAAC,EAAE;YAC5C,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAChE,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAClE,IAAI,CAAC,QAAQ,IAAI,CAAC,UAAU,EAAE;gBAC1B,IAAI,CAAC,QAAQ,EAAE,CAAC;aACnB;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACI,gBAAgB;QACnB,MAAM,YAAY,GAAG,MAAM,CAAC,WAAW,CAAC;QACxC,MAAM,UAAU,GAAG,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC;QAC/F,MAAM,KAAK,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC;QAEzE,IAAI,CAAC,QAAQ,GAAG,UAAU,IAAI,uBAAuB,CAAC;QACtD,IAAI,IAAI,CAAC,QAAQ,EAAE;YACf,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC;SAClE;aAAM;YACH,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC;SACnD;QAED,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE;YACtB,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC;SACpD;QAED,IAAI,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,EAAE;YAC7C,MAAM,WAAW,GAAG,MAAM,CAAC,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC;YACrG,IAAI,WAAW,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,EAAE;gBACjC,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;aAClG;SACJ;IACL,CAAC;IAED;;;;;OAKG;IACI,mBAAmB;QACtB,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE;YAClB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,CAAC,CAAC;SAChG;QAED,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE;YAChB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,KAAK,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,CAAC;SAC5F;aAAM;YACH,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC;SAClG;QAED,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC,CAAC;QAC3H,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC;IAC7E,CAAC;IAED;;OAEG;IACI,wBAAwB;QAC3B,IAAI,CAAC,GAAG,GAAG,SAAS,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC,IAAI,CACvC,IAAI,CAAC,CAAC,CAAC,CACV,CAAC,SAAS,CAAC,GAAG,EAAE;YACb,IAAI,CAAC,QAAQ,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC,IAAI,CACzC,IAAI,CAAC,CAAC,CAAC,CACV,CAAC,SAAS,CAAC,GAAG,EAAE;YACb,IAAI,CAAC,QAAQ,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC,CAAC;IACR,CAAC;IAED;;OAEG;IACI,uBAAuB;QAC1B,IAAI,CAAC,GAAG,CAAC,GAAG,CACR,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC,SAAS,CAAC,CAAC,CAAgB,EAAE,EAAE;YAC/D,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE;gBACrB,KAAK,WAAW;oBACZ,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,eAAe,EAAE,CAAC;oBACpB,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;oBACxB,MAAM;gBACV,KAAK,SAAS;oBACV,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,eAAe,EAAE,CAAC;oBACpB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;oBACtB,MAAM;gBACV,KAAK,OAAO;oBACR,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,eAAe,EAAE,CAAC;oBACpB,IAAI,IAAI,CAAC,aAAa,EAAE;wBACpB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;qBACrC;oBACD,MAAM;gBACV,KAAK,QAAQ;oBACT,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,eAAe,EAAE,CAAC;oBACpB,IAAI,CAAC,QAAQ,EAAE,CAAC;oBAChB,MAAM;aACb;QACL,CAAC,CAAC,CACL,CAAC;IACN,CAAC;IAEM,QAAQ;QACX,IAAI,CAAC,wBAAwB,EAAE,CAAC;QAChC,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE;YACzC,IAAI,CAAC,mBAAmB,EAAE,CAAC;SAC9B;IACL,CAAC;IAEM,eAAe;QAClB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC/B,CAAC;IAEM,WAAW;QACd,IAAI,CAAC,GAAG,EAAE,WAAW,EAAE,CAAC;QACxB,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;QAC/B,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;IACtC,CAAC;;8GAnOQ,iBAAiB,kBAad,QAAQ;kGAbX,iBAAiB,6OC7B9B,opDAkCA;2FDLa,iBAAiB;kBAL7B,SAAS;+BACI,mBAAmB;;0BAiBxB,MAAM;2BAAC,QAAQ;0HAX2B,eAAe;sBAA7D,SAAS;uBAAC,UAAU,EAAE,EAAC,MAAM,EAAE,KAAK,EAAC;gBACrB,WAAW;sBAA3B,MAAM;gBACS,MAAM;sBAArB,KAAK","sourcesContent":["import {\n    AfterViewInit,\n    Component,\n    ElementRef,\n    EventEmitter,\n    Inject,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    Renderer2,\n    ViewChild,\n} from '@angular/core';\nimport { OptionModel } from './dropdown-option.model';\nimport { DropdownOptions } from './dropdown.model';\nimport { fromEvent, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { DOCUMENT } from '@angular/common';\nimport { OptionsService } from './options.service';\n\nconst DEFAULT_DROPDOWN_SIZE = 240;\n// I'm taking additional 16 pixels here, to prevent element sticking to bottom panel\nconst DEFAULT_DROPDOWN_OFFSET = 256;\n\n@Component({\n    selector: 'sdk-dropdown-menu',\n    templateUrl: './dropdown.component.html',\n    styleUrls: ['./dropdown.component.scss']\n})\nexport class DropdownComponent implements OnInit, AfterViewInit, OnDestroy {\n\n    @ViewChild('dropdown', {static: false}) public dropdownElement: ElementRef;\n    @Output() public resultEvent: EventEmitter<any> = new EventEmitter<any>();\n    @Input() public config: DropdownOptions;\n\n    public sub: Subscription;\n    public currentOption: OptionModel;\n    public inverted: boolean;\n    public rects: { bottom?, top?, left?, width? } = {};\n    public highlightedIndex: number;\n\n    constructor(\n        @Inject(DOCUMENT) private document: any,\n        private elem: ElementRef,\n        private renderer: Renderer2,\n        public optionsService: OptionsService\n    ) {\n    }\n\n    public onOptionMouseEnter(o: OptionModel, index: number): void {\n        this.currentOption = o;\n        this.highlightedIndex = index;\n    }\n\n    public onOptionMouseLeave(): void {\n        this.currentOption = undefined;\n        this.highlightedIndex = undefined;\n    }\n\n    public select(ev: any, option: OptionModel): void {\n        this.onResult(option);\n    }\n\n    public onResult(res?: OptionModel): void {\n        this.resultEvent.emit(res);\n        this.resultEvent.complete();\n        this.sub?.unsubscribe();\n    }\n\n    public nextOption(direction: 'up' | 'down'): void {\n        if (!this.highlightedIndex && this.highlightedIndex !== 0) {\n            const selected = this.optionsService.options.find(o => o.selected);\n            if (!!selected) {\n                this.highlightedIndex = this.optionsService.options.indexOf(selected);\n            } else {\n                this.highlightedIndex = direction === 'up' ? 0 : this.optionsService.options.length - 1;\n            }\n        }\n\n        if (this.optionsService.options?.length === 1) {\n            this.currentOption = this.optionsService.options[0];\n            this.highlightedIndex = 0;\n        } else {\n            const isFirstOption = this.highlightedIndex === 0;\n            const optionsLength = this.optionsService.options.length;\n            const isLastOption = this.highlightedIndex === optionsLength - 1;\n            const index = direction === 'down' ? (isLastOption ? 0 : (this.highlightedIndex + 1)) :\n                (isFirstOption ? (optionsLength - 1) : this.highlightedIndex - 1);\n\n            this.currentOption = this.optionsService.options[index];\n            this.highlightedIndex = index;\n        }\n\n        setTimeout(() => this.autoScroll(direction));\n    }\n\n    public autoScroll(direction: 'up' | 'down'): void {\n        const dropdownPaddingTop = 8;\n        /**\n         * okay that padding is constant at the top and the bottom of the element,\n         * so it should be used in calculations when we are counting top offset\n         */\n        const {scrollHeight, scrollTop, offsetHeight} = this.dropdownElement.nativeElement;\n        const currentItem = this.elem.nativeElement.querySelector(`.highlighted`);\n        if (!currentItem) {\n            return;\n        }\n        const itemHeight = currentItem.offsetHeight;\n\n        switch (direction) {\n            case 'up':\n                if (currentItem.offsetTop - dropdownPaddingTop < scrollTop) {\n                    this.dropdownElement.nativeElement.scrollTop = scrollTop - offsetHeight;\n                }\n                if (currentItem.offsetTop + itemHeight > scrollTop + offsetHeight) {\n                    this.dropdownElement.nativeElement.scrollTop = scrollHeight - offsetHeight;\n                }\n                break;\n            default:\n                if (currentItem.offsetTop + itemHeight > offsetHeight + scrollTop) {\n                    this.dropdownElement.nativeElement.scrollTop = scrollTop + itemHeight;\n                } else if (currentItem.offsetTop === dropdownPaddingTop) {\n                    this.dropdownElement.nativeElement.scrollTop = 0;\n                }\n        }\n    }\n\n    /**\n     * click outside subscription if backdrop disabled. also dropdown must be attached during input\n     */\n    public initClickOutsideSub(): void {\n        this.renderer.listen('window', 'click', event => {\n            const isParent = this.config.parentElem?.contains(event.target);\n            const isDropdown = this.elem.nativeElement.contains(event.target);\n            if (!isParent && !isDropdown) {\n                this.onResult();\n            }\n        });\n    }\n\n    /**\n     * detect relative position to prevent dropdown being hidden over `overflow: none`\n     */\n    public getDropdownRects(): void {\n        const windowHeight = window.innerHeight;\n        const pixelsLeft = windowHeight - this.config.triggerRect.height - this.config.triggerRect.top;\n        const rects = this.dropdownElement.nativeElement.getBoundingClientRect();\n\n        this.inverted = pixelsLeft <= DEFAULT_DROPDOWN_OFFSET;\n        if (this.inverted) {\n            this.rects.bottom = windowHeight - this.config.triggerRect.top;\n        } else {\n            this.rects.top = this.config.triggerRect.bottom;\n        }\n\n        if (this.config.fitWidth) {\n            this.rects.width = this.config.triggerRect.width;\n        }\n\n        if (rects.width > this.config.triggerRect.width) {\n            const rightOffset = window.innerWidth - this.config.triggerRect.width - this.config.triggerRect.left;\n            if (rightOffset <= rects.width + 16) {\n                this.rects.left = (this.config.triggerRect.left + this.config.triggerRect.width) - rects.width;\n            }\n        }\n    }\n\n    /**\n     * - if `fitWidth` config options is true\n     * there are width declared depending on its parent element\n     * - if `rects.top` calculated there is enough place to drop it down,\n     *  if it hits `rects.bottom` – show it above the element\n     */\n    public setDropdownPosition(): void {\n        if (this.rects.width) {\n            this.renderer.setStyle(this.dropdownElement.nativeElement, `width`, `${this.rects.width}px`);\n        }\n\n        if (this.rects.top) {\n            this.renderer.setStyle(this.dropdownElement.nativeElement, `top`, `${this.rects.top}px`);\n        } else {\n            this.renderer.setStyle(this.dropdownElement.nativeElement, `bottom`, `${this.rects.bottom}px`);\n        }\n\n        this.renderer.setStyle(this.dropdownElement.nativeElement, `left`, `${this.rects.left || this.config.triggerRect.left}px`);\n        this.renderer.setStyle(this.dropdownElement.nativeElement, `opacity`, 1);\n    }\n\n    /**\n     * detect window resize and scroll to prevent failed dropdown position\n     */\n    public initClosingSubscriptions(): void {\n        this.sub = fromEvent(window, 'scroll').pipe(\n            take(1)\n        ).subscribe(() => {\n            this.onResult();\n        });\n\n        this.sub.add(fromEvent(window, 'resize').pipe(\n            take(1)\n        ).subscribe(() => {\n            this.onResult();\n        }));\n    }\n\n    /**\n     * keyboard events\n     */\n    public initKeydownSubscription(): void {\n        this.sub.add(\n            fromEvent(this.document, 'keydown').subscribe((e: KeyboardEvent) => {\n                switch (e.key || e.code) {\n                    case 'ArrowDown':\n                        e.preventDefault();\n                        e.stopPropagation();\n                        this.nextOption('down');\n                        break;\n                    case 'ArrowUp':\n                        e.preventDefault();\n                        e.stopPropagation();\n                        this.nextOption('up');\n                        break;\n                    case 'Enter':\n                        e.preventDefault();\n                        e.stopPropagation();\n                        if (this.currentOption) {\n                            this.onResult(this.currentOption);\n                        }\n                        break;\n                    case 'Escape':\n                        e.preventDefault();\n                        e.stopPropagation();\n                        this.onResult();\n                        break;\n                }\n            })\n        );\n    }\n\n    public ngOnInit(): void {\n        this.initClosingSubscriptions();\n        this.initKeydownSubscription();\n        if (this.config && this.config.hideBackdrop) {\n            this.initClickOutsideSub();\n        }\n    }\n\n    public ngAfterViewInit(): void {\n        this.getDropdownRects();\n        this.setDropdownPosition();\n    }\n\n    public ngOnDestroy(): void {\n        this.sub?.unsubscribe();\n        this.currentOption = undefined;\n        this.highlightedIndex = undefined;\n    }\n\n}\n","<div class=\"sdk-dropdown-wrap\" [class.no-backdrop]=\"config?.hideBackdrop\">\n    <div class=\"sdk-dropdown-backdrop\" *ngIf=\"!config?.hideBackdrop\" (click)=\"onResult()\"></div>\n    <ul class=\"sdk-dropdown\" #dropdown [class.multi]=\"config?.multi\">\n        <ng-container *ngIf=\"!optionsService.hasOptions\">\n            <li class=\"sdk-dropdown-item\">\n                <a>No options.</a>\n            </li>\n        </ng-container>\n        <ng-container *ngIf=\"optionsService.hasOptions\">\n            <ng-container *ngFor=\"let o of optionsService.optionsObservable | async; index as i\">\n                <li class=\"sdk-dropdown-item\" (click)=\"select($event, o)\"\n                    [ngClass]=\"{\n                        highlighted: o.value === currentOption?.value,\n                        disabled: o.disabled,\n                        active: o.selected && !config?.multi\n                    }\"\n                    (mouseenter)=\"onOptionMouseEnter(o, i)\"\n                    (mouseleave)=\"onOptionMouseLeave()\">\n                    <span class=\"sdk-dropdown-item-checked\" *ngIf=\"config?.multi\">\n                        <ng-container *ngIf=\"o.selected\">\n                            <sdk-icon icon=\"check\" size=\"16\"></sdk-icon>\n                        </ng-container>\n                    </span>\n                    <ng-container *ngIf=\"o.image\">\n                        <img alt=\"{{o.label}}\" class=\"sdk-dropdown-item-image\" [src]=\"o.image\"/>\n                    </ng-container>\n                    <a>{{o.label}}</a>\n                </li>\n            </ng-container>\n        </ng-container>\n    </ul>\n</div>\n\n\n"]}