@leanix/components 0.4.472 → 0.4.473
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/core-ui/components/spinner/spinner.component.mjs +4 -4
- package/esm2022/lib/core-ui/components/tokenizer/tokenizer.component.mjs +40 -1
- package/esm2022/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.mjs +4 -2
- package/esm2022/lib/forms-ui/components/picker/picker.component.mjs +4 -2
- package/esm2022/lib/forms-ui/components/single-select/single-select.component.mjs +12 -1
- package/esm2022/lib/tab-ui/components/tab-group/tab-group.component.mjs +24 -1
- package/fesm2022/leanix-components.mjs +82 -5
- package/fesm2022/leanix-components.mjs.map +1 -1
- package/lib/core-ui/components/spinner/spinner.component.d.ts +3 -3
- package/lib/core-ui/components/tokenizer/tokenizer.component.d.ts +39 -0
- package/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.d.ts +3 -1
- package/lib/forms-ui/components/picker/picker.component.d.ts +3 -1
- package/lib/forms-ui/components/single-select/single-select.component.d.ts +11 -0
- package/lib/tab-ui/components/tab-group/tab-group.component.d.ts +23 -0
- package/package.json +1 -1
@@ -2,10 +2,10 @@ import { Component, HostBinding, Input } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
3
3
|
/**
|
4
4
|
* The following CSS classes should be used to style this component. The use of the inputs is deprecated.
|
5
|
-
* - fullSpace
|
6
|
-
* - fullSpaceFixed
|
5
|
+
* - `fullSpace`: Sets position to absolute with full height and width => fills ancestor with position set.
|
6
|
+
* - `fullSpaceFixed`: Sets position to fixed with full height and width => fills window
|
7
7
|
* (or ancestor with new rendering context, created, for example, with `transform: translate(0, 0)`
|
8
|
-
* - fadeBackground
|
8
|
+
* - `fadeBackground`: Sets background to 60% transparent white. TODO: Always use for fullSpace* classes and remove.
|
9
9
|
*/
|
10
10
|
export class SpinnerComponent {
|
11
11
|
constructor() {
|
@@ -23,4 +23,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
23
23
|
}, {
|
24
24
|
type: Input
|
25
25
|
}] } });
|
26
|
-
//# sourceMappingURL=data:application/json;base64,
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3Bpbm5lci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2NvbXBvbmVudHMvc3JjL2xpYi9jb3JlLXVpL2NvbXBvbmVudHMvc3Bpbm5lci9zcGlubmVyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9zcmMvbGliL2NvcmUtdWkvY29tcG9uZW50cy9zcGlubmVyL3NwaW5uZXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxXQUFXLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUU5RDs7Ozs7O0dBTUc7QUFPSCxNQUFNLE9BQU8sZ0JBQWdCO0lBTjdCO1FBT2dFLG1CQUFjLEdBQVksS0FBSyxDQUFDO0tBQy9GOzhHQUZZLGdCQUFnQjtrR0FBaEIsZ0JBQWdCLHNNQ2Y3Qiw2V0FjQTs7MkZEQ2EsZ0JBQWdCO2tCQU41QixTQUFTOytCQUNFLFlBQVksY0FHVixJQUFJOzhCQUc4QyxjQUFjO3NCQUEzRSxXQUFXO3VCQUFDLCtCQUErQjs7c0JBQUcsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSG9zdEJpbmRpbmcsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogVGhlIGZvbGxvd2luZyBDU1MgY2xhc3NlcyBzaG91bGQgYmUgdXNlZCB0byBzdHlsZSB0aGlzIGNvbXBvbmVudC4gVGhlIHVzZSBvZiB0aGUgaW5wdXRzIGlzIGRlcHJlY2F0ZWQuXG4gKiAtIGBmdWxsU3BhY2VgOiBTZXRzIHBvc2l0aW9uIHRvIGFic29sdXRlIHdpdGggZnVsbCBoZWlnaHQgYW5kIHdpZHRoID0+IGZpbGxzIGFuY2VzdG9yIHdpdGggcG9zaXRpb24gc2V0LlxuICogLSBgZnVsbFNwYWNlRml4ZWRgOiBTZXRzIHBvc2l0aW9uIHRvIGZpeGVkIHdpdGggZnVsbCBoZWlnaHQgYW5kIHdpZHRoID0+IGZpbGxzIHdpbmRvd1xuICogICAob3IgYW5jZXN0b3Igd2l0aCBuZXcgcmVuZGVyaW5nIGNvbnRleHQsIGNyZWF0ZWQsIGZvciBleGFtcGxlLCB3aXRoIGB0cmFuc2Zvcm06IHRyYW5zbGF0ZSgwLCAwKWBcbiAqIC0gYGZhZGVCYWNrZ3JvdW5kYDogU2V0cyBiYWNrZ3JvdW5kIHRvIDYwJSB0cmFuc3BhcmVudCB3aGl0ZS4gVE9ETzogQWx3YXlzIHVzZSBmb3IgZnVsbFNwYWNlKiBjbGFzc2VzIGFuZCByZW1vdmUuXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2x4LXNwaW5uZXInLFxuICB0ZW1wbGF0ZVVybDogJ3NwaW5uZXIuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnc3Bpbm5lci5jb21wb25lbnQuc2NzcyddLFxuICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIFNwaW5uZXJDb21wb25lbnQge1xuICBASG9zdEJpbmRpbmcoJ2NsYXNzLmZhZGVCYWNrZ3JvdW5kRnJvbUlucHV0JykgQElucHV0KCkgcHVibGljIGZhZGVCYWNrZ3JvdW5kOiBib29sZWFuID0gZmFsc2U7XG59XG4iLCI8ZGl2IGNsYXNzPVwiY29udGFpbmVyXCI+XG4gIDxkaXYgY2xhc3M9XCJtaWRkbGVcIj5cbiAgICA8ZGl2IGNsYXNzPVwiY2VudGVyXCI+XG4gICAgICA8ZGl2IGNsYXNzPVwic3Bpbm5lclwiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwicmVjdDFcIj48L2Rpdj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInJlY3QyXCI+PC9kaXY+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJyZWN0M1wiPjwvZGl2PlxuICAgICAgICA8ZGl2IGNsYXNzPVwicmVjdDRcIj48L2Rpdj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInJlY3Q1XCI+PC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICAgIDxuZy1jb250ZW50IC8+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuPC9kaXY+XG4iXX0=
|
@@ -10,6 +10,45 @@ import { TokenComponent } from './token/token.component';
|
|
10
10
|
import * as i0 from "@angular/core";
|
11
11
|
import * as i1 from "../../services/resize-observer.service";
|
12
12
|
import * as i2 from "@ngx-translate/core";
|
13
|
+
/**
|
14
|
+
* The tokenizer component (`lx-tokenizer`) is responsible for rendering a list of items into a container and hiding those that do not fit behind a placeholder. The items are represented as `lx-token`. Each `lx-token` allows to project any content into it, so anything can become a token. The tokenizer calculates how many items fit into the container (based on width), hiding those items that are overflowing and showing a overflow placeholder instead (e.g. "+4").
|
15
|
+
*
|
16
|
+
* ## Usage
|
17
|
+
*
|
18
|
+
* 1. Import `TokenizerComponent` & `TokenComponent` into your module or component.
|
19
|
+
* 2. Use the component in your template. All items should be wrapped in a `lx-token` element for the tokenizer to identify the individual tokens.
|
20
|
+
*
|
21
|
+
* ```ts
|
22
|
+
* import { TokenizerComponent } from '@leanix/components';
|
23
|
+
*
|
24
|
+
* @Component({
|
25
|
+
* selector: 'lx-my-component',
|
26
|
+
* standalone: true,
|
27
|
+
* imports: [TokenizerComponent, TokenComponent],
|
28
|
+
* template: `
|
29
|
+
* <lx-tokenizer>
|
30
|
+
* <lx-token>Token One</lx-token>
|
31
|
+
* <lx-token>Token Two</lx-token>
|
32
|
+
* </lx-tokenizer>
|
33
|
+
* `
|
34
|
+
* })
|
35
|
+
* export class MyComponent {}
|
36
|
+
* ```
|
37
|
+
*
|
38
|
+
* ## API
|
39
|
+
*
|
40
|
+
* ### Inputs
|
41
|
+
*
|
42
|
+
* - `active`: Can be used to activate & deactivate tokenization
|
43
|
+
*
|
44
|
+
* ### Outputs
|
45
|
+
*
|
46
|
+
* - `overflowPlaceholderClick`: Fires when the user clicks on the overflow placeholder
|
47
|
+
*
|
48
|
+
* ### Content projection
|
49
|
+
*
|
50
|
+
* - `lx-token`: All `lx-tokens` will be projected into the tokenizer
|
51
|
+
*/
|
13
52
|
export class TokenizerComponent {
|
14
53
|
constructor(elementRef, destroyRef, changeDetectorRef, resizeObserverService) {
|
15
54
|
this.elementRef = elementRef;
|
@@ -104,4 +143,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
104
143
|
type: ViewChild,
|
105
144
|
args: ['tokensContainer', { read: ElementRef }]
|
106
145
|
}] } });
|
107
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokenizer.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/core-ui/components/tokenizer/tokenizer.component.ts","../../../../../../../../libs/components/src/lib/core-ui/components/tokenizer/tokenizer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgIf } from '@angular/common';\nimport {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  computed,\n  ContentChildren,\n  DestroyRef,\n  effect,\n  ElementRef,\n  EventEmitter,\n  input,\n  Output,\n  QueryList,\n  signal,\n  ViewChild\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { combineLatest } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, filter, map, startWith } from 'rxjs/operators';\nimport { ButtonComponent } from '../../components/button/button.component';\nimport { CounterComponent } from '../../components/counter/counter.component';\nimport { ResizeObserverService } from '../../services/resize-observer.service';\nimport { TokenComponent } from './token/token.component';\n\n@Component({\n  selector: 'lx-tokenizer',\n  standalone: true,\n  templateUrl: './tokenizer.component.html',\n  styleUrls: ['./tokenizer.component.scss'],\n  imports: [NgIf, CounterComponent, ButtonComponent, TranslateModule],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class TokenizerComponent implements AfterContentInit {\n  readonly NAME = 'TokenizerComponent';\n\n  protected overflowItems = signal<TokenComponent[]>([]);\n  protected overflowItemCount = computed(() => this.overflowItems().length);\n  protected showCounter = computed(() => this.active() && this.overflowItemCount() > 0);\n\n  active = input(true);\n  private active$ = toObservable(this.active);\n  private overflowStartIndex = signal<number | null>(null);\n\n  /**\n   * Emitted when the counter button is clicked\n   * @param startIndex The index of the first overflowing element\n   * @param counterElement The counter button element\n   */\n  @Output() overflowPlaceholderClick = new EventEmitter<{ startIndex: number; counterElement: ElementRef<HTMLElement> }>();\n\n  @ContentChildren(TokenComponent) items?: QueryList<TokenComponent>;\n  @ViewChild('counter', { read: ElementRef }) counter?: ElementRef<HTMLElement>;\n  @ViewChild('tokensContainer', { read: ElementRef }) tokensContainer!: ElementRef<HTMLElement>;\n\n  constructor(\n    private elementRef: ElementRef<HTMLElement>,\n    private destroyRef: DestroyRef,\n    private changeDetectorRef: ChangeDetectorRef,\n    private resizeObserverService: ResizeObserverService\n  ) {\n    effect(() => {\n      // Whenever the tokenizer is disabled, ensure all tokens are shown\n      if (!this.active()) {\n        this.items!.forEach((item) => item.show());\n      }\n    });\n  }\n\n  ngAfterContentInit() {\n    const containerWidth = this.resizeObserverService.createResizeObservable(this.elementRef.nativeElement).pipe(\n      map((entry) => entry.contentRect.width ?? 0),\n      debounceTime(16),\n      distinctUntilChanged()\n    );\n    const itemsChanged$ = this.items!.changes;\n\n    combineLatest([\n      this.active$,\n      containerWidth.pipe(startWith(this.elementRef.nativeElement.getBoundingClientRect().width)),\n      itemsChanged$.pipe(startWith(this.items))\n    ])\n      .pipe(\n        filter(([active]) => active),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe(([, containerWidth]) => {\n        this.tokenize(containerWidth);\n      });\n  }\n\n  onCounterClicked(): void {\n    const startIndex = this.overflowStartIndex();\n    if (typeof startIndex === 'number') {\n      this.overflowPlaceholderClick.emit({ startIndex, counterElement: this.counter! });\n    }\n  }\n\n  private tokenize(containerWidth: number): void {\n    if (!this.items) {\n      return;\n    }\n\n    this.items.forEach((item) => item.hide());\n    this.overflowItems.set([]);\n    this.changeDetectorRef.detectChanges();\n\n    let index = 0;\n    for (const item of this.items) {\n      item.show();\n      const usedWidth = this.tokensContainer.nativeElement.getBoundingClientRect().width;\n      if (usedWidth > containerWidth) {\n        item.hide();\n        this.updateOverflowCounter(index);\n\n        const usedWidthIncludingCounter = this.tokensContainer.nativeElement.getBoundingClientRect().width;\n        if (usedWidthIncludingCounter > containerWidth && index >= 1) {\n          const previousItem = this.items.toArray()[index - 1];\n          previousItem?.hide();\n          this.updateOverflowCounter(index - 1);\n        }\n\n        break;\n      }\n      index++;\n    }\n  }\n\n  private updateOverflowCounter(index: number) {\n    this.overflowStartIndex.set(index);\n\n    const overflowingItems = this.items?.toArray().slice(index) ?? [];\n    this.overflowItems.set(overflowingItems);\n    this.changeDetectorRef.detectChanges();\n  }\n}\n","<div #tokensContainer class=\"tokensContainer\">\n  <ng-content select=\"lx-token\" />\n  @if (showCounter()) {\n    <button\n      lx-button\n      class=\"counterButton\"\n      mode=\"ghost\"\n      (click)=\"onCounterClicked()\"\n      [attr.aria-label]=\"\n        NAME + (overflowItemCount() === 1 ? '.showMoreItem' : '.showMoreItems') | translate: { count: overflowItemCount() }\n      \"\n    >\n      <lx-counter #counter aria-hidden=\"true\" [content]=\"'+' + overflowItemCount()\" color=\"gray\" />\n    </button>\n  }\n</div>\n"]}
|
146
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokenizer.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/core-ui/components/tokenizer/tokenizer.component.ts","../../../../../../../../libs/components/src/lib/core-ui/components/tokenizer/tokenizer.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AACvC,OAAO,EAEL,uBAAuB,EAEvB,SAAS,EACT,QAAQ,EACR,eAAe,EAEf,MAAM,EACN,UAAU,EACV,YAAY,EACZ,KAAK,EACL,MAAM,EAEN,MAAM,EACN,SAAS,EACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAC;AAC9E,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,MAAM,CAAC;AACrC,OAAO,EAAE,YAAY,EAAE,oBAAoB,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAC5F,OAAO,EAAE,eAAe,EAAE,MAAM,0CAA0C,CAAC;AAC3E,OAAO,EAAE,gBAAgB,EAAE,MAAM,4CAA4C,CAAC;AAE9E,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;;;;AAEzD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCG;AASH,MAAM,OAAO,kBAAkB;IAsB7B,YACU,UAAmC,EACnC,UAAsB,EACtB,iBAAoC,EACpC,qBAA4C;QAH5C,eAAU,GAAV,UAAU,CAAyB;QACnC,eAAU,GAAV,UAAU,CAAY;QACtB,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,0BAAqB,GAArB,qBAAqB,CAAuB;QAzB7C,SAAI,GAAG,oBAAoB,CAAC;QAE3B,kBAAa,GAAG,MAAM,CAAmB,EAAE,CAAC,CAAC;QAC7C,sBAAiB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC,CAAC;QAChE,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,IAAI,CAAC,iBAAiB,EAAE,GAAG,CAAC,CAAC,CAAC;QAEtF,WAAM,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC;QACb,YAAO,GAAG,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACpC,uBAAkB,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;QAEzD;;;;WAIG;QACO,6BAAwB,GAAG,IAAI,YAAY,EAAmE,CAAC;QAYvH,MAAM,CAAC,GAAG,EAAE;YACV,kEAAkE;YAClE,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;gBACnB,IAAI,CAAC,KAAM,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;YAC7C,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,kBAAkB;QAChB,MAAM,cAAc,GAAG,IAAI,CAAC,qBAAqB,CAAC,sBAAsB,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,IAAI,CAC1G,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,WAAW,CAAC,KAAK,IAAI,CAAC,CAAC,EAC5C,YAAY,CAAC,EAAE,CAAC,EAChB,oBAAoB,EAAE,CACvB,CAAC;QACF,MAAM,aAAa,GAAG,IAAI,CAAC,KAAM,CAAC,OAAO,CAAC;QAE1C,aAAa,CAAC;YACZ,IAAI,CAAC,OAAO;YACZ,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC,KAAK,CAAC,CAAC;YAC3F,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;SAC1C,CAAC;aACC,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,EAC5B,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CACpC;aACA,SAAS,CAAC,CAAC,CAAC,EAAE,cAAc,CAAC,EAAE,EAAE;YAChC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;IACP,CAAC;IAED,gBAAgB;QACd,MAAM,UAAU,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC7C,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE,CAAC;YACnC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,EAAE,UAAU,EAAE,cAAc,EAAE,IAAI,CAAC,OAAQ,EAAE,CAAC,CAAC;QACpF,CAAC;IACH,CAAC;IAEO,QAAQ,CAAC,cAAsB;QACrC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAChB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAC1C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC3B,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QAEvC,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YAC9B,IAAI,CAAC,IAAI,EAAE,CAAC;YACZ,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC,KAAK,CAAC;YACnF,IAAI,SAAS,GAAG,cAAc,EAAE,CAAC;gBAC/B,IAAI,CAAC,IAAI,EAAE,CAAC;gBACZ,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;gBAElC,MAAM,yBAAyB,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC,KAAK,CAAC;gBACnG,IAAI,yBAAyB,GAAG,cAAc,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC;oBAC7D,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;oBACrD,YAAY,EAAE,IAAI,EAAE,CAAC;oBACrB,IAAI,CAAC,qBAAqB,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;gBACxC,CAAC;gBAED,MAAM;YACR,CAAC;YACD,KAAK,EAAE,CAAC;QACV,CAAC;IACH,CAAC;IAEO,qBAAqB,CAAC,KAAa;QACzC,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAEnC,MAAM,gBAAgB,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QAClE,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QACzC,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;IACzC,CAAC;8GArGU,kBAAkB;kGAAlB,kBAAkB,wSAkBZ,cAAc,4GACD,UAAU,6GACF,UAAU,6BC9FlD,6hBAgBA,wQDuDkB,gBAAgB,6FAAE,eAAe,yLAAE,eAAe;;2FAGvD,kBAAkB;kBAR9B,SAAS;+BACE,cAAc,cACZ,IAAI,WAGP,CAAC,IAAI,EAAE,gBAAgB,EAAE,eAAe,EAAE,eAAe,CAAC,mBAClD,uBAAuB,CAAC,MAAM;4KAkBrC,wBAAwB;sBAAjC,MAAM;gBAE0B,KAAK;sBAArC,eAAe;uBAAC,cAAc;gBACa,OAAO;sBAAlD,SAAS;uBAAC,SAAS,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE;gBACU,eAAe;sBAAlE,SAAS;uBAAC,iBAAiB,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE","sourcesContent":["import { NgIf } from '@angular/common';\nimport {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  computed,\n  ContentChildren,\n  DestroyRef,\n  effect,\n  ElementRef,\n  EventEmitter,\n  input,\n  Output,\n  QueryList,\n  signal,\n  ViewChild\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { combineLatest } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, filter, map, startWith } from 'rxjs/operators';\nimport { ButtonComponent } from '../../components/button/button.component';\nimport { CounterComponent } from '../../components/counter/counter.component';\nimport { ResizeObserverService } from '../../services/resize-observer.service';\nimport { TokenComponent } from './token/token.component';\n\n/**\n * The tokenizer component (`lx-tokenizer`) is responsible for rendering a list of items into a container and hiding those that do not fit behind a placeholder. The items are represented as `lx-token`. Each `lx-token` allows to project any content into it, so anything can become a token. The tokenizer calculates how many items fit into the container (based on width), hiding those items that are overflowing and showing a overflow placeholder instead (e.g. \"+4\").\n *\n * ## Usage\n *\n * 1. Import `TokenizerComponent` & `TokenComponent` into your module or component.\n * 2. Use the component in your template. All items should be wrapped in a `lx-token` element for the tokenizer to identify the individual tokens.\n *\n * ```ts\n * import { TokenizerComponent } from '@leanix/components';\n *\n * @Component({\n *   selector: 'lx-my-component',\n *   standalone: true,\n *   imports: [TokenizerComponent, TokenComponent],\n *   template: `\n *     <lx-tokenizer>\n *       <lx-token>Token One</lx-token>\n *       <lx-token>Token Two</lx-token>\n *     </lx-tokenizer>\n *   `\n * })\n * export class MyComponent {}\n * ```\n *\n * ## API\n *\n * ### Inputs\n *\n * - `active`: Can be used to activate & deactivate tokenization\n *\n * ### Outputs\n *\n * - `overflowPlaceholderClick`: Fires when the user clicks on the overflow placeholder\n *\n * ### Content projection\n *\n * - `lx-token`: All `lx-tokens` will be projected into the tokenizer\n */\n@Component({\n  selector: 'lx-tokenizer',\n  standalone: true,\n  templateUrl: './tokenizer.component.html',\n  styleUrls: ['./tokenizer.component.scss'],\n  imports: [NgIf, CounterComponent, ButtonComponent, TranslateModule],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class TokenizerComponent implements AfterContentInit {\n  readonly NAME = 'TokenizerComponent';\n\n  protected overflowItems = signal<TokenComponent[]>([]);\n  protected overflowItemCount = computed(() => this.overflowItems().length);\n  protected showCounter = computed(() => this.active() && this.overflowItemCount() > 0);\n\n  active = input(true);\n  private active$ = toObservable(this.active);\n  private overflowStartIndex = signal<number | null>(null);\n\n  /**\n   * Emitted when the counter button is clicked\n   * @param startIndex The index of the first overflowing element\n   * @param counterElement The counter button element\n   */\n  @Output() overflowPlaceholderClick = new EventEmitter<{ startIndex: number; counterElement: ElementRef<HTMLElement> }>();\n\n  @ContentChildren(TokenComponent) items?: QueryList<TokenComponent>;\n  @ViewChild('counter', { read: ElementRef }) counter?: ElementRef<HTMLElement>;\n  @ViewChild('tokensContainer', { read: ElementRef }) tokensContainer!: ElementRef<HTMLElement>;\n\n  constructor(\n    private elementRef: ElementRef<HTMLElement>,\n    private destroyRef: DestroyRef,\n    private changeDetectorRef: ChangeDetectorRef,\n    private resizeObserverService: ResizeObserverService\n  ) {\n    effect(() => {\n      // Whenever the tokenizer is disabled, ensure all tokens are shown\n      if (!this.active()) {\n        this.items!.forEach((item) => item.show());\n      }\n    });\n  }\n\n  ngAfterContentInit() {\n    const containerWidth = this.resizeObserverService.createResizeObservable(this.elementRef.nativeElement).pipe(\n      map((entry) => entry.contentRect.width ?? 0),\n      debounceTime(16),\n      distinctUntilChanged()\n    );\n    const itemsChanged$ = this.items!.changes;\n\n    combineLatest([\n      this.active$,\n      containerWidth.pipe(startWith(this.elementRef.nativeElement.getBoundingClientRect().width)),\n      itemsChanged$.pipe(startWith(this.items))\n    ])\n      .pipe(\n        filter(([active]) => active),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe(([, containerWidth]) => {\n        this.tokenize(containerWidth);\n      });\n  }\n\n  onCounterClicked(): void {\n    const startIndex = this.overflowStartIndex();\n    if (typeof startIndex === 'number') {\n      this.overflowPlaceholderClick.emit({ startIndex, counterElement: this.counter! });\n    }\n  }\n\n  private tokenize(containerWidth: number): void {\n    if (!this.items) {\n      return;\n    }\n\n    this.items.forEach((item) => item.hide());\n    this.overflowItems.set([]);\n    this.changeDetectorRef.detectChanges();\n\n    let index = 0;\n    for (const item of this.items) {\n      item.show();\n      const usedWidth = this.tokensContainer.nativeElement.getBoundingClientRect().width;\n      if (usedWidth > containerWidth) {\n        item.hide();\n        this.updateOverflowCounter(index);\n\n        const usedWidthIncludingCounter = this.tokensContainer.nativeElement.getBoundingClientRect().width;\n        if (usedWidthIncludingCounter > containerWidth && index >= 1) {\n          const previousItem = this.items.toArray()[index - 1];\n          previousItem?.hide();\n          this.updateOverflowCounter(index - 1);\n        }\n\n        break;\n      }\n      index++;\n    }\n  }\n\n  private updateOverflowCounter(index: number) {\n    this.overflowStartIndex.set(index);\n\n    const overflowingItems = this.items?.toArray().slice(index) ?? [];\n    this.overflowItems.set(overflowingItems);\n    this.changeDetectorRef.detectChanges();\n  }\n}\n","<div #tokensContainer class=\"tokensContainer\">\n  <ng-content select=\"lx-token\" />\n  @if (showCounter()) {\n    <button\n      lx-button\n      class=\"counterButton\"\n      mode=\"ghost\"\n      (click)=\"onCounterClicked()\"\n      [attr.aria-label]=\"\n        NAME + (overflowItemCount() === 1 ? '.showMoreItem' : '.showMoreItems') | translate: { count: overflowItemCount() }\n      \"\n    >\n      <lx-counter #counter aria-hidden=\"true\" [content]=\"'+' + overflowItemCount()\" color=\"gray\" />\n    </button>\n  }\n</div>\n"]}
|
package/esm2022/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.mjs
CHANGED
@@ -14,6 +14,8 @@ import * as i1 from "ngx-infinite-scroll";
|
|
14
14
|
import * as i2 from "@ngx-translate/core";
|
15
15
|
/**
|
16
16
|
* Usage:
|
17
|
+
*
|
18
|
+
* ```
|
17
19
|
* <lx-single-select
|
18
20
|
* #singleSelect
|
19
21
|
* [selection]="value"
|
@@ -28,7 +30,7 @@ import * as i2 from "@ngx-translate/core";
|
|
28
30
|
* (onItemSelected)="singleSelect.selectOption($event)">
|
29
31
|
* </lx-option-group-dropdown>
|
30
32
|
* </lx-single-select>
|
31
|
-
*
|
33
|
+
* ```
|
32
34
|
*/
|
33
35
|
export class OptionGroupDropdownComponent extends KeyboardSelectDirective {
|
34
36
|
constructor() {
|
@@ -125,4 +127,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
125
127
|
type: ViewChild,
|
126
128
|
args: ['selectOrigin', { static: false }]
|
127
129
|
}] } });
|
128
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"option-group-dropdown.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkConnectedOverlay, CdkOverlayOrigin, ConnectedOverlayPositionChange } from '@angular/cdk/overlay';\nimport { AsyncPipe, NgClass, NgFor, NgIf, NgTemplateOutlet, UpperCasePipe } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n  forwardRef\n} from '@angular/core';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { InfiniteScrollModule } from 'ngx-infinite-scroll';\nimport { CounterComponent } from '../../../core-ui/components/counter/counter.component';\nimport { SpinnerComponent } from '../../../core-ui/components/spinner/spinner.component';\nimport { SelectableItemDirective } from '../../directives/selectable-item.directive';\nimport { BasicDropdownItemComponent } from '../basic-dropdown-item/basic-dropdown-item.component';\nimport { BasicDropdownComponent } from '../basic-dropdown/basic-dropdown.component';\nimport { KeyboardSelectDirective } from '../keyboard-select.directive';\n\nexport type OptionGroupOption = { value: string; label: string };\n/**\n * TODO: rename to something unspecific for single selects because multiselects are also supported\n */\nexport type SingleSelectOptionGroup<T = any> = {\n  label: string;\n  options: (OptionGroupOption | T)[];\n};\n\n/**\n * Usage:\n * <lx-single-select\n *    #singleSelect\n *    [selection]=\"value\"\n *    (selectionChange)=\"onSelectionChange($event)\">\n *   <span class=\"selectedOption\">\n *     {{ value.label }}\n *   </span>\n *   <lx-option-group-dropdown\n *     class=\"dropdownComponent\"\n *     [keyboardSelectAction]=\"singleSelect.optionsKeyboardSelectAction$\"\n *     [optionGroups]=\"optionGroups\"\n *     (onItemSelected)=\"singleSelect.selectOption($event)\">\n *    </lx-option-group-dropdown>\n * </lx-single-select>\n *\n */\n@Component({\n  selector: 'lx-option-group-dropdown',\n  templateUrl: 'option-group-dropdown.component.html',\n  styleUrls: ['option-group-dropdown.component.scss'],\n  providers: [{ provide: KeyboardSelectDirective, useExisting: forwardRef(() => OptionGroupDropdownComponent) }],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgIf,\n    CdkOverlayOrigin,\n    CdkConnectedOverlay,\n    NgClass,\n    NgTemplateOutlet,\n    InfiniteScrollModule,\n    SelectableItemDirective,\n    CounterComponent,\n    NgFor,\n    BasicDropdownItemComponent,\n    SpinnerComponent,\n    AsyncPipe,\n    UpperCasePipe,\n    TranslateModule\n  ]\n})\nexport class OptionGroupDropdownComponent extends KeyboardSelectDirective {\n  readonly NAME = 'OptionGroupDropdownComponent';\n  @Input() optionGroups: SingleSelectOptionGroup[] = [];\n  /**\n   * Usually a stream of the current input value of the parent select.\n   */\n  @Input() newOptionLabel?: string;\n  @Input() highlightTerm?: string;\n  @Input() showCreateNewOption: boolean = false;\n  @Input() labelKey: string = 'label';\n  @Input() loading: boolean = false;\n  @Input() trackByProperty?: string;\n  @Input() showNoResultsIfGroupIsEmpty: boolean = false;\n  @Input() override overlayPositioning: boolean = false;\n\n  @Output() onItemSelected = new EventEmitter<OptionGroupOption | any>();\n  @Output() containerScrolled = new EventEmitter();\n  @Output() createNewOption = new EventEmitter<string>();\n\n  @ContentChild('optionTemplate') optionTemplateRef?: TemplateRef<any>;\n  /**\n   * If you provide a <ng-template #noResultsOptionTemplateRef></ng-template> inside the <lx-option-group-dropdown>\n   * it will be used to display the \"no results\" option instead of the default \"No results\" text.\n   * This way you can easily define different \"No results\" texts for different groups too.\n   */\n  @ContentChild('noResultsOptionTemplateRef') noResultsOptionTemplateRef?: TemplateRef<any>;\n  @ViewChild('selectOrigin', { static: false }) selectOrigin!: CdkOverlayOrigin;\n  isTopDropdown = false;\n\n  public get isNewItem() {\n    const options =\n      this.optionGroups?.reduce<(OptionGroupOption | any)[]>((flatOptions, optionGroup) => flatOptions.concat(optionGroup.options), []) ??\n      [];\n    return BasicDropdownComponent.isNewItem(options, this.newOptionLabel ?? '', this.labelKey);\n  }\n\n  selectOption(option: OptionGroupOption | any) {\n    this.onItemSelected.emit(option);\n  }\n\n  trackByLabel(index: number, optionGroup: SingleSelectOptionGroup) {\n    return optionGroup.label + index;\n  }\n\n  public onCreateNewOption() {\n    if (!this.loading && this.newOptionLabel) {\n      this.createNewOption.emit(this.newOptionLabel);\n    }\n  }\n\n  trackByValue(_index: number, option: OptionGroupOption | any) {\n    if (this.trackByProperty) {\n      return option[this.trackByProperty];\n    }\n    return option.value;\n  }\n\n  onPositionChange(event: ConnectedOverlayPositionChange) {\n    this.isTopDropdown = event.connectionPair.originY === 'top';\n  }\n}\n","<ng-container *ngIf=\"overlayPositioning; else dropdown\">\n  <div class=\"overlayOrigin\" cdkOverlayOrigin #selectOrigin=\"cdkOverlayOrigin\"></div>\n  <ng-template\n    cdkConnectedOverlay\n    [cdkConnectedOverlayOrigin]=\"selectOrigin\"\n    [cdkConnectedOverlayOpen]=\"!!(cdkOverlayOpen$ | async)\"\n    [cdkConnectedOverlayWidth]=\"selectOrigin.elementRef.nativeElement.offsetWidth\"\n    (positionChange)=\"onPositionChange($event)\"\n  >\n    <div class=\"overlayDropdown\" [ngClass]=\"{ top: isTopDropdown }\">\n      <ng-container [ngTemplateOutlet]=\"dropdown\" />\n    </div>\n  </ng-template>\n</ng-container>\n\n<ng-template #dropdown>\n  <div\n    #keyboardSelectContainer\n    class=\"scrollingPanel lxThinScrollbar\"\n    infinite-scroll\n    [scrollWindow]=\"false\"\n    [fromRoot]=\"true\"\n    (scrolled)=\"containerScrolled.emit()\"\n  >\n    <ul class=\"options\">\n      <li\n        *ngIf=\"showCreateNewOption && newOptionLabel && isNewItem\"\n        lxSelectableItem\n        [scrollInContainer]=\"keyboardSelectContainer\"\n        #item\n        (click)=\"onCreateNewOption()\"\n        (select)=\"onCreateNewOption()\"\n        [class.highlighted]=\"isItemSelected(item, (selectedItem$ | async)?.element)\"\n        class=\"option\"\n      >\n        <span class=\"newEntryContent\">\n          {{ newOptionLabel }}\n        </span>\n        <lx-counter class=\"lx-margin-left\" size=\"small\" [content]=\"'common.new' | translate | uppercase\" />\n      </li>\n      <ng-container *ngFor=\"let optionGroup of optionGroups; let groupIndex = index; trackBy: trackByLabel\">\n        <li>\n          <div class=\"groupLabel\" *ngIf=\"optionGroup.label\">\n            <span>{{ optionGroup.label | uppercase }}</span>\n          </div>\n          <ul>\n            <ng-container *ngIf=\"showNoResultsIfGroupIsEmpty && optionGroup.options.length === 0; else options\">\n              <li class=\"noResult\">\n                <ng-container *ngIf=\"noResultsOptionTemplateRef; else defaultText\">\n                  <ng-container *ngTemplateOutlet=\"noResultsOptionTemplateRef; context: { group: optionGroup, groupIndex: groupIndex }\" />\n                </ng-container>\n                <ng-template #defaultText>\n                  <span>{{ NAME + '.noResults' | translate }}</span>\n                </ng-template>\n              </li>\n            </ng-container>\n            <ng-template #options>\n              <li\n                *ngFor=\"let option of optionGroup.options; let index = index; trackBy: trackByValue\"\n                lxSelectableItem\n                [scrollInContainer]=\"keyboardSelectContainer\"\n                #item\n                class=\"option\"\n                (click)=\"selectOption(option)\"\n                (select)=\"selectOption(option)\"\n                [class.highlighted]=\"isItemSelected(item, (selectedItem$ | async)?.element)\"\n              >\n                <ng-container *ngIf=\"optionTemplateRef; else basicDropdownOption\">\n                  <ng-container\n                    *ngTemplateOutlet=\"optionTemplateRef; context: { $implicit: option, index: index, groupIndex: groupIndex }\"\n                  />\n                </ng-container>\n                <ng-template #basicDropdownOption>\n                  <lx-basic-dropdown-item [label]=\"option.label\" labelFontWeight=\"normal\" [highlightTerm]=\"highlightTerm\" />\n                </ng-template>\n              </li>\n            </ng-template>\n          </ul>\n        </li>\n      </ng-container>\n      <lx-spinner *ngIf=\"loading\" [fadeBackground]=\"true\" />\n    </ul>\n  </div>\n</ng-template>\n"]}
|
130
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"option-group-dropdown.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/option-group-dropdown/option-group-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkConnectedOverlay, CdkOverlayOrigin, ConnectedOverlayPositionChange } from '@angular/cdk/overlay';\nimport { AsyncPipe, NgClass, NgFor, NgIf, NgTemplateOutlet, UpperCasePipe } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n  forwardRef\n} from '@angular/core';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { InfiniteScrollModule } from 'ngx-infinite-scroll';\nimport { CounterComponent } from '../../../core-ui/components/counter/counter.component';\nimport { SpinnerComponent } from '../../../core-ui/components/spinner/spinner.component';\nimport { SelectableItemDirective } from '../../directives/selectable-item.directive';\nimport { BasicDropdownItemComponent } from '../basic-dropdown-item/basic-dropdown-item.component';\nimport { BasicDropdownComponent } from '../basic-dropdown/basic-dropdown.component';\nimport { KeyboardSelectDirective } from '../keyboard-select.directive';\n\nexport type OptionGroupOption = { value: string; label: string };\n/**\n * TODO: rename to something unspecific for single selects because multiselects are also supported\n */\nexport type SingleSelectOptionGroup<T = any> = {\n  label: string;\n  options: (OptionGroupOption | T)[];\n};\n\n/**\n * Usage:\n *\n * ```\n * <lx-single-select\n *    #singleSelect\n *    [selection]=\"value\"\n *    (selectionChange)=\"onSelectionChange($event)\">\n *   <span class=\"selectedOption\">\n *     {{ value.label }}\n *   </span>\n *   <lx-option-group-dropdown\n *     class=\"dropdownComponent\"\n *     [keyboardSelectAction]=\"singleSelect.optionsKeyboardSelectAction$\"\n *     [optionGroups]=\"optionGroups\"\n *     (onItemSelected)=\"singleSelect.selectOption($event)\">\n *    </lx-option-group-dropdown>\n * </lx-single-select>\n * ```\n */\n@Component({\n  selector: 'lx-option-group-dropdown',\n  templateUrl: 'option-group-dropdown.component.html',\n  styleUrls: ['option-group-dropdown.component.scss'],\n  providers: [{ provide: KeyboardSelectDirective, useExisting: forwardRef(() => OptionGroupDropdownComponent) }],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgIf,\n    CdkOverlayOrigin,\n    CdkConnectedOverlay,\n    NgClass,\n    NgTemplateOutlet,\n    InfiniteScrollModule,\n    SelectableItemDirective,\n    CounterComponent,\n    NgFor,\n    BasicDropdownItemComponent,\n    SpinnerComponent,\n    AsyncPipe,\n    UpperCasePipe,\n    TranslateModule\n  ]\n})\nexport class OptionGroupDropdownComponent extends KeyboardSelectDirective {\n  readonly NAME = 'OptionGroupDropdownComponent';\n  @Input() optionGroups: SingleSelectOptionGroup[] = [];\n  /**\n   * Usually a stream of the current input value of the parent select.\n   */\n  @Input() newOptionLabel?: string;\n  @Input() highlightTerm?: string;\n  @Input() showCreateNewOption: boolean = false;\n  @Input() labelKey: string = 'label';\n  @Input() loading: boolean = false;\n  @Input() trackByProperty?: string;\n  @Input() showNoResultsIfGroupIsEmpty: boolean = false;\n  @Input() override overlayPositioning: boolean = false;\n\n  @Output() onItemSelected = new EventEmitter<OptionGroupOption | any>();\n  @Output() containerScrolled = new EventEmitter();\n  @Output() createNewOption = new EventEmitter<string>();\n\n  @ContentChild('optionTemplate') optionTemplateRef?: TemplateRef<any>;\n  /**\n   * If you provide a <ng-template #noResultsOptionTemplateRef></ng-template> inside the <lx-option-group-dropdown>\n   * it will be used to display the \"no results\" option instead of the default \"No results\" text.\n   * This way you can easily define different \"No results\" texts for different groups too.\n   */\n  @ContentChild('noResultsOptionTemplateRef') noResultsOptionTemplateRef?: TemplateRef<any>;\n  @ViewChild('selectOrigin', { static: false }) selectOrigin!: CdkOverlayOrigin;\n  isTopDropdown = false;\n\n  public get isNewItem() {\n    const options =\n      this.optionGroups?.reduce<(OptionGroupOption | any)[]>((flatOptions, optionGroup) => flatOptions.concat(optionGroup.options), []) ??\n      [];\n    return BasicDropdownComponent.isNewItem(options, this.newOptionLabel ?? '', this.labelKey);\n  }\n\n  selectOption(option: OptionGroupOption | any) {\n    this.onItemSelected.emit(option);\n  }\n\n  trackByLabel(index: number, optionGroup: SingleSelectOptionGroup) {\n    return optionGroup.label + index;\n  }\n\n  public onCreateNewOption() {\n    if (!this.loading && this.newOptionLabel) {\n      this.createNewOption.emit(this.newOptionLabel);\n    }\n  }\n\n  trackByValue(_index: number, option: OptionGroupOption | any) {\n    if (this.trackByProperty) {\n      return option[this.trackByProperty];\n    }\n    return option.value;\n  }\n\n  onPositionChange(event: ConnectedOverlayPositionChange) {\n    this.isTopDropdown = event.connectionPair.originY === 'top';\n  }\n}\n","<ng-container *ngIf=\"overlayPositioning; else dropdown\">\n  <div class=\"overlayOrigin\" cdkOverlayOrigin #selectOrigin=\"cdkOverlayOrigin\"></div>\n  <ng-template\n    cdkConnectedOverlay\n    [cdkConnectedOverlayOrigin]=\"selectOrigin\"\n    [cdkConnectedOverlayOpen]=\"!!(cdkOverlayOpen$ | async)\"\n    [cdkConnectedOverlayWidth]=\"selectOrigin.elementRef.nativeElement.offsetWidth\"\n    (positionChange)=\"onPositionChange($event)\"\n  >\n    <div class=\"overlayDropdown\" [ngClass]=\"{ top: isTopDropdown }\">\n      <ng-container [ngTemplateOutlet]=\"dropdown\" />\n    </div>\n  </ng-template>\n</ng-container>\n\n<ng-template #dropdown>\n  <div\n    #keyboardSelectContainer\n    class=\"scrollingPanel lxThinScrollbar\"\n    infinite-scroll\n    [scrollWindow]=\"false\"\n    [fromRoot]=\"true\"\n    (scrolled)=\"containerScrolled.emit()\"\n  >\n    <ul class=\"options\">\n      <li\n        *ngIf=\"showCreateNewOption && newOptionLabel && isNewItem\"\n        lxSelectableItem\n        [scrollInContainer]=\"keyboardSelectContainer\"\n        #item\n        (click)=\"onCreateNewOption()\"\n        (select)=\"onCreateNewOption()\"\n        [class.highlighted]=\"isItemSelected(item, (selectedItem$ | async)?.element)\"\n        class=\"option\"\n      >\n        <span class=\"newEntryContent\">\n          {{ newOptionLabel }}\n        </span>\n        <lx-counter class=\"lx-margin-left\" size=\"small\" [content]=\"'common.new' | translate | uppercase\" />\n      </li>\n      <ng-container *ngFor=\"let optionGroup of optionGroups; let groupIndex = index; trackBy: trackByLabel\">\n        <li>\n          <div class=\"groupLabel\" *ngIf=\"optionGroup.label\">\n            <span>{{ optionGroup.label | uppercase }}</span>\n          </div>\n          <ul>\n            <ng-container *ngIf=\"showNoResultsIfGroupIsEmpty && optionGroup.options.length === 0; else options\">\n              <li class=\"noResult\">\n                <ng-container *ngIf=\"noResultsOptionTemplateRef; else defaultText\">\n                  <ng-container *ngTemplateOutlet=\"noResultsOptionTemplateRef; context: { group: optionGroup, groupIndex: groupIndex }\" />\n                </ng-container>\n                <ng-template #defaultText>\n                  <span>{{ NAME + '.noResults' | translate }}</span>\n                </ng-template>\n              </li>\n            </ng-container>\n            <ng-template #options>\n              <li\n                *ngFor=\"let option of optionGroup.options; let index = index; trackBy: trackByValue\"\n                lxSelectableItem\n                [scrollInContainer]=\"keyboardSelectContainer\"\n                #item\n                class=\"option\"\n                (click)=\"selectOption(option)\"\n                (select)=\"selectOption(option)\"\n                [class.highlighted]=\"isItemSelected(item, (selectedItem$ | async)?.element)\"\n              >\n                <ng-container *ngIf=\"optionTemplateRef; else basicDropdownOption\">\n                  <ng-container\n                    *ngTemplateOutlet=\"optionTemplateRef; context: { $implicit: option, index: index, groupIndex: groupIndex }\"\n                  />\n                </ng-container>\n                <ng-template #basicDropdownOption>\n                  <lx-basic-dropdown-item [label]=\"option.label\" labelFontWeight=\"normal\" [highlightTerm]=\"highlightTerm\" />\n                </ng-template>\n              </li>\n            </ng-template>\n          </ul>\n        </li>\n      </ng-container>\n      <lx-spinner *ngIf=\"loading\" [fadeBackground]=\"true\" />\n    </ul>\n  </div>\n</ng-template>\n"]}
|
@@ -13,11 +13,12 @@ import * as i1 from "@angular/cdk/bidi";
|
|
13
13
|
* How to use:
|
14
14
|
* Provide the lx-picker with two kinds of ContentChildren:
|
15
15
|
* 1. A lxPickerTrigger, preferably a button element, which we use to open the dropdown on click or enter.
|
16
|
-
* 2. A list of
|
16
|
+
* 2. A list of `<li>` elements with an lx-picker-option attribute that own the option template as content children.
|
17
17
|
*
|
18
18
|
* You can use this component as part of a FormGroup or wire it up yourself by listening on the (select) output of the options.
|
19
19
|
*
|
20
20
|
* Example usage (non FormGroup style):
|
21
|
+
* ```
|
21
22
|
* <lx-picker listBoxAriaLabel="Awesome picker">
|
22
23
|
* <button lxPickerTrigger lx-button mode="outline" size="large" [square]="true" aria-label="Awesome picker" title="Awesome picker">
|
23
24
|
* <lx-icon [name]="selection.label" [color]="selection.value" fontAwsomeStyle="solid"></lx-icon>
|
@@ -26,6 +27,7 @@ import * as i1 from "@angular/cdk/bidi";
|
|
26
27
|
* <lx-icon lx-picker-option [name]="option.label" [color]="option.value" fontAwsomeStyle="solid"></lx-icon>
|
27
28
|
* </li>
|
28
29
|
* </lx-picker>
|
30
|
+
* ```
|
29
31
|
*/
|
30
32
|
export class PickerComponent {
|
31
33
|
get activeDescendantsId() {
|
@@ -198,4 +200,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
198
200
|
type: HostListener,
|
199
201
|
args: ['document:keydown', ['$event']]
|
200
202
|
}] } });
|
201
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"picker.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/picker/picker.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/picker/picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ActiveDescendantKeyManager } from '@angular/cdk/a11y';\nimport { Directionality } from '@angular/cdk/bidi';\nimport { NgIf } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  forwardRef,\n  HostListener,\n  Input,\n  OnDestroy,\n  QueryList\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { AutocloseDirective } from '../../../core-ui/directives/autoclose.directive';\nimport { PickerOptionComponent } from '../picker-option/picker-option.component';\nimport { PickerTriggerDirective } from './picker-trigger.directive';\n\n/**\n * How to use:\n * Provide the lx-picker with two kinds of ContentChildren:\n * 1. A lxPickerTrigger, preferably a button element, which we use to open the dropdown on click or enter.\n * 2. A list of <li> elements with an lx-picker-option attribute that own the option template as content children.\n *\n * You can use this component as part of a FormGroup or wire it up yourself by listening on the (select) output of the options.\n *\n * Example usage (non FormGroup style):\n * <lx-picker listBoxAriaLabel=\"Awesome picker\">\n *   <button lxPickerTrigger lx-button mode=\"outline\" size=\"large\" [square]=\"true\" aria-label=\"Awesome picker\" title=\"Awesome picker\">\n *       <lx-icon [name]=\"selection.label\" [color]=\"selection.value\" fontAwsomeStyle=\"solid\"></lx-icon>\n *    </button>\n *    <li *ngFor=\"let option of options\" lx-picker-option [value]=\"option.value\" [selected]=\"selection === option\" (select)=\"selection = option\" [optionId]=\"option.id\">\n *       <lx-icon lx-picker-option [name]=\"option.label\" [color]=\"option.value\" fontAwsomeStyle=\"solid\"></lx-icon>\n *    </li>\n * </lx-picker>\n */\n@Component({\n  selector: 'lx-picker',\n  templateUrl: 'picker.component.html',\n  styleUrls: ['picker.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => PickerComponent)\n    }\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [NgIf, AutocloseDirective]\n})\nexport class PickerComponent implements AfterViewInit, ControlValueAccessor, OnDestroy {\n  /**\n   * A human-readable string value which identifies the listbox.\n   * If there's a visible label, then aria-labelledby should be used instead to refer to that label.\n   */\n  @Input() listBoxAriaLabel?: string;\n  /**\n   * Identifies the visible element which identifies the listbox.\n   * If there's no visible label, then aria-label should be used instead to include a label.\n   */\n  @Input() listBoxAriaLabelledBy?: string;\n  @Input() openDirection: 'left' | 'right' = 'right';\n\n  get activeDescendantsId() {\n    return this._activeDescendantsId;\n  }\n  set activeDescendantsId(value: string | undefined) {\n    this._activeDescendantsId = value;\n    this.cdRef.markForCheck();\n  }\n\n  get open() {\n    return this._open;\n  }\n  set open(value: boolean) {\n    this._open = value;\n    this.cdRef.markForCheck();\n  }\n\n  /**\n   * Returns a negative margin, that is used to move the picker dropdown to the left side of the picker trigger.\n   */\n  get leftOffset() {\n    const dropdownWidth = 185;\n    return this.openDirection === 'left' ? `-${dropdownWidth - this._trigger.width}px` : '0px';\n  }\n\n  @ContentChildren(PickerOptionComponent)\n  private options!: QueryList<PickerOptionComponent>;\n  keyManager!: ActiveDescendantKeyManager<PickerOptionComponent>;\n\n  @ContentChild(PickerTriggerDirective)\n  private _trigger!: PickerTriggerDirective;\n  private _isFormControl = false;\n  private _open = false;\n  private _activeDescendantsId?: string;\n  private _lastSelectWasByKeyManager = false;\n\n  readonly destroyed$ = new Subject<void>();\n\n  constructor(\n    private dir: Directionality,\n    private cdRef: ChangeDetectorRef\n  ) {}\n\n  @HostListener('document:keydown', ['$event'])\n  manageKeyboardEvents(event: KeyboardEvent) {\n    if (this.open) {\n      const activeIndex = this.keyManager.activeItemIndex || 0;\n      switch (event.key) {\n        case 'Enter':\n          setTimeout(() => {\n            this.closePicker();\n          });\n          break;\n        case 'Escape':\n        case 'Tab':\n          this.setActivePickerOption(-1);\n          this.closePicker();\n          break;\n        case 'ArrowDown':\n          event.preventDefault();\n          const newDownIndex = this.keyManager.activeItemIndex === -1 ? 0 : activeIndex + 5;\n          this.setActivePickerOption(newDownIndex < this.options.length ? newDownIndex : activeIndex);\n          break;\n        case 'ArrowUp':\n          event.preventDefault();\n          const newUpIndex = this.keyManager.activeItemIndex === -1 ? this.options.length - 1 : activeIndex - 5;\n          this.setActivePickerOption(newUpIndex >= 0 ? newUpIndex : activeIndex);\n          break;\n        case 'ArrowLeft':\n          if (this.keyManager.activeItemIndex === -1) {\n            this.setActivePickerOption(this.options.length - 1);\n          } else {\n            this.keyManager.onKeydown(event);\n            this.activeDescendantsId = this.keyManager.activeItem?.optionId;\n          }\n          break;\n        case 'Home':\n          event.preventDefault();\n          this.setActivePickerOption(0);\n          break;\n        case 'End':\n          event.preventDefault();\n          this.setActivePickerOption(this.options.length - 1);\n          break;\n        default:\n          // matches ArrowRight\n          this.keyManager.onKeydown(event);\n          this.activeDescendantsId = this.keyManager.activeItem?.optionId;\n      }\n      if (this.keyManager.activeItem) {\n        this._lastSelectWasByKeyManager = true;\n        this.propagateChange(this.keyManager.activeItem.value);\n        if (!this._isFormControl) {\n          this.keyManager.activeItem.selectOption();\n        }\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.keyManager = new ActiveDescendantKeyManager(this.options).withHorizontalOrientation(this.dir.value).withVerticalOrientation(true);\n\n    this.options.map((item: PickerOptionComponent) => {\n      item.select.pipe(takeUntil(this.destroyed$)).subscribe((value: any) => {\n        this.propagateChange(value);\n        if (value !== this.keyManager.activeItem?.value || !this._lastSelectWasByKeyManager) {\n          this.closePicker();\n        }\n        this._lastSelectWasByKeyManager = false;\n      });\n    });\n\n    this._trigger.clickEvent$.pipe(takeUntil(this.destroyed$)).subscribe(() => {\n      this.toggleOpen();\n    });\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next();\n  }\n\n  closePicker() {\n    this.setActivePickerOption(-1);\n    this.open = false;\n  }\n\n  toggleOpen() {\n    this.open = !this.open;\n    if (this.open) {\n      this.onTouched();\n    }\n    this.setActivePickerOption(this.open === false ? -1 : this.options.toArray().findIndex((option) => option.selected));\n  }\n\n  private setActivePickerOption(index: number) {\n    this.keyManager.setActiveItem(index);\n    if (this.keyManager.activeItem) {\n      this.activeDescendantsId = this.keyManager.activeItem.optionId;\n    } else {\n      this.activeDescendantsId = undefined;\n    }\n  }\n\n  //#region Angular Forms API. This is only used when using the Picker inside a FormGroup\n\n  writeValue(value: any) {\n    this.options?.toArray().forEach((option) => (option.selected = option.value === value));\n  }\n\n  registerOnChange(fn: any): void {\n    this.propagateChange = fn;\n    this._isFormControl = true;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  private propagateChange(_value: any) {}\n  private onTouched() {}\n\n  //#endregion\n}\n","<div class=\"pickerContainer\">\n  <ng-content select=\"[lxPickerTrigger]\" />\n  <ul\n    class=\"pickerOptions\"\n    *ngIf=\"open\"\n    lxAutoclose\n    (autoclose)=\"closePicker()\"\n    role=\"listbox\"\n    [style.margin-left]=\"leftOffset\"\n    [attr.aria-activedescendant]=\"activeDescendantsId\"\n    [attr.aria-label]=\"listBoxAriaLabel\"\n    [attr.aria-labelledby]=\"listBoxAriaLabelledBy\"\n  >\n    <ng-content />\n  </ul>\n</div>\n"]}
|
203
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"picker.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/picker/picker.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/picker/picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ActiveDescendantKeyManager } from '@angular/cdk/a11y';\nimport { Directionality } from '@angular/cdk/bidi';\nimport { NgIf } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  forwardRef,\n  HostListener,\n  Input,\n  OnDestroy,\n  QueryList\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { AutocloseDirective } from '../../../core-ui/directives/autoclose.directive';\nimport { PickerOptionComponent } from '../picker-option/picker-option.component';\nimport { PickerTriggerDirective } from './picker-trigger.directive';\n\n/**\n * How to use:\n * Provide the lx-picker with two kinds of ContentChildren:\n * 1. A lxPickerTrigger, preferably a button element, which we use to open the dropdown on click or enter.\n * 2. A list of `<li>` elements with an lx-picker-option attribute that own the option template as content children.\n *\n * You can use this component as part of a FormGroup or wire it up yourself by listening on the (select) output of the options.\n *\n * Example usage (non FormGroup style):\n * ```\n * <lx-picker listBoxAriaLabel=\"Awesome picker\">\n *   <button lxPickerTrigger lx-button mode=\"outline\" size=\"large\" [square]=\"true\" aria-label=\"Awesome picker\" title=\"Awesome picker\">\n *       <lx-icon [name]=\"selection.label\" [color]=\"selection.value\" fontAwsomeStyle=\"solid\"></lx-icon>\n *    </button>\n *    <li *ngFor=\"let option of options\" lx-picker-option [value]=\"option.value\" [selected]=\"selection === option\" (select)=\"selection = option\" [optionId]=\"option.id\">\n *       <lx-icon lx-picker-option [name]=\"option.label\" [color]=\"option.value\" fontAwsomeStyle=\"solid\"></lx-icon>\n *    </li>\n * </lx-picker>\n * ```\n */\n@Component({\n  selector: 'lx-picker',\n  templateUrl: 'picker.component.html',\n  styleUrls: ['picker.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => PickerComponent)\n    }\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [NgIf, AutocloseDirective]\n})\nexport class PickerComponent implements AfterViewInit, ControlValueAccessor, OnDestroy {\n  /**\n   * A human-readable string value which identifies the listbox.\n   * If there's a visible label, then aria-labelledby should be used instead to refer to that label.\n   */\n  @Input() listBoxAriaLabel?: string;\n  /**\n   * Identifies the visible element which identifies the listbox.\n   * If there's no visible label, then aria-label should be used instead to include a label.\n   */\n  @Input() listBoxAriaLabelledBy?: string;\n  @Input() openDirection: 'left' | 'right' = 'right';\n\n  get activeDescendantsId() {\n    return this._activeDescendantsId;\n  }\n  set activeDescendantsId(value: string | undefined) {\n    this._activeDescendantsId = value;\n    this.cdRef.markForCheck();\n  }\n\n  get open() {\n    return this._open;\n  }\n  set open(value: boolean) {\n    this._open = value;\n    this.cdRef.markForCheck();\n  }\n\n  /**\n   * Returns a negative margin, that is used to move the picker dropdown to the left side of the picker trigger.\n   */\n  get leftOffset() {\n    const dropdownWidth = 185;\n    return this.openDirection === 'left' ? `-${dropdownWidth - this._trigger.width}px` : '0px';\n  }\n\n  @ContentChildren(PickerOptionComponent)\n  private options!: QueryList<PickerOptionComponent>;\n  keyManager!: ActiveDescendantKeyManager<PickerOptionComponent>;\n\n  @ContentChild(PickerTriggerDirective)\n  private _trigger!: PickerTriggerDirective;\n  private _isFormControl = false;\n  private _open = false;\n  private _activeDescendantsId?: string;\n  private _lastSelectWasByKeyManager = false;\n\n  readonly destroyed$ = new Subject<void>();\n\n  constructor(\n    private dir: Directionality,\n    private cdRef: ChangeDetectorRef\n  ) {}\n\n  @HostListener('document:keydown', ['$event'])\n  manageKeyboardEvents(event: KeyboardEvent) {\n    if (this.open) {\n      const activeIndex = this.keyManager.activeItemIndex || 0;\n      switch (event.key) {\n        case 'Enter':\n          setTimeout(() => {\n            this.closePicker();\n          });\n          break;\n        case 'Escape':\n        case 'Tab':\n          this.setActivePickerOption(-1);\n          this.closePicker();\n          break;\n        case 'ArrowDown':\n          event.preventDefault();\n          const newDownIndex = this.keyManager.activeItemIndex === -1 ? 0 : activeIndex + 5;\n          this.setActivePickerOption(newDownIndex < this.options.length ? newDownIndex : activeIndex);\n          break;\n        case 'ArrowUp':\n          event.preventDefault();\n          const newUpIndex = this.keyManager.activeItemIndex === -1 ? this.options.length - 1 : activeIndex - 5;\n          this.setActivePickerOption(newUpIndex >= 0 ? newUpIndex : activeIndex);\n          break;\n        case 'ArrowLeft':\n          if (this.keyManager.activeItemIndex === -1) {\n            this.setActivePickerOption(this.options.length - 1);\n          } else {\n            this.keyManager.onKeydown(event);\n            this.activeDescendantsId = this.keyManager.activeItem?.optionId;\n          }\n          break;\n        case 'Home':\n          event.preventDefault();\n          this.setActivePickerOption(0);\n          break;\n        case 'End':\n          event.preventDefault();\n          this.setActivePickerOption(this.options.length - 1);\n          break;\n        default:\n          // matches ArrowRight\n          this.keyManager.onKeydown(event);\n          this.activeDescendantsId = this.keyManager.activeItem?.optionId;\n      }\n      if (this.keyManager.activeItem) {\n        this._lastSelectWasByKeyManager = true;\n        this.propagateChange(this.keyManager.activeItem.value);\n        if (!this._isFormControl) {\n          this.keyManager.activeItem.selectOption();\n        }\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.keyManager = new ActiveDescendantKeyManager(this.options).withHorizontalOrientation(this.dir.value).withVerticalOrientation(true);\n\n    this.options.map((item: PickerOptionComponent) => {\n      item.select.pipe(takeUntil(this.destroyed$)).subscribe((value: any) => {\n        this.propagateChange(value);\n        if (value !== this.keyManager.activeItem?.value || !this._lastSelectWasByKeyManager) {\n          this.closePicker();\n        }\n        this._lastSelectWasByKeyManager = false;\n      });\n    });\n\n    this._trigger.clickEvent$.pipe(takeUntil(this.destroyed$)).subscribe(() => {\n      this.toggleOpen();\n    });\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next();\n  }\n\n  closePicker() {\n    this.setActivePickerOption(-1);\n    this.open = false;\n  }\n\n  toggleOpen() {\n    this.open = !this.open;\n    if (this.open) {\n      this.onTouched();\n    }\n    this.setActivePickerOption(this.open === false ? -1 : this.options.toArray().findIndex((option) => option.selected));\n  }\n\n  private setActivePickerOption(index: number) {\n    this.keyManager.setActiveItem(index);\n    if (this.keyManager.activeItem) {\n      this.activeDescendantsId = this.keyManager.activeItem.optionId;\n    } else {\n      this.activeDescendantsId = undefined;\n    }\n  }\n\n  //#region Angular Forms API. This is only used when using the Picker inside a FormGroup\n\n  writeValue(value: any) {\n    this.options?.toArray().forEach((option) => (option.selected = option.value === value));\n  }\n\n  registerOnChange(fn: any): void {\n    this.propagateChange = fn;\n    this._isFormControl = true;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  private propagateChange(_value: any) {}\n  private onTouched() {}\n\n  //#endregion\n}\n","<div class=\"pickerContainer\">\n  <ng-content select=\"[lxPickerTrigger]\" />\n  <ul\n    class=\"pickerOptions\"\n    *ngIf=\"open\"\n    lxAutoclose\n    (autoclose)=\"closePicker()\"\n    role=\"listbox\"\n    [style.margin-left]=\"leftOffset\"\n    [attr.aria-activedescendant]=\"activeDescendantsId\"\n    [attr.aria-label]=\"listBoxAriaLabel\"\n    [attr.aria-labelledby]=\"listBoxAriaLabelledBy\"\n  >\n    <ng-content />\n  </ul>\n</div>\n"]}
|
@@ -14,6 +14,17 @@ import { BasicDropdownComponent } from '../basic-dropdown/basic-dropdown.compone
|
|
14
14
|
import { KeyboardSelectAction } from '../keyboard-select.directive';
|
15
15
|
import * as i0 from "@angular/core";
|
16
16
|
import * as i1 from "@angular/forms";
|
17
|
+
/**
|
18
|
+
* ## Usage
|
19
|
+
*
|
20
|
+
* 1. Import the `LxFormsModule` module from `@leanix/components` in your module where you want to use the component.
|
21
|
+
*
|
22
|
+
* ```ts
|
23
|
+
* import { LxFormsModule } from '@leanix/components';
|
24
|
+
* ```
|
25
|
+
*
|
26
|
+
* 2. Use the component in your template.
|
27
|
+
*/
|
17
28
|
export class SingleSelectComponent extends BaseSelectDirective {
|
18
29
|
static calculateNewCursorPostionOnKeyboardNavigation(cursorPosition, keyCode) {
|
19
30
|
switch (keyCode) {
|
@@ -282,4 +293,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
282
293
|
type: ViewChild,
|
283
294
|
args: ['queryInput', { static: true }]
|
284
295
|
}] } });
|
285
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/single-select/single-select.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/single-select/single-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  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  TemplateRef,\n  ViewChild,\n  booleanAttribute,\n  forwardRef\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { get, includes, isEqual } from 'lodash/fp';\nimport { Subject, fromEvent } 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 { SelectedOptionDirective } from '../../directives/selected-option.directive';\nimport { ARROW_LEFT, ARROW_RIGHT, BACKSPACE, ESCAPE, TAB } from '../../helpers/key-codes.constants';\nimport { getKeyboardNavigationEvents } from '../../helpers/keyboard-navigation.helpers';\nimport { BaseSelectDirective } from '../../models/base-select.directive';\nimport { SingleSelectPadding } from '../../models/single-select-padding.interface';\nimport { BasicDropdownComponent } from '../basic-dropdown/basic-dropdown.component';\nimport { KeyboardSelectAction } from '../keyboard-select.directive';\n\nexport type SingleSelectSize = 'default' | 'small' | 'select2' | 'large';\n\n@Component({\n  selector: 'lx-single-select',\n  templateUrl: 'single-select.component.html',\n  styleUrls: ['single-select.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => SingleSelectComponent)\n    }\n  ],\n  standalone: true,\n  imports: [NgIf, MarkInvalidDirective, NgTemplateOutlet, FormsModule, ReactiveFormsModule, AsyncPipe]\n})\nexport class SingleSelectComponent extends BaseSelectDirective implements OnDestroy, AfterViewInit, ControlValueAccessor {\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() selection: any;\n  @Input() selectionBackground: 'white' | 'gray' = 'white';\n  @Input() size: SingleSelectSize = 'default';\n  @Input() dropdownWidthScale: '1x' | '1.5x' | '2x' = '1x';\n  @Input() padding?: SingleSelectPadding = 'default';\n  @Input() inputId?: string;\n  @Input({ transform: booleanAttribute }) required: boolean = false;\n\n  @Output() selectionChange = new EventEmitter<any>();\n  @Output() blur = new EventEmitter<FocusEvent>();\n\n  @ContentChild(SelectDropdownDirective, { read: TemplateRef, static: true }) explicitDropdown?: TemplateRef<any>;\n  @ContentChild(SelectedOptionDirective, { read: TemplateRef, static: true }) explicitSelectedOption?: TemplateRef<any>;\n  @ContentChild(BasicDropdownComponent, { static: false }) private dropdownComponent?: BasicDropdownComponent;\n\n  @ViewChild('selectedOption', { static: true }) implicitSelectedOption!: TemplateRef<any>;\n  @ViewChild('dropdown', { static: true }) implicitDropdown!: TemplateRef<any>;\n\n  get selectedOptionTmpl() {\n    return this.explicitSelectedOption || this.implicitSelectedOption;\n  }\n\n  get dropdownTmpl() {\n    return this.explicitDropdown || this.implicitDropdown;\n  }\n\n  @Input() allowClear = true;\n  @Input() tabIndex: number = 0;\n  @Input() markInvalid: boolean = false;\n\n  @ViewChild('queryInput', { static: true }) queryInput!: ElementRef<HTMLInputElement>;\n\n  override readonly destroyed$ = new Subject<void>();\n  isInputFocusedViaTab = false;\n\n  constructor(private cd: ChangeDetectorRef) {\n    super();\n  }\n\n  get cursorStyle() {\n    if (this.disabled) {\n      return 'default';\n    } else if (this.allowQuery) {\n      return 'text';\n    } else {\n      return 'pointer';\n    }\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.queryControl.reset('', { emitEvent: false });\n  }\n\n  focus() {\n    this.queryInput.nativeElement.focus();\n  }\n\n  focusAndOpen() {\n    this.queryInput.nativeElement.focus();\n    this.open.next(true);\n  }\n\n  override ngOnDestroy() {\n    super.ngOnDestroy();\n    this.destroyed$.next();\n  }\n\n  override ngAfterViewInit() {\n    super.ngAfterViewInit();\n    this.open\n      .pipe(\n        distinctUntilChanged(),\n        filter((open) => open && !!this.dropdownComponent),\n        map(() => this.selection),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((selection) => {\n        if (selection) {\n          const index = this.determineHighlightIndex(this.dropdownComponent!.options, selection, this.dropdownComponent!.itemKey);\n          if (index > -1) {\n            this.dropdownComponent!.select(index);\n          } else {\n            this.selectFirstOption();\n          }\n        } else {\n          this.selectFirstOption();\n        }\n      });\n\n    // keyboard event handling\n    const sourceElement = this.queryInput.nativeElement;\n    // handle keyboard navigation for the selection\n    fromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        map((event) => this.mapToEventSet(event, sourceElement, this.virtualCursorPosition)),\n        tap((eventSet) => {\n          if (!this.allowQuery && eventSet.event.keyCode !== BACKSPACE) {\n            // additional check for even.code for userEvent API compatibility\n            if (eventSet.event.keyCode !== TAB && eventSet.event.code !== 'Tab') {\n              // if querying is disabled, only the tab key press event will be propagated, so that you can still defocus.\n              eventSet.event.preventDefault();\n            }\n            return;\n          }\n          if (\n            eventSet.virtualCursorPosition === 0 &&\n            eventSet.inputCursorPosition === 0 &&\n            eventSet.event.keyCode === BACKSPACE &&\n            this.selection &&\n            this.allowClear\n          ) {\n            this.removeSelection();\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) => this.allowQuery && 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 = SingleSelectComponent.calculateNewCursorPostionOnKeyboardNavigation(\n            this.virtualCursorPosition,\n            event.keyCode\n          );\n          return {\n            cursorPosition: this.virtualCursorPosition,\n            event\n          };\n        }),\n        map((positionSet) => {\n          const action = SingleSelectComponent.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 keyboardEvents = getKeyboardNavigationEvents(sourceElement, this.destroyed$);\n\n    keyboardEvents.verticalNavigation$.subscribe((event) => {\n      event.preventDefault();\n    });\n\n    keyboardEvents.close$.subscribe((event) => {\n      if (event.keyCode === ESCAPE && this.dropdownOpen) {\n        event.preventDefault();\n        event.stopImmediatePropagation();\n      }\n      this.open.next(false);\n    });\n\n    keyboardEvents.dropdownKeyboardEvent$.subscribe(this.optionsKeyboardSelectAction$);\n\n    keyboardEvents.enter$.subscribe(() => this.open.next(!this.dropdownOpen));\n    keyboardEvents.space$.pipe(filter(() => !this.allowQuery)).subscribe(() => {\n      if (this.dropdownOpen) {\n        this.optionsKeyboardSelectAction$.next(KeyboardSelectAction.EXECUTE);\n      }\n      this.open.next(!this.dropdownOpen);\n    });\n\n    keyboardEvents.openOnArrowDown$.subscribe(() => this.open.next(true));\n  }\n\n  selectOption(option: any) {\n    this.selection = option;\n\n    this.selectionChange.emit(option);\n    this.propagateChange(option);\n\n    this.resetSelectState();\n\n    // This timeout makes sure that the dropdown will be closed at the end\n    // of all the callbacks in the open subscriptions.\n    setTimeout(() => {\n      this.open.next(false);\n    });\n  }\n\n  removeSelection() {\n    this.selection = null;\n    this.cd.markForCheck();\n    this.cd.detectChanges();\n\n    this.selectionChange.emit(this.selection);\n    this.propagateChange(this.selection);\n\n    this.resetSelectState();\n  }\n\n  propagateChange(_value: any) {}\n\n  writeValue(newSelection: any) {\n    this.selection = newSelection;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n  }\n\n  registerOnChange(fn: any) {\n    this.propagateChange = fn;\n  }\n\n  registerOnTouched(_fn: any) {}\n\n  focusedViaTab() {\n    this.isInputFocusedViaTab = true;\n  }\n\n  private determineHighlightIndex(options: any[], selection: any, itemKey?: string) {\n    return itemKey\n      ? options.findIndex((option) => isEqual(get(itemKey, selection), get(itemKey, option)) || isEqual(selection, get(itemKey, option)))\n      : options.findIndex((option) => isEqual(selection, option));\n  }\n}\n","<div *ngIf=\"dropdownOpen\" class=\"backdrop\" (click)=\"handleBackdropClick($event)\"></div>\n<div\n  [lxMarkInvalid]=\"markInvalid\"\n  class=\"selectContainer {{ padding }}Padding\"\n  [class.open]=\"dropdownOpen\"\n  [class.top]=\"(dropdownDirection$ | async) === 'TOP'\"\n  [class.bottom]=\"(dropdownDirection$ | async) === 'BOTTOM'\"\n  [class.disabled]=\"disabled\"\n  [class.grayBackground]=\"selectionBackground === 'gray'\"\n  [class.defaultSize]=\"size === 'default'\"\n  [class.smallSize]=\"size === 'small'\"\n  [class.select2Size]=\"size === 'select2'\"\n  [class.largeSize]=\"size === 'large'\"\n  [style.cursor]=\"cursorStyle\"\n  [class.focused]=\"isInputFocused\"\n  [class.focusedVisible]=\"isInputFocusedViaTab\"\n  (click)=\"handleClick($event.target === toggle)\"\n>\n  <div class=\"selectionContainer\">\n    <div class=\"inputContainer\">\n      <div *ngIf=\"!selection && selection !== 0 && !queryInput?.value\" class=\"placeholder\" [attr.title]=\"placeholder\">\n        {{ placeholder }}\n      </div>\n      <div\n        *ngIf=\"!allowQuery || !queryInput?.value\"\n        class=\"selection\"\n        [class.lowerOpacity]=\"allowQuery && !queryInput?.value && dropdownOpen\"\n        [class.lightgrayColor]=\"!allowQuery && dropdownOpen\"\n        (click)=\"$event.stopPropagation(); focus()\"\n      >\n        <ng-container *ngTemplateOutlet=\"selectedOptionTmpl\" />\n      </div>\n      <input\n        #queryInput\n        class=\"queryInput\"\n        type=\"text\"\n        [attr.id]=\"inputId\"\n        [attr.required]=\"required ? 'true' : null\"\n        [class.isHidden]=\"!allowQuery\"\n        (keydown.enter)=\"$event.preventDefault()\"\n        (focus)=\"isInputFocused = true\"\n        (blur)=\"isInputFocused = false; isInputFocusedViaTab = false; blur.emit($event)\"\n        (keyup.tab)=\"focusedViaTab()\"\n        (keyup.shift.tab)=\"focusedViaTab()\"\n        [tabIndex]=\"disabled ? -1 : tabIndex\"\n        [formControl]=\"queryControl\"\n        [readOnly]=\"!allowQuery\"\n      />\n    </div>\n    <div class=\"iconContainer\">\n      <i *ngIf=\"(selection || selection === 0) && allowClear && !disabled\" class=\"far fa-times\" (click)=\"removeSelection()\"></i>\n      <i #toggle class=\"fas fa-angle-down\" aria-hidden=\"true\"></i>\n    </div>\n  </div>\n  <div class=\"optionsContainer\" [style.width]=\"dropdownWidth\" #optionsContainer>\n    <ng-container *ngIf=\"dropdownOpen\">\n      <ng-container *ngTemplateOutlet=\"dropdownTmpl\" />\n    </ng-container>\n  </div>\n</div>\n\n<ng-template #selectedOption>\n  <ng-content select=\".selectedOption\" />\n</ng-template>\n<ng-template #dropdown>\n  <ng-content select=\".dropdownComponent\" />\n</ng-template>\n"]}
|
296
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/forms-ui/components/single-select/single-select.component.ts","../../../../../../../../libs/components/src/lib/forms-ui/components/single-select/single-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  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  TemplateRef,\n  ViewChild,\n  booleanAttribute,\n  forwardRef\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { get, includes, isEqual } from 'lodash/fp';\nimport { Subject, fromEvent } 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 { SelectedOptionDirective } from '../../directives/selected-option.directive';\nimport { ARROW_LEFT, ARROW_RIGHT, BACKSPACE, ESCAPE, TAB } from '../../helpers/key-codes.constants';\nimport { getKeyboardNavigationEvents } from '../../helpers/keyboard-navigation.helpers';\nimport { BaseSelectDirective } from '../../models/base-select.directive';\nimport { SingleSelectPadding } from '../../models/single-select-padding.interface';\nimport { BasicDropdownComponent } from '../basic-dropdown/basic-dropdown.component';\nimport { KeyboardSelectAction } from '../keyboard-select.directive';\n\nexport type SingleSelectSize = 'default' | 'small' | 'select2' | 'large';\n\n/**\n * ## Usage\n *\n * 1. Import the `LxFormsModule` module from `@leanix/components` in your module where you want to use the component.\n *\n * ```ts\n * import { LxFormsModule } from '@leanix/components';\n * ```\n *\n * 2. Use the component in your template.\n */\n@Component({\n  selector: 'lx-single-select',\n  templateUrl: 'single-select.component.html',\n  styleUrls: ['single-select.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => SingleSelectComponent)\n    }\n  ],\n  standalone: true,\n  imports: [NgIf, MarkInvalidDirective, NgTemplateOutlet, FormsModule, ReactiveFormsModule, AsyncPipe]\n})\nexport class SingleSelectComponent extends BaseSelectDirective implements OnDestroy, AfterViewInit, ControlValueAccessor {\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() selection: any;\n  @Input() selectionBackground: 'white' | 'gray' = 'white';\n  @Input() size: SingleSelectSize = 'default';\n  @Input() dropdownWidthScale: '1x' | '1.5x' | '2x' = '1x';\n  @Input() padding?: SingleSelectPadding = 'default';\n  @Input() inputId?: string;\n  @Input({ transform: booleanAttribute }) required: boolean = false;\n\n  @Output() selectionChange = new EventEmitter<any>();\n  @Output() blur = new EventEmitter<FocusEvent>();\n\n  @ContentChild(SelectDropdownDirective, { read: TemplateRef, static: true }) explicitDropdown?: TemplateRef<any>;\n  @ContentChild(SelectedOptionDirective, { read: TemplateRef, static: true }) explicitSelectedOption?: TemplateRef<any>;\n  @ContentChild(BasicDropdownComponent, { static: false }) private dropdownComponent?: BasicDropdownComponent;\n\n  @ViewChild('selectedOption', { static: true }) implicitSelectedOption!: TemplateRef<any>;\n  @ViewChild('dropdown', { static: true }) implicitDropdown!: TemplateRef<any>;\n\n  get selectedOptionTmpl() {\n    return this.explicitSelectedOption || this.implicitSelectedOption;\n  }\n\n  get dropdownTmpl() {\n    return this.explicitDropdown || this.implicitDropdown;\n  }\n\n  @Input() allowClear = true;\n  @Input() tabIndex: number = 0;\n  @Input() markInvalid: boolean = false;\n\n  @ViewChild('queryInput', { static: true }) queryInput!: ElementRef<HTMLInputElement>;\n\n  override readonly destroyed$ = new Subject<void>();\n  isInputFocusedViaTab = false;\n\n  constructor(private cd: ChangeDetectorRef) {\n    super();\n  }\n\n  get cursorStyle() {\n    if (this.disabled) {\n      return 'default';\n    } else if (this.allowQuery) {\n      return 'text';\n    } else {\n      return 'pointer';\n    }\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.queryControl.reset('', { emitEvent: false });\n  }\n\n  focus() {\n    this.queryInput.nativeElement.focus();\n  }\n\n  focusAndOpen() {\n    this.queryInput.nativeElement.focus();\n    this.open.next(true);\n  }\n\n  override ngOnDestroy() {\n    super.ngOnDestroy();\n    this.destroyed$.next();\n  }\n\n  override ngAfterViewInit() {\n    super.ngAfterViewInit();\n    this.open\n      .pipe(\n        distinctUntilChanged(),\n        filter((open) => open && !!this.dropdownComponent),\n        map(() => this.selection),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((selection) => {\n        if (selection) {\n          const index = this.determineHighlightIndex(this.dropdownComponent!.options, selection, this.dropdownComponent!.itemKey);\n          if (index > -1) {\n            this.dropdownComponent!.select(index);\n          } else {\n            this.selectFirstOption();\n          }\n        } else {\n          this.selectFirstOption();\n        }\n      });\n\n    // keyboard event handling\n    const sourceElement = this.queryInput.nativeElement;\n    // handle keyboard navigation for the selection\n    fromEvent<KeyboardEvent>(sourceElement, 'keydown')\n      .pipe(\n        map((event) => this.mapToEventSet(event, sourceElement, this.virtualCursorPosition)),\n        tap((eventSet) => {\n          if (!this.allowQuery && eventSet.event.keyCode !== BACKSPACE) {\n            // additional check for even.code for userEvent API compatibility\n            if (eventSet.event.keyCode !== TAB && eventSet.event.code !== 'Tab') {\n              // if querying is disabled, only the tab key press event will be propagated, so that you can still defocus.\n              eventSet.event.preventDefault();\n            }\n            return;\n          }\n          if (\n            eventSet.virtualCursorPosition === 0 &&\n            eventSet.inputCursorPosition === 0 &&\n            eventSet.event.keyCode === BACKSPACE &&\n            this.selection &&\n            this.allowClear\n          ) {\n            this.removeSelection();\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) => this.allowQuery && 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 = SingleSelectComponent.calculateNewCursorPostionOnKeyboardNavigation(\n            this.virtualCursorPosition,\n            event.keyCode\n          );\n          return {\n            cursorPosition: this.virtualCursorPosition,\n            event\n          };\n        }),\n        map((positionSet) => {\n          const action = SingleSelectComponent.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 keyboardEvents = getKeyboardNavigationEvents(sourceElement, this.destroyed$);\n\n    keyboardEvents.verticalNavigation$.subscribe((event) => {\n      event.preventDefault();\n    });\n\n    keyboardEvents.close$.subscribe((event) => {\n      if (event.keyCode === ESCAPE && this.dropdownOpen) {\n        event.preventDefault();\n        event.stopImmediatePropagation();\n      }\n      this.open.next(false);\n    });\n\n    keyboardEvents.dropdownKeyboardEvent$.subscribe(this.optionsKeyboardSelectAction$);\n\n    keyboardEvents.enter$.subscribe(() => this.open.next(!this.dropdownOpen));\n    keyboardEvents.space$.pipe(filter(() => !this.allowQuery)).subscribe(() => {\n      if (this.dropdownOpen) {\n        this.optionsKeyboardSelectAction$.next(KeyboardSelectAction.EXECUTE);\n      }\n      this.open.next(!this.dropdownOpen);\n    });\n\n    keyboardEvents.openOnArrowDown$.subscribe(() => this.open.next(true));\n  }\n\n  selectOption(option: any) {\n    this.selection = option;\n\n    this.selectionChange.emit(option);\n    this.propagateChange(option);\n\n    this.resetSelectState();\n\n    // This timeout makes sure that the dropdown will be closed at the end\n    // of all the callbacks in the open subscriptions.\n    setTimeout(() => {\n      this.open.next(false);\n    });\n  }\n\n  removeSelection() {\n    this.selection = null;\n    this.cd.markForCheck();\n    this.cd.detectChanges();\n\n    this.selectionChange.emit(this.selection);\n    this.propagateChange(this.selection);\n\n    this.resetSelectState();\n  }\n\n  propagateChange(_value: any) {}\n\n  writeValue(newSelection: any) {\n    this.selection = newSelection;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n  }\n\n  registerOnChange(fn: any) {\n    this.propagateChange = fn;\n  }\n\n  registerOnTouched(_fn: any) {}\n\n  focusedViaTab() {\n    this.isInputFocusedViaTab = true;\n  }\n\n  private determineHighlightIndex(options: any[], selection: any, itemKey?: string) {\n    return itemKey\n      ? options.findIndex((option) => isEqual(get(itemKey, selection), get(itemKey, option)) || isEqual(selection, get(itemKey, option)))\n      : options.findIndex((option) => isEqual(selection, option));\n  }\n}\n","<div *ngIf=\"dropdownOpen\" class=\"backdrop\" (click)=\"handleBackdropClick($event)\"></div>\n<div\n  [lxMarkInvalid]=\"markInvalid\"\n  class=\"selectContainer {{ padding }}Padding\"\n  [class.open]=\"dropdownOpen\"\n  [class.top]=\"(dropdownDirection$ | async) === 'TOP'\"\n  [class.bottom]=\"(dropdownDirection$ | async) === 'BOTTOM'\"\n  [class.disabled]=\"disabled\"\n  [class.grayBackground]=\"selectionBackground === 'gray'\"\n  [class.defaultSize]=\"size === 'default'\"\n  [class.smallSize]=\"size === 'small'\"\n  [class.select2Size]=\"size === 'select2'\"\n  [class.largeSize]=\"size === 'large'\"\n  [style.cursor]=\"cursorStyle\"\n  [class.focused]=\"isInputFocused\"\n  [class.focusedVisible]=\"isInputFocusedViaTab\"\n  (click)=\"handleClick($event.target === toggle)\"\n>\n  <div class=\"selectionContainer\">\n    <div class=\"inputContainer\">\n      <div *ngIf=\"!selection && selection !== 0 && !queryInput?.value\" class=\"placeholder\" [attr.title]=\"placeholder\">\n        {{ placeholder }}\n      </div>\n      <div\n        *ngIf=\"!allowQuery || !queryInput?.value\"\n        class=\"selection\"\n        [class.lowerOpacity]=\"allowQuery && !queryInput?.value && dropdownOpen\"\n        [class.lightgrayColor]=\"!allowQuery && dropdownOpen\"\n        (click)=\"$event.stopPropagation(); focus()\"\n      >\n        <ng-container *ngTemplateOutlet=\"selectedOptionTmpl\" />\n      </div>\n      <input\n        #queryInput\n        class=\"queryInput\"\n        type=\"text\"\n        [attr.id]=\"inputId\"\n        [attr.required]=\"required ? 'true' : null\"\n        [class.isHidden]=\"!allowQuery\"\n        (keydown.enter)=\"$event.preventDefault()\"\n        (focus)=\"isInputFocused = true\"\n        (blur)=\"isInputFocused = false; isInputFocusedViaTab = false; blur.emit($event)\"\n        (keyup.tab)=\"focusedViaTab()\"\n        (keyup.shift.tab)=\"focusedViaTab()\"\n        [tabIndex]=\"disabled ? -1 : tabIndex\"\n        [formControl]=\"queryControl\"\n        [readOnly]=\"!allowQuery\"\n      />\n    </div>\n    <div class=\"iconContainer\">\n      <i *ngIf=\"(selection || selection === 0) && allowClear && !disabled\" class=\"far fa-times\" (click)=\"removeSelection()\"></i>\n      <i #toggle class=\"fas fa-angle-down\" aria-hidden=\"true\"></i>\n    </div>\n  </div>\n  <div class=\"optionsContainer\" [style.width]=\"dropdownWidth\" #optionsContainer>\n    <ng-container *ngIf=\"dropdownOpen\">\n      <ng-container *ngTemplateOutlet=\"dropdownTmpl\" />\n    </ng-container>\n  </div>\n</div>\n\n<ng-template #selectedOption>\n  <ng-content select=\".selectedOption\" />\n</ng-template>\n<ng-template #dropdown>\n  <ng-content select=\".dropdownComponent\" />\n</ng-template>\n"]}
|