@ni/nimble-angular 17.4.1 → 17.4.2

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.
@@ -1,6 +1,7 @@
1
1
  import { AfterViewChecked, ElementRef, Renderer2 } from '@angular/core';
2
2
  import { ControlValueAccessor } from '@angular/forms';
3
- import type { ListOption } from '../../public-api';
3
+ import type { ListOption } from '@ni/nimble-components/dist/esm/list-option';
4
+ import type { Combobox } from '../../public-api';
4
5
  import * as i0 from "@angular/core";
5
6
  /**
6
7
  * @description
@@ -25,7 +26,8 @@ export declare class NimbleComboboxControlValueAccessorDirective implements Cont
25
26
  * checking for changes.
26
27
  */
27
28
  set compareWith(fn: (o1: unknown, o2: unknown) => boolean);
28
- private readonly _optionMap;
29
+ private readonly _displayTextToOptionsMap;
30
+ private readonly _optionToModelValueMap;
29
31
  private _modelValue;
30
32
  private _optionUpdateQueue;
31
33
  private _compareWith;
@@ -40,7 +42,7 @@ export declare class NimbleComboboxControlValueAccessorDirective implements Cont
40
42
  * @nodoc
41
43
  */
42
44
  private onTouched;
43
- constructor(_renderer: Renderer2, _elementRef: ElementRef);
45
+ constructor(_renderer: Renderer2, _elementRef: ElementRef<Combobox>);
44
46
  ngAfterViewChecked(): void;
45
47
  /**
46
48
  * Updates the underlying nimble-combobox value with the expected display string.
@@ -68,12 +70,15 @@ export declare class NimbleComboboxControlValueAccessorDirective implements Cont
68
70
  /**
69
71
  * @internal
70
72
  */
71
- addOption(displayValue: string, modelValue: unknown): void;
73
+ addOption(modelValue: unknown, option: ListOption): void;
72
74
  /**
73
75
  * @internal
74
76
  */
75
- removeOption(displayValue: string): void;
76
- queueOptionUpdate(listOption: ListOption, modelValue: unknown): void;
77
+ removeOption(option: ListOption): void;
78
+ /**
79
+ * @internal
80
+ */
81
+ queueOptionUpdate(modelValue: unknown, listOption: ListOption): void;
77
82
  private updateDisplayValue;
78
83
  private getValueStringFromValue;
79
84
  /**
@@ -18,7 +18,6 @@ export declare class NimbleComboboxListOptionDirective implements AfterViewInit,
18
18
  */
19
19
  set ngValue(value: unknown);
20
20
  private _modelValue;
21
- private _currentTextContent;
22
21
  constructor(elementRef: ElementRef<ListOption>, renderer: Renderer2, combobox?: NimbleComboboxControlValueAccessorDirective | undefined);
23
22
  ngAfterViewInit(): void;
24
23
  ngOnDestroy(): void;
@@ -18,7 +18,8 @@ export class NimbleComboboxControlValueAccessorDirective {
18
18
  constructor(_renderer, _elementRef) {
19
19
  this._renderer = _renderer;
20
20
  this._elementRef = _elementRef;
21
- this._optionMap = new Map();
21
+ this._displayTextToOptionsMap = new Map();
22
+ this._optionToModelValueMap = new Map();
22
23
  this._optionUpdateQueue = [];
23
24
  this._compareWith = Object.is;
24
25
  }
@@ -35,7 +36,7 @@ export class NimbleComboboxControlValueAccessorDirective {
35
36
  }
36
37
  ngAfterViewChecked() {
37
38
  for (const updateValue of this._optionUpdateQueue) {
38
- this.addOption(updateValue.listOption.text, updateValue.modelValue);
39
+ this.addOption(updateValue.modelValue, updateValue.listOption);
39
40
  }
40
41
  this._optionUpdateQueue = [];
41
42
  }
@@ -53,9 +54,11 @@ export class NimbleComboboxControlValueAccessorDirective {
53
54
  */
54
55
  registerOnChange(fn) {
55
56
  this.onChange = (valueString) => {
56
- const modelValue = this._optionMap.has(valueString) ? this._optionMap.get(valueString) : OPTION_NOT_FOUND;
57
- this._modelValue = modelValue;
58
- fn(modelValue);
57
+ const options = this._displayTextToOptionsMap.get(valueString);
58
+ this._modelValue = options
59
+ ? this._optionToModelValueMap.get(options[0])
60
+ : OPTION_NOT_FOUND;
61
+ fn(this._modelValue);
59
62
  };
60
63
  }
61
64
  /**
@@ -78,18 +81,43 @@ export class NimbleComboboxControlValueAccessorDirective {
78
81
  /**
79
82
  * @internal
80
83
  */
81
- addOption(displayValue, modelValue) {
82
- this._optionMap.set(displayValue, modelValue);
84
+ addOption(modelValue, option) {
85
+ this._optionToModelValueMap.set(option, modelValue);
86
+ const options = this._displayTextToOptionsMap.get(option.text);
87
+ if (options) {
88
+ const optionIndex = options.indexOf(option);
89
+ if (optionIndex < 0) {
90
+ options.push(option);
91
+ }
92
+ }
93
+ else {
94
+ this._displayTextToOptionsMap.set(option.text, [option]);
95
+ }
83
96
  this.updateDisplayValue();
84
97
  }
85
98
  /**
86
99
  * @internal
87
100
  */
88
- removeOption(displayValue) {
89
- this._optionMap.delete(displayValue);
101
+ removeOption(option) {
102
+ const options = this._displayTextToOptionsMap.get(option.text);
103
+ if (options) {
104
+ if (options.length > 1) {
105
+ const removeIndex = options.indexOf(option);
106
+ if (removeIndex >= 0) {
107
+ options.splice(removeIndex, 1);
108
+ }
109
+ }
110
+ else {
111
+ this._displayTextToOptionsMap.delete(option.text);
112
+ }
113
+ }
114
+ this._optionToModelValueMap.delete(option);
90
115
  }
91
- queueOptionUpdate(listOption, modelValue) {
92
- this.removeOption(listOption.text);
116
+ /**
117
+ * @internal
118
+ */
119
+ queueOptionUpdate(modelValue, listOption) {
120
+ this.removeOption(listOption);
93
121
  this._optionUpdateQueue.push({ listOption, modelValue });
94
122
  }
95
123
  updateDisplayValue() {
@@ -97,9 +125,13 @@ export class NimbleComboboxControlValueAccessorDirective {
97
125
  this.setProperty('value', valueAsString ?? '');
98
126
  }
99
127
  getValueStringFromValue(value) {
100
- for (const [optionKey, optionValue] of this._optionMap.entries()) {
101
- if (this._compareWith(optionValue, value)) {
102
- return optionKey;
128
+ for (const [option, modelValue] of this._optionToModelValueMap.entries()) {
129
+ if (this._compareWith(modelValue, value)) {
130
+ for (const [displayText, options] of this._displayTextToOptionsMap.entries()) {
131
+ if (options.includes(option)) {
132
+ return displayText;
133
+ }
134
+ }
103
135
  }
104
136
  }
105
137
  return undefined;
@@ -141,4 +173,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
141
173
  type: HostListener,
142
174
  args: ['blur']
143
175
  }] } });
144
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nimble-combobox-control-value-accessor.directive.js","sourceRoot":"","sources":["../../../../../../projects/ni/nimble-angular/src/directives/combobox/nimble-combobox-control-value-accessor.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewChecked, Directive, ElementRef, forwardRef, HostListener, Input, Renderer2 } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport type { ListOption } from '../../public-api';\n\n/**\n* @description\n* This symbol instance will be returned when the value input of the Combobox is set\n* to a value not found in the set of options.\n*/\nexport const OPTION_NOT_FOUND: unique symbol = Symbol('not found');\nexport type OptionNotFound = typeof OPTION_NOT_FOUND;\n\n/**\n * Control Value Accessor implementation to target combobox inputs.\n * @description\n * The expectation when binding value via 'ngModel', is that the content in each list-option be\n * unique. When this isn't the case the behavior is undefined. Additionally, it is expected\n * that when using 'ngModel' that each list-option bind a value via 'ngValue', and not 'value'.\n */\n@Directive({\n    selector:\n      'nimble-combobox[formControlName],nimble-combobox[formControl],nimble-combobox[ngModel]',\n    providers: [{\n        provide: NG_VALUE_ACCESSOR,\n        useExisting: forwardRef(() => NimbleComboboxControlValueAccessorDirective),\n        multi: true\n    }]\n})\nexport class NimbleComboboxControlValueAccessorDirective implements ControlValueAccessor, AfterViewChecked {\n    /**\n     * @description\n     * Tracks the option comparison algorithm for tracking identities when\n     * checking for changes.\n     */\n    @Input()\n    public set compareWith(fn: (o1: unknown, o2: unknown) => boolean) {\n        if (typeof fn !== 'function') {\n            throw new Error(`compareWith must be a function, but received ${JSON.stringify(fn)}`);\n        }\n        this._compareWith = fn;\n    }\n\n    private readonly _optionMap: Map<string, unknown> = new Map<string, unknown>();\n\n    private _modelValue: unknown;\n\n    private _optionUpdateQueue: { listOption: ListOption, modelValue: unknown }[] = [];\n\n    private _compareWith: (o1: unknown, o2: unknown) => boolean = Object.is;\n\n    /**\n     * The registered callback function called when a change or input event occurs on the input\n     * element.\n     * @nodoc\n     */\n    @HostListener('input', ['$event.target.control.value]'])\n    @HostListener('change', ['$event.target.value]'])\n    private onChange: (_: string) => void;\n\n    /**\n      * The registered callback function called when a blur event occurs on the input element.\n      * @nodoc\n      */\n    @HostListener('blur')\n    private onTouched: () => void;\n\n    public constructor(\n        private readonly _renderer: Renderer2,\n        private readonly _elementRef: ElementRef\n    ) {}\n\n    public ngAfterViewChecked(): void {\n        for (const updateValue of this._optionUpdateQueue) {\n            this.addOption(updateValue.listOption.text, updateValue.modelValue);\n        }\n        this._optionUpdateQueue = [];\n    }\n\n    /**\n     * Updates the underlying nimble-combobox value with the expected display string.\n     * @param value The ngValue set on the nimble-combobox\n     */\n    public writeValue(value: unknown): void {\n        this._modelValue = value;\n        this.updateDisplayValue();\n    }\n\n    /**\n     * Registers a function called when the control value changes.\n     * @nodoc\n     */\n    public registerOnChange(fn: (value: unknown) => void): void {\n        this.onChange = (valueString: string): void => {\n            const modelValue = this._optionMap.has(valueString) ? this._optionMap.get(valueString) : OPTION_NOT_FOUND;\n            this._modelValue = modelValue;\n            fn(modelValue);\n        };\n    }\n\n    /**\n     * Registers a function called when the control is touched.\n     * @nodoc\n     */\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    /**\n     * @description\n     * Function that is called by the forms API when the control status changes to\n     * or from 'DISABLED'. Depending on the status, it enables or disables the\n     * combobox.\n     * @param isDisabled The disabled status to set on the combobox\n     */\n    public setDisabledState(isDisabled: boolean): void {\n        this.setProperty('disabled', isDisabled);\n    }\n\n    /**\n     * @internal\n     */\n    public addOption(displayValue: string, modelValue: unknown): void {\n        this._optionMap.set(displayValue, modelValue);\n        this.updateDisplayValue();\n    }\n\n    /**\n     * @internal\n     */\n    public removeOption(displayValue: string): void {\n        this._optionMap.delete(displayValue);\n    }\n\n    public queueOptionUpdate(listOption: ListOption, modelValue: unknown): void {\n        this.removeOption(listOption.text);\n        this._optionUpdateQueue.push({ listOption, modelValue });\n    }\n\n    private updateDisplayValue(): void {\n        const valueAsString = this.getValueStringFromValue(this._modelValue);\n        this.setProperty('value', valueAsString ?? '');\n    }\n\n    private getValueStringFromValue(value: unknown): string | undefined {\n        for (const [optionKey, optionValue] of this._optionMap.entries()) {\n            if (this._compareWith(optionValue, value)) {\n                return optionKey;\n            }\n        }\n        return undefined;\n    }\n\n    /**\n     * Helper method that sets a property on a target element using the current Renderer\n     * implementation.\n     * @nodoc\n     */\n    private setProperty(key: string, value: unknown): void {\n        this._renderer.setProperty(this._elementRef.nativeElement, key, value);\n    }\n}\n"]}
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nimble-combobox-control-value-accessor.directive.js","sourceRoot":"","sources":["../../../../../../projects/ni/nimble-angular/src/directives/combobox/nimble-combobox-control-value-accessor.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewChecked, Directive, ElementRef, forwardRef, HostListener, Input, Renderer2 } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport type { ListOption } from '@ni/nimble-components/dist/esm/list-option';\nimport type { Combobox } from '../../public-api';\n\n/**\n* @description\n* This symbol instance will be returned when the value input of the Combobox is set\n* to a value not found in the set of options.\n*/\nexport const OPTION_NOT_FOUND: unique symbol = Symbol('not found');\nexport type OptionNotFound = typeof OPTION_NOT_FOUND;\n\n/**\n * Control Value Accessor implementation to target combobox inputs.\n * @description\n * The expectation when binding value via 'ngModel', is that the content in each list-option be\n * unique. When this isn't the case the behavior is undefined. Additionally, it is expected\n * that when using 'ngModel' that each list-option bind a value via 'ngValue', and not 'value'.\n */\n@Directive({\n    selector:\n      'nimble-combobox[formControlName],nimble-combobox[formControl],nimble-combobox[ngModel]',\n    providers: [{\n        provide: NG_VALUE_ACCESSOR,\n        useExisting: forwardRef(() => NimbleComboboxControlValueAccessorDirective),\n        multi: true\n    }]\n})\nexport class NimbleComboboxControlValueAccessorDirective implements ControlValueAccessor, AfterViewChecked {\n    /**\n     * @description\n     * Tracks the option comparison algorithm for tracking identities when\n     * checking for changes.\n     */\n    @Input()\n    public set compareWith(fn: (o1: unknown, o2: unknown) => boolean) {\n        if (typeof fn !== 'function') {\n            throw new Error(`compareWith must be a function, but received ${JSON.stringify(fn)}`);\n        }\n        this._compareWith = fn;\n    }\n\n    private readonly _displayTextToOptionsMap: Map<string, ListOption[]> = new Map<string, ListOption[]>();\n    private readonly _optionToModelValueMap: Map<ListOption, unknown> = new Map<ListOption, unknown>();\n\n    private _modelValue: unknown;\n\n    private _optionUpdateQueue: { listOption: ListOption, modelValue: unknown }[] = [];\n\n    private _compareWith: (o1: unknown, o2: unknown) => boolean = Object.is;\n\n    /**\n     * The registered callback function called when a change or input event occurs on the input\n     * element.\n     * @nodoc\n     */\n    @HostListener('input', ['$event.target.control.value]'])\n    @HostListener('change', ['$event.target.value]'])\n    private onChange: (_: string) => void;\n\n    /**\n      * The registered callback function called when a blur event occurs on the input element.\n      * @nodoc\n      */\n    @HostListener('blur')\n    private onTouched: () => void;\n\n    public constructor(private readonly _renderer: Renderer2, private readonly _elementRef: ElementRef<Combobox>) {}\n\n    public ngAfterViewChecked(): void {\n        for (const updateValue of this._optionUpdateQueue) {\n            this.addOption(updateValue.modelValue, updateValue.listOption);\n        }\n        this._optionUpdateQueue = [];\n    }\n\n    /**\n     * Updates the underlying nimble-combobox value with the expected display string.\n     * @param value The ngValue set on the nimble-combobox\n     */\n    public writeValue(value: unknown): void {\n        this._modelValue = value;\n        this.updateDisplayValue();\n    }\n\n    /**\n     * Registers a function called when the control value changes.\n     * @nodoc\n     */\n    public registerOnChange(fn: (value: unknown) => void): void {\n        this.onChange = (valueString: string): void => {\n            const options = this._displayTextToOptionsMap.get(valueString);\n            this._modelValue = options\n                ? this._optionToModelValueMap.get(options[0])\n                : OPTION_NOT_FOUND;\n            fn(this._modelValue);\n        };\n    }\n\n    /**\n     * Registers a function called when the control is touched.\n     * @nodoc\n     */\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    /**\n     * @description\n     * Function that is called by the forms API when the control status changes to\n     * or from 'DISABLED'. Depending on the status, it enables or disables the\n     * combobox.\n     * @param isDisabled The disabled status to set on the combobox\n     */\n    public setDisabledState(isDisabled: boolean): void {\n        this.setProperty('disabled', isDisabled);\n    }\n\n    /**\n     * @internal\n     */\n    public addOption(modelValue: unknown, option: ListOption): void {\n        this._optionToModelValueMap.set(option, modelValue);\n        const options = this._displayTextToOptionsMap.get(option.text);\n        if (options) {\n            const optionIndex = options.indexOf(option);\n            if (optionIndex < 0) {\n                options.push(option);\n            }\n        } else {\n            this._displayTextToOptionsMap.set(option.text, [option]);\n        }\n        this.updateDisplayValue();\n    }\n\n    /**\n     * @internal\n     */\n    public removeOption(option: ListOption): void {\n        const options = this._displayTextToOptionsMap.get(option.text);\n        if (options) {\n            if (options.length > 1) {\n                const removeIndex = options.indexOf(option);\n                if (removeIndex >= 0) {\n                    options.splice(removeIndex, 1);\n                }\n            } else {\n                this._displayTextToOptionsMap.delete(option.text);\n            }\n        }\n        this._optionToModelValueMap.delete(option);\n    }\n\n    /**\n     * @internal\n     */\n    public queueOptionUpdate(modelValue: unknown, listOption: ListOption): void {\n        this.removeOption(listOption);\n        this._optionUpdateQueue.push({ listOption, modelValue });\n    }\n\n    private updateDisplayValue(): void {\n        const valueAsString = this.getValueStringFromValue(this._modelValue);\n        this.setProperty('value', valueAsString ?? '');\n    }\n\n    private getValueStringFromValue(value: unknown): string | undefined {\n        for (const [option, modelValue] of this._optionToModelValueMap.entries()) {\n            if (this._compareWith(modelValue, value)) {\n                for (const [displayText, options] of this._displayTextToOptionsMap.entries()) {\n                    if (options.includes(option)) {\n                        return displayText;\n                    }\n                }\n            }\n        }\n        return undefined;\n    }\n\n    /**\n     * Helper method that sets a property on a target element using the current Renderer\n     * implementation.\n     * @nodoc\n     */\n    private setProperty(key: string, value: unknown): void {\n        this._renderer.setProperty(this._elementRef.nativeElement, key, value);\n    }\n}\n"]}
@@ -31,17 +31,16 @@ export class NimbleComboboxListOptionDirective {
31
31
  }
32
32
  ngAfterViewInit() {
33
33
  if (this.combobox) {
34
- this._currentTextContent = this.elementRef.nativeElement.text;
35
- this.combobox.addOption(this._currentTextContent, this._modelValue);
34
+ this.combobox.addOption(this._modelValue, this.elementRef.nativeElement);
36
35
  }
37
36
  }
38
37
  ngOnDestroy() {
39
38
  if (this.combobox) {
40
- this.combobox.removeOption(this._currentTextContent);
39
+ this.combobox.removeOption(this.elementRef.nativeElement);
41
40
  }
42
41
  }
43
42
  updateComboboxValue(value) {
44
- this.combobox.queueOptionUpdate(this.elementRef.nativeElement, value);
43
+ this.combobox.queueOptionUpdate(value, this.elementRef.nativeElement);
45
44
  }
46
45
  }
47
46
  NimbleComboboxListOptionDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NimbleComboboxListOptionDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: NimbleComboboxControlValueAccessorDirective, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Directive });
@@ -63,4 +62,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
63
62
  }], ngValue: [{
64
63
  type: Input
65
64
  }] } });
66
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmltYmxlLWNvbWJvYm94LWxpc3Qtb3B0aW9uLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25pL25pbWJsZS1hbmd1bGFyL3NyYy9kaXJlY3RpdmVzL2xpc3Qtb3B0aW9uL25pbWJsZS1jb21ib2JveC1saXN0LW9wdGlvbi5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBYyxJQUFJLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQXVDLE1BQU0sZUFBZSxDQUFDO0FBRTFILE9BQU8sRUFBMkIsaUJBQWlCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUNuRyxPQUFPLEVBQUUsMkNBQTJDLEVBQUUsTUFBTSw4REFBOEQsQ0FBQzs7O0FBRTNIOztHQUVHO0FBSUgsTUFBTSxPQUFPLGlDQUFpQztJQXdCMUMsWUFDcUIsVUFBa0MsRUFDbEMsUUFBbUIsRUFDc0QsUUFBc0Q7UUFGL0gsZUFBVSxHQUFWLFVBQVUsQ0FBd0I7UUFDbEMsYUFBUSxHQUFSLFFBQVEsQ0FBVztRQUNzRCxhQUFRLEdBQVIsUUFBUSxDQUE4QztRQU41SSxnQkFBVyxHQUFZLFNBQVMsQ0FBQztJQU9yQyxDQUFDO0lBM0JMLElBQVcsUUFBUTtRQUNmLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDO0lBQ2xELENBQUM7SUFFRCxJQUFvQixRQUFRLENBQUMsS0FBOEI7UUFDdkQsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLEVBQUUsVUFBVSxFQUFFLGlCQUFpQixDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7SUFDbkcsQ0FBQztJQUVEOzs7T0FHRztJQUNILElBQ1csT0FBTyxDQUFDLEtBQWM7UUFDN0IsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2YsSUFBSSxDQUFDLFdBQVcsR0FBRyxLQUFLLENBQUM7WUFDekIsSUFBSSxDQUFDLG1CQUFtQixDQUFDLEtBQUssQ0FBQyxDQUFDO1NBQ25DO0lBQ0wsQ0FBQztJQVdNLGVBQWU7UUFDbEIsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2YsSUFBSSxDQUFDLG1CQUFtQixHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQztZQUM5RCxJQUFJLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1NBQ3ZFO0lBQ0wsQ0FBQztJQUVNLFdBQVc7UUFDZCxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDZixJQUFJLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsbUJBQW1CLENBQUMsQ0FBQztTQUN4RDtJQUNMLENBQUM7SUFFTyxtQkFBbUIsQ0FBQyxLQUFjO1FBQ3RDLElBQUksQ0FBQyxRQUFTLENBQUMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDM0UsQ0FBQzs7OEhBN0NRLGlDQUFpQyxxRUEyQjlCLDJDQUEyQztrSEEzQjlDLGlDQUFpQzsyRkFBakMsaUNBQWlDO2tCQUg3QyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSxvQkFBb0I7aUJBQ2pDOzswQkE0QlEsTUFBTTsyQkFBQywyQ0FBMkM7OzBCQUFHLFFBQVE7OzBCQUFJLElBQUk7NENBdEJ0RCxRQUFRO3NCQUEzQixLQUFLO2dCQVNLLE9BQU87c0JBRGpCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEVsZW1lbnRSZWYsIEhvc3QsIEluamVjdCwgSW5wdXQsIE9wdGlvbmFsLCBSZW5kZXJlcjIsIEFmdGVyVmlld0luaXQsIE9uRGVzdHJveSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHR5cGUgeyBMaXN0T3B0aW9uIH0gZnJvbSAnQG5pL25pbWJsZS1jb21wb25lbnRzL2Rpc3QvZXNtL2xpc3Qtb3B0aW9uJztcbmltcG9ydCB7IEJvb2xlYW5WYWx1ZU9yQXR0cmlidXRlLCB0b0Jvb2xlYW5Qcm9wZXJ0eSB9IGZyb20gJ0BuaS9uaW1ibGUtYW5ndWxhci9pbnRlcm5hbC11dGlsaXRpZXMnO1xuaW1wb3J0IHsgTmltYmxlQ29tYm9ib3hDb250cm9sVmFsdWVBY2Nlc3NvckRpcmVjdGl2ZSB9IGZyb20gJy4uL2NvbWJvYm94L25pbWJsZS1jb21ib2JveC1jb250cm9sLXZhbHVlLWFjY2Vzc29yLmRpcmVjdGl2ZSc7XG5cbi8qKlxuICogRGlyZWN0aXZlIHRvIHByb3ZpZGUgQW5ndWxhciBpbnRlZ3JhdGlvbiBmb3IgdGhlIGxpc3Qgb3B0aW9uIHdoZW4gdXNlZCB3aXRoIGEgY29tYm9ib3guXG4gKi9cbkBEaXJlY3RpdmUoe1xuICAgIHNlbGVjdG9yOiAnbmltYmxlLWxpc3Qtb3B0aW9uJ1xufSlcbmV4cG9ydCBjbGFzcyBOaW1ibGVDb21ib2JveExpc3RPcHRpb25EaXJlY3RpdmUgaW1wbGVtZW50cyBBZnRlclZpZXdJbml0LCBPbkRlc3Ryb3kge1xuICAgIHB1YmxpYyBnZXQgZGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgICAgIHJldHVybiB0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudC5kaXNhYmxlZDtcbiAgICB9XG5cbiAgICBASW5wdXQoKSBwdWJsaWMgc2V0IGRpc2FibGVkKHZhbHVlOiBCb29sZWFuVmFsdWVPckF0dHJpYnV0ZSkge1xuICAgICAgICB0aGlzLnJlbmRlcmVyLnNldFByb3BlcnR5KHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50LCAnZGlzYWJsZWQnLCB0b0Jvb2xlYW5Qcm9wZXJ0eSh2YWx1ZSkpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIEBkZXNjcmlwdGlvblxuICAgICAqIFRyYWNrcyB0aGUgdmFsdWUgYm91bmQgdG8gdGhlIG9wdGlvbiBlbGVtZW50LlxuICAgICAqL1xuICAgIEBJbnB1dCgpXG4gICAgcHVibGljIHNldCBuZ1ZhbHVlKHZhbHVlOiB1bmtub3duKSB7XG4gICAgICAgIGlmICh0aGlzLmNvbWJvYm94KSB7XG4gICAgICAgICAgICB0aGlzLl9tb2RlbFZhbHVlID0gdmFsdWU7XG4gICAgICAgICAgICB0aGlzLnVwZGF0ZUNvbWJvYm94VmFsdWUodmFsdWUpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBfbW9kZWxWYWx1ZTogdW5rbm93biA9IHVuZGVmaW5lZDtcbiAgICBwcml2YXRlIF9jdXJyZW50VGV4dENvbnRlbnQ6IHN0cmluZztcblxuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihcbiAgICAgICAgcHJpdmF0ZSByZWFkb25seSBlbGVtZW50UmVmOiBFbGVtZW50UmVmPExpc3RPcHRpb24+LFxuICAgICAgICBwcml2YXRlIHJlYWRvbmx5IHJlbmRlcmVyOiBSZW5kZXJlcjIsXG4gICAgICAgIEBJbmplY3QoTmltYmxlQ29tYm9ib3hDb250cm9sVmFsdWVBY2Nlc3NvckRpcmVjdGl2ZSkgQE9wdGlvbmFsKCkgQEhvc3QoKSBwcml2YXRlIHJlYWRvbmx5IGNvbWJvYm94PzogTmltYmxlQ29tYm9ib3hDb250cm9sVmFsdWVBY2Nlc3NvckRpcmVjdGl2ZVxuICAgICkgeyB9XG5cbiAgICBwdWJsaWMgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5jb21ib2JveCkge1xuICAgICAgICAgICAgdGhpcy5fY3VycmVudFRleHRDb250ZW50ID0gdGhpcy5lbGVtZW50UmVmLm5hdGl2ZUVsZW1lbnQudGV4dDtcbiAgICAgICAgICAgIHRoaXMuY29tYm9ib3guYWRkT3B0aW9uKHRoaXMuX2N1cnJlbnRUZXh0Q29udGVudCwgdGhpcy5fbW9kZWxWYWx1ZSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmNvbWJvYm94KSB7XG4gICAgICAgICAgICB0aGlzLmNvbWJvYm94LnJlbW92ZU9wdGlvbih0aGlzLl9jdXJyZW50VGV4dENvbnRlbnQpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgcHJpdmF0ZSB1cGRhdGVDb21ib2JveFZhbHVlKHZhbHVlOiB1bmtub3duKTogdm9pZCB7XG4gICAgICAgIHRoaXMuY29tYm9ib3ghLnF1ZXVlT3B0aW9uVXBkYXRlKHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50LCB2YWx1ZSk7XG4gICAgfVxufVxuIl19
65
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmltYmxlLWNvbWJvYm94LWxpc3Qtb3B0aW9uLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25pL25pbWJsZS1hbmd1bGFyL3NyYy9kaXJlY3RpdmVzL2xpc3Qtb3B0aW9uL25pbWJsZS1jb21ib2JveC1saXN0LW9wdGlvbi5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBYyxJQUFJLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQXVDLE1BQU0sZUFBZSxDQUFDO0FBRTFILE9BQU8sRUFBMkIsaUJBQWlCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUNuRyxPQUFPLEVBQUUsMkNBQTJDLEVBQUUsTUFBTSw4REFBOEQsQ0FBQzs7O0FBRTNIOztHQUVHO0FBSUgsTUFBTSxPQUFPLGlDQUFpQztJQXVCMUMsWUFDcUIsVUFBa0MsRUFDbEMsUUFBbUIsRUFDc0QsUUFBc0Q7UUFGL0gsZUFBVSxHQUFWLFVBQVUsQ0FBd0I7UUFDbEMsYUFBUSxHQUFSLFFBQVEsQ0FBVztRQUNzRCxhQUFRLEdBQVIsUUFBUSxDQUE4QztRQUw1SSxnQkFBVyxHQUFZLFNBQVMsQ0FBQztJQU1yQyxDQUFDO0lBMUJMLElBQVcsUUFBUTtRQUNmLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDO0lBQ2xELENBQUM7SUFFRCxJQUFvQixRQUFRLENBQUMsS0FBOEI7UUFDdkQsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLEVBQUUsVUFBVSxFQUFFLGlCQUFpQixDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7SUFDbkcsQ0FBQztJQUVEOzs7T0FHRztJQUNILElBQ1csT0FBTyxDQUFDLEtBQWM7UUFDN0IsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2YsSUFBSSxDQUFDLFdBQVcsR0FBRyxLQUFLLENBQUM7WUFDekIsSUFBSSxDQUFDLG1CQUFtQixDQUFDLEtBQUssQ0FBQyxDQUFDO1NBQ25DO0lBQ0wsQ0FBQztJQVVNLGVBQWU7UUFDbEIsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2YsSUFBSSxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1NBQzVFO0lBQ0wsQ0FBQztJQUVNLFdBQVc7UUFDZCxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDZixJQUFJLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1NBQzdEO0lBQ0wsQ0FBQztJQUVPLG1CQUFtQixDQUFDLEtBQWM7UUFDdEMsSUFBSSxDQUFDLFFBQVMsQ0FBQyxpQkFBaUIsQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUMzRSxDQUFDOzs4SEEzQ1EsaUNBQWlDLHFFQTBCOUIsMkNBQTJDO2tIQTFCOUMsaUNBQWlDOzJGQUFqQyxpQ0FBaUM7a0JBSDdDLFNBQVM7bUJBQUM7b0JBQ1AsUUFBUSxFQUFFLG9CQUFvQjtpQkFDakM7OzBCQTJCUSxNQUFNOzJCQUFDLDJDQUEyQzs7MEJBQUcsUUFBUTs7MEJBQUksSUFBSTs0Q0FyQnRELFFBQVE7c0JBQTNCLEtBQUs7Z0JBU0ssT0FBTztzQkFEakIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgRWxlbWVudFJlZiwgSG9zdCwgSW5qZWN0LCBJbnB1dCwgT3B0aW9uYWwsIFJlbmRlcmVyMiwgQWZ0ZXJWaWV3SW5pdCwgT25EZXN0cm95IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgdHlwZSB7IExpc3RPcHRpb24gfSBmcm9tICdAbmkvbmltYmxlLWNvbXBvbmVudHMvZGlzdC9lc20vbGlzdC1vcHRpb24nO1xuaW1wb3J0IHsgQm9vbGVhblZhbHVlT3JBdHRyaWJ1dGUsIHRvQm9vbGVhblByb3BlcnR5IH0gZnJvbSAnQG5pL25pbWJsZS1hbmd1bGFyL2ludGVybmFsLXV0aWxpdGllcyc7XG5pbXBvcnQgeyBOaW1ibGVDb21ib2JveENvbnRyb2xWYWx1ZUFjY2Vzc29yRGlyZWN0aXZlIH0gZnJvbSAnLi4vY29tYm9ib3gvbmltYmxlLWNvbWJvYm94LWNvbnRyb2wtdmFsdWUtYWNjZXNzb3IuZGlyZWN0aXZlJztcblxuLyoqXG4gKiBEaXJlY3RpdmUgdG8gcHJvdmlkZSBBbmd1bGFyIGludGVncmF0aW9uIGZvciB0aGUgbGlzdCBvcHRpb24gd2hlbiB1c2VkIHdpdGggYSBjb21ib2JveC5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gICAgc2VsZWN0b3I6ICduaW1ibGUtbGlzdC1vcHRpb24nXG59KVxuZXhwb3J0IGNsYXNzIE5pbWJsZUNvbWJvYm94TGlzdE9wdGlvbkRpcmVjdGl2ZSBpbXBsZW1lbnRzIEFmdGVyVmlld0luaXQsIE9uRGVzdHJveSB7XG4gICAgcHVibGljIGdldCBkaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50LmRpc2FibGVkO1xuICAgIH1cblxuICAgIEBJbnB1dCgpIHB1YmxpYyBzZXQgZGlzYWJsZWQodmFsdWU6IEJvb2xlYW5WYWx1ZU9yQXR0cmlidXRlKSB7XG4gICAgICAgIHRoaXMucmVuZGVyZXIuc2V0UHJvcGVydHkodGhpcy5lbGVtZW50UmVmLm5hdGl2ZUVsZW1lbnQsICdkaXNhYmxlZCcsIHRvQm9vbGVhblByb3BlcnR5KHZhbHVlKSk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQGRlc2NyaXB0aW9uXG4gICAgICogVHJhY2tzIHRoZSB2YWx1ZSBib3VuZCB0byB0aGUgb3B0aW9uIGVsZW1lbnQuXG4gICAgICovXG4gICAgQElucHV0KClcbiAgICBwdWJsaWMgc2V0IG5nVmFsdWUodmFsdWU6IHVua25vd24pIHtcbiAgICAgICAgaWYgKHRoaXMuY29tYm9ib3gpIHtcbiAgICAgICAgICAgIHRoaXMuX21vZGVsVmFsdWUgPSB2YWx1ZTtcbiAgICAgICAgICAgIHRoaXMudXBkYXRlQ29tYm9ib3hWYWx1ZSh2YWx1ZSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwcml2YXRlIF9tb2RlbFZhbHVlOiB1bmtub3duID0gdW5kZWZpbmVkO1xuXG4gICAgcHVibGljIGNvbnN0cnVjdG9yKFxuICAgICAgICBwcml2YXRlIHJlYWRvbmx5IGVsZW1lbnRSZWY6IEVsZW1lbnRSZWY8TGlzdE9wdGlvbj4sXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgcmVuZGVyZXI6IFJlbmRlcmVyMixcbiAgICAgICAgQEluamVjdChOaW1ibGVDb21ib2JveENvbnRyb2xWYWx1ZUFjY2Vzc29yRGlyZWN0aXZlKSBAT3B0aW9uYWwoKSBASG9zdCgpIHByaXZhdGUgcmVhZG9ubHkgY29tYm9ib3g/OiBOaW1ibGVDb21ib2JveENvbnRyb2xWYWx1ZUFjY2Vzc29yRGlyZWN0aXZlXG4gICAgKSB7IH1cblxuICAgIHB1YmxpYyBuZ0FmdGVyVmlld0luaXQoKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmNvbWJvYm94KSB7XG4gICAgICAgICAgICB0aGlzLmNvbWJvYm94LmFkZE9wdGlvbih0aGlzLl9tb2RlbFZhbHVlLCB0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmNvbWJvYm94KSB7XG4gICAgICAgICAgICB0aGlzLmNvbWJvYm94LnJlbW92ZU9wdGlvbih0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwcml2YXRlIHVwZGF0ZUNvbWJvYm94VmFsdWUodmFsdWU6IHVua25vd24pOiB2b2lkIHtcbiAgICAgICAgdGhpcy5jb21ib2JveCEucXVldWVPcHRpb25VcGRhdGUodmFsdWUsIHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50KTtcbiAgICB9XG59XG4iXX0=
@@ -1241,7 +1241,8 @@ class NimbleComboboxControlValueAccessorDirective {
1241
1241
  constructor(_renderer, _elementRef) {
1242
1242
  this._renderer = _renderer;
1243
1243
  this._elementRef = _elementRef;
1244
- this._optionMap = new Map();
1244
+ this._displayTextToOptionsMap = new Map();
1245
+ this._optionToModelValueMap = new Map();
1245
1246
  this._optionUpdateQueue = [];
1246
1247
  this._compareWith = Object.is;
1247
1248
  }
@@ -1258,7 +1259,7 @@ class NimbleComboboxControlValueAccessorDirective {
1258
1259
  }
1259
1260
  ngAfterViewChecked() {
1260
1261
  for (const updateValue of this._optionUpdateQueue) {
1261
- this.addOption(updateValue.listOption.text, updateValue.modelValue);
1262
+ this.addOption(updateValue.modelValue, updateValue.listOption);
1262
1263
  }
1263
1264
  this._optionUpdateQueue = [];
1264
1265
  }
@@ -1276,9 +1277,11 @@ class NimbleComboboxControlValueAccessorDirective {
1276
1277
  */
1277
1278
  registerOnChange(fn) {
1278
1279
  this.onChange = (valueString) => {
1279
- const modelValue = this._optionMap.has(valueString) ? this._optionMap.get(valueString) : OPTION_NOT_FOUND;
1280
- this._modelValue = modelValue;
1281
- fn(modelValue);
1280
+ const options = this._displayTextToOptionsMap.get(valueString);
1281
+ this._modelValue = options
1282
+ ? this._optionToModelValueMap.get(options[0])
1283
+ : OPTION_NOT_FOUND;
1284
+ fn(this._modelValue);
1282
1285
  };
1283
1286
  }
1284
1287
  /**
@@ -1301,18 +1304,43 @@ class NimbleComboboxControlValueAccessorDirective {
1301
1304
  /**
1302
1305
  * @internal
1303
1306
  */
1304
- addOption(displayValue, modelValue) {
1305
- this._optionMap.set(displayValue, modelValue);
1307
+ addOption(modelValue, option) {
1308
+ this._optionToModelValueMap.set(option, modelValue);
1309
+ const options = this._displayTextToOptionsMap.get(option.text);
1310
+ if (options) {
1311
+ const optionIndex = options.indexOf(option);
1312
+ if (optionIndex < 0) {
1313
+ options.push(option);
1314
+ }
1315
+ }
1316
+ else {
1317
+ this._displayTextToOptionsMap.set(option.text, [option]);
1318
+ }
1306
1319
  this.updateDisplayValue();
1307
1320
  }
1308
1321
  /**
1309
1322
  * @internal
1310
1323
  */
1311
- removeOption(displayValue) {
1312
- this._optionMap.delete(displayValue);
1324
+ removeOption(option) {
1325
+ const options = this._displayTextToOptionsMap.get(option.text);
1326
+ if (options) {
1327
+ if (options.length > 1) {
1328
+ const removeIndex = options.indexOf(option);
1329
+ if (removeIndex >= 0) {
1330
+ options.splice(removeIndex, 1);
1331
+ }
1332
+ }
1333
+ else {
1334
+ this._displayTextToOptionsMap.delete(option.text);
1335
+ }
1336
+ }
1337
+ this._optionToModelValueMap.delete(option);
1313
1338
  }
1314
- queueOptionUpdate(listOption, modelValue) {
1315
- this.removeOption(listOption.text);
1339
+ /**
1340
+ * @internal
1341
+ */
1342
+ queueOptionUpdate(modelValue, listOption) {
1343
+ this.removeOption(listOption);
1316
1344
  this._optionUpdateQueue.push({ listOption, modelValue });
1317
1345
  }
1318
1346
  updateDisplayValue() {
@@ -1320,9 +1348,13 @@ class NimbleComboboxControlValueAccessorDirective {
1320
1348
  this.setProperty('value', valueAsString !== null && valueAsString !== void 0 ? valueAsString : '');
1321
1349
  }
1322
1350
  getValueStringFromValue(value) {
1323
- for (const [optionKey, optionValue] of this._optionMap.entries()) {
1324
- if (this._compareWith(optionValue, value)) {
1325
- return optionKey;
1351
+ for (const [option, modelValue] of this._optionToModelValueMap.entries()) {
1352
+ if (this._compareWith(modelValue, value)) {
1353
+ for (const [displayText, options] of this._displayTextToOptionsMap.entries()) {
1354
+ if (options.includes(option)) {
1355
+ return displayText;
1356
+ }
1357
+ }
1326
1358
  }
1327
1359
  }
1328
1360
  return undefined;
@@ -6540,17 +6572,16 @@ class NimbleComboboxListOptionDirective {
6540
6572
  }
6541
6573
  ngAfterViewInit() {
6542
6574
  if (this.combobox) {
6543
- this._currentTextContent = this.elementRef.nativeElement.text;
6544
- this.combobox.addOption(this._currentTextContent, this._modelValue);
6575
+ this.combobox.addOption(this._modelValue, this.elementRef.nativeElement);
6545
6576
  }
6546
6577
  }
6547
6578
  ngOnDestroy() {
6548
6579
  if (this.combobox) {
6549
- this.combobox.removeOption(this._currentTextContent);
6580
+ this.combobox.removeOption(this.elementRef.nativeElement);
6550
6581
  }
6551
6582
  }
6552
6583
  updateComboboxValue(value) {
6553
- this.combobox.queueOptionUpdate(this.elementRef.nativeElement, value);
6584
+ this.combobox.queueOptionUpdate(value, this.elementRef.nativeElement);
6554
6585
  }
6555
6586
  }
6556
6587
  NimbleComboboxListOptionDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NimbleComboboxListOptionDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: NimbleComboboxControlValueAccessorDirective, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Directive });