UNPKG

@engie-group/fluid-design-system-angular

Version:

Fluid Design System Angular

334 lines (331 loc) 47.8 kB
import { CommonModule, DOCUMENT } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChild, ContentChildren, forwardRef, Inject, Input, ViewChild, ViewEncapsulation, } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { race, Subject, takeUntil } from 'rxjs'; import { selectAnimations } from '../../shared/animations'; import { CustomLabelDirective } from '../custom-label/custom-label.directive'; import { FormFieldDirective } from '../form-field/form-field.directive'; import { FormItemComponent } from '../form-item/form-item.component'; import { ListGroupComponent } from '../list-group/list-group.component'; import { ListItemComponent } from '../list-item/list-item.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class SelectComponent extends FormItemComponent { static { this.ESCAPE_CODE = 'Escape'; } static { this.ENTER_CODE = 'Enter'; } static { this.UP_CODE = 'ArrowUp'; } static { this.DOWN_CODE = 'ArrowDown'; } /* Regex matching every alpha-numeric characters. \d : every digits \p{Letter} : every letters in the latin alphabet including letters with diacritics The "u" flag enables unicode mode required to use `\p{Letter}`. See : - https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions/Unicode_Property_Escapes#general_categories - https://unicode.org/reports/tr18/#General_Category_Property */ static { this.ALPHA_NUMERIC_REGEX = /^[\d\p{Letter}]$/u; } constructor(element, cdr, document) { super(); this.element = element; this.cdr = cdr; this.document = document; /** * @ignore */ this._onChange = (_) => { }; /** * @ignore */ this._onTouched = () => { }; /** * Notifier used to stop items click event subscription. * @ignore */ this.unsubscribe = new Subject(); this.childOptionsChange = new Subject(); /** * @ignore */ this.isOpen = false; /** * @ignore */ this.selectedValue = ''; /** * @ignore */ this.selectedIndex = -1; this.iconName = 'keyboard_arrow_down'; } ngAfterViewInit() { setTimeout(() => { this.setInputsAndListenersOnOptions(); this.selectOptions?.changes .pipe(takeUntil(this.unsubscribe)) .subscribe(() => { setTimeout(() => { this.setInputsAndListenersOnOptions(); }); }); }); } ngOnDestroy() { this.unsubscribe.next(); this.unsubscribe.complete(); this.childOptionsChange.complete(); } setInputsAndListenersOnOptions() { this.childOptionsChange.next(); const unsubscribeCond$ = race(this.unsubscribe, this.childOptionsChange); this.selectOptions?.forEach((item) => { item.role = 'option'; if (this.selectedValue?.trim() !== '') { item.updateSelected(this.selectedValue === item.getValue()); } item.itemClick .pipe(takeUntil(unsubscribeCond$)) .subscribe(() => { const value = item.getValue(); this.writeValue(value); this._onChange(value); this.closeList(); setTimeout(() => { this.buttonEl?.nativeElement.focus(); }); }); }); // Get selected index on mount based on current value this.selectedIndex = this.selectOptions?.toArray().findIndex(opt => { return opt.getValue() === this.selectedValue; }); this.cdr.markForCheck(); } /** * @ignore */ getAdditionalClass() { const classes = ['nj-form-item--select', 'nj-form-item--custom-list']; if (this.isOpen) { classes.push('nj-form-item--open'); } if (this.customLabel?.templateRef) { classes.push('nj-form-item--custom-label'); } return classes.join(' '); } getSubscriptId() { return `${this.inputId}-subscript`; } getInstructionsId() { return `${this.inputId}-instructions`; } getDescriptionId() { return `${this.getSubscriptId()} ${this.getInstructionsId()}`; } /** * Get index of the selected value */ indexForValue(value) { return this.selectOptions ?.toArray() .findIndex((item) => item.getValue() === value); } openList() { this.isOpen = true; this.focusedIndex = this.selectedIndex; setTimeout(() => { if (this.selectedIndex === -1) { // Focus the `ul` element this.listEl?.rootEl.nativeElement.focus(); // The scrolling element is not the `ul` node but the `nj-list-group` this.listEl?.element.nativeElement.scrollTo({ top: 0 }); } }); } closeList() { this.isOpen = false; } toggleIsOpen() { if (this.isOpen) { this.closeList(); } else { this.openList(); } } /** * Index of the currently focused option. */ get focusedIndex() { return this.selectOptions ?.toArray() .findIndex((item) => this.document.activeElement === item.el.nativeElement); } set focusedIndex(value) { this.selectOptions?.forEach((el, i) => { el.ariaSelected = i === value; }); setTimeout(() => { if (value !== -1) { this.selectOptions?.get(value).el.nativeElement.focus(); } }); } handleListKeydown(e) { // Escape key closes the list and focuses the button if (e.code === SelectComponent.ESCAPE_CODE) { this.closeList(); setTimeout(() => { this.buttonEl?.nativeElement.focus(); }); } // Navigate between options and set `focusedIndex` if (e.code === SelectComponent.UP_CODE) { e.preventDefault(); // Dont loop back to the end of the list if (this.focusedIndex > 0) { this.focusedIndex -= 1; } } if (e.code === SelectComponent.DOWN_CODE) { e.preventDefault(); // Dont loop back to the beginning of the list if (this.focusedIndex < this.selectOptions?.length - 1) { this.focusedIndex += 1; } } // Select the current `focusedIndex` option if (e.code === SelectComponent.ENTER_CODE) { e.preventDefault(); if (this.focusedIndex !== -1) { const value = this.selectOptions?.get(this.focusedIndex).getValue(); this.writeValue(value); this._onChange(value); } this.closeList(); setTimeout(() => { this.buttonEl?.nativeElement.focus(); }); } // Jump to first option matching first letter if (SelectComponent.ALPHA_NUMERIC_REGEX.test(e.key)) { const goToIndex = this.selectOptions ?.toArray() .findIndex((item) => item.getValue()[0].toLowerCase() === e.key.toLowerCase()); if (goToIndex !== -1) { this.focusedIndex = goToIndex; } } } handleFocusout(e) { if (!this.element.nativeElement?.contains(e.relatedTarget)) { this.closeList(); if (this._onTouched) { this._onTouched(); } } } /** * Implemented as part of ControlValueAccessor. * @ignore */ registerOnChange(fn) { this._onChange = fn; } /** * Implemented as part of ControlValueAccessor. * @ignore */ registerOnTouched(fn) { this._onTouched = fn; } /** * Implemented as part of ControlValueAccessor. * @ignore */ setDisabledState(isDisabled) { if (!this.selectedValue) { return; } this.isDisabled = isDisabled; } /** * Implemented as part of ControlValueAccessor. * @ignore */ writeValue(value) { this.selectedValue = value; this.selectedIndex = this.indexForValue(value); this.selectOptions?.forEach((item) => { item.updateSelected(item.getValue() === value); }); this.cdr.markForCheck(); } get customLabelContext() { const value = this.selectedValue; const index = this.selectedIndex; return { $implicit: value, value, index }; } /** * Label (≠ value) of selected option * @ignore */ get selectedLabel() { return this.selectOptions?.get(this.selectedIndex)?.getLabel() ?? ''; } /** * Aria-label for the trigger button element. * @ignore */ get buttonLabel() { return `${this.fieldLabel} - ${this.customLabelEl?.nativeElement.innerText || this.selectedValue || this.buttonDefaultValueLabel}`; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SelectComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: SelectComponent, isStandalone: true, selector: "nj-select", inputs: { iconName: "iconName", fieldLabel: "fieldLabel", listNavigationLabel: "listNavigationLabel", buttonDefaultValueLabel: "buttonDefaultValueLabel" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SelectComponent), multi: true, }, ], queries: [{ propertyName: "customLabel", first: true, predicate: CustomLabelDirective, descendants: true }, { propertyName: "selectOptions", predicate: ListItemComponent, descendants: true }], viewQueries: [{ propertyName: "buttonEl", first: true, predicate: ["button"], descendants: true }, { propertyName: "customLabelEl", first: true, predicate: ["customLabelEl"], descendants: true }, { propertyName: "listEl", first: true, predicate: ListGroupComponent, descendants: true }], usesInheritance: true, ngImport: i0, template: "<nj-form-item\n [hasError]=\"hasError\"\n [hasSuccess]=\"hasSuccess\"\n [hasHint]=\"hasHint\"\n [isDisabled]=\"isDisabled\"\n [hasCustomIcon]=\"hasCustomIcon\"\n [isFloatingLabel]=\"isFloatingLabel\"\n [iconName]=\"iconName\"\n [size]=\"size\"\n [isSelect]=\"true\"\n [additionalClass]=\"getAdditionalClass()\"\n [inputId]=\"inputId\"\n (focusout)=\"handleFocusout($event)\"\n>\n <input\n type=\"text\"\n readonly\n [value]=\"selectedLabel\"\n [attr.id]=\"inputId\"\n [disabled]=\"isDisabled\"\n tabindex=\"-1\"\n aria-hidden=\"true\"\n njFormField\n />\n <ng-content njFormLabel select=\"[njFormLabel]\"></ng-content>\n <ng-content njFormSubscript select=\"[njFormSubscript]\"></ng-content>\n <ng-container njFormAdditionalContent>\n <div\n *ngIf=\"selectedValue && customLabel?.templateRef\"\n #customLabelEl\n class=\"nj-form-item__custom-label\"\n aria-hidden=\"true\"\n njFormField>\n <ng-container [ngTemplateOutlet]=\"customLabel.templateRef\"\n [ngTemplateOutletContext]=\"customLabelContext\"></ng-container>\n </div>\n <p [id]=\"getInstructionsId()\" hidden>{{ listNavigationLabel }}</p>\n <button\n #button\n type=\"button\"\n class=\"nj-form-item__custom-list-button\"\n aria-haspopup=\"listbox\"\n [attr.aria-expanded]=\"isOpen\"\n role=\"combobox\"\n [attr.aria-label]=\"buttonLabel\"\n [attr.tabindex]=\"isOpen ? -1 : null\"\n [attr.aria-describedby]=\"getDescriptionId()\"\n (click)=\"toggleIsOpen()\"\n ></button>\n <nj-list-group\n class=\"nj-form-item__list nj-form-item__list--no-animate\"\n [hidden]=\"!isOpen\"\n [@transformList]=\"isOpen ? 'open': 'void'\"\n [isDense]=\"true\"\n [hasBorder]=\"false\"\n [isClickable]=\"true\"\n [isCustomSelectList]=\"true\"\n tabindex=\"-1\"\n [ariaLabel]=\"fieldLabel\"\n (keydown)=\"handleListKeydown($event)\"\n >\n <ng-content select=\"[njSelectOptions]\"></ng-content>\n </nj-list-group>\n </ng-container>\n</nj-form-item>\n", styles: [":host{display:block;width:100%;height:100%}.nj-form-item__custom-list-button{outline:none}.nj-form-item--custom-label input{z-index:-1;transform:scale(0);opacity:0}.nj-form-item--custom-label div.nj-form-item__custom-label{position:absolute;inset:0}\n"], dependencies: [{ kind: "component", type: ListGroupComponent, selector: "nj-list-group", inputs: ["listId", "isClickable", "isCheckboxList", "hasBorder", "isDense", "isCustomSelectList", "ariaLabel", "isMultiSelect"] }, { kind: "component", type: FormItemComponent, selector: "nj-form-item", inputs: ["inputId", "size", "isFloatingLabel", "isDisabled", "isRequired", "hasSuccess", "hasError", "hasHint", "hasCustomIcon", "isIconClickable", "iconName", "additionalClass", "passwordButtonLabelShow", "passwordButtonLabelHide", "passwordNoticeIsVisible", "passwordNoticeIsHidden", "isSelect"], outputs: ["iconClick", "iconKeydown", "wrapperClick"] }, { kind: "directive", type: FormFieldDirective, selector: "input[njFormField], textarea[njFormField], select[njFormField], nj-select[njFormField], div[njFormField]", exportAs: ["njFormField"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], animations: [selectAnimations.transformList], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SelectComponent, decorators: [{ type: Component, args: [{ selector: 'nj-select', changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SelectComponent), multi: true, }, ], animations: [selectAnimations.transformList], encapsulation: ViewEncapsulation.None, standalone: true, imports: [ListGroupComponent, FormItemComponent, FormFieldDirective, CommonModule], template: "<nj-form-item\n [hasError]=\"hasError\"\n [hasSuccess]=\"hasSuccess\"\n [hasHint]=\"hasHint\"\n [isDisabled]=\"isDisabled\"\n [hasCustomIcon]=\"hasCustomIcon\"\n [isFloatingLabel]=\"isFloatingLabel\"\n [iconName]=\"iconName\"\n [size]=\"size\"\n [isSelect]=\"true\"\n [additionalClass]=\"getAdditionalClass()\"\n [inputId]=\"inputId\"\n (focusout)=\"handleFocusout($event)\"\n>\n <input\n type=\"text\"\n readonly\n [value]=\"selectedLabel\"\n [attr.id]=\"inputId\"\n [disabled]=\"isDisabled\"\n tabindex=\"-1\"\n aria-hidden=\"true\"\n njFormField\n />\n <ng-content njFormLabel select=\"[njFormLabel]\"></ng-content>\n <ng-content njFormSubscript select=\"[njFormSubscript]\"></ng-content>\n <ng-container njFormAdditionalContent>\n <div\n *ngIf=\"selectedValue && customLabel?.templateRef\"\n #customLabelEl\n class=\"nj-form-item__custom-label\"\n aria-hidden=\"true\"\n njFormField>\n <ng-container [ngTemplateOutlet]=\"customLabel.templateRef\"\n [ngTemplateOutletContext]=\"customLabelContext\"></ng-container>\n </div>\n <p [id]=\"getInstructionsId()\" hidden>{{ listNavigationLabel }}</p>\n <button\n #button\n type=\"button\"\n class=\"nj-form-item__custom-list-button\"\n aria-haspopup=\"listbox\"\n [attr.aria-expanded]=\"isOpen\"\n role=\"combobox\"\n [attr.aria-label]=\"buttonLabel\"\n [attr.tabindex]=\"isOpen ? -1 : null\"\n [attr.aria-describedby]=\"getDescriptionId()\"\n (click)=\"toggleIsOpen()\"\n ></button>\n <nj-list-group\n class=\"nj-form-item__list nj-form-item__list--no-animate\"\n [hidden]=\"!isOpen\"\n [@transformList]=\"isOpen ? 'open': 'void'\"\n [isDense]=\"true\"\n [hasBorder]=\"false\"\n [isClickable]=\"true\"\n [isCustomSelectList]=\"true\"\n tabindex=\"-1\"\n [ariaLabel]=\"fieldLabel\"\n (keydown)=\"handleListKeydown($event)\"\n >\n <ng-content select=\"[njSelectOptions]\"></ng-content>\n </nj-list-group>\n </ng-container>\n</nj-form-item>\n", styles: [":host{display:block;width:100%;height:100%}.nj-form-item__custom-list-button{outline:none}.nj-form-item--custom-label input{z-index:-1;transform:scale(0);opacity:0}.nj-form-item--custom-label div.nj-form-item__custom-label{position:absolute;inset:0}\n"] }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT] }] }]; }, propDecorators: { iconName: [{ type: Input }], fieldLabel: [{ type: Input }], listNavigationLabel: [{ type: Input }], buttonDefaultValueLabel: [{ type: Input }], buttonEl: [{ type: ViewChild, args: ['button'] }], customLabelEl: [{ type: ViewChild, args: ['customLabelEl'] }], listEl: [{ type: ViewChild, args: [ListGroupComponent] }], customLabel: [{ type: ContentChild, args: [CustomLabelDirective] }], selectOptions: [{ type: ContentChildren, args: [ListItemComponent, { descendants: true }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../src/components/select/select.component.ts","../../../../src/components/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {CommonModule, DOCUMENT} from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  ElementRef,\n  forwardRef,\n  Inject,\n  Input,\n  OnDestroy,\n  QueryList,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\nimport {finalize, race, Subject, takeUntil} from 'rxjs';\nimport {selectAnimations} from '../../shared/animations';\nimport {CustomLabelDirective} from '../custom-label/custom-label.directive';\nimport {FormFieldDirective} from '../form-field/form-field.directive';\nimport {FormItemComponent} from '../form-item/form-item.component';\nimport {ListGroupComponent} from '../list-group/list-group.component';\nimport {ListItemComponent} from '../list-item/list-item.component';\nimport {SelectCustomLabelContext} from './select-custom-label-context.model';\n\n@Component({\n  selector: 'nj-select',\n  templateUrl: './select.component.html',\n  styleUrls: ['./select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => SelectComponent),\n      multi: true,\n    },\n  ],\n  animations: [selectAnimations.transformList],\n  encapsulation: ViewEncapsulation.None,\n  standalone: true,\n  imports: [ListGroupComponent, FormItemComponent, FormFieldDirective, CommonModule]\n})\nexport class SelectComponent\n  extends FormItemComponent\n  implements AfterViewInit, ControlValueAccessor, OnDestroy {\n  private static readonly ESCAPE_CODE = 'Escape';\n  private static readonly ENTER_CODE = 'Enter';\n  private static readonly UP_CODE = 'ArrowUp';\n  private static readonly DOWN_CODE = 'ArrowDown';\n  /*\n    Regex matching every alpha-numeric characters.\n\n    \\d : every digits\n    \\p{Letter} : every letters in the latin alphabet including letters with diacritics\n\n    The \"u\" flag enables unicode mode required to use `\\p{Letter}`.\n\n    See :\n    - https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions/Unicode_Property_Escapes#general_categories\n    - https://unicode.org/reports/tr18/#General_Category_Property\n  */\n  private static readonly ALPHA_NUMERIC_REGEX = /^[\\d\\p{Letter}]$/u;\n\n  /**\n   * @ignore\n   */\n  private _onChange = (_: any): void => {\n  };\n\n  /**\n   * @ignore\n   */\n  private _onTouched = (): void => {\n  };\n\n  /**\n   * Notifier used to stop items click event subscription.\n   * @ignore\n   */\n  private unsubscribe = new Subject<void>();\n\n  private childOptionsChange = new Subject<void>();\n\n  /**\n   * @ignore\n   */\n  isOpen = false;\n\n  /**\n   * @ignore\n   */\n  selectedValue = '';\n\n  /**\n   * @ignore\n   */\n  selectedIndex = -1;\n\n  @Input() iconName = 'keyboard_arrow_down';\n\n  /**\n   * Label used for accessibility related attributes on button and list.\n   * Should be the same value (text only) as the `<label>` element\n   */\n  @Input() fieldLabel: string;\n\n  /**\n   * Instructions on how to navigate the list. It is append after the input label.\n   * @example \"Use up and down arrows and Enter to select a value\"\n   */\n  @Input() listNavigationLabel: string;\n\n  /**\n   * Button default label when no value is selected. It is append after the input label.\n   * @example \"Select a value\"\n   */\n  @Input() buttonDefaultValueLabel: string;\n\n  /**\n   * Trigger button to toggle the list\n   * @ignore\n   */\n  @ViewChild('button') buttonEl: ElementRef<HTMLButtonElement>;\n\n  @ViewChild('customLabelEl') protected customLabelEl: ElementRef<HTMLElement>;\n\n  /**\n   * List containing options\n   * @ignore\n   */\n  @ViewChild(ListGroupComponent) listEl: ListGroupComponent;\n\n  /**\n   * Label to display instead of raw text value\n   * @ignore\n   * @example\n   * <ng-template njCustomLabel let-value let-index=\"index\">\n   *  Value: {{value}} - Index: {{index}}\n   * </ng-template>\n   *\n   * @example\n   * <span *njCustomLabel=\"let value;let index=index\">\n   *  Value: {{value}} - Index: {{index}}\n   * </span>\n   */\n  @ContentChild(CustomLabelDirective) protected customLabel?: CustomLabelDirective<SelectCustomLabelContext>;\n\n  /**\n   * Option items\n   * @ignore\n   */\n  @ContentChildren(ListItemComponent, {descendants: true})\n  selectOptions: QueryList<ListItemComponent>;\n\n  constructor(\n    private readonly element: ElementRef<HTMLElement>,\n    private readonly cdr: ChangeDetectorRef,\n    @Inject(DOCUMENT) private document\n  ) {\n    super();\n  }\n\n  ngAfterViewInit() {\n    setTimeout(() => {\n      this.setInputsAndListenersOnOptions();\n\n      this.selectOptions?.changes\n        .pipe(takeUntil(this.unsubscribe))\n        .subscribe(() => {\n          setTimeout(() => {\n            this.setInputsAndListenersOnOptions();\n          });\n        });\n    });\n  }\n\n  ngOnDestroy() {\n    this.unsubscribe.next();\n    this.unsubscribe.complete();\n    this.childOptionsChange.complete();\n  }\n\n  setInputsAndListenersOnOptions() {\n    this.childOptionsChange.next();\n    const unsubscribeCond$ = race(this.unsubscribe, this.childOptionsChange);\n\n    this.selectOptions?.forEach((item) => {\n      item.role = 'option';\n\n      if (this.selectedValue?.trim() !== '') {\n        item.updateSelected(this.selectedValue === item.getValue());\n      }\n\n\n      item.itemClick\n        .pipe(\n          takeUntil(unsubscribeCond$),\n        )\n        .subscribe(() => {\n          const value = item.getValue();\n          this.writeValue(value);\n          this._onChange(value);\n          this.closeList();\n\n          setTimeout(() => {\n            this.buttonEl?.nativeElement.focus();\n          });\n        });\n    });\n\n    // Get selected index on mount based on current value\n    this.selectedIndex = this.selectOptions?.toArray().findIndex(opt => {\n      return opt.getValue() === this.selectedValue;\n    });\n\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * @ignore\n   */\n  getAdditionalClass(): string {\n    const classes = ['nj-form-item--select', 'nj-form-item--custom-list'];\n    if (this.isOpen) {\n      classes.push('nj-form-item--open');\n    }\n\n    if (this.customLabel?.templateRef) {\n      classes.push('nj-form-item--custom-label');\n    }\n    return classes.join(' ');\n  }\n\n  getSubscriptId(): string {\n    return `${this.inputId}-subscript`;\n  }\n\n  getInstructionsId(): string {\n    return `${this.inputId}-instructions`;\n  }\n\n  getDescriptionId(): string {\n    return `${this.getSubscriptId()} ${this.getInstructionsId()}`;\n  }\n\n  /**\n   * Get index of the selected value\n   */\n  private indexForValue(value: string): number {\n    return this.selectOptions\n      ?.toArray()\n      .findIndex((item) => item.getValue() === value);\n  }\n\n  private openList() {\n    this.isOpen = true;\n    this.focusedIndex = this.selectedIndex;\n\n    setTimeout(() => {\n      if (this.selectedIndex === -1) {\n        // Focus the `ul` element\n        this.listEl?.rootEl.nativeElement.focus();\n        // The scrolling element is not the `ul` node but the `nj-list-group`\n        this.listEl?.element.nativeElement.scrollTo({top: 0});\n      }\n    });\n  }\n\n  private closeList() {\n    this.isOpen = false;\n  }\n\n  toggleIsOpen() {\n    if (this.isOpen) {\n      this.closeList();\n    } else {\n      this.openList();\n    }\n  }\n\n  /**\n   * Index of the currently focused option.\n   */\n  private get focusedIndex(): number {\n    return this.selectOptions\n      ?.toArray()\n      .findIndex(\n        (item) => this.document.activeElement === item.el.nativeElement\n      );\n  }\n\n  private set focusedIndex(value: number) {\n    this.selectOptions?.forEach((el, i) => {\n      el.ariaSelected = i === value;\n    });\n\n    setTimeout(() => {\n      if (value !== -1) {\n        this.selectOptions?.get(value).el.nativeElement.focus();\n      }\n    });\n  }\n\n  handleListKeydown(e: KeyboardEvent) {\n    // Escape key closes the list and focuses the button\n    if (e.code === SelectComponent.ESCAPE_CODE) {\n      this.closeList();\n      setTimeout(() => {\n        this.buttonEl?.nativeElement.focus();\n      });\n    }\n\n    // Navigate between options and set `focusedIndex`\n    if (e.code === SelectComponent.UP_CODE) {\n      e.preventDefault();\n      // Dont loop back to the end of the list\n      if (this.focusedIndex > 0) {\n        this.focusedIndex -= 1;\n      }\n    }\n\n    if (e.code === SelectComponent.DOWN_CODE) {\n      e.preventDefault();\n      // Dont loop back to the beginning of the list\n      if (this.focusedIndex < this.selectOptions?.length - 1) {\n        this.focusedIndex += 1;\n      }\n    }\n\n    // Select the current `focusedIndex` option\n    if (e.code === SelectComponent.ENTER_CODE) {\n      e.preventDefault();\n      if (this.focusedIndex !== -1) {\n        const value = this.selectOptions?.get(this.focusedIndex).getValue();\n        this.writeValue(value);\n        this._onChange(value);\n      }\n      this.closeList();\n\n      setTimeout(() => {\n        this.buttonEl?.nativeElement.focus();\n      });\n    }\n\n    // Jump to first option matching first letter\n    if (SelectComponent.ALPHA_NUMERIC_REGEX.test(e.key)) {\n      const goToIndex = this.selectOptions\n        ?.toArray()\n        .findIndex(\n          (item) => item.getValue()[0].toLowerCase() === e.key.toLowerCase()\n        );\n\n      if (goToIndex !== -1) {\n        this.focusedIndex = goToIndex;\n      }\n    }\n  }\n\n  handleFocusout(e: FocusEvent) {\n    if (!this.element.nativeElement?.contains(e.relatedTarget as Node)) {\n      this.closeList();\n\n      if (this._onTouched) {\n        this._onTouched();\n      }\n    }\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   * @ignore\n   */\n  registerOnChange(fn: any): void {\n    this._onChange = fn;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   * @ignore\n   */\n  registerOnTouched(fn: any): void {\n    this._onTouched = fn;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   * @ignore\n   */\n  setDisabledState(isDisabled: boolean): void {\n    if (!this.selectedValue) {\n      return;\n    }\n    this.isDisabled = isDisabled;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   * @ignore\n   */\n  writeValue(value: string): void {\n    this.selectedValue = value;\n    this.selectedIndex = this.indexForValue(value);\n    this.selectOptions?.forEach((item) => {\n      item.updateSelected(item.getValue() === value);\n    });\n    this.cdr.markForCheck();\n  }\n\n  protected get customLabelContext(): SelectCustomLabelContext {\n    const value = this.selectedValue;\n    const index = this.selectedIndex;\n    return {$implicit: value, value, index};\n  }\n\n  /**\n   * Label (≠ value) of selected option\n   * @ignore\n   */\n  get selectedLabel(): string {\n    return this.selectOptions?.get(this.selectedIndex)?.getLabel() ?? '';\n  }\n\n  /**\n   * Aria-label for the trigger button element.\n   * @ignore\n   */\n  get buttonLabel(): string {\n    return `${this.fieldLabel} - ${\n      this.customLabelEl?.nativeElement.innerText || this.selectedValue || this.buttonDefaultValueLabel\n    }`;\n  }\n}\n","<nj-form-item\n  [hasError]=\"hasError\"\n  [hasSuccess]=\"hasSuccess\"\n  [hasHint]=\"hasHint\"\n  [isDisabled]=\"isDisabled\"\n  [hasCustomIcon]=\"hasCustomIcon\"\n  [isFloatingLabel]=\"isFloatingLabel\"\n  [iconName]=\"iconName\"\n  [size]=\"size\"\n  [isSelect]=\"true\"\n  [additionalClass]=\"getAdditionalClass()\"\n  [inputId]=\"inputId\"\n  (focusout)=\"handleFocusout($event)\"\n>\n  <input\n    type=\"text\"\n    readonly\n    [value]=\"selectedLabel\"\n    [attr.id]=\"inputId\"\n    [disabled]=\"isDisabled\"\n    tabindex=\"-1\"\n    aria-hidden=\"true\"\n    njFormField\n  />\n  <ng-content njFormLabel select=\"[njFormLabel]\"></ng-content>\n  <ng-content njFormSubscript select=\"[njFormSubscript]\"></ng-content>\n  <ng-container njFormAdditionalContent>\n    <div\n      *ngIf=\"selectedValue && customLabel?.templateRef\"\n      #customLabelEl\n      class=\"nj-form-item__custom-label\"\n      aria-hidden=\"true\"\n      njFormField>\n      <ng-container [ngTemplateOutlet]=\"customLabel.templateRef\"\n                    [ngTemplateOutletContext]=\"customLabelContext\"></ng-container>\n    </div>\n    <p [id]=\"getInstructionsId()\" hidden>{{ listNavigationLabel }}</p>\n    <button\n      #button\n      type=\"button\"\n      class=\"nj-form-item__custom-list-button\"\n      aria-haspopup=\"listbox\"\n      [attr.aria-expanded]=\"isOpen\"\n      role=\"combobox\"\n      [attr.aria-label]=\"buttonLabel\"\n      [attr.tabindex]=\"isOpen ? -1 : null\"\n      [attr.aria-describedby]=\"getDescriptionId()\"\n      (click)=\"toggleIsOpen()\"\n    ></button>\n    <nj-list-group\n      class=\"nj-form-item__list nj-form-item__list--no-animate\"\n      [hidden]=\"!isOpen\"\n      [@transformList]=\"isOpen ? 'open': 'void'\"\n      [isDense]=\"true\"\n      [hasBorder]=\"false\"\n      [isClickable]=\"true\"\n      [isCustomSelectList]=\"true\"\n      tabindex=\"-1\"\n      [ariaLabel]=\"fieldLabel\"\n      (keydown)=\"handleListKeydown($event)\"\n    >\n      <ng-content select=\"[njSelectOptions]\"></ng-content>\n    </nj-list-group>\n  </ng-container>\n</nj-form-item>\n"]}