UNPKG

ngx-bootstrap

Version:
311 lines 49.8 kB
import { ChangeDetectorRef, Component, ElementRef, HostListener, QueryList, Renderer2, ViewChild, ViewChildren, Output, EventEmitter } from '@angular/core'; import { Utils } from 'ngx-bootstrap/utils'; import { PositioningService } from 'ngx-bootstrap/positioning'; import { Subscription } from 'rxjs'; import { latinize } from './typeahead-utils'; import { typeaheadAnimation } from './typeahead-animations'; import * as i0 from "@angular/core"; import * as i1 from "ngx-bootstrap/positioning"; import * as i2 from "@angular/common"; let nextWindowId = 0; export class TypeaheadContainerComponent { get typeaheadTemplateMethods() { return { selectMatch: this.selectMatch.bind(this), selectActive: this.selectActive.bind(this), isActive: this.isActive.bind(this) }; } constructor(positionService, renderer, element, changeDetectorRef) { this.positionService = positionService; this.renderer = renderer; this.element = element; this.changeDetectorRef = changeDetectorRef; // eslint-disable-next-line @angular-eslint/no-output-rename this.activeChangeEvent = new EventEmitter(); this.isFocused = false; this.positionServiceSubscription = new Subscription(); this.height = 0; this.popupId = `ngb-typeahead-${nextWindowId++}`; this._matches = []; this.renderer.setAttribute(this.element.nativeElement, 'id', this.popupId); this.positionServiceSubscription.add(this.positionService.event$?.subscribe(() => { if (this.isAnimated) { this.animationState = this.isTopPosition ? 'animated-up' : 'animated-down'; this.changeDetectorRef.detectChanges(); return; } this.animationState = 'unanimated'; this.changeDetectorRef.detectChanges(); })); } get active() { return this._active; } set active(active) { this._active = active; this.activeChanged(); } get matches() { return this._matches; } set matches(value) { this.positionService.setOptions({ modifiers: { flip: { enabled: this.adaptivePosition } }, allowedPositions: ['top', 'bottom'] }); this._matches = value; this.needScrollbar = this.typeaheadScrollable && this.typeaheadOptionsInScrollableView < this.matches.length; if (this.typeaheadScrollable) { setTimeout(() => { this.setScrollableMode(); }); } if (this.typeaheadIsFirstItemActive && this._matches.length > 0) { this.setActive(this._matches[0]); if (this._active?.isHeader()) { this.nextActiveMatch(); } } if (this._active && !this.typeaheadIsFirstItemActive) { const concurrency = this._matches.find(match => match.value === this._active?.value); if (concurrency) { this.selectActive(concurrency); return; } this.active = void 0; } } get isTopPosition() { return this.element.nativeElement.classList.contains('top'); } get optionsListTemplate() { return this.parent ? this.parent.optionsListTemplate : undefined; } get isAnimated() { return this.parent ? this.parent.isAnimated : false; } get adaptivePosition() { return this.parent ? this.parent.adaptivePosition : false; } get typeaheadScrollable() { return this.parent ? this.parent.typeaheadScrollable : false; } get typeaheadOptionsInScrollableView() { return this.parent ? this.parent.typeaheadOptionsInScrollableView : 5; } get typeaheadIsFirstItemActive() { return this.parent ? this.parent.typeaheadIsFirstItemActive : true; } // eslint-disable-next-line @typescript-eslint/no-explicit-any get itemTemplate() { return this.parent ? this.parent.typeaheadItemTemplate : undefined; } get canSelectItemsOnBlur() { return !!this.parent?.selectItemOnBlur; } selectActiveMatch(isActiveItemChanged) { if (this._active && this.parent?.typeaheadSelectFirstItem) { this.selectMatch(this._active); } if (!this.parent?.typeaheadSelectFirstItem && isActiveItemChanged) { this.selectMatch(this._active); } } activeChanged() { if (!this._active) { return; } const index = this.matches.indexOf(this._active); this.activeChangeEvent.emit(`${this.popupId}-${index}`); } prevActiveMatch() { if (!this._active) { return; } const index = this.matches.indexOf(this._active); this.setActive(this.matches[index - 1 < 0 ? this.matches.length - 1 : index - 1]); if (this._active.isHeader()) { this.prevActiveMatch(); } if (this.typeaheadScrollable) { this.scrollPrevious(index); } } nextActiveMatch() { const index = this._active ? this.matches.indexOf(this._active) : -1; this.setActive(this.matches[index + 1 > this.matches.length - 1 ? 0 : index + 1]); if (this._active?.isHeader()) { this.nextActiveMatch(); } if (this.typeaheadScrollable) { this.scrollNext(index); } } selectActive(value) { this.isFocused = true; this.setActive(value); } highlight(match, query) { let itemStr = match.value; let itemStrHelper = (this.parent && this.parent.typeaheadLatinize ? latinize(itemStr) : itemStr).toLowerCase(); let startIdx; let tokenLen; // Replaces the capture string with the same string inside of a "strong" tag if (typeof query === 'object') { const queryLen = query.length; for (let i = 0; i < queryLen; i += 1) { // query[i] is already latinized and lower case startIdx = itemStrHelper.indexOf(query[i]); tokenLen = query[i].length; if (startIdx >= 0 && tokenLen > 0) { itemStr = `${itemStr.substring(0, startIdx)}<strong>${itemStr.substring(startIdx, startIdx + tokenLen)}</strong>` + `${itemStr.substring(startIdx + tokenLen)}`; itemStrHelper = `${itemStrHelper.substring(0, startIdx)} ${' '.repeat(tokenLen)} ` + `${itemStrHelper.substring(startIdx + tokenLen)}`; } } } else if (query) { // query is already latinized and lower case startIdx = itemStrHelper.indexOf(query); tokenLen = query.length; if (startIdx >= 0 && tokenLen > 0) { itemStr = `${itemStr.substring(0, startIdx)}<strong>${itemStr.substring(startIdx, startIdx + tokenLen)}</strong>` + `${itemStr.substring(startIdx + tokenLen)}`; } } return itemStr; } focusLost() { this.isFocused = false; if (!this.canSelectItemsOnBlur) { this.setActive(void 0); } } isActive(value) { return this.active === value; } selectMatch(value, event) { if (event) { event.stopPropagation(); event.preventDefault(); } this.parent?.changeModel(value); setTimeout(() => this.parent?.typeaheadOnSelect.emit(value), 0); return false; } setScrollableMode() { if (!this.ulElement) { this.ulElement = this.element; } if (this.liElements?.first) { const ulStyles = Utils.getStyles(this.ulElement.nativeElement); const liStyles = Utils.getStyles(this.liElements.first.nativeElement); const ulPaddingBottom = parseFloat((ulStyles['padding-bottom'] ? ulStyles['padding-bottom'] : '') .replace('px', '')); const ulPaddingTop = parseFloat((ulStyles['padding-top'] ? ulStyles['padding-top'] : '0') .replace('px', '')); const optionHeight = parseFloat((liStyles.height ? liStyles.height : '0') .replace('px', '')); const height = this.typeaheadOptionsInScrollableView * optionHeight; this.guiHeight = `${height + ulPaddingTop + ulPaddingBottom}px`; } this.renderer.setStyle(this.element.nativeElement, 'visibility', 'visible'); } scrollPrevious(index) { if (index === 0) { this.scrollToBottom(); return; } if (this.liElements && this.ulElement) { const liElement = this.liElements.toArray()[index - 1]; if (liElement && !this.isScrolledIntoView(liElement.nativeElement)) { this.ulElement.nativeElement.scrollTop = liElement.nativeElement.offsetTop; } } } scrollNext(index) { if (index + 1 > this.matches.length - 1) { this.scrollToTop(); return; } if (this.liElements && this.ulElement) { const liElement = this.liElements.toArray()[index + 1]; if (liElement && !this.isScrolledIntoView(liElement.nativeElement)) { this.ulElement.nativeElement.scrollTop = liElement.nativeElement.offsetTop - Number(this.ulElement.nativeElement.offsetHeight) + Number(liElement.nativeElement.offsetHeight); } } } ngOnDestroy() { this.positionServiceSubscription.unsubscribe(); } setActive(value) { this._active = value; let preview; if (!(this._active == null || this._active.isHeader())) { preview = value; } this.parent?.typeaheadOnPreview.emit(preview); } isScrolledIntoView(elem) { if (!this.ulElement) { return false; } const containerViewTop = this.ulElement.nativeElement.scrollTop; const containerViewBottom = containerViewTop + Number(this.ulElement.nativeElement.offsetHeight); const elemTop = elem.offsetTop; const elemBottom = elemTop + elem.offsetHeight; return ((elemBottom <= containerViewBottom) && (elemTop >= containerViewTop)); } ; scrollToBottom() { if (!this.ulElement?.nativeElement) { return; } this.ulElement.nativeElement.scrollTop = this.ulElement.nativeElement.scrollHeight; } scrollToTop() { if (!this.ulElement?.nativeElement) { return; } this.ulElement.nativeElement.scrollTop = 0; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.4", ngImport: i0, type: TypeaheadContainerComponent, deps: [{ token: i1.PositioningService }, { token: i0.Renderer2 }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.4", type: TypeaheadContainerComponent, selector: "typeahead-container", outputs: { activeChangeEvent: "activeChange" }, host: { listeners: { "mouseleave": "focusLost()", "blur": "focusLost()" }, properties: { "style.height": "needScrollbar ? guiHeight: 'auto'", "style.visibility": "'inherit'", "class.dropup": "dropup", "attr.role": "'listbox'" }, styleAttribute: "position: absolute;display: block;", classAttribute: "dropdown open bottom dropdown-menu" }, viewQueries: [{ propertyName: "ulElement", first: true, predicate: ["ulElement"], descendants: true }, { propertyName: "liElements", predicate: ["liElements"], descendants: true }], ngImport: i0, template: "<!-- inject options list template -->\n<ng-template [ngTemplateOutlet]=\"optionsListTemplate || bs4Template\"\n [ngTemplateOutletContext]=\"{\n matches: matches,\n itemTemplate: itemTemplate || bsItemTemplate,\n query: query,\n $implicit: typeaheadTemplateMethods\n }\">\n</ng-template>\n\n<!-- default options item template -->\n<ng-template #bsItemTemplate let-match=\"match\" let-query=\"query\">\n <span [innerHtml]=\"highlight(match, query)\"></span>\n</ng-template>\n\n<!-- Bootstrap 4 options list template -->\n<ng-template #bs4Template>\n <ng-template ngFor let-match let-i=\"index\" [ngForOf]=\"matches\">\n <h6 *ngIf=\"match.isHeader()\" class=\"dropdown-header\">{{ match }}</h6>\n <ng-template [ngIf]=\"!match.isHeader()\">\n <button #liElements\n [id]=\"popupId + '-' + i\"\n role=\"option\"\n [@typeaheadAnimation]=\"animationState\"\n class=\"dropdown-item\"\n (click)=\"selectMatch(match, $event)\"\n (mouseenter)=\"selectActive(match)\"\n [class.active]=\"isActive(match)\">\n <ng-template [ngTemplateOutlet]=\"itemTemplate || bsItemTemplate\"\n [ngTemplateOutletContext]=\"{item: match.item, index: i, match: match, query: query}\">\n </ng-template>\n </button>\n </ng-template>\n </ng-template>\n</ng-template>\n", styles: [":host.dropdown{z-index:1000}:host.dropdown-menu,.dropdown-menu{overflow-y:auto;height:100px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], animations: [typeaheadAnimation] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.4", ngImport: i0, type: TypeaheadContainerComponent, decorators: [{ type: Component, args: [{ selector: 'typeahead-container', host: { class: 'dropdown open bottom dropdown-menu', '[style.height]': `needScrollbar ? guiHeight: 'auto'`, '[style.visibility]': `'inherit'`, '[class.dropup]': 'dropup', style: 'position: absolute;display: block;', '[attr.role]': `'listbox'` }, animations: [typeaheadAnimation], template: "<!-- inject options list template -->\n<ng-template [ngTemplateOutlet]=\"optionsListTemplate || bs4Template\"\n [ngTemplateOutletContext]=\"{\n matches: matches,\n itemTemplate: itemTemplate || bsItemTemplate,\n query: query,\n $implicit: typeaheadTemplateMethods\n }\">\n</ng-template>\n\n<!-- default options item template -->\n<ng-template #bsItemTemplate let-match=\"match\" let-query=\"query\">\n <span [innerHtml]=\"highlight(match, query)\"></span>\n</ng-template>\n\n<!-- Bootstrap 4 options list template -->\n<ng-template #bs4Template>\n <ng-template ngFor let-match let-i=\"index\" [ngForOf]=\"matches\">\n <h6 *ngIf=\"match.isHeader()\" class=\"dropdown-header\">{{ match }}</h6>\n <ng-template [ngIf]=\"!match.isHeader()\">\n <button #liElements\n [id]=\"popupId + '-' + i\"\n role=\"option\"\n [@typeaheadAnimation]=\"animationState\"\n class=\"dropdown-item\"\n (click)=\"selectMatch(match, $event)\"\n (mouseenter)=\"selectActive(match)\"\n [class.active]=\"isActive(match)\">\n <ng-template [ngTemplateOutlet]=\"itemTemplate || bsItemTemplate\"\n [ngTemplateOutletContext]=\"{item: match.item, index: i, match: match, query: query}\">\n </ng-template>\n </button>\n </ng-template>\n </ng-template>\n</ng-template>\n", styles: [":host.dropdown{z-index:1000}:host.dropdown-menu,.dropdown-menu{overflow-y:auto;height:100px}\n"] }] }], ctorParameters: () => [{ type: i1.PositioningService }, { type: i0.Renderer2 }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }], propDecorators: { activeChangeEvent: [{ type: Output, args: ['activeChange'] }], ulElement: [{ type: ViewChild, args: ['ulElement', { static: false }] }], liElements: [{ type: ViewChildren, args: ['liElements'] }], focusLost: [{ type: HostListener, args: ['mouseleave'] }, { type: HostListener, args: ['blur'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead-container.component.js","sourceRoot":"","sources":["../../../../src/typeahead/typeahead-container.component.ts","../../../../src/typeahead/typeahead-container.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,iBAAiB,EACjB,SAAS,EACT,UAAU,EACV,YAAY,EAEZ,SAAS,EACT,SAAS,EAET,SAAS,EACT,YAAY,EACZ,MAAM,EACN,YAAY,EACb,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,KAAK,EAAE,MAAM,qBAAqB,CAAC;AAC5C,OAAO,EAAE,kBAAkB,EAAE,MAAM,2BAA2B,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAEpC,OAAO,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAG7C,OAAO,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;;;;AAG5D,IAAI,YAAY,GAAG,CAAC,CAAC;AA6BrB,MAAM,OAAO,2BAA2B;IAmBtC,IAAI,wBAAwB;QAC1B,OAAO;YACL,WAAW,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC;YACxC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC;YAC1C,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC;SACnC,CAAC;IACJ,CAAC;IAWD,YACU,eAAmC,EACnC,QAAmB,EACpB,OAAmB,EAClB,iBAAoC;QAHpC,oBAAe,GAAf,eAAe,CAAoB;QACnC,aAAQ,GAAR,QAAQ,CAAW;QACpB,YAAO,GAAP,OAAO,CAAY;QAClB,sBAAiB,GAAjB,iBAAiB,CAAmB;QAvC9C,4DAA4D;QACpC,sBAAiB,GAAG,IAAI,YAAY,EAAE,CAAC;QAI/D,cAAS,GAAG,KAAK,CAAC;QASlB,gCAA2B,GAAG,IAAI,YAAY,EAAE,CAAC;QACjD,WAAM,GAAG,CAAC,CAAC;QACX,YAAO,GAAG,iBAAiB,YAAY,EAAE,EAAE,CAAC;QAWlC,aAAQ,GAAqB,EAAE,CAAC;QAcxC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAC3E,IAAI,CAAC,2BAA2B,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,SAAS,CACzE,GAAG,EAAE;YACH,IAAI,IAAI,CAAC,UAAU,EAAE;gBACnB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,eAAe,CAAC;gBAC3E,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;gBAEvC,OAAO;aACR;YAED,IAAI,CAAC,cAAc,GAAG,YAAY,CAAC;YACnC,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QACzC,CAAC,CACF,CAAC,CAAC;IACL,CAAC;IAED,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;IAED,IAAI,MAAM,CAAC,MAAkC;QAC3C,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC;QACtB,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED,IAAI,OAAO,CAAC,KAAuB;QACjC,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC;YAC9B,SAAS,EAAE,EAAE,IAAI,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,gBAAgB,EAAE,EAAE;YACvD,gBAAgB,EAAE,CAAC,KAAK,EAAE,QAAQ,CAAC;SACpC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QAEtB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,mBAAmB,IAAI,IAAI,CAAC,gCAAgC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC;QAE7G,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC5B,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,CAAC,CAAC,CAAC;SACJ;QAED,IAAI,IAAI,CAAC,0BAA0B,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YAEjC,IAAI,IAAI,CAAC,OAAO,EAAE,QAAQ,EAAE,EAAE;gBAC5B,IAAI,CAAC,eAAe,EAAE,CAAC;aACxB;SACF;QAED,IAAI,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,0BAA0B,EAAE;YACpD,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,KAAK,KAAK,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;YAErF,IAAI,WAAW,EAAE;gBACf,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;gBAE/B,OAAO;aACR;YAED,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC,CAAC;SACtB;IACH,CAAC;IAED,IAAI,aAAa;QACf,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC,CAAC,SAAS,CAAC;IACnE,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,KAAK,CAAC;IACtD,CAAC;IAED,IAAI,gBAAgB;QAClB,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC;IAC5D,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC,CAAC,KAAK,CAAC;IAC/D,CAAC;IAED,IAAI,gCAAgC;QAClC,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,gCAAgC,CAAC,CAAC,CAAC,CAAC,CAAC;IACxE,CAAC;IAED,IAAI,0BAA0B;QAC5B,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC;IACrE,CAAC;IACD,8DAA8D;IAC9D,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC,CAAC,SAAS,CAAC;IACrE,CAAC;IAED,IAAI,oBAAoB;QACtB,OAAO,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,gBAAgB,CAAC;IACzC,CAAC;IAED,iBAAiB,CAAC,mBAA6B;QAC7C,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,EAAE,wBAAwB,EAAE;YACzD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;SAChC;QAED,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,wBAAwB,IAAI,mBAAmB,EAAE;YACjE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;SAChC;IACH,CAAC;IAED,aAAa;QACX,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,OAAO;SACR;QACD,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACjD,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,OAAO,IAAI,KAAK,EAAE,CAAC,CAAC;IAC1D,CAAC;IAED,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,OAAO;SACR;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACjD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CACzB,KAAK,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAClD,CAAC,CAAC;QAEL,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE;YAC3B,IAAI,CAAC,eAAe,EAAE,CAAC;SACxB;QAED,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC5B,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;SAC5B;IACH,CAAC;IAED,eAAe;QACb,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CACzB,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAClD,CAAC,CAAC;QAEL,IAAI,IAAI,CAAC,OAAO,EAAE,QAAQ,EAAE,EAAE;YAC5B,IAAI,CAAC,eAAe,EAAE,CAAC;SACxB;QAED,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC5B,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;SACxB;IACH,CAAC;IAED,YAAY,CAAC,KAAqB;QAChC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IACxB,CAAC;IAED,SAAS,CAAC,KAAqB,EAAE,KAAwB;QACvD,IAAI,OAAO,GAAW,KAAK,CAAC,KAAK,CAAC;QAClC,IAAI,aAAa,GAAW,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,iBAAiB;YACvE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC;YACnB,CAAC,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAC;QAC3B,IAAI,QAAgB,CAAC;QACrB,IAAI,QAAgB,CAAC;QACrB,4EAA4E;QAC5E,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;YAC7B,MAAM,QAAQ,GAAW,KAAK,CAAC,MAAM,CAAC;YACtC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,EAAE,CAAC,IAAI,CAAC,EAAE;gBACpC,+CAA+C;gBAC/C,QAAQ,GAAG,aAAa,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;gBAC3C,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;gBAC3B,IAAI,QAAQ,IAAI,CAAC,IAAI,QAAQ,GAAG,CAAC,EAAE;oBACjC,OAAO;wBACL,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,WAAW,OAAO,CAAC,SAAS,CAAC,QAAQ,EAAE,QAAQ,GAAG,QAAQ,CAAC,WAAW;4BACvG,GAAG,OAAO,CAAC,SAAS,CAAC,QAAQ,GAAG,QAAQ,CAAC,EAAE,CAAC;oBAC9C,aAAa;wBACX,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,WAAW,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,WAAW;4BACjF,GAAG,aAAa,CAAC,SAAS,CAAC,QAAQ,GAAG,QAAQ,CAAC,EAAE,CAAC;iBACrD;aACF;SACF;aAAM,IAAI,KAAK,EAAE;YAChB,4CAA4C;YAC5C,QAAQ,GAAG,aAAa,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YACxC,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC;YACxB,IAAI,QAAQ,IAAI,CAAC,IAAI,QAAQ,GAAG,CAAC,EAAE;gBACjC,OAAO;oBACL,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,WAAW,OAAO,CAAC,SAAS,CAAC,QAAQ,EAAE,QAAQ,GAAG,QAAQ,CAAC,WAAW;wBACvG,GAAG,OAAO,CAAC,SAAS,CAAC,QAAQ,GAAG,QAAQ,CAAC,EAAE,CAAC;aAC/C;SACF;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAID,SAAS;QACP,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE;YAC9B,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;SACxB;IACH,CAAC;IAED,QAAQ,CAAC,KAAqB;QAC5B,OAAO,IAAI,CAAC,MAAM,KAAK,KAAK,CAAC;IAC/B,CAAC;IAED,WAAW,CAAC,KAAsB,EAAE,KAAa;QAC/C,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE,CAAC;YACxB,KAAK,CAAC,cAAc,EAAE,CAAC;SACxB;QACD,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,KAAK,CAAC,CAAC;QAChC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;QAEhE,OAAO,KAAK,CAAC;IACf,CAAC;IAED,iBAAiB;QACf,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YACnB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC;SAC/B;QAED,IAAI,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE;YAC1B,MAAM,QAAQ,GAAG,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;YAC/D,MAAM,QAAQ,GAAG,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACtE,MAAM,eAAe,GAAG,UAAU,CAAC,CAAC,QAAQ,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;iBAC9F,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;YACtB,MAAM,YAAY,GAAG,UAAU,CAAC,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;iBACtF,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;YACtB,MAAM,YAAY,GAAG,UAAU,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC;iBACtE,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;YACtB,MAAM,MAAM,GAAG,IAAI,CAAC,gCAAgC,GAAG,YAAY,CAAC;YACpE,IAAI,CAAC,SAAS,GAAG,GAAG,MAAM,GAAG,YAAY,GAAG,eAAe,IAAI,CAAC;SACjE;QAED,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,YAAY,EAAE,SAAS,CAAC,CAAC;IAC9E,CAAC;IAED,cAAc,CAAC,KAAa;QAC1B,IAAI,KAAK,KAAK,CAAC,EAAE;YACf,IAAI,CAAC,cAAc,EAAE,CAAC;YAEtB,OAAO;SACR;QACD,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,SAAS,EAAE;YACrC,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YACvD,IAAI,SAAS,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,aAAa,CAAC,EAAE;gBAClE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,SAAS,GAAG,SAAS,CAAC,aAAa,CAAC,SAAS,CAAC;aAC5E;SACF;IACH,CAAC;IAED,UAAU,CAAC,KAAa;QACtB,IAAI,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;YACvC,IAAI,CAAC,WAAW,EAAE,CAAC;YAEnB,OAAO;SACR;QACD,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,SAAS,EAAE;YACrC,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YACvD,IAAI,SAAS,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,aAAa,CAAC,EAAE;gBAClE,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,SAAS;oBACpC,SAAS,CAAC,aAAa,CAAC,SAAS;wBACjC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,YAAY,CAAC;wBACjD,MAAM,CAAC,SAAS,CAAC,aAAa,CAAC,YAAY,CAAC,CAAC;aAChD;SACF;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,2BAA2B,CAAC,WAAW,EAAE,CAAC;IACjD,CAAC;IAES,SAAS,CAAC,KAAsB;QACxC,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,OAAO,CAAC;QACZ,IAAI,CAAC,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,EAAE;YACtD,OAAO,GAAG,KAAK,CAAC;SACjB;QACD,IAAI,CAAC,MAAM,EAAE,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChD,CAAC;IAEO,kBAAkB,CAAC,IAAiB;QAC1C,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YACnB,OAAO,KAAK,CAAC;SACd;QACD,MAAM,gBAAgB,GAAW,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,SAAS,CAAC;QACxE,MAAM,mBAAmB,GAAG,gBAAgB,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,YAAY,CAAC,CAAC;QACjG,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC;QAC/B,MAAM,UAAU,GAAG,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC;QAE/C,OAAO,CAAC,CAAC,UAAU,IAAI,mBAAmB,CAAC,IAAI,CAAC,OAAO,IAAI,gBAAgB,CAAC,CAAC,CAAC;IAChF,CAAC;IAAA,CAAC;IAEM,cAAc;QACpB,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,aAAa,EAAE;YAClC,OAAO;SACR;QACD,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,YAAY,CAAC;IACrF,CAAC;IAEO,WAAW;QACjB,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,aAAa,EAAE;YAClC,OAAO;SACR;QACD,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,SAAS,GAAG,CAAC,CAAC;IAC7C,CAAC;8GA/VU,2BAA2B;kGAA3B,2BAA2B,onBCtDxC,q7CAmCA,khBDgBc,CAAC,kBAAkB,CAAC;;2FAGrB,2BAA2B;kBA3BvC,SAAS;+BACE,qBAAqB,QAGzB;wBACJ,KAAK,EAAE,oCAAoC;wBAC3C,gBAAgB,EAAE,mCAAmC;wBACrD,oBAAoB,EAAE,WAAW;wBACjC,gBAAgB,EAAE,QAAQ;wBAC1B,KAAK,EAAE,oCAAoC;wBAC3C,aAAa,EAAE,WAAW;qBAC3B,cAaW,CAAC,kBAAkB,CAAC;wKAKR,iBAAiB;sBAAxC,MAAM;uBAAC,cAAc;gBA6Bd,SAAS;sBADhB,SAAS;uBAAC,WAAW,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE;gBAIjC,UAAU;sBADjB,YAAY;uBAAC,YAAY;gBA+M1B,SAAS;sBAFR,YAAY;uBAAC,YAAY;;sBACzB,YAAY;uBAAC,MAAM","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  HostListener,\n  OnDestroy,\n  QueryList,\n  Renderer2,\n  TemplateRef,\n  ViewChild,\n  ViewChildren,\n  Output,\n  EventEmitter\n} from '@angular/core';\n\nimport { Utils } from 'ngx-bootstrap/utils';\nimport { PositioningService } from 'ngx-bootstrap/positioning';\nimport { Subscription } from 'rxjs';\n\nimport { latinize } from './typeahead-utils';\nimport { TypeaheadMatch } from './typeahead-match.class';\nimport { TypeaheadDirective } from './typeahead.directive';\nimport { typeaheadAnimation } from './typeahead-animations';\nimport { TypeaheadOptionItemContext, TypeaheadOptionListContext, TypeaheadTemplateMethods } from './models';\n\nlet nextWindowId = 0;\n\n@Component({\n  selector: 'typeahead-container',\n  templateUrl: './typeahead-container.component.html',\n  // eslint-disable-next-line @angular-eslint/no-host-metadata-property\n  host: {\n    class: 'dropdown open bottom dropdown-menu',\n    '[style.height]': `needScrollbar ? guiHeight: 'auto'`,\n    '[style.visibility]': `'inherit'`,\n    '[class.dropup]': 'dropup',\n    style: 'position: absolute;display: block;',\n    '[attr.role]': `'listbox'`\n  },\n  styles: [\n    `\n    :host.dropdown {\n      z-index: 1000;\n    }\n\n    :host.dropdown-menu, .dropdown-menu {\n      overflow-y: auto;\n      height: 100px;\n    }\n  `\n  ],\n  animations: [typeaheadAnimation]\n})\n\nexport class TypeaheadContainerComponent implements OnDestroy {\n  // eslint-disable-next-line @angular-eslint/no-output-rename\n  @Output('activeChange') activeChangeEvent = new EventEmitter();\n\n  parent?: TypeaheadDirective;\n  query?: string[] | string;\n  isFocused = false;\n  top?: string;\n  left?: string;\n  display?: string;\n  placemen?: string;\n  dropup?: boolean;\n  guiHeight?: string;\n  needScrollbar?: boolean;\n  animationState?: string;\n  positionServiceSubscription = new Subscription();\n  height = 0;\n  popupId = `ngb-typeahead-${nextWindowId++}`;\n\n  get typeaheadTemplateMethods(): TypeaheadTemplateMethods {\n    return {\n      selectMatch: this.selectMatch.bind(this),\n      selectActive: this.selectActive.bind(this),\n      isActive: this.isActive.bind(this)\n    };\n  }\n\n  protected _active?: TypeaheadMatch;\n  protected _matches: TypeaheadMatch[] = [];\n\n  @ViewChild('ulElement', { static: false })\n  private ulElement?: ElementRef;\n\n  @ViewChildren('liElements')\n  private liElements?: QueryList<ElementRef>;\n\n  constructor(\n    private positionService: PositioningService,\n    private renderer: Renderer2,\n    public element: ElementRef,\n    private changeDetectorRef: ChangeDetectorRef\n  ) {\n    this.renderer.setAttribute(this.element.nativeElement, 'id', this.popupId);\n    this.positionServiceSubscription.add(this.positionService.event$?.subscribe(\n      () => {\n        if (this.isAnimated) {\n          this.animationState = this.isTopPosition ? 'animated-up' : 'animated-down';\n          this.changeDetectorRef.detectChanges();\n\n          return;\n        }\n\n        this.animationState = 'unanimated';\n        this.changeDetectorRef.detectChanges();\n      }\n    ));\n  }\n\n  get active(): TypeaheadMatch | undefined {\n    return this._active;\n  }\n\n  set active(active: TypeaheadMatch | undefined) {\n    this._active = active;\n    this.activeChanged();\n  }\n\n  get matches(): TypeaheadMatch[] {\n    return this._matches;\n  }\n\n  set matches(value: TypeaheadMatch[]) {\n    this.positionService.setOptions({\n      modifiers: { flip: { enabled: this.adaptivePosition } },\n      allowedPositions: ['top', 'bottom']\n    });\n\n    this._matches = value;\n\n    this.needScrollbar = this.typeaheadScrollable && this.typeaheadOptionsInScrollableView < this.matches.length;\n\n    if (this.typeaheadScrollable) {\n      setTimeout(() => {\n        this.setScrollableMode();\n      });\n    }\n\n    if (this.typeaheadIsFirstItemActive && this._matches.length > 0) {\n      this.setActive(this._matches[0]);\n\n      if (this._active?.isHeader()) {\n        this.nextActiveMatch();\n      }\n    }\n\n    if (this._active && !this.typeaheadIsFirstItemActive) {\n      const concurrency = this._matches.find(match => match.value === this._active?.value);\n\n      if (concurrency) {\n        this.selectActive(concurrency);\n\n        return;\n      }\n\n      this.active = void 0;\n    }\n  }\n\n  get isTopPosition(): boolean {\n    return this.element.nativeElement.classList.contains('top');\n  }\n\n  get optionsListTemplate(): TemplateRef<TypeaheadOptionListContext> | undefined {\n    return this.parent ? this.parent.optionsListTemplate : undefined;\n  }\n\n  get isAnimated(): boolean {\n    return this.parent ? this.parent.isAnimated : false;\n  }\n\n  get adaptivePosition(): boolean {\n    return this.parent ? this.parent.adaptivePosition : false;\n  }\n\n  get typeaheadScrollable(): boolean {\n    return this.parent ? this.parent.typeaheadScrollable : false;\n  }\n\n  get typeaheadOptionsInScrollableView(): number {\n    return this.parent ? this.parent.typeaheadOptionsInScrollableView : 5;\n  }\n\n  get typeaheadIsFirstItemActive(): boolean {\n    return this.parent ? this.parent.typeaheadIsFirstItemActive : true;\n  }\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  get itemTemplate(): TemplateRef<TypeaheadOptionItemContext> | undefined {\n    return this.parent ? this.parent.typeaheadItemTemplate : undefined;\n  }\n\n  get canSelectItemsOnBlur(): boolean {\n    return !!this.parent?.selectItemOnBlur;\n  }\n\n  selectActiveMatch(isActiveItemChanged?: boolean): void {\n    if (this._active && this.parent?.typeaheadSelectFirstItem) {\n      this.selectMatch(this._active);\n    }\n\n    if (!this.parent?.typeaheadSelectFirstItem && isActiveItemChanged) {\n      this.selectMatch(this._active);\n    }\n  }\n\n  activeChanged(): void {\n    if (!this._active) {\n      return;\n    }\n    const index = this.matches.indexOf(this._active);\n    this.activeChangeEvent.emit(`${this.popupId}-${index}`);\n  }\n\n  prevActiveMatch(): void {\n    if (!this._active) {\n      return;\n    }\n\n    const index = this.matches.indexOf(this._active);\n    this.setActive(this.matches[\n      index - 1 < 0 ? this.matches.length - 1 : index - 1\n      ]);\n\n    if (this._active.isHeader()) {\n      this.prevActiveMatch();\n    }\n\n    if (this.typeaheadScrollable) {\n      this.scrollPrevious(index);\n    }\n  }\n\n  nextActiveMatch(): void {\n    const index = this._active ? this.matches.indexOf(this._active) : -1;\n    this.setActive(this.matches[\n      index + 1 > this.matches.length - 1 ? 0 : index + 1\n      ]);\n\n    if (this._active?.isHeader()) {\n      this.nextActiveMatch();\n    }\n\n    if (this.typeaheadScrollable) {\n      this.scrollNext(index);\n    }\n  }\n\n  selectActive(value: TypeaheadMatch): void {\n    this.isFocused = true;\n    this.setActive(value);\n  }\n\n  highlight(match: TypeaheadMatch, query: string[] | string): string {\n    let itemStr: string = match.value;\n    let itemStrHelper: string = (this.parent && this.parent.typeaheadLatinize\n      ? latinize(itemStr)\n      : itemStr).toLowerCase();\n    let startIdx: number;\n    let tokenLen: number;\n    // Replaces the capture string with the same string inside of a \"strong\" tag\n    if (typeof query === 'object') {\n      const queryLen: number = query.length;\n      for (let i = 0; i < queryLen; i += 1) {\n        // query[i] is already latinized and lower case\n        startIdx = itemStrHelper.indexOf(query[i]);\n        tokenLen = query[i].length;\n        if (startIdx >= 0 && tokenLen > 0) {\n          itemStr =\n            `${itemStr.substring(0, startIdx)}<strong>${itemStr.substring(startIdx, startIdx + tokenLen)}</strong>` +\n            `${itemStr.substring(startIdx + tokenLen)}`;\n          itemStrHelper =\n            `${itemStrHelper.substring(0, startIdx)}        ${' '.repeat(tokenLen)}         ` +\n            `${itemStrHelper.substring(startIdx + tokenLen)}`;\n        }\n      }\n    } else if (query) {\n      // query is already latinized and lower case\n      startIdx = itemStrHelper.indexOf(query);\n      tokenLen = query.length;\n      if (startIdx >= 0 && tokenLen > 0) {\n        itemStr =\n          `${itemStr.substring(0, startIdx)}<strong>${itemStr.substring(startIdx, startIdx + tokenLen)}</strong>` +\n          `${itemStr.substring(startIdx + tokenLen)}`;\n      }\n    }\n\n    return itemStr;\n  }\n\n  @HostListener('mouseleave')\n  @HostListener('blur')\n  focusLost(): void {\n    this.isFocused = false;\n    if (!this.canSelectItemsOnBlur) {\n      this.setActive(void 0);\n    }\n  }\n\n  isActive(value: TypeaheadMatch): boolean {\n    return this.active === value;\n  }\n\n  selectMatch(value?: TypeaheadMatch, event?: Event): boolean {\n    if (event) {\n      event.stopPropagation();\n      event.preventDefault();\n    }\n    this.parent?.changeModel(value);\n    setTimeout(() => this.parent?.typeaheadOnSelect.emit(value), 0);\n\n    return false;\n  }\n\n  setScrollableMode(): void {\n    if (!this.ulElement) {\n      this.ulElement = this.element;\n    }\n\n    if (this.liElements?.first) {\n      const ulStyles = Utils.getStyles(this.ulElement.nativeElement);\n      const liStyles = Utils.getStyles(this.liElements.first.nativeElement);\n      const ulPaddingBottom = parseFloat((ulStyles['padding-bottom'] ? ulStyles['padding-bottom'] : '')\n        .replace('px', ''));\n      const ulPaddingTop = parseFloat((ulStyles['padding-top'] ? ulStyles['padding-top'] : '0')\n        .replace('px', ''));\n      const optionHeight = parseFloat((liStyles.height ? liStyles.height : '0')\n        .replace('px', ''));\n      const height = this.typeaheadOptionsInScrollableView * optionHeight;\n      this.guiHeight = `${height + ulPaddingTop + ulPaddingBottom}px`;\n    }\n\n    this.renderer.setStyle(this.element.nativeElement, 'visibility', 'visible');\n  }\n\n  scrollPrevious(index: number): void {\n    if (index === 0) {\n      this.scrollToBottom();\n\n      return;\n    }\n    if (this.liElements && this.ulElement) {\n      const liElement = this.liElements.toArray()[index - 1];\n      if (liElement && !this.isScrolledIntoView(liElement.nativeElement)) {\n        this.ulElement.nativeElement.scrollTop = liElement.nativeElement.offsetTop;\n      }\n    }\n  }\n\n  scrollNext(index: number): void {\n    if (index + 1 > this.matches.length - 1) {\n      this.scrollToTop();\n\n      return;\n    }\n    if (this.liElements && this.ulElement) {\n      const liElement = this.liElements.toArray()[index + 1];\n      if (liElement && !this.isScrolledIntoView(liElement.nativeElement)) {\n        this.ulElement.nativeElement.scrollTop =\n          liElement.nativeElement.offsetTop -\n          Number(this.ulElement.nativeElement.offsetHeight) +\n          Number(liElement.nativeElement.offsetHeight);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.positionServiceSubscription.unsubscribe();\n  }\n\n  protected setActive(value?: TypeaheadMatch): void {\n    this._active = value;\n    let preview;\n    if (!(this._active == null || this._active.isHeader())) {\n      preview = value;\n    }\n    this.parent?.typeaheadOnPreview.emit(preview);\n  }\n\n  private isScrolledIntoView(elem: HTMLElement): boolean {\n    if (!this.ulElement) {\n      return false;\n    }\n    const containerViewTop: number = this.ulElement.nativeElement.scrollTop;\n    const containerViewBottom = containerViewTop + Number(this.ulElement.nativeElement.offsetHeight);\n    const elemTop = elem.offsetTop;\n    const elemBottom = elemTop + elem.offsetHeight;\n\n    return ((elemBottom <= containerViewBottom) && (elemTop >= containerViewTop));\n  };\n\n  private scrollToBottom(): void {\n    if (!this.ulElement?.nativeElement) {\n      return;\n    }\n    this.ulElement.nativeElement.scrollTop = this.ulElement.nativeElement.scrollHeight;\n  }\n\n  private scrollToTop(): void {\n    if (!this.ulElement?.nativeElement) {\n      return;\n    }\n    this.ulElement.nativeElement.scrollTop = 0;\n  }\n}\n","<!-- inject options list template -->\n<ng-template [ngTemplateOutlet]=\"optionsListTemplate || bs4Template\"\n             [ngTemplateOutletContext]=\"{\n               matches: matches,\n               itemTemplate: itemTemplate || bsItemTemplate,\n               query: query,\n               $implicit: typeaheadTemplateMethods\n             }\">\n</ng-template>\n\n<!-- default options item template -->\n<ng-template #bsItemTemplate let-match=\"match\" let-query=\"query\">\n  <span [innerHtml]=\"highlight(match, query)\"></span>\n</ng-template>\n\n<!-- Bootstrap 4 options list template -->\n<ng-template #bs4Template>\n  <ng-template ngFor let-match let-i=\"index\" [ngForOf]=\"matches\">\n    <h6 *ngIf=\"match.isHeader()\" class=\"dropdown-header\">{{ match }}</h6>\n    <ng-template [ngIf]=\"!match.isHeader()\">\n      <button #liElements\n              [id]=\"popupId + '-' + i\"\n              role=\"option\"\n              [@typeaheadAnimation]=\"animationState\"\n              class=\"dropdown-item\"\n              (click)=\"selectMatch(match, $event)\"\n              (mouseenter)=\"selectActive(match)\"\n              [class.active]=\"isActive(match)\">\n        <ng-template [ngTemplateOutlet]=\"itemTemplate || bsItemTemplate\"\n                     [ngTemplateOutletContext]=\"{item: match.item, index: i, match: match, query: query}\">\n        </ng-template>\n      </button>\n    </ng-template>\n  </ng-template>\n</ng-template>\n"]}