@leanix/components 0.4.97 → 0.4.99
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/forms-ui/components/multi-select/multi-select.component.mjs +13 -4
- package/esm2022/lib/tab-ui/components/tab/tab.component.mjs +9 -44
- package/esm2022/lib/tab-ui/components/tab-group/tab-group.component.mjs +15 -80
- package/fesm2022/leanix-components.mjs +38 -129
- package/fesm2022/leanix-components.mjs.map +1 -1
- package/lib/forms-ui/components/multi-select/multi-select.component.d.ts +6 -1
- package/lib/tab-ui/components/tab/tab.component.d.ts +5 -16
- package/lib/tab-ui/components/tab-group/tab-group.component.d.ts +5 -14
- package/package.json +1 -1
@@ -1,8 +1,8 @@
|
|
1
1
|
import { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';
|
2
|
-
import { Component, ContentChild, EventEmitter,
|
2
|
+
import { Component, ContentChild, EventEmitter, Input, Output, TemplateRef, ViewChild, forwardRef } from '@angular/core';
|
3
3
|
import { FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';
|
4
4
|
import { filter as _filter, findIndex, includes, last } from 'lodash/fp';
|
5
|
-
import { fromEvent as observableFromEvent
|
5
|
+
import { Subject, fromEvent as observableFromEvent } from 'rxjs';
|
6
6
|
import { distinctUntilChanged, filter, map, takeUntil, tap } from 'rxjs/operators';
|
7
7
|
import { MarkInvalidDirective } from '../../directives/mark-invalid.directive';
|
8
8
|
import { SelectDropdownDirective } from '../../directives/select-dropdown.directive';
|
@@ -171,12 +171,21 @@ export class MultiSelectComponent extends BaseSelectDirective {
|
|
171
171
|
.pipe(filter((event) => includes(event.keyCode, [ENTER, ARROW_DOWN])), takeUntil(this.destroyed$))
|
172
172
|
.subscribe(() => this.open.next(true));
|
173
173
|
}
|
174
|
-
|
174
|
+
/**
|
175
|
+
* Adds the given option to the selection and, by default, resets the input field.
|
176
|
+
* @param option The option to add to the selection.
|
177
|
+
* @param keepSearchQueryAndScrollPosition If true, the search query and scroll position of the dropdown will be kept, unless there are no options left.
|
178
|
+
*/
|
179
|
+
addItemToSelection(option, keepSearchQueryAndScrollPosition = false) {
|
175
180
|
if (option !== null) {
|
176
181
|
this.selection = [...(this.selection || [])].concat(option);
|
177
182
|
this.selectionChange.emit([...this.selection]);
|
178
183
|
this.propagateChange([...this.selection]);
|
179
184
|
}
|
185
|
+
const filteredOptionsInDropdownBeforeSelection = this.dropdownBase?._items.length ?? 0;
|
186
|
+
if (keepSearchQueryAndScrollPosition && filteredOptionsInDropdownBeforeSelection > 1) {
|
187
|
+
return;
|
188
|
+
}
|
180
189
|
this.resetSelectState();
|
181
190
|
}
|
182
191
|
removeItem(optionToRemove) {
|
@@ -251,4 +260,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImpor
|
|
251
260
|
type: ViewChild,
|
252
261
|
args: ['dropdown', { static: true }]
|
253
262
|
}] } });
|
254
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/multi-select/multi-select.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/multi-select/multi-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  EventEmitter,\n  forwardRef,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { filter as _filter, findIndex, includes, last } from 'lodash/fp';\nimport { fromEvent as observableFromEvent, Subject } from 'rxjs';\nimport { distinctUntilChanged, filter, map, takeUntil, tap } from 'rxjs/operators';\nimport { MarkInvalidDirective } from '../../directives/mark-invalid.directive';\nimport { SelectDropdownDirective } from '../../directives/select-dropdown.directive';\nimport { ARROW_DOWN, ARROW_LEFT, ARROW_RIGHT, ARROW_UP, BACKSPACE, ENTER, ESCAPE, TAB } from '../../helpers/key-codes.constants';\nimport { BaseSelectDirective, EventSet } from '../../models/base-select.directive';\nimport { KeyboardSelectAction } from '../keyboard-select.directive';\nimport { ResponsiveInputComponent } from '../responsive-input/responsive-input.component';\n\nexport type MultiSelectSize = 'default' | 'small' | 'large' | 'select2';\n\nexport type MultiSelectRemoveItemEvent = {\n  item: any;\n  /**\n   * Is true when the item was not removed by pressing backspace on the keyboard rather than clicking the 'x'.\n   */\n  isMouse: boolean;\n};\n\n@Component({\n  selector: 'lx-multi-select',\n  templateUrl: 'multi-select.component.html',\n  styleUrls: ['multi-select.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectComponent),\n      multi: true\n    }\n  ],\n  standalone: true,\n  imports: [NgIf, MarkInvalidDirective, ResponsiveInputComponent, FormsModule, ReactiveFormsModule, NgTemplateOutlet, AsyncPipe]\n})\nexport class MultiSelectComponent extends BaseSelectDirective implements AfterViewInit, ControlValueAccessor {\n  public static isStillPossibleMoveToLeft(eventSet: EventSet): boolean {\n    return (\n      !(\n        eventSet.virtualCursorPosition === 0 &&\n        eventSet.inputCursorPosition === 0 &&\n        eventSet.event.keyCode === ARROW_LEFT &&\n        eventSet.selectionLength === 0\n      ) &&\n      !(\n        eventSet.event.keyCode === ARROW_LEFT &&\n        typeof eventSet.selectionLength === 'number' &&\n        eventSet.selectionLength > 0 &&\n        Math.abs(eventSet.virtualCursorPosition) === eventSet.selectionLength\n      )\n    );\n  }\n\n  public static isMovingfromZeroToRight(eventSet: EventSet) {\n    return eventSet.virtualCursorPosition === 0 && eventSet.inputCursorPosition === 0 && eventSet.event.keyCode === ARROW_RIGHT;\n  }\n\n  public static calculateNewCursorPostionOnKeyboardNavigation(cursorPosition: number, keyCode: number) {\n    switch (keyCode) {\n      case ARROW_LEFT:\n        cursorPosition--;\n        break;\n      case ARROW_RIGHT:\n        cursorPosition++;\n        break;\n    }\n    return cursorPosition;\n  }\n\n  public static getKeyboardSelectAction(cursorPosition: number, keyCode: number): KeyboardSelectAction | null {\n    if (cursorPosition <= -1 && keyCode === ARROW_RIGHT) {\n      return KeyboardSelectAction.NEXT;\n    }\n    if (cursorPosition === -1 && keyCode === ARROW_LEFT) {\n      return KeyboardSelectAction.LAST;\n    }\n    if (cursorPosition < -1 && keyCode === ARROW_LEFT) {\n      return KeyboardSelectAction.PREV;\n    }\n    if (cursorPosition <= -1 && keyCode === BACKSPACE) {\n      return KeyboardSelectAction.EXECUTE;\n    }\n    return null;\n  }\n\n  @Input() markInvalid: boolean = false;\n  @Input() selection?: any[] | null = [];\n  @Input() size: MultiSelectSize = 'default';\n  @Input() dropdownWidthScale: '1x' | '1.5x' | '2x' = '1x';\n\n  @Output() selectionChange = new EventEmitter<any[]>();\n  @Output() blur = new EventEmitter<void>();\n\n  @ContentChild(SelectDropdownDirective, { read: TemplateRef, static: true }) explicitDropdown?: TemplateRef<any>;\n\n  @ViewChild(ResponsiveInputComponent, { static: true }) queryInput!: ResponsiveInputComponent;\n  @ViewChild('dropdown', { static: true }) implicitDropdown!: TemplateRef<any>;\n\n  get dropdownTmpl() {\n    return this.explicitDropdown || this.implicitDropdown;\n  }\n\n  override readonly destroyed$ = new Subject<void>();\n  isInputFocusedViaTab = false;\n\n  constructor(private cd: ChangeDetectorRef) {\n    super();\n  }\n\n  get dropdownWidth(): string | undefined {\n    switch (this.dropdownWidthScale) {\n      case '1x':\n        return undefined;\n      case '1.5x':\n        return '150%';\n      case '2x':\n        return '200%';\n    }\n  }\n\n  resetInput() {\n    this.queryInput.resetInput();\n    this.queryControl.setValue('');\n  }\n\n  focus() {\n    this.queryInput.focusInput();\n  }\n\n  focusAndOpen() {\n    this.queryInput.focusInput();\n    this.open.next(true);\n  }\n\n  override ngAfterViewInit() {\n    super.ngAfterViewInit();\n\n    this.open\n      .pipe(\n        distinctUntilChanged(),\n        filter((open) => open),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => {\n        this.selectFirstOption();\n      });\n\n    // keyboard event handling\n    const sourceElement = this.queryInput.responsiveInput.nativeElement;\n\n    observableFromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        filter((event) => includes(event.keyCode, [ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => this.open.next(true));\n\n    // handle keyboard navigation for the selection\n    observableFromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        map((event) =>\n          this.mapToEventSet(\n            event,\n            this.queryInput.responsiveInput.nativeElement,\n            this.virtualCursorPosition,\n            this.selection ? this.selection.length : 0\n          )\n        ),\n        filter((eventSet) => {\n          return !MultiSelectComponent.isMovingfromZeroToRight(eventSet) && MultiSelectComponent.isStillPossibleMoveToLeft(eventSet);\n        }),\n        tap((eventSet) => {\n          if (\n            eventSet.virtualCursorPosition === 0 &&\n            eventSet.inputCursorPosition === 0 &&\n            eventSet.event.keyCode === BACKSPACE &&\n            typeof eventSet.selectionLength === 'number' &&\n            eventSet.selectionLength > 0\n          ) {\n            const lastItem = last(this.selection);\n            this.removeItem({ item: lastItem, isMouse: false });\n            eventSet.inputCursorPosition = undefined; // set to null to prevent continuing in the stream\n          }\n          if (typeof eventSet.inputCursorPosition === 'number' && eventSet.inputCursorPosition > 0) {\n            this.virtualCursorPosition = 0;\n          } else if (eventSet.virtualCursorPosition === -1 && eventSet.event.keyCode === ARROW_RIGHT) {\n            eventSet.event.preventDefault();\n            this.selectionKeyboardSelectAction$.next(KeyboardSelectAction.UNSELECT);\n            this.virtualCursorPosition = 0;\n            eventSet.inputCursorPosition = undefined; // set to null to prevent continuing in the stream\n          }\n        }),\n        filter((eventSet) => eventSet.inputCursorPosition === 0),\n        map((eventSet) => eventSet.event),\n        filter((event) => includes(event.keyCode, [ARROW_RIGHT, ARROW_LEFT, BACKSPACE])),\n        map((event) => {\n          this.virtualCursorPosition = MultiSelectComponent.calculateNewCursorPostionOnKeyboardNavigation(\n            this.virtualCursorPosition,\n            event.keyCode\n          );\n          return {\n            cursorPosition: this.virtualCursorPosition,\n            event\n          };\n        }),\n        map((positionSet) => {\n          const action = MultiSelectComponent.getKeyboardSelectAction(positionSet.cursorPosition, positionSet.event.keyCode);\n          if (action === KeyboardSelectAction.NEXT) {\n            // prevent cursor from moving in the input field, while we are still in the selection items\n            positionSet.event.preventDefault();\n          }\n          return action;\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(this.selectionKeyboardSelectAction$);\n\n    const inputKeyboardEvents$ = observableFromEvent<KeyboardEvent>(sourceElement, 'keydown');\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ARROW_UP, ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((event) => {\n        event.preventDefault();\n      });\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => event.keyCode === ESCAPE || event.keyCode === TAB),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((event) => {\n        if (event.keyCode === ESCAPE && this.dropdownOpen) {\n          event.preventDefault();\n          event.stopImmediatePropagation();\n        }\n        this.open.next(false);\n      });\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ENTER, ARROW_UP, ARROW_DOWN])),\n        map((event) => {\n          switch (event.keyCode) {\n            case ENTER:\n              return KeyboardSelectAction.EXECUTE;\n            case ARROW_UP:\n              return KeyboardSelectAction.PREV;\n            case ARROW_DOWN:\n              return KeyboardSelectAction.NEXT;\n            default:\n              return null;\n          }\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(this.optionsKeyboardSelectAction$);\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ENTER, ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => this.open.next(true));\n  }\n\n  addItemToSelection(option: any) {\n    if (option !== null) {\n      this.selection = [...(this.selection || [])].concat(option);\n      this.selectionChange.emit([...this.selection]);\n      this.propagateChange([...this.selection]);\n    }\n    this.resetSelectState();\n  }\n\n  removeItem(optionToRemove: MultiSelectRemoveItemEvent) {\n    if (this.disabled) {\n      return;\n    }\n    const prevOptionIndex = findIndex((option) => option === optionToRemove.item, this.selection);\n    const prevSelectionLenght = this.selection?.length ?? 0;\n    this.selection = _filter((option) => option !== optionToRemove.item, this.selection);\n    this.cd.markForCheck();\n    this.cd.detectChanges();\n    this.selectionChange.emit([...this.selection]);\n    this.propagateChange([...this.selection]);\n\n    if (prevOptionIndex === prevSelectionLenght - 1 || optionToRemove.isMouse) {\n      this.resetSelectState();\n    } else {\n      this.virtualCursorPosition++;\n    }\n  }\n\n  propagateChange(_value: any[]) {}\n\n  writeValue(value: any[]) {\n    this.selection = value;\n  }\n\n  registerOnChange(fn: any) {\n    this.propagateChange = fn;\n  }\n\n  registerOnTouched(_fn: any) {}\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  focusedViaTab() {\n    this.isInputFocusedViaTab = true;\n  }\n}\n","<div *ngIf=\"dropdownOpen\" class=\"backdrop\" (click)=\"handleBackdropClick($event)\"></div>\n<div\n  [lxMarkInvalid]=\"markInvalid\"\n  class=\"selectContainer\"\n  [class.open]=\"dropdownOpen\"\n  [class.top]=\"(dropdownDirection$ | async) === 'TOP'\"\n  [class.bottom]=\"(dropdownDirection$ | async) === 'BOTTOM'\"\n  [class.focused]=\"isInputFocused\"\n  [class.focusedVisible]=\"isInputFocusedViaTab\"\n  [class.hasSelection]=\"selection && selection.length > 0\"\n  [class.smallSize]=\"size === 'small'\"\n  [class.disabled]=\"disabled\"\n  (click)=\"handleClick($event.target === toggle)\"\n>\n  <div class=\"selectedChoicesContainer\">\n    <div *ngIf=\"(selection?.length === 0 || !selection) && !queryControl?.value\" class=\"placeholder\" [attr.title]=\"placeholder\">\n      {{ placeholder }}\n    </div>\n    <div class=\"selection\">\n      <ng-content select=\".pills\"></ng-content>\n    </div>\n    <lx-responsive-input\n      (focus)=\"isInputFocused = true\"\n      (blur)=\"isInputFocused = false; isInputFocusedViaTab = false; blur.emit()\"\n      (focusViaTab)=\"focusedViaTab()\"\n      [formControl]=\"queryControl\"\n      (keydown.enter)=\"$event.preventDefault()\"\n    ></lx-responsive-input>\n    <i #toggle [hidden]=\"disabled\" class=\"fas fa-angle-down\" aria-hidden=\"true\"></i>\n  </div>\n  <div class=\"optionsContainer\" [style.width]=\"dropdownWidth\" #optionsContainer>\n    <ng-container *ngIf=\"dropdownOpen\">\n      <ng-container *ngTemplateOutlet=\"dropdownTmpl\"></ng-container>\n    </ng-container>\n  </div>\n</div>\n\n<ng-template #dropdown>\n  <ng-content select=\".dropdownComponent\"></ng-content>\n</ng-template>\n"]}
|
263
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/multi-select/multi-select.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/multi-select/multi-select.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AACpE,OAAO,EAGL,SAAS,EACT,YAAY,EACZ,YAAY,EACZ,KAAK,EACL,MAAM,EACN,WAAW,EACX,SAAS,EACT,UAAU,EACX,MAAM,eAAe,CAAC;AACvB,OAAO,EAAwB,WAAW,EAAE,iBAAiB,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC3G,OAAO,EAAE,MAAM,IAAI,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACzE,OAAO,EAAE,OAAO,EAAE,SAAS,IAAI,mBAAmB,EAAE,MAAM,MAAM,CAAC;AACjE,OAAO,EAAE,oBAAoB,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACnF,OAAO,EAAE,oBAAoB,EAAE,MAAM,yCAAyC,CAAC;AAC/E,OAAO,EAAE,uBAAuB,EAAE,MAAM,4CAA4C,CAAC;AACrF,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,WAAW,EAAE,QAAQ,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,mCAAmC,CAAC;AACjI,OAAO,EAAE,mBAAmB,EAAY,MAAM,oCAAoC,CAAC;AACnF,OAAO,EAAE,oBAAoB,EAAE,MAAM,8BAA8B,CAAC;AACpE,OAAO,EAAE,wBAAwB,EAAE,MAAM,gDAAgD,CAAC;;;AA0B1F,MAAM,OAAO,oBAAqB,SAAQ,mBAAmB;IACpD,MAAM,CAAC,yBAAyB,CAAC,QAAkB;QACxD,OAAO,CACL,CAAC,CACC,QAAQ,CAAC,qBAAqB,KAAK,CAAC;YACpC,QAAQ,CAAC,mBAAmB,KAAK,CAAC;YAClC,QAAQ,CAAC,KAAK,CAAC,OAAO,KAAK,UAAU;YACrC,QAAQ,CAAC,eAAe,KAAK,CAAC,CAC/B;YACD,CAAC,CACC,QAAQ,CAAC,KAAK,CAAC,OAAO,KAAK,UAAU;gBACrC,OAAO,QAAQ,CAAC,eAAe,KAAK,QAAQ;gBAC5C,QAAQ,CAAC,eAAe,GAAG,CAAC;gBAC5B,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,qBAAqB,CAAC,KAAK,QAAQ,CAAC,eAAe,CACtE,CACF,CAAC;IACJ,CAAC;IAEM,MAAM,CAAC,uBAAuB,CAAC,QAAkB;QACtD,OAAO,QAAQ,CAAC,qBAAqB,KAAK,CAAC,IAAI,QAAQ,CAAC,mBAAmB,KAAK,CAAC,IAAI,QAAQ,CAAC,KAAK,CAAC,OAAO,KAAK,WAAW,CAAC;IAC9H,CAAC;IAEM,MAAM,CAAC,6CAA6C,CAAC,cAAsB,EAAE,OAAe;QACjG,QAAQ,OAAO,EAAE;YACf,KAAK,UAAU;gBACb,cAAc,EAAE,CAAC;gBACjB,MAAM;YACR,KAAK,WAAW;gBACd,cAAc,EAAE,CAAC;gBACjB,MAAM;SACT;QACD,OAAO,cAAc,CAAC;IACxB,CAAC;IAEM,MAAM,CAAC,uBAAuB,CAAC,cAAsB,EAAE,OAAe;QAC3E,IAAI,cAAc,IAAI,CAAC,CAAC,IAAI,OAAO,KAAK,WAAW,EAAE;YACnD,OAAO,oBAAoB,CAAC,IAAI,CAAC;SAClC;QACD,IAAI,cAAc,KAAK,CAAC,CAAC,IAAI,OAAO,KAAK,UAAU,EAAE;YACnD,OAAO,oBAAoB,CAAC,IAAI,CAAC;SAClC;QACD,IAAI,cAAc,GAAG,CAAC,CAAC,IAAI,OAAO,KAAK,UAAU,EAAE;YACjD,OAAO,oBAAoB,CAAC,IAAI,CAAC;SAClC;QACD,IAAI,cAAc,IAAI,CAAC,CAAC,IAAI,OAAO,KAAK,SAAS,EAAE;YACjD,OAAO,oBAAoB,CAAC,OAAO,CAAC;SACrC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAeD,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,gBAAgB,CAAC;IACxD,CAAC;IAKD,YAAoB,EAAqB;QACvC,KAAK,EAAE,CAAC;QADU,OAAE,GAAF,EAAE,CAAmB;QApBhC,gBAAW,GAAY,KAAK,CAAC;QAC7B,cAAS,GAAkB,EAAE,CAAC;QAC9B,SAAI,GAAoB,SAAS,CAAC;QAClC,uBAAkB,GAAyB,IAAI,CAAC;QAE/C,oBAAe,GAAG,IAAI,YAAY,EAAS,CAAC;QAC5C,SAAI,GAAG,IAAI,YAAY,EAAQ,CAAC;QAWxB,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;QACnD,yBAAoB,GAAG,KAAK,CAAC;IAI7B,CAAC;IAED,IAAI,aAAa;QACf,QAAQ,IAAI,CAAC,kBAAkB,EAAE;YAC/B,KAAK,IAAI;gBACP,OAAO,SAAS,CAAC;YACnB,KAAK,MAAM;gBACT,OAAO,MAAM,CAAC;YAChB,KAAK,IAAI;gBACP,OAAO,MAAM,CAAC;SACjB;IACH,CAAC;IAED,UAAU;QACR,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;QAC7B,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;IACjC,CAAC;IAED,KAAK;QACH,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;IAC/B,CAAC;IAED,YAAY;QACV,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;QAC7B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACvB,CAAC;IAEQ,eAAe;QACtB,KAAK,CAAC,eAAe,EAAE,CAAC;QAExB,IAAI,CAAC,IAAI;aACN,IAAI,CACH,oBAAoB,EAAE,EACtB,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EACtB,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;QAEL,0BAA0B;QAC1B,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC;QAEpE,mBAAmB,CAAgB,aAAa,EAAE,SAAS,CAAC;aACzD,IAAI,CACH,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC,EACxD,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;QAEzC,+CAA+C;QAC/C,mBAAmB,CAAgB,aAAa,EAAE,SAAS,CAAC;aACzD,IAAI,CACH,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CACZ,IAAI,CAAC,aAAa,CAChB,KAAK,EACL,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,aAAa,EAC7C,IAAI,CAAC,qBAAqB,EAC1B,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAC3C,CACF,EACD,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE;YAClB,OAAO,CAAC,oBAAoB,CAAC,uBAAuB,CAAC,QAAQ,CAAC,IAAI,oBAAoB,CAAC,yBAAyB,CAAC,QAAQ,CAAC,CAAC;QAC7H,CAAC,CAAC,EACF,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE;YACf,IACE,QAAQ,CAAC,qBAAqB,KAAK,CAAC;gBACpC,QAAQ,CAAC,mBAAmB,KAAK,CAAC;gBAClC,QAAQ,CAAC,KAAK,CAAC,OAAO,KAAK,SAAS;gBACpC,OAAO,QAAQ,CAAC,eAAe,KAAK,QAAQ;gBAC5C,QAAQ,CAAC,eAAe,GAAG,CAAC,EAC5B;gBACA,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACtC,IAAI,CAAC,UAAU,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;gBACpD,QAAQ,CAAC,mBAAmB,GAAG,SAAS,CAAC,CAAC,kDAAkD;aAC7F;YACD,IAAI,OAAO,QAAQ,CAAC,mBAAmB,KAAK,QAAQ,IAAI,QAAQ,CAAC,mBAAmB,GAAG,CAAC,EAAE;gBACxF,IAAI,CAAC,qBAAqB,GAAG,CAAC,CAAC;aAChC;iBAAM,IAAI,QAAQ,CAAC,qBAAqB,KAAK,CAAC,CAAC,IAAI,QAAQ,CAAC,KAAK,CAAC,OAAO,KAAK,WAAW,EAAE;gBAC1F,QAAQ,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC;gBAChC,IAAI,CAAC,8BAA8B,CAAC,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;gBACxE,IAAI,CAAC,qBAAqB,GAAG,CAAC,CAAC;gBAC/B,QAAQ,CAAC,mBAAmB,GAAG,SAAS,CAAC,CAAC,kDAAkD;aAC7F;QACH,CAAC,CAAC,EACF,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,mBAAmB,KAAK,CAAC,CAAC,EACxD,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,EACjC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,WAAW,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC,CAAC,EAChF,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;YACZ,IAAI,CAAC,qBAAqB,GAAG,oBAAoB,CAAC,6CAA6C,CAC7F,IAAI,CAAC,qBAAqB,EAC1B,KAAK,CAAC,OAAO,CACd,CAAC;YACF,OAAO;gBACL,cAAc,EAAE,IAAI,CAAC,qBAAqB;gBAC1C,KAAK;aACN,CAAC;QACJ,CAAC,CAAC,EACF,GAAG,CAAC,CAAC,WAAW,EAAE,EAAE;YAClB,MAAM,MAAM,GAAG,oBAAoB,CAAC,uBAAuB,CAAC,WAAW,CAAC,cAAc,EAAE,WAAW,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YACnH,IAAI,MAAM,KAAK,oBAAoB,CAAC,IAAI,EAAE;gBACxC,2FAA2F;gBAC3F,WAAW,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC;aACpC;YACD,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;QAElD,MAAM,oBAAoB,GAAG,mBAAmB,CAAgB,aAAa,EAAE,SAAS,CAAC,CAAC;QAE1F,oBAAoB;aACjB,IAAI,CACH,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,EAClE,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,KAAK,CAAC,cAAc,EAAE,CAAC;QACzB,CAAC,CAAC,CAAC;QAEL,oBAAoB;aACjB,IAAI,CACH,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,OAAO,KAAK,MAAM,IAAI,KAAK,CAAC,OAAO,KAAK,GAAG,CAAC,EACpE,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,IAAI,KAAK,CAAC,OAAO,KAAK,MAAM,IAAI,IAAI,CAAC,YAAY,EAAE;gBACjD,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,KAAK,CAAC,wBAAwB,EAAE,CAAC;aAClC;YACD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxB,CAAC,CAAC,CAAC;QAEL,oBAAoB;aACjB,IAAI,CACH,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,EACzE,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;YACZ,QAAQ,KAAK,CAAC,OAAO,EAAE;gBACrB,KAAK,KAAK;oBACR,OAAO,oBAAoB,CAAC,OAAO,CAAC;gBACtC,KAAK,QAAQ;oBACX,OAAO,oBAAoB,CAAC,IAAI,CAAC;gBACnC,KAAK,UAAU;oBACb,OAAO,oBAAoB,CAAC,IAAI,CAAC;gBACnC;oBACE,OAAO,IAAI,CAAC;aACf;QACH,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC;QAEhD,oBAAoB;aACjB,IAAI,CACH,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,EAC/D,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IAC3C,CAAC;IAED;;;;OAIG;IACH,kBAAkB,CAAC,MAAW,EAAE,gCAAgC,GAAG,KAAK;QACtE,IAAI,MAAM,KAAK,IAAI,EAAE;YACnB,IAAI,CAAC,SAAS,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAC5D,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;YAC/C,IAAI,CAAC,eAAe,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;SAC3C;QAED,MAAM,wCAAwC,GAAG,IAAI,CAAC,YAAY,EAAE,MAAM,CAAC,MAAM,IAAI,CAAC,CAAC;QACvF,IAAI,gCAAgC,IAAI,wCAAwC,GAAG,CAAC,EAAE;YACpF,OAAO;SACR;QAED,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,UAAU,CAAC,cAA0C;QACnD,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,OAAO;SACR;QACD,MAAM,eAAe,GAAG,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,KAAK,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;QAC9F,MAAM,mBAAmB,GAAG,IAAI,CAAC,SAAS,EAAE,MAAM,IAAI,CAAC,CAAC;QACxD,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,KAAK,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;QACrF,IAAI,CAAC,EAAE,CAAC,YAAY,EAAE,CAAC;QACvB,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,CAAC;QACxB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;QAC/C,IAAI,CAAC,eAAe,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;QAE1C,IAAI,eAAe,KAAK,mBAAmB,GAAG,CAAC,IAAI,cAAc,CAAC,OAAO,EAAE;YACzE,IAAI,CAAC,gBAAgB,EAAE,CAAC;SACzB;aAAM;YACL,IAAI,CAAC,qBAAqB,EAAE,CAAC;SAC9B;IACH,CAAC;IAED,eAAe,CAAC,MAAa,IAAG,CAAC;IAEjC,UAAU,CAAC,KAAY;QACrB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;IACzB,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;IAC5B,CAAC;IAED,iBAAiB,CAAC,GAAQ,IAAG,CAAC;IAE9B,gBAAgB,CAAC,UAAmB;QAClC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;IAC7B,CAAC;IAED,aAAa;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;IACnC,CAAC;8GAlSU,oBAAoB;kGAApB,oBAAoB,qPAVpB;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,oBAAoB,CAAC;gBACnD,KAAK,EAAE,IAAI;aACZ;SACF,wEA8Da,uBAAuB,2BAAU,WAAW,uFAE/C,wBAAwB,kMC5GrC,qoDAwCA,wuEDMY,IAAI,6FAAE,oBAAoB,uFAAE,wBAAwB,0GAAE,WAAW,sIAAE,mBAAmB,kNAAE,gBAAgB,+IAAE,SAAS;;2FAElH,oBAAoB;kBAdhC,SAAS;+BACE,iBAAiB,aAGhB;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,qBAAqB,CAAC;4BACnD,KAAK,EAAE,IAAI;yBACZ;qBACF,cACW,IAAI,WACP,CAAC,IAAI,EAAE,oBAAoB,EAAE,wBAAwB,EAAE,WAAW,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,SAAS,CAAC;sFAoDrH,WAAW;sBAAnB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBAEI,eAAe;sBAAxB,MAAM;gBACG,IAAI;sBAAb,MAAM;gBAEqE,gBAAgB;sBAA3F,YAAY;uBAAC,uBAAuB,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE;gBAEnB,UAAU;sBAAhE,SAAS;uBAAC,wBAAwB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACZ,gBAAgB;sBAAxD,SAAS;uBAAC,UAAU,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n  forwardRef\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { filter as _filter, findIndex, includes, last } from 'lodash/fp';\nimport { Subject, fromEvent as observableFromEvent } from 'rxjs';\nimport { distinctUntilChanged, filter, map, takeUntil, tap } from 'rxjs/operators';\nimport { MarkInvalidDirective } from '../../directives/mark-invalid.directive';\nimport { SelectDropdownDirective } from '../../directives/select-dropdown.directive';\nimport { ARROW_DOWN, ARROW_LEFT, ARROW_RIGHT, ARROW_UP, BACKSPACE, ENTER, ESCAPE, TAB } from '../../helpers/key-codes.constants';\nimport { BaseSelectDirective, EventSet } from '../../models/base-select.directive';\nimport { KeyboardSelectAction } from '../keyboard-select.directive';\nimport { ResponsiveInputComponent } from '../responsive-input/responsive-input.component';\n\nexport type MultiSelectSize = 'default' | 'small' | 'large' | 'select2';\n\nexport type MultiSelectRemoveItemEvent = {\n  item: any;\n  /**\n   * Is true when the item was not removed by pressing backspace on the keyboard rather than clicking the 'x'.\n   */\n  isMouse: boolean;\n};\n\n@Component({\n  selector: 'lx-multi-select',\n  templateUrl: 'multi-select.component.html',\n  styleUrls: ['multi-select.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectComponent),\n      multi: true\n    }\n  ],\n  standalone: true,\n  imports: [NgIf, MarkInvalidDirective, ResponsiveInputComponent, FormsModule, ReactiveFormsModule, NgTemplateOutlet, AsyncPipe]\n})\nexport class MultiSelectComponent extends BaseSelectDirective implements AfterViewInit, ControlValueAccessor {\n  public static isStillPossibleMoveToLeft(eventSet: EventSet): boolean {\n    return (\n      !(\n        eventSet.virtualCursorPosition === 0 &&\n        eventSet.inputCursorPosition === 0 &&\n        eventSet.event.keyCode === ARROW_LEFT &&\n        eventSet.selectionLength === 0\n      ) &&\n      !(\n        eventSet.event.keyCode === ARROW_LEFT &&\n        typeof eventSet.selectionLength === 'number' &&\n        eventSet.selectionLength > 0 &&\n        Math.abs(eventSet.virtualCursorPosition) === eventSet.selectionLength\n      )\n    );\n  }\n\n  public static isMovingfromZeroToRight(eventSet: EventSet) {\n    return eventSet.virtualCursorPosition === 0 && eventSet.inputCursorPosition === 0 && eventSet.event.keyCode === ARROW_RIGHT;\n  }\n\n  public static calculateNewCursorPostionOnKeyboardNavigation(cursorPosition: number, keyCode: number) {\n    switch (keyCode) {\n      case ARROW_LEFT:\n        cursorPosition--;\n        break;\n      case ARROW_RIGHT:\n        cursorPosition++;\n        break;\n    }\n    return cursorPosition;\n  }\n\n  public static getKeyboardSelectAction(cursorPosition: number, keyCode: number): KeyboardSelectAction | null {\n    if (cursorPosition <= -1 && keyCode === ARROW_RIGHT) {\n      return KeyboardSelectAction.NEXT;\n    }\n    if (cursorPosition === -1 && keyCode === ARROW_LEFT) {\n      return KeyboardSelectAction.LAST;\n    }\n    if (cursorPosition < -1 && keyCode === ARROW_LEFT) {\n      return KeyboardSelectAction.PREV;\n    }\n    if (cursorPosition <= -1 && keyCode === BACKSPACE) {\n      return KeyboardSelectAction.EXECUTE;\n    }\n    return null;\n  }\n\n  @Input() markInvalid: boolean = false;\n  @Input() selection?: any[] | null = [];\n  @Input() size: MultiSelectSize = 'default';\n  @Input() dropdownWidthScale: '1x' | '1.5x' | '2x' = '1x';\n\n  @Output() selectionChange = new EventEmitter<any[]>();\n  @Output() blur = new EventEmitter<void>();\n\n  @ContentChild(SelectDropdownDirective, { read: TemplateRef, static: true }) explicitDropdown?: TemplateRef<any>;\n\n  @ViewChild(ResponsiveInputComponent, { static: true }) queryInput!: ResponsiveInputComponent;\n  @ViewChild('dropdown', { static: true }) implicitDropdown!: TemplateRef<any>;\n\n  get dropdownTmpl() {\n    return this.explicitDropdown || this.implicitDropdown;\n  }\n\n  override readonly destroyed$ = new Subject<void>();\n  isInputFocusedViaTab = false;\n\n  constructor(private cd: ChangeDetectorRef) {\n    super();\n  }\n\n  get dropdownWidth(): string | undefined {\n    switch (this.dropdownWidthScale) {\n      case '1x':\n        return undefined;\n      case '1.5x':\n        return '150%';\n      case '2x':\n        return '200%';\n    }\n  }\n\n  resetInput() {\n    this.queryInput.resetInput();\n    this.queryControl.setValue('');\n  }\n\n  focus() {\n    this.queryInput.focusInput();\n  }\n\n  focusAndOpen() {\n    this.queryInput.focusInput();\n    this.open.next(true);\n  }\n\n  override ngAfterViewInit() {\n    super.ngAfterViewInit();\n\n    this.open\n      .pipe(\n        distinctUntilChanged(),\n        filter((open) => open),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => {\n        this.selectFirstOption();\n      });\n\n    // keyboard event handling\n    const sourceElement = this.queryInput.responsiveInput.nativeElement;\n\n    observableFromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        filter((event) => includes(event.keyCode, [ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => this.open.next(true));\n\n    // handle keyboard navigation for the selection\n    observableFromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        map((event) =>\n          this.mapToEventSet(\n            event,\n            this.queryInput.responsiveInput.nativeElement,\n            this.virtualCursorPosition,\n            this.selection ? this.selection.length : 0\n          )\n        ),\n        filter((eventSet) => {\n          return !MultiSelectComponent.isMovingfromZeroToRight(eventSet) && MultiSelectComponent.isStillPossibleMoveToLeft(eventSet);\n        }),\n        tap((eventSet) => {\n          if (\n            eventSet.virtualCursorPosition === 0 &&\n            eventSet.inputCursorPosition === 0 &&\n            eventSet.event.keyCode === BACKSPACE &&\n            typeof eventSet.selectionLength === 'number' &&\n            eventSet.selectionLength > 0\n          ) {\n            const lastItem = last(this.selection);\n            this.removeItem({ item: lastItem, isMouse: false });\n            eventSet.inputCursorPosition = undefined; // set to null to prevent continuing in the stream\n          }\n          if (typeof eventSet.inputCursorPosition === 'number' && eventSet.inputCursorPosition > 0) {\n            this.virtualCursorPosition = 0;\n          } else if (eventSet.virtualCursorPosition === -1 && eventSet.event.keyCode === ARROW_RIGHT) {\n            eventSet.event.preventDefault();\n            this.selectionKeyboardSelectAction$.next(KeyboardSelectAction.UNSELECT);\n            this.virtualCursorPosition = 0;\n            eventSet.inputCursorPosition = undefined; // set to null to prevent continuing in the stream\n          }\n        }),\n        filter((eventSet) => eventSet.inputCursorPosition === 0),\n        map((eventSet) => eventSet.event),\n        filter((event) => includes(event.keyCode, [ARROW_RIGHT, ARROW_LEFT, BACKSPACE])),\n        map((event) => {\n          this.virtualCursorPosition = MultiSelectComponent.calculateNewCursorPostionOnKeyboardNavigation(\n            this.virtualCursorPosition,\n            event.keyCode\n          );\n          return {\n            cursorPosition: this.virtualCursorPosition,\n            event\n          };\n        }),\n        map((positionSet) => {\n          const action = MultiSelectComponent.getKeyboardSelectAction(positionSet.cursorPosition, positionSet.event.keyCode);\n          if (action === KeyboardSelectAction.NEXT) {\n            // prevent cursor from moving in the input field, while we are still in the selection items\n            positionSet.event.preventDefault();\n          }\n          return action;\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(this.selectionKeyboardSelectAction$);\n\n    const inputKeyboardEvents$ = observableFromEvent<KeyboardEvent>(sourceElement, 'keydown');\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ARROW_UP, ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((event) => {\n        event.preventDefault();\n      });\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => event.keyCode === ESCAPE || event.keyCode === TAB),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((event) => {\n        if (event.keyCode === ESCAPE && this.dropdownOpen) {\n          event.preventDefault();\n          event.stopImmediatePropagation();\n        }\n        this.open.next(false);\n      });\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ENTER, ARROW_UP, ARROW_DOWN])),\n        map((event) => {\n          switch (event.keyCode) {\n            case ENTER:\n              return KeyboardSelectAction.EXECUTE;\n            case ARROW_UP:\n              return KeyboardSelectAction.PREV;\n            case ARROW_DOWN:\n              return KeyboardSelectAction.NEXT;\n            default:\n              return null;\n          }\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(this.optionsKeyboardSelectAction$);\n\n    inputKeyboardEvents$\n      .pipe(\n        filter((event) => includes(event.keyCode, [ENTER, ARROW_DOWN])),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe(() => this.open.next(true));\n  }\n\n  /**\n   * Adds the given option to the selection and, by default, resets the input field.\n   * @param option The option to add to the selection.\n   * @param keepSearchQueryAndScrollPosition If true, the search query and scroll position of the dropdown will be kept, unless there are no options left.\n   */\n  addItemToSelection(option: any, keepSearchQueryAndScrollPosition = false) {\n    if (option !== null) {\n      this.selection = [...(this.selection || [])].concat(option);\n      this.selectionChange.emit([...this.selection]);\n      this.propagateChange([...this.selection]);\n    }\n\n    const filteredOptionsInDropdownBeforeSelection = this.dropdownBase?._items.length ?? 0;\n    if (keepSearchQueryAndScrollPosition && filteredOptionsInDropdownBeforeSelection > 1) {\n      return;\n    }\n\n    this.resetSelectState();\n  }\n\n  removeItem(optionToRemove: MultiSelectRemoveItemEvent) {\n    if (this.disabled) {\n      return;\n    }\n    const prevOptionIndex = findIndex((option) => option === optionToRemove.item, this.selection);\n    const prevSelectionLenght = this.selection?.length ?? 0;\n    this.selection = _filter((option) => option !== optionToRemove.item, this.selection);\n    this.cd.markForCheck();\n    this.cd.detectChanges();\n    this.selectionChange.emit([...this.selection]);\n    this.propagateChange([...this.selection]);\n\n    if (prevOptionIndex === prevSelectionLenght - 1 || optionToRemove.isMouse) {\n      this.resetSelectState();\n    } else {\n      this.virtualCursorPosition++;\n    }\n  }\n\n  propagateChange(_value: any[]) {}\n\n  writeValue(value: any[]) {\n    this.selection = value;\n  }\n\n  registerOnChange(fn: any) {\n    this.propagateChange = fn;\n  }\n\n  registerOnTouched(_fn: any) {}\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  focusedViaTab() {\n    this.isInputFocusedViaTab = true;\n  }\n}\n","<div *ngIf=\"dropdownOpen\" class=\"backdrop\" (click)=\"handleBackdropClick($event)\"></div>\n<div\n  [lxMarkInvalid]=\"markInvalid\"\n  class=\"selectContainer\"\n  [class.open]=\"dropdownOpen\"\n  [class.top]=\"(dropdownDirection$ | async) === 'TOP'\"\n  [class.bottom]=\"(dropdownDirection$ | async) === 'BOTTOM'\"\n  [class.focused]=\"isInputFocused\"\n  [class.focusedVisible]=\"isInputFocusedViaTab\"\n  [class.hasSelection]=\"selection && selection.length > 0\"\n  [class.smallSize]=\"size === 'small'\"\n  [class.disabled]=\"disabled\"\n  (click)=\"handleClick($event.target === toggle)\"\n>\n  <div class=\"selectedChoicesContainer\">\n    <div *ngIf=\"(selection?.length === 0 || !selection) && !queryControl?.value\" class=\"placeholder\" [attr.title]=\"placeholder\">\n      {{ placeholder }}\n    </div>\n    <div class=\"selection\">\n      <ng-content select=\".pills\"></ng-content>\n    </div>\n    <lx-responsive-input\n      (focus)=\"isInputFocused = true\"\n      (blur)=\"isInputFocused = false; isInputFocusedViaTab = false; blur.emit()\"\n      (focusViaTab)=\"focusedViaTab()\"\n      [formControl]=\"queryControl\"\n      (keydown.enter)=\"$event.preventDefault()\"\n    ></lx-responsive-input>\n    <i #toggle [hidden]=\"disabled\" class=\"fas fa-angle-down\" aria-hidden=\"true\"></i>\n  </div>\n  <div class=\"optionsContainer\" [style.width]=\"dropdownWidth\" #optionsContainer>\n    <ng-container *ngIf=\"dropdownOpen\">\n      <ng-container *ngTemplateOutlet=\"dropdownTmpl\"></ng-container>\n    </ng-container>\n  </div>\n</div>\n\n<ng-template #dropdown>\n  <ng-content select=\".dropdownComponent\"></ng-content>\n</ng-template>\n"]}
|
@@ -5,12 +5,11 @@ import { RouterLink, RouterLinkActive } from '@angular/router';
|
|
5
5
|
import { uniqueId } from 'lodash/fp';
|
6
6
|
import { BadgeComponent } from '../../../core-ui/components/badge/badge.component';
|
7
7
|
import * as i0 from "@angular/core";
|
8
|
-
import * as i1 from "@angular/
|
9
|
-
import * as i2 from "@angular/cdk/portal";
|
8
|
+
import * as i1 from "@angular/cdk/portal";
|
10
9
|
export class TabComponent {
|
11
10
|
set isActive(value) {
|
12
11
|
this._isActive = value;
|
13
|
-
this.cd
|
12
|
+
this.cd.markForCheck();
|
14
13
|
}
|
15
14
|
get isActive() {
|
16
15
|
if (this.routerLinkActive) {
|
@@ -18,9 +17,7 @@ export class TabComponent {
|
|
18
17
|
}
|
19
18
|
return this._isActive;
|
20
19
|
}
|
21
|
-
constructor(
|
22
|
-
this.router = router;
|
23
|
-
this.activatedRoute = activatedRoute;
|
20
|
+
constructor(cd) {
|
24
21
|
this.cd = cd;
|
25
22
|
this.label = '';
|
26
23
|
this.routerLinkActiveOptions = { exact: true };
|
@@ -30,47 +27,20 @@ export class TabComponent {
|
|
30
27
|
this.background = 'white';
|
31
28
|
this.disabled = false;
|
32
29
|
this.switch = new EventEmitter();
|
33
|
-
this.keyDownAction = new EventEmitter();
|
34
30
|
this._isActive = false;
|
35
31
|
this.tabPanelId = uniqueId('panelId');
|
36
32
|
this.tabId = uniqueId('tab');
|
37
33
|
}
|
38
34
|
select() {
|
39
|
-
|
40
|
-
const formattedTabLink = typeof this.tabLink === 'string' ? [this.tabLink] : this.tabLink;
|
41
|
-
this.router.navigate(formattedTabLink, { relativeTo: this.activatedRoute }).then(() => {
|
42
|
-
this.switch.emit({ tabId: this.tabId, tabPanelId: this.tabPanelId });
|
43
|
-
});
|
44
|
-
}
|
45
|
-
else {
|
46
|
-
this.switch.emit({ tabId: this.tabId, tabPanelId: this.tabPanelId });
|
47
|
-
}
|
48
|
-
this.isActive = true;
|
35
|
+
this.switch.emit({ tabId: this.tabId, tabPanelId: this.tabPanelId });
|
49
36
|
}
|
50
|
-
|
51
|
-
|
52
|
-
}
|
53
|
-
handleKeyDown(event) {
|
54
|
-
switch (event.code) {
|
55
|
-
case 'Enter':
|
56
|
-
case 'Space':
|
57
|
-
/**
|
58
|
-
* We are preventing the default browser behavior when the `SPACE` key is pressed, in which
|
59
|
-
* case the browser would scroll down the page.
|
60
|
-
*/
|
61
|
-
event.preventDefault();
|
62
|
-
this.select();
|
63
|
-
break;
|
64
|
-
}
|
65
|
-
this.keyDownAction.emit(event);
|
66
|
-
}
|
67
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabComponent, deps: [{ token: i1.Router }, { token: i1.ActivatedRoute }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
68
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.2", type: TabComponent, isStandalone: true, selector: "lx-tab", inputs: { icon: "icon", label: "label", title: "title", tabLink: "tabLink", counter: "counter", routerLinkActiveOptions: "routerLinkActiveOptions", counterBadgeSize: "counterBadgeSize", noMargin: "noMargin", noLeftMarginForFirstTab: "noLeftMarginForFirstTab", background: "background", disabled: "disabled" }, outputs: { switch: "switch", keyDownAction: "keyDownAction" }, viewQueries: [{ propertyName: "content", first: true, predicate: ["contentTemplate"], descendants: true, static: true }, { propertyName: "routerLinkActive", first: true, predicate: RouterLinkActive, descendants: true }, { propertyName: "tabElement", first: true, predicate: ["tabElement"], descendants: true }], ngImport: i0, template: "<li\n #tabElement\n class=\"tab\"\n [class.active]=\"isActive\"\n [class.withIcon]=\"icon\"\n [class.noMargin]=\"noMargin\"\n [class.noLeftMarginForFirstTab]=\"noLeftMarginForFirstTab\"\n [class.backgroundGray]=\"background === 'gray'\"\n [class.disabled]=\"disabled\"\n [class.portalTab]=\"!tabLink\"\n [class.routerLinkTab]=\"tabLink\"\n [attr.title]=\"title\"\n (click)=\"select()\"\n (keydown)=\"handleKeyDown($event)\"\n role=\"tab\"\n [attr.id]=\"tabId\"\n [attr.aria-selected]=\"isActive\"\n [attr.aria-controls]=\"tabPanelId\"\n [attr.tabindex]=\"isActive ? '0' : '-1'\"\n>\n <a\n *ngIf=\"tabLink; else portalTab\"\n tabindex=\"-1\"\n [routerLink]=\"tabLink\"\n routerLinkActive\n [routerLinkActiveOptions]=\"routerLinkActiveOptions\"\n >\n <ng-container *ngTemplateOutlet=\"portalTab\"></ng-container>\n </a>\n\n <ng-template #portalTab>\n <i *ngIf=\"icon\" class=\"icon {{ icon }}\"></i>\n <span *ngIf=\"label\" class=\"tabLabel\">{{ label }}</span>\n <lx-badge *ngIf=\"counter\" [size]=\"counterBadgeSize\" [content]=\"counter\" [color]=\"'gray'\"></lx-badge>\n </ng-template>\n\n <ng-template cdkPortal #contentTemplate=\"cdkPortal\">\n <div *ngIf=\"!tabLink\" class=\"content\" role=\"tabpanel\" [attr.id]=\"tabPanelId\">\n <ng-content></ng-content>\n </div>\n </ng-template>\n</li>\n", styles: [":host{display:inline-block}.tab{position:relative;display:inline-block;margin:0 8px;text-align:center;transition:border-bottom .5s;cursor:pointer}.tab:after{position:absolute;content:\"\";left:0;right:0;bottom:-1px;height:2px;background-color:transparent}.tab:hover:after,.tab.active:after{transition:background-color .1s;background-color:#1666ee}.tab.active{cursor:default}.tab.disabled{opacity:.3;cursor:default}.tab:hover .tabLabel,.tab.active .tabLabel{color:#2a303d;transition:color .1s}.portalTab,.routerLinkTab a{display:inline-block;padding:8px 8px 12px;line-height:20px}.portalTab.withIcon,.routerLinkTab a.withIcon{padding:10px 14px;line-height:16px}.routerLinkTab a:focus{outline:0}.routerLinkTab.active a,.routerLinkTab.disabled a{cursor:default}.icon{font-size:16px;opacity:.6}.noMargin{margin:0}.noLeftMarginForFirstTab{margin-left:0}.backgroundGray{background-color:#c2c9d6}.backgroundGray:not(.active):not(.disabled){background-color:#cfd5df}.tabLabel{color:#61779d;display:inline-block;font-size:13px;text-decoration:none}.content{height:100%}lx-badge{margin-left:8px}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: BadgeComponent, selector: "lx-badge", inputs: ["content", "size", "color"] }, { kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i2.CdkPortal, selector: "[cdkPortal]", exportAs: ["cdkPortal"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
37
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
38
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.2", type: TabComponent, isStandalone: true, selector: "lx-tab", inputs: { icon: "icon", label: "label", title: "title", tabLink: "tabLink", counter: "counter", routerLinkActiveOptions: "routerLinkActiveOptions", counterBadgeSize: "counterBadgeSize", noMargin: "noMargin", noLeftMarginForFirstTab: "noLeftMarginForFirstTab", background: "background", disabled: "disabled" }, outputs: { switch: "switch" }, viewQueries: [{ propertyName: "content", first: true, predicate: ["contentTemplate"], descendants: true, static: true }, { propertyName: "routerLinkActive", first: true, predicate: RouterLinkActive, descendants: true }], ngImport: i0, template: "<li\n class=\"tab\"\n [class.active]=\"isActive\"\n [class.withIcon]=\"icon\"\n [class.noMargin]=\"noMargin\"\n [class.noLeftMarginForFirstTab]=\"noLeftMarginForFirstTab\"\n [class.backgroundGray]=\"background === 'gray'\"\n [class.disabled]=\"disabled\"\n [class.portalTab]=\"!tabLink\"\n [class.routerLinkTab]=\"tabLink\"\n [attr.title]=\"title\"\n (click)=\"select()\"\n (keyup.enter)=\"select()\"\n (keyup.space)=\"select()\"\n [attr.role]=\"tabLink ? 'presentation' : 'tab'\"\n [attr.id]=\"tabLink ? null : tabId\"\n [attr.aria-selected]=\"tabLink ? null : isActive\"\n [attr.aria-controls]=\"tabLink ? null : tabPanelId\"\n [attr.tabindex]=\"tabLink ? null : isActive || disabled || tabLink ? '-1' : '0'\"\n>\n <a\n *ngIf=\"tabLink; else portalTab\"\n role=\"tab\"\n [attr.id]=\"tabId\"\n [attr.aria-selected]=\"isActive\"\n [attr.aria-controls]=\"tabPanelId\"\n [routerLink]=\"tabLink\"\n routerLinkActive\n [routerLinkActiveOptions]=\"routerLinkActiveOptions\"\n >\n <ng-container *ngTemplateOutlet=\"portalTab\"></ng-container>\n </a>\n\n <ng-template #portalTab>\n <i *ngIf=\"icon\" class=\"icon {{ icon }}\"></i>\n <span *ngIf=\"label\" class=\"tabLabel\">{{ label }}</span>\n <lx-badge *ngIf=\"counter\" [size]=\"counterBadgeSize\" [content]=\"counter\" [color]=\"'gray'\"></lx-badge>\n </ng-template>\n\n <ng-template cdkPortal #contentTemplate=\"cdkPortal\">\n <div *ngIf=\"!tabLink\" class=\"content\" role=\"tabpanel\" [attr.id]=\"tabPanelId\">\n <ng-content></ng-content>\n </div>\n </ng-template>\n</li>\n", styles: [":host{display:inline-block}.tab{position:relative;display:inline-block;margin:0 8px;text-align:center;transition:border-bottom .5s;cursor:pointer}.tab:after{position:absolute;content:\"\";left:0;right:0;bottom:-1px;height:2px;background-color:transparent}.tab:hover:after,.tab.active:after{transition:background-color .1s;background-color:#1666ee;outline:0}.tab:hover,.tab.active{outline:0}.tab.active{cursor:default}.tab.disabled{opacity:.3;cursor:default}.tab:hover .tabLabel,.tab.active .tabLabel{color:#2a303d;transition:color .1s}.portalTab,.routerLinkTab a{display:inline-block;padding:8px 8px 12px;line-height:20px}.portalTab.withIcon,.routerLinkTab a.withIcon{padding:10px 14px;line-height:16px}.routerLinkTab a:focus{outline:0}.routerLinkTab.active a,.routerLinkTab.disabled a{cursor:default}.icon{font-size:16px;opacity:.6}.noMargin{margin:0}.noLeftMarginForFirstTab{margin-left:0}.backgroundGray{background-color:#c2c9d6}.backgroundGray:not(.active):not(.disabled){background-color:#cfd5df}.tabLabel{color:#61779d;display:inline-block;font-size:13px;text-decoration:none}.content{height:100%}lx-badge{margin-left:8px}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: BadgeComponent, selector: "lx-badge", inputs: ["content", "size", "color"] }, { kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i1.CdkPortal, selector: "[cdkPortal]", exportAs: ["cdkPortal"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
69
39
|
}
|
70
40
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabComponent, decorators: [{
|
71
41
|
type: Component,
|
72
|
-
args: [{ selector: 'lx-tab', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, RouterLinkActive, RouterLink, NgTemplateOutlet, BadgeComponent, PortalModule], template: "<li\n
|
73
|
-
}], ctorParameters: () => [{ type:
|
42
|
+
args: [{ selector: 'lx-tab', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, RouterLinkActive, RouterLink, NgTemplateOutlet, BadgeComponent, PortalModule], template: "<li\n class=\"tab\"\n [class.active]=\"isActive\"\n [class.withIcon]=\"icon\"\n [class.noMargin]=\"noMargin\"\n [class.noLeftMarginForFirstTab]=\"noLeftMarginForFirstTab\"\n [class.backgroundGray]=\"background === 'gray'\"\n [class.disabled]=\"disabled\"\n [class.portalTab]=\"!tabLink\"\n [class.routerLinkTab]=\"tabLink\"\n [attr.title]=\"title\"\n (click)=\"select()\"\n (keyup.enter)=\"select()\"\n (keyup.space)=\"select()\"\n [attr.role]=\"tabLink ? 'presentation' : 'tab'\"\n [attr.id]=\"tabLink ? null : tabId\"\n [attr.aria-selected]=\"tabLink ? null : isActive\"\n [attr.aria-controls]=\"tabLink ? null : tabPanelId\"\n [attr.tabindex]=\"tabLink ? null : isActive || disabled || tabLink ? '-1' : '0'\"\n>\n <a\n *ngIf=\"tabLink; else portalTab\"\n role=\"tab\"\n [attr.id]=\"tabId\"\n [attr.aria-selected]=\"isActive\"\n [attr.aria-controls]=\"tabPanelId\"\n [routerLink]=\"tabLink\"\n routerLinkActive\n [routerLinkActiveOptions]=\"routerLinkActiveOptions\"\n >\n <ng-container *ngTemplateOutlet=\"portalTab\"></ng-container>\n </a>\n\n <ng-template #portalTab>\n <i *ngIf=\"icon\" class=\"icon {{ icon }}\"></i>\n <span *ngIf=\"label\" class=\"tabLabel\">{{ label }}</span>\n <lx-badge *ngIf=\"counter\" [size]=\"counterBadgeSize\" [content]=\"counter\" [color]=\"'gray'\"></lx-badge>\n </ng-template>\n\n <ng-template cdkPortal #contentTemplate=\"cdkPortal\">\n <div *ngIf=\"!tabLink\" class=\"content\" role=\"tabpanel\" [attr.id]=\"tabPanelId\">\n <ng-content></ng-content>\n </div>\n </ng-template>\n</li>\n", styles: [":host{display:inline-block}.tab{position:relative;display:inline-block;margin:0 8px;text-align:center;transition:border-bottom .5s;cursor:pointer}.tab:after{position:absolute;content:\"\";left:0;right:0;bottom:-1px;height:2px;background-color:transparent}.tab:hover:after,.tab.active:after{transition:background-color .1s;background-color:#1666ee;outline:0}.tab:hover,.tab.active{outline:0}.tab.active{cursor:default}.tab.disabled{opacity:.3;cursor:default}.tab:hover .tabLabel,.tab.active .tabLabel{color:#2a303d;transition:color .1s}.portalTab,.routerLinkTab a{display:inline-block;padding:8px 8px 12px;line-height:20px}.portalTab.withIcon,.routerLinkTab a.withIcon{padding:10px 14px;line-height:16px}.routerLinkTab a:focus{outline:0}.routerLinkTab.active a,.routerLinkTab.disabled a{cursor:default}.icon{font-size:16px;opacity:.6}.noMargin{margin:0}.noLeftMarginForFirstTab{margin-left:0}.backgroundGray{background-color:#c2c9d6}.backgroundGray:not(.active):not(.disabled){background-color:#cfd5df}.tabLabel{color:#61779d;display:inline-block;font-size:13px;text-decoration:none}.content{height:100%}lx-badge{margin-left:8px}\n"] }]
|
43
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { icon: [{
|
74
44
|
type: Input
|
75
45
|
}], label: [{
|
76
46
|
type: Input
|
@@ -94,16 +64,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImpor
|
|
94
64
|
type: Input
|
95
65
|
}], switch: [{
|
96
66
|
type: Output
|
97
|
-
}], keyDownAction: [{
|
98
|
-
type: Output
|
99
67
|
}], content: [{
|
100
68
|
type: ViewChild,
|
101
69
|
args: ['contentTemplate', { static: true }]
|
102
70
|
}], routerLinkActive: [{
|
103
71
|
type: ViewChild,
|
104
72
|
args: [RouterLinkActive]
|
105
|
-
}], tabElement: [{
|
106
|
-
type: ViewChild,
|
107
|
-
args: ['tabElement']
|
108
73
|
}] } });
|
109
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/tab-ui/components/tab/tab.component.ts","../../../../../../../../libs/components/src/lib/tab-ui/components/tab/tab.component.html"],"names":[],"mappings":"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","sourcesContent":["import { PortalModule, TemplatePortal } from '@angular/cdk/portal';\nimport { NgIf, NgTemplateOutlet } from '@angular/common';\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core';\nimport { ActivatedRoute, Router, RouterLink, RouterLinkActive } from '@angular/router';\nimport { uniqueId } from 'lodash/fp';\nimport { BadgeComponent } from '../../../core-ui/components/badge/badge.component';\n\n@Component({\n  selector: 'lx-tab',\n  templateUrl: 'tab.component.html',\n  styleUrls: ['tab.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [NgIf, RouterLinkActive, RouterLink, NgTemplateOutlet, BadgeComponent, PortalModule]\n})\nexport class TabComponent {\n  /**\n   * The icon input takes a font awesome selector\n   * WARNING: Currently this component does not support using an icon together with a label or counter.\n   */\n  @Input() icon?: string;\n  @Input() label: string = '';\n  @Input() title?: string;\n  /**\n   * In some parts of the application, we are passing a plain string to the tabLink input, whereas in\n   * most parts, the input is passed as an array. The RouterLink directive also accepts both types,\n   * therefore we are using type of the input as string | any[].\n   */\n  @Input() tabLink?: string | any[];\n  @Input() counter?: number;\n  @Input() routerLinkActiveOptions: { exact: boolean } = { exact: true };\n  @Input() counterBadgeSize: 'default' | 'small' = 'default';\n  @Input() noMargin: boolean = false;\n  @Input() noLeftMarginForFirstTab: boolean = false;\n  @Input() background: 'white' | 'gray' = 'white';\n  @Input() disabled: boolean = false;\n\n  @Output() switch = new EventEmitter();\n  @Output() keyDownAction = new EventEmitter<KeyboardEvent>();\n\n  tabPanelId: string;\n  tabId: string;\n\n  @ViewChild('contentTemplate', { static: true })\n  content?: TemplatePortal<any>;\n\n  @ViewChild(RouterLinkActive) routerLinkActive?: RouterLinkActive;\n\n  @ViewChild('tabElement') tabElement?: ElementRef<HTMLElement>;\n\n  set isActive(value: boolean) {\n    this._isActive = value;\n    this.cd?.markForCheck();\n  }\n  get isActive() {\n    if (this.routerLinkActive) {\n      return this.routerLinkActive.isActive;\n    }\n    return this._isActive;\n  }\n  private _isActive = false;\n\n  constructor(\n    private router: Router,\n    private activatedRoute: ActivatedRoute,\n    private cd: ChangeDetectorRef\n  ) {\n    this.tabPanelId = uniqueId('panelId');\n    this.tabId = uniqueId('tab');\n  }\n\n  select(): void {\n    if (this.tabLink) {\n      const formattedTabLink = typeof this.tabLink === 'string' ? [this.tabLink] : this.tabLink;\n      this.router.navigate(formattedTabLink, { relativeTo: this.activatedRoute }).then(() => {\n        this.switch.emit({ tabId: this.tabId, tabPanelId: this.tabPanelId });\n      });\n    } else {\n      this.switch.emit({ tabId: this.tabId, tabPanelId: this.tabPanelId });\n    }\n    this.isActive = true;\n  }\n\n  setFocus(): void {\n    this.tabElement?.nativeElement?.focus();\n  }\n\n  handleKeyDown(event: KeyboardEvent) {\n    switch (event.code) {\n      case 'Enter':\n      case 'Space':\n        /**\n         * We are preventing the default browser behavior when the `SPACE` key is pressed, in which\n         * case the browser would scroll down the page.\n         */\n        event.preventDefault();\n        this.select();\n        break;\n    }\n    this.keyDownAction.emit(event);\n  }\n}\n","<li\n  #tabElement\n  class=\"tab\"\n  [class.active]=\"isActive\"\n  [class.withIcon]=\"icon\"\n  [class.noMargin]=\"noMargin\"\n  [class.noLeftMarginForFirstTab]=\"noLeftMarginForFirstTab\"\n  [class.backgroundGray]=\"background === 'gray'\"\n  [class.disabled]=\"disabled\"\n  [class.portalTab]=\"!tabLink\"\n  [class.routerLinkTab]=\"tabLink\"\n  [attr.title]=\"title\"\n  (click)=\"select()\"\n  (keydown)=\"handleKeyDown($event)\"\n  role=\"tab\"\n  [attr.id]=\"tabId\"\n  [attr.aria-selected]=\"isActive\"\n  [attr.aria-controls]=\"tabPanelId\"\n  [attr.tabindex]=\"isActive ? '0' : '-1'\"\n>\n  <a\n    *ngIf=\"tabLink; else portalTab\"\n    tabindex=\"-1\"\n    [routerLink]=\"tabLink\"\n    routerLinkActive\n    [routerLinkActiveOptions]=\"routerLinkActiveOptions\"\n  >\n    <ng-container *ngTemplateOutlet=\"portalTab\"></ng-container>\n  </a>\n\n  <ng-template #portalTab>\n    <i *ngIf=\"icon\" class=\"icon {{ icon }}\"></i>\n    <span *ngIf=\"label\" class=\"tabLabel\">{{ label }}</span>\n    <lx-badge *ngIf=\"counter\" [size]=\"counterBadgeSize\" [content]=\"counter\" [color]=\"'gray'\"></lx-badge>\n  </ng-template>\n\n  <ng-template cdkPortal #contentTemplate=\"cdkPortal\">\n    <div *ngIf=\"!tabLink\" class=\"content\" role=\"tabpanel\" [attr.id]=\"tabPanelId\">\n      <ng-content></ng-content>\n    </div>\n  </ng-template>\n</li>\n"]}
|
74
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFiLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9zcmMvbGliL3RhYi11aS9jb21wb25lbnRzL3RhYi90YWIuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL3NyYy9saWIvdGFiLXVpL2NvbXBvbmVudHMvdGFiL3RhYi5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFrQixNQUFNLHFCQUFxQixDQUFDO0FBQ25FLE9BQU8sRUFBRSxJQUFJLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsdUJBQXVCLEVBQXFCLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDOUgsT0FBTyxFQUFFLFVBQVUsRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9ELE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFDckMsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1EQUFtRCxDQUFDOzs7QUFVbkYsTUFBTSxPQUFPLFlBQVk7SUEyQnZCLElBQUksUUFBUSxDQUFDLEtBQWM7UUFDekIsSUFBSSxDQUFDLFNBQVMsR0FBRyxLQUFLLENBQUM7UUFDdkIsSUFBSSxDQUFDLEVBQUUsQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUN6QixDQUFDO0lBQ0QsSUFBSSxRQUFRO1FBQ1YsSUFBSSxJQUFJLENBQUMsZ0JBQWdCLEVBQUU7WUFDekIsT0FBTyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDO1NBQ3ZDO1FBQ0QsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDO0lBQ3hCLENBQUM7SUFHRCxZQUFvQixFQUFxQjtRQUFyQixPQUFFLEdBQUYsRUFBRSxDQUFtQjtRQWpDaEMsVUFBSyxHQUFXLEVBQUUsQ0FBQztRQUluQiw0QkFBdUIsR0FBdUIsRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLENBQUM7UUFDOUQscUJBQWdCLEdBQXdCLFNBQVMsQ0FBQztRQUNsRCxhQUFRLEdBQVksS0FBSyxDQUFDO1FBQzFCLDRCQUF1QixHQUFZLEtBQUssQ0FBQztRQUN6QyxlQUFVLEdBQXFCLE9BQU8sQ0FBQztRQUN2QyxhQUFRLEdBQVksS0FBSyxDQUFDO1FBRXpCLFdBQU0sR0FBRyxJQUFJLFlBQVksRUFBRSxDQUFDO1FBb0I5QixjQUFTLEdBQUcsS0FBSyxDQUFDO1FBR3hCLElBQUksQ0FBQyxVQUFVLEdBQUcsUUFBUSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ3RDLElBQUksQ0FBQyxLQUFLLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQy9CLENBQUM7SUFFRCxNQUFNO1FBQ0osSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUUsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDLENBQUM7SUFDdkUsQ0FBQzs4R0E5Q1UsWUFBWTtrR0FBWixZQUFZLG9qQkF5QlosZ0JBQWdCLGdEQ3hDN0IsNmpEQTZDQSxvcUNEaENZLElBQUksNkZBQUUsZ0JBQWdCLDhNQUFFLFVBQVUsNE5BQUUsZ0JBQWdCLG9KQUFFLGNBQWMsMEZBQUUsWUFBWTs7MkZBRWpGLFlBQVk7a0JBUnhCLFNBQVM7K0JBQ0UsUUFBUSxtQkFHRCx1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLElBQUksRUFBRSxnQkFBZ0IsRUFBRSxVQUFVLEVBQUUsZ0JBQWdCLEVBQUUsY0FBYyxFQUFFLFlBQVksQ0FBQztzRkFPcEYsSUFBSTtzQkFBWixLQUFLO2dCQUNHLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUs7Z0JBQ0csT0FBTztzQkFBZixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDRyx1QkFBdUI7c0JBQS9CLEtBQUs7Z0JBQ0csZ0JBQWdCO3NCQUF4QixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ0csdUJBQXVCO3NCQUEvQixLQUFLO2dCQUNHLFVBQVU7c0JBQWxCLEtBQUs7Z0JBQ0csUUFBUTtzQkFBaEIsS0FBSztnQkFFSSxNQUFNO3NCQUFmLE1BQU07Z0JBTVAsT0FBTztzQkFETixTQUFTO3VCQUFDLGlCQUFpQixFQUFFLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRTtnQkFHakIsZ0JBQWdCO3NCQUE1QyxTQUFTO3VCQUFDLGdCQUFnQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBvcnRhbE1vZHVsZSwgVGVtcGxhdGVQb3J0YWwgfSBmcm9tICdAYW5ndWxhci9jZGsvcG9ydGFsJztcbmltcG9ydCB7IE5nSWYsIE5nVGVtcGxhdGVPdXRsZXQgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENoYW5nZURldGVjdG9yUmVmLCBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5wdXQsIE91dHB1dCwgVmlld0NoaWxkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBSb3V0ZXJMaW5rLCBSb3V0ZXJMaW5rQWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7IHVuaXF1ZUlkIH0gZnJvbSAnbG9kYXNoL2ZwJztcbmltcG9ydCB7IEJhZGdlQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vLi4vY29yZS11aS9jb21wb25lbnRzL2JhZGdlL2JhZGdlLmNvbXBvbmVudCc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2x4LXRhYicsXG4gIHRlbXBsYXRlVXJsOiAndGFiLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJ3RhYi5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW05nSWYsIFJvdXRlckxpbmtBY3RpdmUsIFJvdXRlckxpbmssIE5nVGVtcGxhdGVPdXRsZXQsIEJhZGdlQ29tcG9uZW50LCBQb3J0YWxNb2R1bGVdXG59KVxuZXhwb3J0IGNsYXNzIFRhYkNvbXBvbmVudCB7XG4gIC8qKlxuICAgKiBUaGUgaWNvbiBpbnB1dCB0YWtlcyBhIGZvbnQgYXdlc29tZSBzZWxlY3RvclxuICAgKiBXQVJOSU5HOiBDdXJyZW50bHkgdGhpcyBjb21wb25lbnQgZG9lcyBub3Qgc3VwcG9ydCB1c2luZyBhbiBpY29uIHRvZ2V0aGVyIHdpdGggYSBsYWJlbCBvciBjb3VudGVyLlxuICAgKi9cbiAgQElucHV0KCkgaWNvbj86IHN0cmluZztcbiAgQElucHV0KCkgbGFiZWw6IHN0cmluZyA9ICcnO1xuICBASW5wdXQoKSB0aXRsZT86IHN0cmluZztcbiAgQElucHV0KCkgdGFiTGluaz86IHN0cmluZztcbiAgQElucHV0KCkgY291bnRlcj86IG51bWJlcjtcbiAgQElucHV0KCkgcm91dGVyTGlua0FjdGl2ZU9wdGlvbnM6IHsgZXhhY3Q6IGJvb2xlYW4gfSA9IHsgZXhhY3Q6IHRydWUgfTtcbiAgQElucHV0KCkgY291bnRlckJhZGdlU2l6ZTogJ2RlZmF1bHQnIHwgJ3NtYWxsJyA9ICdkZWZhdWx0JztcbiAgQElucHV0KCkgbm9NYXJnaW46IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCkgbm9MZWZ0TWFyZ2luRm9yRmlyc3RUYWI6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCkgYmFja2dyb3VuZDogJ3doaXRlJyB8ICdncmF5JyA9ICd3aGl0ZSc7XG4gIEBJbnB1dCgpIGRpc2FibGVkOiBib29sZWFuID0gZmFsc2U7XG5cbiAgQE91dHB1dCgpIHN3aXRjaCA9IG5ldyBFdmVudEVtaXR0ZXIoKTtcblxuICB0YWJQYW5lbElkOiBzdHJpbmc7XG4gIHRhYklkOiBzdHJpbmc7XG5cbiAgQFZpZXdDaGlsZCgnY29udGVudFRlbXBsYXRlJywgeyBzdGF0aWM6IHRydWUgfSlcbiAgY29udGVudD86IFRlbXBsYXRlUG9ydGFsPGFueT47XG5cbiAgQFZpZXdDaGlsZChSb3V0ZXJMaW5rQWN0aXZlKSByb3V0ZXJMaW5rQWN0aXZlPzogUm91dGVyTGlua0FjdGl2ZTtcblxuICBzZXQgaXNBY3RpdmUodmFsdWU6IGJvb2xlYW4pIHtcbiAgICB0aGlzLl9pc0FjdGl2ZSA9IHZhbHVlO1xuICAgIHRoaXMuY2QubWFya0ZvckNoZWNrKCk7XG4gIH1cbiAgZ2V0IGlzQWN0aXZlKCkge1xuICAgIGlmICh0aGlzLnJvdXRlckxpbmtBY3RpdmUpIHtcbiAgICAgIHJldHVybiB0aGlzLnJvdXRlckxpbmtBY3RpdmUuaXNBY3RpdmU7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9pc0FjdGl2ZTtcbiAgfVxuICBwcml2YXRlIF9pc0FjdGl2ZSA9IGZhbHNlO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgY2Q6IENoYW5nZURldGVjdG9yUmVmKSB7XG4gICAgdGhpcy50YWJQYW5lbElkID0gdW5pcXVlSWQoJ3BhbmVsSWQnKTtcbiAgICB0aGlzLnRhYklkID0gdW5pcXVlSWQoJ3RhYicpO1xuICB9XG5cbiAgc2VsZWN0KCk6IHZvaWQge1xuICAgIHRoaXMuc3dpdGNoLmVtaXQoeyB0YWJJZDogdGhpcy50YWJJZCwgdGFiUGFuZWxJZDogdGhpcy50YWJQYW5lbElkIH0pO1xuICB9XG59XG4iLCI8bGlcbiAgY2xhc3M9XCJ0YWJcIlxuICBbY2xhc3MuYWN0aXZlXT1cImlzQWN0aXZlXCJcbiAgW2NsYXNzLndpdGhJY29uXT1cImljb25cIlxuICBbY2xhc3Mubm9NYXJnaW5dPVwibm9NYXJnaW5cIlxuICBbY2xhc3Mubm9MZWZ0TWFyZ2luRm9yRmlyc3RUYWJdPVwibm9MZWZ0TWFyZ2luRm9yRmlyc3RUYWJcIlxuICBbY2xhc3MuYmFja2dyb3VuZEdyYXldPVwiYmFja2dyb3VuZCA9PT0gJ2dyYXknXCJcbiAgW2NsYXNzLmRpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgW2NsYXNzLnBvcnRhbFRhYl09XCIhdGFiTGlua1wiXG4gIFtjbGFzcy5yb3V0ZXJMaW5rVGFiXT1cInRhYkxpbmtcIlxuICBbYXR0ci50aXRsZV09XCJ0aXRsZVwiXG4gIChjbGljayk9XCJzZWxlY3QoKVwiXG4gIChrZXl1cC5lbnRlcik9XCJzZWxlY3QoKVwiXG4gIChrZXl1cC5zcGFjZSk9XCJzZWxlY3QoKVwiXG4gIFthdHRyLnJvbGVdPVwidGFiTGluayA/ICdwcmVzZW50YXRpb24nIDogJ3RhYidcIlxuICBbYXR0ci5pZF09XCJ0YWJMaW5rID8gbnVsbCA6IHRhYklkXCJcbiAgW2F0dHIuYXJpYS1zZWxlY3RlZF09XCJ0YWJMaW5rID8gbnVsbCA6IGlzQWN0aXZlXCJcbiAgW2F0dHIuYXJpYS1jb250cm9sc109XCJ0YWJMaW5rID8gbnVsbCA6IHRhYlBhbmVsSWRcIlxuICBbYXR0ci50YWJpbmRleF09XCJ0YWJMaW5rID8gbnVsbCA6IGlzQWN0aXZlIHx8IGRpc2FibGVkIHx8IHRhYkxpbmsgPyAnLTEnIDogJzAnXCJcbj5cbiAgPGFcbiAgICAqbmdJZj1cInRhYkxpbms7IGVsc2UgcG9ydGFsVGFiXCJcbiAgICByb2xlPVwidGFiXCJcbiAgICBbYXR0ci5pZF09XCJ0YWJJZFwiXG4gICAgW2F0dHIuYXJpYS1zZWxlY3RlZF09XCJpc0FjdGl2ZVwiXG4gICAgW2F0dHIuYXJpYS1jb250cm9sc109XCJ0YWJQYW5lbElkXCJcbiAgICBbcm91dGVyTGlua109XCJ0YWJMaW5rXCJcbiAgICByb3V0ZXJMaW5rQWN0aXZlXG4gICAgW3JvdXRlckxpbmtBY3RpdmVPcHRpb25zXT1cInJvdXRlckxpbmtBY3RpdmVPcHRpb25zXCJcbiAgPlxuICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJwb3J0YWxUYWJcIj48L25nLWNvbnRhaW5lcj5cbiAgPC9hPlxuXG4gIDxuZy10ZW1wbGF0ZSAjcG9ydGFsVGFiPlxuICAgIDxpICpuZ0lmPVwiaWNvblwiIGNsYXNzPVwiaWNvbiB7eyBpY29uIH19XCI+PC9pPlxuICAgIDxzcGFuICpuZ0lmPVwibGFiZWxcIiBjbGFzcz1cInRhYkxhYmVsXCI+e3sgbGFiZWwgfX08L3NwYW4+XG4gICAgPGx4LWJhZGdlICpuZ0lmPVwiY291bnRlclwiIFtzaXplXT1cImNvdW50ZXJCYWRnZVNpemVcIiBbY29udGVudF09XCJjb3VudGVyXCIgW2NvbG9yXT1cIidncmF5J1wiPjwvbHgtYmFkZ2U+XG4gIDwvbmctdGVtcGxhdGU+XG5cbiAgPG5nLXRlbXBsYXRlIGNka1BvcnRhbCAjY29udGVudFRlbXBsYXRlPVwiY2RrUG9ydGFsXCI+XG4gICAgPGRpdiAqbmdJZj1cIiF0YWJMaW5rXCIgY2xhc3M9XCJjb250ZW50XCIgcm9sZT1cInRhYnBhbmVsXCIgW2F0dHIuaWRdPVwidGFiUGFuZWxJZFwiPlxuICAgICAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuICAgIDwvZGl2PlxuICA8L25nLXRlbXBsYXRlPlxuPC9saT5cbiJdfQ==
|
@@ -1,5 +1,5 @@
|
|
1
1
|
import { PortalModule } from '@angular/cdk/portal';
|
2
|
-
import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Input, Output
|
2
|
+
import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Input, Output } from '@angular/core';
|
3
3
|
import { Subject, merge } from 'rxjs';
|
4
4
|
import { map, startWith, switchMap, takeUntil } from 'rxjs/operators';
|
5
5
|
import { TabComponent } from '../tab/tab.component';
|
@@ -12,22 +12,13 @@ export class TabGroupComponent {
|
|
12
12
|
get activeTabPortal() {
|
13
13
|
return this.tabs[this.selectedIndex]?.content;
|
14
14
|
}
|
15
|
-
|
16
|
-
return this.tabs.map((t) => t.tabId);
|
17
|
-
}
|
18
|
-
constructor(cd, ngZone) {
|
15
|
+
constructor(cd) {
|
19
16
|
this.cd = cd;
|
20
|
-
this.ngZone = ngZone;
|
21
17
|
this.isCentered = false;
|
22
18
|
this.selectedIndex = 0;
|
23
|
-
this.ariaTabPattern = 'manualActivation';
|
24
19
|
this.indexChange = new EventEmitter();
|
25
|
-
this.focusedIndex = 0;
|
26
20
|
this.destroyed$ = new Subject();
|
27
21
|
}
|
28
|
-
ngOnInit() {
|
29
|
-
this.focusedIndex = this.selectedIndex;
|
30
|
-
}
|
31
22
|
ngOnChanges(changes) {
|
32
23
|
const tab = this.tabs[this.selectedIndex];
|
33
24
|
if (changes['selectedIndex'] && !changes['selectedIndex'].isFirstChange() && tab) {
|
@@ -38,103 +29,47 @@ export class TabGroupComponent {
|
|
38
29
|
if (this.tabs[0]) {
|
39
30
|
this.tabs[0].noLeftMarginForFirstTab = true;
|
40
31
|
}
|
41
|
-
this.
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
});
|
46
|
-
});
|
32
|
+
const tab = this.tabs[this.selectedIndex];
|
33
|
+
if (tab) {
|
34
|
+
this.switchTo(tab);
|
35
|
+
}
|
47
36
|
const listenToTabsSwitch = (tabs) => merge(...tabs.map((tab) => tab.switch.asObservable().pipe(map(() => tab))));
|
48
37
|
this.tabsQueryList?.changes
|
49
38
|
.pipe(startWith(null), switchMap(() => listenToTabsSwitch(this.tabs)), takeUntil(this.destroyed$))
|
50
39
|
.subscribe((tab) => {
|
51
40
|
this.switchTo(tab);
|
52
41
|
});
|
53
|
-
this.tabs.forEach((tab) => {
|
54
|
-
tab.keyDownAction.pipe(takeUntil(this.destroyed$)).subscribe((event) => {
|
55
|
-
this.handleKeyDown(tab, event);
|
56
|
-
});
|
57
|
-
});
|
58
42
|
}
|
59
43
|
ngOnDestroy() {
|
60
44
|
this.destroyed$.next();
|
61
|
-
this.destroyed$.complete();
|
62
45
|
}
|
63
|
-
|
64
|
-
|
65
|
-
// resync focused index on selected index when leaving tabs
|
66
|
-
this.focusedIndex = this.selectedIndex;
|
67
|
-
}
|
46
|
+
tabIds() {
|
47
|
+
return this.tabs.map((t) => t.tabId);
|
68
48
|
}
|
69
49
|
switchTo(tab) {
|
70
50
|
this.tabsQueryList?.forEach((t, index) => {
|
71
|
-
t.isActive = t
|
51
|
+
t.isActive = t === tab;
|
72
52
|
if (t.isActive) {
|
73
53
|
this.selectedIndex = index;
|
74
|
-
this.focusedIndex = index;
|
75
54
|
this.indexChange.emit(index);
|
76
55
|
}
|
77
56
|
});
|
78
57
|
this.cd.markForCheck();
|
79
58
|
}
|
80
|
-
|
81
|
-
|
82
|
-
case 'ArrowRight':
|
83
|
-
const newTabIndex = (this.focusedIndex + 1) % this.tabs.length;
|
84
|
-
this.navigateToTab(newTabIndex);
|
85
|
-
break;
|
86
|
-
case 'ArrowLeft':
|
87
|
-
const previousTabIndex = this.focusedIndex === 0 ? this.tabs.length - 1 : this.focusedIndex - 1;
|
88
|
-
this.navigateToTab(previousTabIndex);
|
89
|
-
break;
|
90
|
-
case 'Home':
|
91
|
-
event.preventDefault();
|
92
|
-
this.navigateToTab(0);
|
93
|
-
break;
|
94
|
-
case 'End':
|
95
|
-
event.preventDefault();
|
96
|
-
this.navigateToTab(this.tabs.length - 1);
|
97
|
-
break;
|
98
|
-
}
|
99
|
-
}
|
100
|
-
navigateToTab(tabIndex) {
|
101
|
-
const tab = this.tabs[tabIndex];
|
102
|
-
if (!tab) {
|
103
|
-
return;
|
104
|
-
}
|
105
|
-
tab.setFocus();
|
106
|
-
this.focusedIndex = tabIndex;
|
107
|
-
if (this.ariaTabPattern === 'automaticActivation') {
|
108
|
-
tab.select();
|
109
|
-
}
|
110
|
-
}
|
111
|
-
navigateToInitialTab() {
|
112
|
-
const alreadyActivedTabWithLink = this.tabs.find((t) => {
|
113
|
-
return t.tabLink && t.isActive;
|
114
|
-
});
|
115
|
-
// Activated tab by URL takes priority over initial `selectedIndex` input value
|
116
|
-
const initialTab = alreadyActivedTabWithLink || this.tabs[this.selectedIndex];
|
117
|
-
initialTab?.select();
|
118
|
-
}
|
119
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabGroupComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
|
120
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.2", type: TabGroupComponent, isStandalone: true, selector: "lx-tab-group", inputs: { isCentered: "isCentered", selectedIndex: "selectedIndex", ariaTabPattern: "ariaTabPattern" }, outputs: { indexChange: "indexChange" }, queries: [{ propertyName: "tabsQueryList", predicate: TabComponent, descendants: true }], viewQueries: [{ propertyName: "tabListElement", first: true, predicate: ["tabListElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<ul\n #tabListElement\n class=\"tabs\"\n role=\"tablist\"\n (focusout)=\"onFocusOut($event)\"\n [attr.aria-owns]=\"tabIds.join(' ')\"\n [class.centered]=\"isCentered\"\n>\n <ng-content></ng-content>\n</ul>\n\n<ng-template [cdkPortalOutlet]=\"activeTabPortal\"></ng-template>\n", styles: [".tabs{display:inline-block;list-style:none;border-bottom:solid 1px #cfd5df;width:100%;margin:0;padding:0}.tabs.centered{text-align:center}\n"], dependencies: [{ kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i1.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
59
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabGroupComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
60
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.2", type: TabGroupComponent, isStandalone: true, selector: "lx-tab-group", inputs: { isCentered: "isCentered", selectedIndex: "selectedIndex" }, outputs: { indexChange: "indexChange" }, queries: [{ propertyName: "tabsQueryList", predicate: TabComponent }], usesOnChanges: true, ngImport: i0, template: "<ul class=\"tabs\" role=\"tablist\" [attr.aria-owns]=\"tabIds().join(' ')\" [class.centered]=\"isCentered\">\n <ng-content></ng-content>\n</ul>\n\n<ng-template [cdkPortalOutlet]=\"activeTabPortal\"></ng-template>\n", styles: [".tabs{display:inline-block;list-style:none;border-bottom:solid 1px #cfd5df;width:100%;margin:0;padding:0}.tabs.centered{text-align:center}\n"], dependencies: [{ kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i1.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
121
61
|
}
|
122
62
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: TabGroupComponent, decorators: [{
|
123
63
|
type: Component,
|
124
|
-
args: [{ selector: 'lx-tab-group', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [PortalModule], template: "<ul
|
125
|
-
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }
|
64
|
+
args: [{ selector: 'lx-tab-group', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [PortalModule], template: "<ul class=\"tabs\" role=\"tablist\" [attr.aria-owns]=\"tabIds().join(' ')\" [class.centered]=\"isCentered\">\n <ng-content></ng-content>\n</ul>\n\n<ng-template [cdkPortalOutlet]=\"activeTabPortal\"></ng-template>\n", styles: [".tabs{display:inline-block;list-style:none;border-bottom:solid 1px #cfd5df;width:100%;margin:0;padding:0}.tabs.centered{text-align:center}\n"] }]
|
65
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { isCentered: [{
|
126
66
|
type: Input
|
127
67
|
}], selectedIndex: [{
|
128
68
|
type: Input
|
129
|
-
}], ariaTabPattern: [{
|
130
|
-
type: Input
|
131
69
|
}], indexChange: [{
|
132
70
|
type: Output
|
133
71
|
}], tabsQueryList: [{
|
134
72
|
type: ContentChildren,
|
135
|
-
args: [TabComponent
|
136
|
-
}], tabListElement: [{
|
137
|
-
type: ViewChild,
|
138
|
-
args: ['tabListElement']
|
73
|
+
args: [TabComponent]
|
139
74
|
}] } });
|
140
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-group.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/tab-ui/components/tab-group/tab-group.component.ts","../../../../../../../../libs/components/src/lib/tab-ui/components/tab-group/tab-group.component.html"],"names":[],"mappings":"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","sourcesContent":["import { PortalModule } from '@angular/cdk/portal';\nimport {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  QueryList,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { Subject, merge } from 'rxjs';\nimport { map, startWith, switchMap, takeUntil } from 'rxjs/operators';\nimport { TabComponent } from '../tab/tab.component';\n\n@Component({\n  selector: 'lx-tab-group',\n  templateUrl: 'tab-group.component.html',\n  styleUrls: ['tab-group.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [PortalModule]\n})\nexport class TabGroupComponent implements OnInit, OnChanges, AfterContentInit, OnDestroy {\n  @Input() isCentered: boolean = false;\n  @Input() selectedIndex: number = 0;\n  @Input() ariaTabPattern: 'manualActivation' | 'automaticActivation' = 'manualActivation';\n  @Output() indexChange = new EventEmitter<number>();\n\n  @ContentChildren(TabComponent, { descendants: true }) tabsQueryList?: QueryList<TabComponent>;\n  @ViewChild('tabListElement') tabListElement?: ElementRef<HTMLElement>;\n\n  focusedIndex: number = 0;\n\n  private get tabs(): TabComponent[] {\n    return this.tabsQueryList?.toArray() || [];\n  }\n\n  get activeTabPortal() {\n    return this.tabs[this.selectedIndex]?.content;\n  }\n\n  get tabIds(): string[] {\n    return this.tabs.map((t) => t.tabId);\n  }\n\n  readonly destroyed$ = new Subject<void>();\n\n  constructor(\n    private cd: ChangeDetectorRef,\n    private ngZone: NgZone\n  ) {}\n\n  ngOnInit() {\n    this.focusedIndex = this.selectedIndex;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const tab = this.tabs[this.selectedIndex];\n    if (changes['selectedIndex'] && !changes['selectedIndex'].isFirstChange() && tab) {\n      this.switchTo(tab);\n    }\n  }\n\n  ngAfterContentInit() {\n    if (this.tabs[0]) {\n      this.tabs[0].noLeftMarginForFirstTab = true;\n    }\n\n    this.ngZone.runOutsideAngular(() => {\n      // Use setTimeout to let RouterLinkActive enable isActive method (used on tabs)\n      setTimeout(() => {\n        this.navigateToInitialTab();\n      });\n    });\n\n    const listenToTabsSwitch = (tabs: TabComponent[]) => merge(...tabs.map((tab) => tab.switch.asObservable().pipe(map(() => tab))));\n\n    this.tabsQueryList?.changes\n      .pipe(\n        startWith(null),\n        switchMap(() => listenToTabsSwitch(this.tabs)),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((tab) => {\n        this.switchTo(tab);\n      });\n\n    this.tabs.forEach((tab: TabComponent) => {\n      tab.keyDownAction.pipe(takeUntil(this.destroyed$)).subscribe((event) => {\n        this.handleKeyDown(tab, event);\n      });\n    });\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  onFocusOut(event: any) {\n    if (!this.tabListElement?.nativeElement?.contains(event.relatedTarget)) {\n      // resync focused index on selected index when leaving tabs\n      this.focusedIndex = this.selectedIndex;\n    }\n  }\n\n  switchTo(tab: TabComponent) {\n    this.tabsQueryList?.forEach((t, index) => {\n      t.isActive = t.tabId === tab.tabId;\n      if (t.isActive) {\n        this.selectedIndex = index;\n        this.focusedIndex = index;\n        this.indexChange.emit(index);\n      }\n    });\n    this.cd.markForCheck();\n  }\n\n  handleKeyDown(_tab: TabComponent, event: KeyboardEvent) {\n    switch (event.code) {\n      case 'ArrowRight':\n        const newTabIndex = (this.focusedIndex + 1) % this.tabs.length;\n        this.navigateToTab(newTabIndex);\n        break;\n      case 'ArrowLeft':\n        const previousTabIndex = this.focusedIndex === 0 ? this.tabs.length - 1 : this.focusedIndex - 1;\n        this.navigateToTab(previousTabIndex);\n        break;\n      case 'Home':\n        event.preventDefault();\n        this.navigateToTab(0);\n        break;\n      case 'End':\n        event.preventDefault();\n        this.navigateToTab(this.tabs.length - 1);\n        break;\n    }\n  }\n\n  private navigateToTab(tabIndex: number): void {\n    const tab = this.tabs[tabIndex];\n    if (!tab) {\n      return;\n    }\n\n    tab.setFocus();\n    this.focusedIndex = tabIndex;\n\n    if (this.ariaTabPattern === 'automaticActivation') {\n      tab.select();\n    }\n  }\n\n  private navigateToInitialTab(): void {\n    const alreadyActivedTabWithLink = this.tabs.find((t) => {\n      return t.tabLink && t.isActive;\n    });\n\n    // Activated tab by URL takes priority over initial `selectedIndex` input value\n    const initialTab = alreadyActivedTabWithLink || this.tabs[this.selectedIndex];\n\n    initialTab?.select();\n  }\n}\n","<ul\n  #tabListElement\n  class=\"tabs\"\n  role=\"tablist\"\n  (focusout)=\"onFocusOut($event)\"\n  [attr.aria-owns]=\"tabIds.join(' ')\"\n  [class.centered]=\"isCentered\"\n>\n  <ng-content></ng-content>\n</ul>\n\n<ng-template [cdkPortalOutlet]=\"activeTabPortal\"></ng-template>\n"]}
|
75
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFiLWdyb3VwLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9zcmMvbGliL3RhYi11aS9jb21wb25lbnRzL3RhYi1ncm91cC90YWItZ3JvdXAuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL3NyYy9saWIvdGFiLXVpL2NvbXBvbmVudHMvdGFiLWdyb3VwL3RhYi1ncm91cC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDbkQsT0FBTyxFQUVMLHVCQUF1QixFQUV2QixTQUFTLEVBQ1QsZUFBZSxFQUNmLFlBQVksRUFDWixLQUFLLEVBR0wsTUFBTSxFQUdQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQ3RDLE9BQU8sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxTQUFTLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUN0RSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7OztBQVVwRCxNQUFNLE9BQU8saUJBQWlCO0lBTzVCLElBQVksSUFBSTtRQUNkLE9BQU8sSUFBSSxDQUFDLGFBQWEsRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLENBQUM7SUFDN0MsQ0FBQztJQUVELElBQUksZUFBZTtRQUNqQixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxFQUFFLE9BQU8sQ0FBQztJQUNoRCxDQUFDO0lBSUQsWUFBb0IsRUFBcUI7UUFBckIsT0FBRSxHQUFGLEVBQUUsQ0FBbUI7UUFoQmhDLGVBQVUsR0FBWSxLQUFLLENBQUM7UUFDNUIsa0JBQWEsR0FBVyxDQUFDLENBQUM7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QixhQUFhO3NCQUEzQyxlQUFlO3VCQUFDLFlBQVkiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQb3J0YWxNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jZGsvcG9ydGFsJztcbmltcG9ydCB7XG4gIEFmdGVyQ29udGVudEluaXQsXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBDaGFuZ2VEZXRlY3RvclJlZixcbiAgQ29tcG9uZW50LFxuICBDb250ZW50Q2hpbGRyZW4sXG4gIEV2ZW50RW1pdHRlcixcbiAgSW5wdXQsXG4gIE9uQ2hhbmdlcyxcbiAgT25EZXN0cm95LFxuICBPdXRwdXQsXG4gIFF1ZXJ5TGlzdCxcbiAgU2ltcGxlQ2hhbmdlc1xufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFN1YmplY3QsIG1lcmdlIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBtYXAsIHN0YXJ0V2l0aCwgc3dpdGNoTWFwLCB0YWtlVW50aWwgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQgeyBUYWJDb21wb25lbnQgfSBmcm9tICcuLi90YWIvdGFiLmNvbXBvbmVudCc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2x4LXRhYi1ncm91cCcsXG4gIHRlbXBsYXRlVXJsOiAndGFiLWdyb3VwLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJ3RhYi1ncm91cC5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW1BvcnRhbE1vZHVsZV1cbn0pXG5leHBvcnQgY2xhc3MgVGFiR3JvdXBDb21wb25lbnQgaW1wbGVtZW50cyBPbkNoYW5nZXMsIEFmdGVyQ29udGVudEluaXQsIE9uRGVzdHJveSB7XG4gIEBJbnB1dCgpIGlzQ2VudGVyZWQ6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCkgc2VsZWN0ZWRJbmRleDogbnVtYmVyID0gMDtcbiAgQE91dHB1dCgpIGluZGV4Q2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxudW1iZXI+KCk7XG5cbiAgQENvbnRlbnRDaGlsZHJlbihUYWJDb21wb25lbnQpIHRhYnNRdWVyeUxpc3Q/OiBRdWVyeUxpc3Q8VGFiQ29tcG9uZW50PjtcblxuICBwcml2YXRlIGdldCB0YWJzKCk6IFRhYkNvbXBvbmVudFtdIHtcbiAgICByZXR1cm4gdGhpcy50YWJzUXVlcnlMaXN0Py50b0FycmF5KCkgfHwgW107XG4gIH1cblxuICBnZXQgYWN0aXZlVGFiUG9ydGFsKCkge1xuICAgIHJldHVybiB0aGlzLnRhYnNbdGhpcy5zZWxlY3RlZEluZGV4XT8uY29udGVudDtcbiAgfVxuXG4gIHJlYWRvbmx5IGRlc3Ryb3llZCQgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgY2Q6IENoYW5nZURldGVjdG9yUmVmKSB7fVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpIHtcbiAgICBjb25zdCB0YWIgPSB0aGlzLnRhYnNbdGhpcy5zZWxlY3RlZEluZGV4XTtcbiAgICBpZiAoY2hhbmdlc1snc2VsZWN0ZWRJbmRleCddICYmICFjaGFuZ2VzWydzZWxlY3RlZEluZGV4J10uaXNGaXJzdENoYW5nZSgpICYmIHRhYikge1xuICAgICAgdGhpcy5zd2l0Y2hUbyh0YWIpO1xuICAgIH1cbiAgfVxuXG4gIG5nQWZ0ZXJDb250ZW50SW5pdCgpIHtcbiAgICBpZiAodGhpcy50YWJzWzBdKSB7XG4gICAgICB0aGlzLnRhYnNbMF0ubm9MZWZ0TWFyZ2luRm9yRmlyc3RUYWIgPSB0cnVlO1xuICAgIH1cbiAgICBjb25zdCB0YWIgPSB0aGlzLnRhYnNbdGhpcy5zZWxlY3RlZEluZGV4XTtcbiAgICBpZiAodGFiKSB7XG4gICAgICB0aGlzLnN3aXRjaFRvKHRhYik7XG4gICAgfVxuXG4gICAgY29uc3QgbGlzdGVuVG9UYWJzU3dpdGNoID0gKHRhYnM6IFRhYkNvbXBvbmVudFtdKSA9PiBtZXJnZSguLi50YWJzLm1hcCgodGFiKSA9PiB0YWIuc3dpdGNoLmFzT2JzZXJ2YWJsZSgpLnBpcGUobWFwKCgpID0+IHRhYikpKSk7XG5cbiAgICB0aGlzLnRhYnNRdWVyeUxpc3Q/LmNoYW5nZXNcbiAgICAgIC5waXBlKFxuICAgICAgICBzdGFydFdpdGgobnVsbCksXG4gICAgICAgIHN3aXRjaE1hcCgoKSA9PiBsaXN0ZW5Ub1RhYnNTd2l0Y2godGhpcy50YWJzKSksXG4gICAgICAgIHRha2VVbnRpbCh0aGlzLmRlc3Ryb3llZCQpXG4gICAgICApXG4gICAgICAuc3Vic2NyaWJlKCh0YWIpID0+IHtcbiAgICAgICAgdGhpcy5zd2l0Y2hUbyh0YWIpO1xuICAgICAgfSk7XG4gIH1cblxuICBuZ09uRGVzdHJveSgpIHtcbiAgICB0aGlzLmRlc3Ryb3llZCQubmV4dCgpO1xuICB9XG5cbiAgdGFiSWRzKCk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gdGhpcy50YWJzLm1hcCgodCkgPT4gdC50YWJJZCk7XG4gIH1cblxuICBzd2l0Y2hUbyh0YWI6IFRhYkNvbXBvbmVudCkge1xuICAgIHRoaXMudGFic1F1ZXJ5TGlzdD8uZm9yRWFjaCgodCwgaW5kZXgpID0+IHtcbiAgICAgIHQuaXNBY3RpdmUgPSB0ID09PSB0YWI7XG4gICAgICBpZiAodC5pc0FjdGl2ZSkge1xuICAgICAgICB0aGlzLnNlbGVjdGVkSW5kZXggPSBpbmRleDtcbiAgICAgICAgdGhpcy5pbmRleENoYW5nZS5lbWl0KGluZGV4KTtcbiAgICAgIH1cbiAgICB9KTtcbiAgICB0aGlzLmNkLm1hcmtGb3JDaGVjaygpO1xuICB9XG59XG4iLCI8dWwgY2xhc3M9XCJ0YWJzXCIgcm9sZT1cInRhYmxpc3RcIiBbYXR0ci5hcmlhLW93bnNdPVwidGFiSWRzKCkuam9pbignICcpXCIgW2NsYXNzLmNlbnRlcmVkXT1cImlzQ2VudGVyZWRcIj5cbiAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuPC91bD5cblxuPG5nLXRlbXBsYXRlIFtjZGtQb3J0YWxPdXRsZXRdPVwiYWN0aXZlVGFiUG9ydGFsXCI+PC9uZy10ZW1wbGF0ZT5cbiJdfQ==
|