UNPKG

@angular/cdk

Version:

Angular Material Component Development Kit

862 lines 107 kB
/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import { ChangeDetectorRef, ContentChildren, Directive, ElementRef, forwardRef, inject, Input, Output, QueryList, } from '@angular/core'; import { ActiveDescendantKeyManager } from '@angular/cdk/a11y'; import { A, DOWN_ARROW, END, ENTER, hasModifierKey, HOME, LEFT_ARROW, RIGHT_ARROW, SPACE, UP_ARROW, } from '@angular/cdk/keycodes'; import { coerceArray, coerceBooleanProperty } from '@angular/cdk/coercion'; import { SelectionModel } from '@angular/cdk/collections'; import { defer, merge, Subject } from 'rxjs'; import { filter, map, startWith, switchMap, takeUntil } from 'rxjs/operators'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { Directionality } from '@angular/cdk/bidi'; import * as i0 from "@angular/core"; /** The next id to use for creating unique DOM IDs. */ let nextId = 0; /** * An implementation of SelectionModel that internally always represents the selection as a * multi-selection. This is necessary so that we can recover the full selection if the user * switches the listbox from single-selection to multi-selection after initialization. * * This selection model may report multiple selected values, even if it is in single-selection * mode. It is up to the user (CdkListbox) to check for invalid selections. */ class ListboxSelectionModel extends SelectionModel { constructor(multiple = false, initiallySelectedValues, emitChanges = true, compareWith) { super(true, initiallySelectedValues, emitChanges, compareWith); this.multiple = multiple; } isMultipleSelection() { return this.multiple; } select(...values) { // The super class is always in multi-selection mode, so we need to override the behavior if // this selection model actually belongs to a single-selection listbox. if (this.multiple) { return super.select(...values); } else { return super.setSelection(...values); } } } /** A selectable option in a listbox. */ class CdkOption { constructor() { this._generatedId = `cdk-option-${nextId++}`; this._disabled = false; /** The option's host element */ this.element = inject(ElementRef).nativeElement; /** The parent listbox this option belongs to. */ this.listbox = inject(CdkListbox); /** Emits when the option is destroyed. */ this.destroyed = new Subject(); /** Emits when the option is clicked. */ this._clicked = new Subject(); } /** The id of the option's host element. */ get id() { return this._id || this._generatedId; } set id(value) { this._id = value; } /** Whether this option is disabled. */ get disabled() { return this.listbox.disabled || this._disabled; } set disabled(value) { this._disabled = coerceBooleanProperty(value); } /** The tabindex of the option when it is enabled. */ get enabledTabIndex() { return this._enabledTabIndex === undefined ? this.listbox.enabledTabIndex : this._enabledTabIndex; } set enabledTabIndex(value) { this._enabledTabIndex = value; } ngOnDestroy() { this.destroyed.next(); this.destroyed.complete(); } /** Whether this option is selected. */ isSelected() { return this.listbox.isSelected(this); } /** Whether this option is active. */ isActive() { return this.listbox.isActive(this); } /** Toggle the selected state of this option. */ toggle() { this.listbox.toggle(this); } /** Select this option if it is not selected. */ select() { this.listbox.select(this); } /** Deselect this option if it is selected. */ deselect() { this.listbox.deselect(this); } /** Focus this option. */ focus() { this.element.focus(); } /** Get the label for this element which is required by the FocusableOption interface. */ getLabel() { return (this.typeaheadLabel ?? this.element.textContent?.trim()) || ''; } /** * No-op implemented as a part of `Highlightable`. * @docs-private */ setActiveStyles() { } /** * No-op implemented as a part of `Highlightable`. * @docs-private */ setInactiveStyles() { } /** Handle focus events on the option. */ _handleFocus() { // Options can wind up getting focused in active descendant mode if the user clicks on them. // In this case, we push focus back to the parent listbox to prevent an extra tab stop when // the user performs a shift+tab. if (this.listbox.useActiveDescendant) { this.listbox._setActiveOption(this); this.listbox.focus(); } } /** Get the tabindex for this option. */ _getTabIndex() { if (this.listbox.useActiveDescendant || this.disabled) { return -1; } return this.isActive() ? this.enabledTabIndex : -1; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkOption, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.0", type: CdkOption, isStandalone: true, selector: "[cdkOption]", inputs: { id: "id", value: ["cdkOption", "value"], typeaheadLabel: ["cdkOptionTypeaheadLabel", "typeaheadLabel"], disabled: ["cdkOptionDisabled", "disabled"], enabledTabIndex: ["tabindex", "enabledTabIndex"] }, host: { attributes: { "role": "option" }, listeners: { "click": "_clicked.next($event)", "focus": "_handleFocus()" }, properties: { "id": "id", "attr.aria-selected": "isSelected()", "attr.tabindex": "_getTabIndex()", "attr.aria-disabled": "disabled", "class.cdk-option-active": "isActive()" }, classAttribute: "cdk-option" }, exportAs: ["cdkOption"], ngImport: i0 }); } } export { CdkOption }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkOption, decorators: [{ type: Directive, args: [{ selector: '[cdkOption]', standalone: true, exportAs: 'cdkOption', host: { 'role': 'option', 'class': 'cdk-option', '[id]': 'id', '[attr.aria-selected]': 'isSelected()', '[attr.tabindex]': '_getTabIndex()', '[attr.aria-disabled]': 'disabled', '[class.cdk-option-active]': 'isActive()', '(click)': '_clicked.next($event)', '(focus)': '_handleFocus()', }, }] }], propDecorators: { id: [{ type: Input }], value: [{ type: Input, args: ['cdkOption'] }], typeaheadLabel: [{ type: Input, args: ['cdkOptionTypeaheadLabel'] }], disabled: [{ type: Input, args: ['cdkOptionDisabled'] }], enabledTabIndex: [{ type: Input, args: ['tabindex'] }] } }); class CdkListbox { constructor() { this._generatedId = `cdk-listbox-${nextId++}`; this._disabled = false; this._useActiveDescendant = false; this._orientation = 'vertical'; this._navigationWrapDisabled = false; this._navigateDisabledOptions = false; /** Emits when the selected value(s) in the listbox change. */ this.valueChange = new Subject(); /** The selection model used by the listbox. */ this.selectionModel = new ListboxSelectionModel(); /** Emits when the listbox is destroyed. */ this.destroyed = new Subject(); /** The host element of the listbox. */ this.element = inject(ElementRef).nativeElement; /** The change detector for this listbox. */ this.changeDetectorRef = inject(ChangeDetectorRef); /** Whether the currently selected value in the selection model is invalid. */ this._invalid = false; /** The last user-triggered option. */ this._lastTriggered = null; /** Callback called when the listbox has been touched */ this._onTouched = () => { }; /** Callback called when the listbox value changes */ this._onChange = () => { }; /** Emits when an option has been clicked. */ this._optionClicked = defer(() => this.options.changes.pipe(startWith(this.options), switchMap(options => merge(...options.map(option => option._clicked.pipe(map(event => ({ option, event })))))))); /** The directionality of the page. */ this._dir = inject(Directionality, { optional: true }); /** A predicate that skips disabled options. */ this._skipDisabledPredicate = (option) => option.disabled; /** A predicate that does not skip any options. */ this._skipNonePredicate = () => false; /** Whether the listbox currently has focus. */ this._hasFocus = false; } /** The id of the option's host element. */ get id() { return this._id || this._generatedId; } set id(value) { this._id = value; } /** The tabindex to use when the listbox is enabled. */ get enabledTabIndex() { return this._enabledTabIndex === undefined ? 0 : this._enabledTabIndex; } set enabledTabIndex(value) { this._enabledTabIndex = value; } /** The value selected in the listbox, represented as an array of option values. */ get value() { return this._invalid ? [] : this.selectionModel.selected; } set value(value) { this._setSelection(value); } /** * Whether the listbox allows multiple options to be selected. If the value switches from `true` * to `false`, and more than one option is selected, all options are deselected. */ get multiple() { return this.selectionModel.multiple; } set multiple(value) { this.selectionModel.multiple = coerceBooleanProperty(value); if (this.options) { this._updateInternalValue(); } } /** Whether the listbox is disabled. */ get disabled() { return this._disabled; } set disabled(value) { this._disabled = coerceBooleanProperty(value); } /** Whether the listbox will use active descendant or will move focus onto the options. */ get useActiveDescendant() { return this._useActiveDescendant; } set useActiveDescendant(shouldUseActiveDescendant) { this._useActiveDescendant = coerceBooleanProperty(shouldUseActiveDescendant); } /** The orientation of the listbox. Only affects keyboard interaction, not visual layout. */ get orientation() { return this._orientation; } set orientation(value) { this._orientation = value === 'horizontal' ? 'horizontal' : 'vertical'; if (value === 'horizontal') { this.listKeyManager?.withHorizontalOrientation(this._dir?.value || 'ltr'); } else { this.listKeyManager?.withVerticalOrientation(); } } /** The function used to compare option values. */ get compareWith() { return this.selectionModel.compareWith; } set compareWith(fn) { this.selectionModel.compareWith = fn; } /** * Whether the keyboard navigation should wrap when the user presses arrow down on the last item * or arrow up on the first item. */ get navigationWrapDisabled() { return this._navigationWrapDisabled; } set navigationWrapDisabled(wrap) { this._navigationWrapDisabled = coerceBooleanProperty(wrap); this.listKeyManager?.withWrap(!this._navigationWrapDisabled); } /** Whether keyboard navigation should skip over disabled items. */ get navigateDisabledOptions() { return this._navigateDisabledOptions; } set navigateDisabledOptions(skip) { this._navigateDisabledOptions = coerceBooleanProperty(skip); this.listKeyManager?.skipPredicate(this._navigateDisabledOptions ? this._skipNonePredicate : this._skipDisabledPredicate); } ngAfterContentInit() { if (typeof ngDevMode === 'undefined' || ngDevMode) { this._verifyNoOptionValueCollisions(); this._verifyOptionValues(); } this._initKeyManager(); // Update the internal value whenever the options or the model value changes. merge(this.selectionModel.changed, this.options.changes) .pipe(startWith(null), takeUntil(this.destroyed)) .subscribe(() => this._updateInternalValue()); this._optionClicked .pipe(filter(({ option }) => !option.disabled), takeUntil(this.destroyed)) .subscribe(({ option, event }) => this._handleOptionClicked(option, event)); } ngOnDestroy() { this.listKeyManager?.destroy(); this.destroyed.next(); this.destroyed.complete(); } /** * Toggle the selected state of the given option. * @param option The option to toggle */ toggle(option) { this.toggleValue(option.value); } /** * Toggle the selected state of the given value. * @param value The value to toggle */ toggleValue(value) { if (this._invalid) { this.selectionModel.clear(false); } this.selectionModel.toggle(value); } /** * Select the given option. * @param option The option to select */ select(option) { this.selectValue(option.value); } /** * Select the given value. * @param value The value to select */ selectValue(value) { if (this._invalid) { this.selectionModel.clear(false); } this.selectionModel.select(value); } /** * Deselect the given option. * @param option The option to deselect */ deselect(option) { this.deselectValue(option.value); } /** * Deselect the given value. * @param value The value to deselect */ deselectValue(value) { if (this._invalid) { this.selectionModel.clear(false); } this.selectionModel.deselect(value); } /** * Set the selected state of all options. * @param isSelected The new selected state to set */ setAllSelected(isSelected) { if (!isSelected) { this.selectionModel.clear(); } else { if (this._invalid) { this.selectionModel.clear(false); } this.selectionModel.select(...this.options.map(option => option.value)); } } /** * Get whether the given option is selected. * @param option The option to get the selected state of */ isSelected(option) { return this.isValueSelected(option.value); } /** * Get whether the given option is active. * @param option The option to get the active state of */ isActive(option) { return !!(this.listKeyManager?.activeItem === option); } /** * Get whether the given value is selected. * @param value The value to get the selected state of */ isValueSelected(value) { if (this._invalid) { return false; } return this.selectionModel.isSelected(value); } /** * Registers a callback to be invoked when the listbox's value changes from user input. * @param fn The callback to register * @docs-private */ registerOnChange(fn) { this._onChange = fn; } /** * Registers a callback to be invoked when the listbox is blurred by the user. * @param fn The callback to register * @docs-private */ registerOnTouched(fn) { this._onTouched = fn; } /** * Sets the listbox's value. * @param value The new value of the listbox * @docs-private */ writeValue(value) { this._setSelection(value); this._verifyOptionValues(); } /** * Sets the disabled state of the listbox. * @param isDisabled The new disabled state * @docs-private */ setDisabledState(isDisabled) { this.disabled = isDisabled; } /** Focus the listbox's host element. */ focus() { this.element.focus(); } /** * Triggers the given option in response to user interaction. * - In single selection mode: selects the option and deselects any other selected option. * - In multi selection mode: toggles the selected state of the option. * @param option The option to trigger */ triggerOption(option) { if (option && !option.disabled) { this._lastTriggered = option; const changed = this.multiple ? this.selectionModel.toggle(option.value) : this.selectionModel.select(option.value); if (changed) { this._onChange(this.value); this.valueChange.next({ value: this.value, listbox: this, option: option, }); } } } /** * Trigger the given range of options in response to user interaction. * Should only be called in multi-selection mode. * @param trigger The option that was triggered * @param from The start index of the options to toggle * @param to The end index of the options to toggle * @param on Whether to toggle the option range on */ triggerRange(trigger, from, to, on) { if (this.disabled || (trigger && trigger.disabled)) { return; } this._lastTriggered = trigger; const isEqual = this.compareWith ?? Object.is; const updateValues = [...this.options] .slice(Math.max(0, Math.min(from, to)), Math.min(this.options.length, Math.max(from, to) + 1)) .filter(option => !option.disabled) .map(option => option.value); const selected = [...this.value]; for (const updateValue of updateValues) { const selectedIndex = selected.findIndex(selectedValue => isEqual(selectedValue, updateValue)); if (on && selectedIndex === -1) { selected.push(updateValue); } else if (!on && selectedIndex !== -1) { selected.splice(selectedIndex, 1); } } let changed = this.selectionModel.setSelection(...selected); if (changed) { this._onChange(this.value); this.valueChange.next({ value: this.value, listbox: this, option: trigger, }); } } /** * Sets the given option as active. * @param option The option to make active */ _setActiveOption(option) { this.listKeyManager.setActiveItem(option); } /** Called when the listbox receives focus. */ _handleFocus() { if (!this.useActiveDescendant) { if (this.selectionModel.selected.length > 0) { this._setNextFocusToSelectedOption(); } else { this.listKeyManager.setNextItemActive(); } this._focusActiveOption(); } } /** Called when the user presses keydown on the listbox. */ _handleKeydown(event) { if (this._disabled) { return; } const { keyCode } = event; const previousActiveIndex = this.listKeyManager.activeItemIndex; const ctrlKeys = ['ctrlKey', 'metaKey']; if (this.multiple && keyCode === A && hasModifierKey(event, ...ctrlKeys)) { // Toggle all options off if they're all selected, otherwise toggle them all on. this.triggerRange(null, 0, this.options.length - 1, this.options.length !== this.value.length); event.preventDefault(); return; } if (this.multiple && (keyCode === SPACE || keyCode === ENTER) && hasModifierKey(event, 'shiftKey')) { if (this.listKeyManager.activeItem && this.listKeyManager.activeItemIndex != null) { this.triggerRange(this.listKeyManager.activeItem, this._getLastTriggeredIndex() ?? this.listKeyManager.activeItemIndex, this.listKeyManager.activeItemIndex, !this.listKeyManager.activeItem.isSelected()); } event.preventDefault(); return; } if (this.multiple && keyCode === HOME && hasModifierKey(event, ...ctrlKeys) && hasModifierKey(event, 'shiftKey')) { const trigger = this.listKeyManager.activeItem; if (trigger) { const from = this.listKeyManager.activeItemIndex; this.listKeyManager.setFirstItemActive(); this.triggerRange(trigger, from, this.listKeyManager.activeItemIndex, !trigger.isSelected()); } event.preventDefault(); return; } if (this.multiple && keyCode === END && hasModifierKey(event, ...ctrlKeys) && hasModifierKey(event, 'shiftKey')) { const trigger = this.listKeyManager.activeItem; if (trigger) { const from = this.listKeyManager.activeItemIndex; this.listKeyManager.setLastItemActive(); this.triggerRange(trigger, from, this.listKeyManager.activeItemIndex, !trigger.isSelected()); } event.preventDefault(); return; } if (keyCode === SPACE || keyCode === ENTER) { this.triggerOption(this.listKeyManager.activeItem); event.preventDefault(); return; } const isNavKey = keyCode === UP_ARROW || keyCode === DOWN_ARROW || keyCode === LEFT_ARROW || keyCode === RIGHT_ARROW || keyCode === HOME || keyCode === END; this.listKeyManager.onKeydown(event); // Will select an option if shift was pressed while navigating to the option if (isNavKey && event.shiftKey && previousActiveIndex !== this.listKeyManager.activeItemIndex) { this.triggerOption(this.listKeyManager.activeItem); } } /** Called when a focus moves into the listbox. */ _handleFocusIn() { // Note that we use a `focusin` handler for this instead of the existing `focus` handler, // because focus won't land on the listbox if `useActiveDescendant` is enabled. this._hasFocus = true; } /** * Called when the focus leaves an element in the listbox. * @param event The focusout event */ _handleFocusOut(event) { const otherElement = event.relatedTarget; if (this.element !== otherElement && !this.element.contains(otherElement)) { this._onTouched(); this._hasFocus = false; this._setNextFocusToSelectedOption(); } } /** Get the id of the active option if active descendant is being used. */ _getAriaActiveDescendant() { return this._useActiveDescendant ? this.listKeyManager?.activeItem?.id : null; } /** Get the tabindex for the listbox. */ _getTabIndex() { if (this.disabled) { return -1; } return this.useActiveDescendant || !this.listKeyManager.activeItem ? this.enabledTabIndex : -1; } /** Initialize the key manager. */ _initKeyManager() { this.listKeyManager = new ActiveDescendantKeyManager(this.options) .withWrap(!this._navigationWrapDisabled) .withTypeAhead() .withHomeAndEnd() .withAllowedModifierKeys(['shiftKey']) .skipPredicate(this._navigateDisabledOptions ? this._skipNonePredicate : this._skipDisabledPredicate); if (this.orientation === 'vertical') { this.listKeyManager.withVerticalOrientation(); } else { this.listKeyManager.withHorizontalOrientation(this._dir?.value || 'ltr'); } if (this.selectionModel.selected.length) { Promise.resolve().then(() => this._setNextFocusToSelectedOption()); } this.listKeyManager.change.subscribe(() => this._focusActiveOption()); } /** Focus the active option. */ _focusActiveOption() { if (!this.useActiveDescendant) { this.listKeyManager.activeItem?.focus(); } this.changeDetectorRef.markForCheck(); } /** * Set the selected values. * @param value The list of new selected values. */ _setSelection(value) { if (this._invalid) { this.selectionModel.clear(false); } this.selectionModel.setSelection(...this._coerceValue(value)); if (!this._hasFocus) { this._setNextFocusToSelectedOption(); } } /** Sets the first selected option as first in the keyboard focus order. */ _setNextFocusToSelectedOption() { // Null check the options since they only get defined after `ngAfterContentInit`. const selected = this.options?.find(option => option.isSelected()); if (selected) { this.listKeyManager.updateActiveItem(selected); } } /** Update the internal value of the listbox based on the selection model. */ _updateInternalValue() { const indexCache = new Map(); this.selectionModel.sort((a, b) => { const aIndex = this._getIndexForValue(indexCache, a); const bIndex = this._getIndexForValue(indexCache, b); return aIndex - bIndex; }); const selected = this.selectionModel.selected; this._invalid = (!this.multiple && selected.length > 1) || !!this._getInvalidOptionValues(selected).length; this.changeDetectorRef.markForCheck(); } /** * Gets the index of the given value in the given list of options. * @param cache The cache of indices found so far * @param value The value to find * @return The index of the value in the options list */ _getIndexForValue(cache, value) { const isEqual = this.compareWith || Object.is; if (!cache.has(value)) { let index = -1; for (let i = 0; i < this.options.length; i++) { if (isEqual(value, this.options.get(i).value)) { index = i; break; } } cache.set(value, index); } return cache.get(value); } /** * Handle the user clicking an option. * @param option The option that was clicked. */ _handleOptionClicked(option, event) { event.preventDefault(); this.listKeyManager.setActiveItem(option); if (event.shiftKey && this.multiple) { this.triggerRange(option, this._getLastTriggeredIndex() ?? this.listKeyManager.activeItemIndex, this.listKeyManager.activeItemIndex, !option.isSelected()); } else { this.triggerOption(option); } } /** Verifies that no two options represent the same value under the compareWith function. */ _verifyNoOptionValueCollisions() { this.options.changes.pipe(startWith(this.options), takeUntil(this.destroyed)).subscribe(() => { const isEqual = this.compareWith ?? Object.is; for (let i = 0; i < this.options.length; i++) { const option = this.options.get(i); let duplicate = null; for (let j = i + 1; j < this.options.length; j++) { const other = this.options.get(j); if (isEqual(option.value, other.value)) { duplicate = other; break; } } if (duplicate) { // TODO(mmalerba): Link to docs about this. if (this.compareWith) { console.warn(`Found multiple CdkOption representing the same value under the given compareWith function`, { option1: option.element, option2: duplicate.element, compareWith: this.compareWith, }); } else { console.warn(`Found multiple CdkOption with the same value`, { option1: option.element, option2: duplicate.element, }); } return; } } }); } /** Verifies that the option values are valid. */ _verifyOptionValues() { if (this.options && (typeof ngDevMode === 'undefined' || ngDevMode)) { const selected = this.selectionModel.selected; const invalidValues = this._getInvalidOptionValues(selected); if (!this.multiple && selected.length > 1) { throw Error('Listbox cannot have more than one selected value in multi-selection mode.'); } if (invalidValues.length) { throw Error('Listbox has selected values that do not match any of its options.'); } } } /** * Coerces a value into an array representing a listbox selection. * @param value The value to coerce * @return An array */ _coerceValue(value) { return value == null ? [] : coerceArray(value); } /** * Get the sublist of values that do not represent valid option values in this listbox. * @param values The list of values * @return The sublist of values that are not valid option values */ _getInvalidOptionValues(values) { const isEqual = this.compareWith || Object.is; const validValues = (this.options || []).map(option => option.value); return values.filter(value => !validValues.some(validValue => isEqual(value, validValue))); } /** Get the index of the last triggered option. */ _getLastTriggeredIndex() { const index = this.options.toArray().indexOf(this._lastTriggered); return index === -1 ? null : index; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkListbox, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.0", type: CdkListbox, isStandalone: true, selector: "[cdkListbox]", inputs: { id: "id", enabledTabIndex: ["tabindex", "enabledTabIndex"], value: ["cdkListboxValue", "value"], multiple: ["cdkListboxMultiple", "multiple"], disabled: ["cdkListboxDisabled", "disabled"], useActiveDescendant: ["cdkListboxUseActiveDescendant", "useActiveDescendant"], orientation: ["cdkListboxOrientation", "orientation"], compareWith: ["cdkListboxCompareWith", "compareWith"], navigationWrapDisabled: ["cdkListboxNavigationWrapDisabled", "navigationWrapDisabled"], navigateDisabledOptions: ["cdkListboxNavigatesDisabledOptions", "navigateDisabledOptions"] }, outputs: { valueChange: "cdkListboxValueChange" }, host: { attributes: { "role": "listbox" }, listeners: { "focus": "_handleFocus()", "keydown": "_handleKeydown($event)", "focusout": "_handleFocusOut($event)", "focusin": "_handleFocusIn()" }, properties: { "id": "id", "attr.tabindex": "_getTabIndex()", "attr.aria-disabled": "disabled", "attr.aria-multiselectable": "multiple", "attr.aria-activedescendant": "_getAriaActiveDescendant()", "attr.aria-orientation": "orientation" }, classAttribute: "cdk-listbox" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => CdkListbox), multi: true, }, ], queries: [{ propertyName: "options", predicate: CdkOption, descendants: true }], exportAs: ["cdkListbox"], ngImport: i0 }); } } export { CdkListbox }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CdkListbox, decorators: [{ type: Directive, args: [{ selector: '[cdkListbox]', standalone: true, exportAs: 'cdkListbox', host: { 'role': 'listbox', 'class': 'cdk-listbox', '[id]': 'id', '[attr.tabindex]': '_getTabIndex()', '[attr.aria-disabled]': 'disabled', '[attr.aria-multiselectable]': 'multiple', '[attr.aria-activedescendant]': '_getAriaActiveDescendant()', '[attr.aria-orientation]': 'orientation', '(focus)': '_handleFocus()', '(keydown)': '_handleKeydown($event)', '(focusout)': '_handleFocusOut($event)', '(focusin)': '_handleFocusIn()', }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => CdkListbox), multi: true, }, ], }] }], propDecorators: { id: [{ type: Input }], enabledTabIndex: [{ type: Input, args: ['tabindex'] }], value: [{ type: Input, args: ['cdkListboxValue'] }], multiple: [{ type: Input, args: ['cdkListboxMultiple'] }], disabled: [{ type: Input, args: ['cdkListboxDisabled'] }], useActiveDescendant: [{ type: Input, args: ['cdkListboxUseActiveDescendant'] }], orientation: [{ type: Input, args: ['cdkListboxOrientation'] }], compareWith: [{ type: Input, args: ['cdkListboxCompareWith'] }], navigationWrapDisabled: [{ type: Input, args: ['cdkListboxNavigationWrapDisabled'] }], navigateDisabledOptions: [{ type: Input, args: ['cdkListboxNavigatesDisabledOptions'] }], valueChange: [{ type: Output, args: ['cdkListboxValueChange'] }], options: [{ type: ContentChildren, args: [CdkOption, { descendants: true }] }] } }); //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdGJveC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9jZGsvbGlzdGJveC9saXN0Ym94LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUVILE9BQU8sRUFFTCxpQkFBaUIsRUFDakIsZUFBZSxFQUNmLFNBQVMsRUFDVCxVQUFVLEVBQ1YsVUFBVSxFQUNWLE1BQU0sRUFDTixLQUFLLEVBRUwsTUFBTSxFQUNOLFNBQVMsR0FDVixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUMsMEJBQTBCLEVBQXNDLE1BQU0sbUJBQW1CLENBQUM7QUFDbEcsT0FBTyxFQUNMLENBQUMsRUFDRCxVQUFVLEVBQ1YsR0FBRyxFQUNILEtBQUssRUFDTCxjQUFjLEVBQ2QsSUFBSSxFQUNKLFVBQVUsRUFDVixXQUFXLEVBQ1gsS0FBSyxFQUNMLFFBQVEsR0FDVCxNQUFNLHVCQUF1QixDQUFDO0FBQy9CLE9BQU8sRUFBZSxXQUFXLEVBQUUscUJBQXFCLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQUN2RixPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFDeEQsT0FBTyxFQUFDLEtBQUssRUFBRSxLQUFLLEVBQWMsT0FBTyxFQUFDLE1BQU0sTUFBTSxDQUFDO0FBQ3ZELE9BQU8sRUFBQyxNQUFNLEVBQUUsR0FBRyxFQUFFLFNBQVMsRUFBRSxTQUFTLEVBQUUsU0FBUyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFDNUUsT0FBTyxFQUF1QixpQkFBaUIsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBQ3ZFLE9BQU8sRUFBQyxjQUFjLEVBQUMsTUFBTSxtQkFBbUIsQ0FBQzs7QUFFakQsc0RBQXNEO0FBQ3RELElBQUksTUFBTSxHQUFHLENBQUMsQ0FBQztBQUVmOzs7Ozs7O0dBT0c7QUFDSCxNQUFNLHFCQUF5QixTQUFRLGNBQWlCO0lBQ3RELFlBQ1MsV0FBVyxLQUFLLEVBQ3ZCLHVCQUE2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