src/modules/dropdown/components/dropdown.component.ts
Component to display dropdown list based on the options passed as parameters. The dropdown component is based on the Angular Material MatSelect.
OnInit
OnChanges
OnInit
OnDestroy
ControlValueAccessor
MatFormFieldControl
Validator
encapsulation | ViewEncapsulation.None |
host | { |
providers |
{
provide: NG_VALIDATORS, useExisting: StarkDropdownComponent, multi: true
}
{
provide: MatFormFieldControl, useExisting: StarkDropdownComponent
}
|
selector | stark-dropdown |
templateUrl | ./dropdown.component.html |
Properties |
|
Methods |
|
Inputs |
Outputs |
HostBindings |
Accessors |
Public
constructor(logger: StarkLoggingService, renderer: Renderer2, elementRef: ElementRef, fm: FocusMonitor, injector: Injector, translateService: TranslateService, formField?: MatFormField)
|
||||||||||||||||||||||||||||||||
Class constructor
Parameters :
|
defaultBlank | |
Type : boolean
|
|
If the dropdown will contain a default blank (optional) |
disabled | |
Type : boolean
|
|
Default value : false
|
|
Whether the dropdown is disabled |
dropdownId | |
Type : string
|
|
Default value : "undefined"
|
|
HTML "id" attribute of the element. |
dropdownName | |
Type : string
|
|
Default value : "undefined"
|
|
HTML "name" attribute of the element. |
multiSelect | |
Type : boolean
|
|
Allows multiple option selection. Setting the attribute to "true" or empty will enable this feature. (optional) |
optionIdProperty | |
Type : string
|
|
Property name to be used as id of the options defined. If this parameter as well as the optionLabelProperty are omitted, then the options parameter is considered to be an array of simple types. (optional) |
optionLabelProperty | |
Type : string
|
|
Property name to be used as label of the options defined (dynamically translated via the @ngx-translate service if the provided text is defined in the translation keys). If this parameter as well as the optionIdProperty are omitted, then the options parameter is considered to be an array of simple types. (optional) |
options | |
Type : any[]
|
|
Default value : []
|
|
Array of options to be included in the dropdown list. This parameter is a one-way binding (one-directional). |
placeholder | |
Type : string
|
|
Default value : ""
|
|
Text to be displayed as the dropdown's placeholder. Dynamically translated via the @ngx-translate service if the provided text is defined in the translation keys. |
required | |
Type : boolean
|
|
If the dropdown is required or not. by default, the dropdown is not required |
value | |
Type : any | any[]
|
|
Source object to be bound to the dropdown ngModel. |
color | |
Type : string
|
|
Inherited from
AbstractStarkUiComponent
|
|
Defined in
AbstractStarkUiComponent:16
|
|
Color theme |
selectionChanged | |
Type : EventEmitter
|
|
This will emit the newly selected value. |
attr.aria-describedby |
Type : string
|
Default value : ""
|
ARIA |
class.floating |
Type : boolean
|
Whether the label should float above the field to let the user type a value |
id |
Type : string
|
Default value : `stark-dropdown-input-${StarkDropdownComponent.nextId++}`
|
HTML id of this component |
Public areSimpleTypes |
areSimpleTypes()
|
Whether the type of the option elements is a primitive type (string, number, boolean...) In case none of the optionId and optionLabel parameters are provided, then a simple data types array is assumed
Returns :
boolean
|
Public getOptionLabel | ||||||||
getOptionLabel(option: any)
|
||||||||
Return the label of the option or the option itself if it is a simple type.
Parameters :
Returns :
any
The value to return |
Public getOptionValue | ||||||||
getOptionValue(option: any)
|
||||||||
Return the id of the option or the option itself if it is a simple type.
Parameters :
Returns :
any
The value to return |
Public ngOnChanges | ||||||||
ngOnChanges(changes: SimpleChanges)
|
||||||||
Component lifecycle hook
Parameters :
Returns :
void
|
Public ngOnDestroy |
ngOnDestroy()
|
Component lifecycle hook
Returns :
void
|
Public ngOnInit |
ngOnInit()
|
Inherited from
AbstractStarkUiComponent
|
Defined in
AbstractStarkUiComponent:331
|
Component lifecycle hook
Returns :
void
|
Public onContainerClick |
onContainerClick()
|
Method implemented to use MatFormFieldControl It handles a click on the control's container.
Returns :
void
|
Public onInternalValueChange | ||||||||
onInternalValueChange(value: any | any[])
|
||||||||
Trigger required methods
Parameters :
Returns :
void
|
Public onOpenedChange |
onOpenedChange()
|
Method which set the MatFormField as touched once the mat-select has been clicked.
Returns :
void
|
Public onSelectionChange | ||||||||
onSelectionChange(change: MatSelectChange)
|
||||||||
Handled when a
Parameters :
Returns :
void
|
Public registerOnChange | ||||||||
registerOnChange(fn: (_: any) => void)
|
||||||||
Registers a function called when the control value changes.
Parameters :
Returns :
void
|
Public registerOnTouched | ||||||||
registerOnTouched(fn: () => void)
|
||||||||
Registers a function called when the control is touched.
Parameters :
Returns :
void
|
Public registerOnValidatorChange | ||||||||
registerOnValidatorChange(fn: () => void)
|
||||||||
Registers a callback function to call when the validator inputs change.
Parameters :
Returns :
void
|
Public setDefaultBlank |
setDefaultBlank()
|
set blank by default in the dropdown component
Returns :
void
|
Public setDescribedByIds | ||||||||
setDescribedByIds(ids: string[])
|
||||||||
Method implemented to use MatFormFieldControl Sets the list of element IDs that currently describe this control.
Parameters :
Returns :
void
|
Public setDisabledState | ||||||||
setDisabledState(isDisabled: boolean)
|
||||||||
Sets the "disabled" property on the input element.
Parameters :
Returns :
void
|
Public writeValue | ||||||||
writeValue(obj: any)
|
||||||||
Sets the "value" property on the input element.
Parameters :
Returns :
void
|
Private _onValidatorChange |
Type : function
|
Default value : () => {...}
|
The registered callback function called when the validator inputs change. |
Public describedBy |
Type : string
|
Default value : ""
|
Decorators :
@HostBinding('attr.aria-describedby')
|
ARIA |
Public id |
Default value : `stark-dropdown-input-${StarkDropdownComponent.nextId++}`
|
Decorators :
@HostBinding()
|
HTML id of this component |
Public logger |
Type : StarkLoggingService
|
Decorators :
@Inject(STARK_LOGGING_SERVICE)
|
- The `StarkLoggingService` instance of the application.
|
Private Optional multiSelectElement |
Type : MatSelect
|
Decorators :
@ViewChild('multiSelectEl')
|
Reference to the multi MatSelect element embedded in this component |
Static nextId |
Type : number
|
Default value : 0
|
Variable that will be incremented automatically to serve as unique id for every new instance of this component |
Static ngAcceptInputType_multiSelect |
Type : BooleanInput
|
Static ngAcceptInputType_required |
Type : BooleanInput
|
Private Optional singleSelectElement |
Type : MatSelect
|
Decorators :
@ViewChild('singleSelectEl')
|
Reference to the single MatSelect element embedded in this component |
Public stateChanges |
Type : Subject<void>
|
Default value : new Subject<void>()
|
Variable to define to use MatFormFieldControl.
Stream that emits whenever the state of the control changes such that the parent |
shouldLabelFloat |
getshouldLabelFloat()
|
Whether the label should float above the field to let the user type a value
Returns :
boolean
|
multiSelect | ||||||
getmultiSelect()
|
||||||
Allows multiple option selection. Setting the attribute to "true" or empty will enable this feature. (optional)
Returns :
boolean
|
||||||
setmultiSelect(value: boolean)
|
||||||
Parameters :
Returns :
void
|
required | ||||||
getrequired()
|
||||||
If the dropdown is required or not. by default, the dropdown is not required
Returns :
boolean
|
||||||
setrequired(value: boolean)
|
||||||
Parameters :
Returns :
void
|
value | ||||||
getvalue()
|
||||||
Source object to be bound to the dropdown ngModel.
Returns :
any | []
|
||||||
setvalue(value: any | any[])
|
||||||
Parameters :
Returns :
void
|
errorState |
geterrorState()
|
Variable to define to use MatFormFieldControl Whether the control is in an error state.
Returns :
boolean
|
empty |
getempty()
|
Variable to define to use MatFormFieldControl Whether the control is empty.
Returns :
boolean
|
import {
Component,
ElementRef,
EventEmitter,
HostBinding,
Inject,
Injector,
Input,
OnChanges,
OnDestroy,
OnInit,
Optional,
Output,
Renderer2,
SimpleChanges,
ViewChild,
ViewEncapsulation
} from "@angular/core";
import { STARK_LOGGING_SERVICE, StarkLoggingService } from "@nationalbankbelgium/stark-core";
import { AbstractStarkUiComponent } from "@nationalbankbelgium/stark-ui/src/internal-common";
import { AbstractControl, ControlValueAccessor, NG_VALIDATORS, NgControl, ValidationErrors, Validator, Validators } from "@angular/forms";
import { Subject, Subscription } from "rxjs";
import { MatLegacyFormField as MatFormField, MatLegacyFormFieldControl as MatFormFieldControl } from "@angular/material/legacy-form-field";
import { FocusMonitor, FocusOrigin } from "@angular/cdk/a11y";
import { MatLegacySelect as MatSelect, MatLegacySelectChange as MatSelectChange } from "@angular/material/legacy-select";
import { BooleanInput, coerceBooleanProperty } from "@angular/cdk/coercion";
import { TranslateService } from "@ngx-translate/core";
import isEqual from "lodash-es/isEqual";
/**
* @ignore
*/
const componentName = "stark-dropdown";
/**
* Component to display dropdown list based on the options passed as parameters. The dropdown component is based
* on the Angular Material MatSelect.
*/
@Component({
selector: "stark-dropdown",
templateUrl: "./dropdown.component.html",
encapsulation: ViewEncapsulation.None,
// We need to use host instead of @HostBinding: https://github.com/NationalBankBelgium/stark/issues/664
host: {
class: componentName
},
providers: [
{
provide: NG_VALIDATORS,
useExisting: StarkDropdownComponent,
multi: true
},
{
// This implementation has been made thanks to the official documentation.
// See: https://v7.material.angular.io/guide/creating-a-custom-form-field-control
provide: MatFormFieldControl,
useExisting: StarkDropdownComponent
}
]
})
export class StarkDropdownComponent
extends AbstractStarkUiComponent
implements OnInit, OnChanges, OnInit, OnDestroy, ControlValueAccessor, MatFormFieldControl<any | any[]>, Validator
{
/**
* Variable that will be incremented automatically to serve as unique id for every new instance of this component
*/
public static nextId = 0;
/**
* HTML id of this component
*/
@HostBinding()
public id = `stark-dropdown-input-${StarkDropdownComponent.nextId++}`;
/**
* ARIA `describedBy` attribute to be set to the DOM element where this component is attached to
*/
@HostBinding("attr.aria-describedby")
public describedBy = "";
/**
* Whether the label should float above the field to let the user type a value
*/
@HostBinding("class.floating")
public get shouldLabelFloat(): boolean {
return this.focused || !this.empty;
}
/**
* If the dropdown will contain a default blank (optional)
*/
@Input()
public defaultBlank?: boolean;
/**
* HTML "id" attribute of the element.
*/
@Input()
public dropdownId = "undefined";
/**
* HTML "name" attribute of the element.
*/
@Input()
public dropdownName = "undefined";
/**
* header Text to be displayed as the dropdown's header. Such header will be shown at the top of the
* dropdown options list(dynamically translated via the $translate service if the provided text is defined in the translation keys).
*/
// TODO re-enable once a solution for the replacement of md-select-header as been found
// @Input()
// public header?: string;
/**
* Whether the dropdown is disabled
*/
@Input()
public disabled = false;
/**
* Allows multiple option selection. Setting the attribute to "true" or empty
* will enable this feature. (optional)
*/
@Input()
public get multiSelect(): boolean {
return this._multiSelect;
}
public set multiSelect(value: boolean) {
this._multiSelect = coerceBooleanProperty(value);
}
// Information about boolean coercion https://angular.io/guide/template-typecheck#input-setter-coercion
public static ngAcceptInputType_multiSelect: BooleanInput;
/**
* @ignore
* @internal
*/
private _multiSelect = false;
/**
* Array of options to be included in the dropdown list. This parameter is a one-way
* binding (one-directional).
*/
@Input()
public options: any[] = [];
/**
* Property name to be used as id of the options defined. If this parameter as
* well as the optionLabelProperty are omitted, then the options parameter is considered to be an array of simple types. (optional)
*/
@Input()
public optionIdProperty?: string;
/**
* Property name to be used as label of the options defined (dynamically translated via
* the @ngx-translate service if the provided text is defined in the translation keys). If this parameter as well as the
* optionIdProperty are omitted, then the options parameter is considered to be an array of simple types. (optional)
*/
@Input()
public optionLabelProperty?: string;
/**
* Text to be displayed as the dropdown's placeholder.
* Dynamically translated via the @ngx-translate service if the provided text is defined in the translation keys.
*/
@Input()
public placeholder = "";
/**
* If the dropdown is required or not. by default, the dropdown is not required
*/
@Input()
public get required(): boolean {
return this._required;
}
public set required(value: boolean) {
this._required = coerceBooleanProperty(value);
this.setDefaultBlank();
this.stateChanges.next();
this._onValidatorChange();
}
// Information about boolean coercion https://angular.io/guide/template-typecheck#input-setter-coercion
public static ngAcceptInputType_required: BooleanInput;
/**
* @ignore
* @internal
*/
private _required = false;
/**
* Source object to be bound to the dropdown ngModel.
*/
@Input()
public get value(): any | any[] {
return this._value;
}
public set value(value: any | any[]) {
if (!isEqual(this._value, value)) {
this._value = value;
this.stateChanges.next();
}
}
/**
* @ignore
* @internal
*/
private _value: any | any[];
/**
* This will emit the newly selected value.
*/
@Output()
public readonly selectionChanged = new EventEmitter<any>();
/**
* Reference to the single MatSelect element embedded in this component
*/
@ViewChild("singleSelectEl")
private singleSelectElement?: MatSelect;
/**
* Reference to the multi MatSelect element embedded in this component
*/
@ViewChild("multiSelectEl")
private multiSelectElement?: MatSelect;
/**
* @ignore
* @internal
*/
public optionsAreSimpleTypes?: boolean;
/**
* @ignore
* @internal
*/
// eslint-disable-next-line no-null/no-null
public ngControl: NgControl | null = null;
/**
* Variable to define to use MatFormFieldControl.
* Stream that emits whenever the state of the control changes such that the parent `MatFormField`
* needs to run change detection.
*/
public stateChanges: Subject<void> = new Subject<void>();
/**
* @ignore
* @internal
*/
public focused = false;
/**
* Variable to define to use MatFormFieldControl
* Whether the control is in an error state.
*/
public get errorState(): boolean {
/**
* We decided to handle errors only when ngControl is defined, thanks to a `formControl` or an `NgModel`.
* Otherwise, we're able to switch the state to "error" and display the `alert` color on the mat-form-field.
* But there is no way for the developer to detect there is an error coming from the `stark-dropdown` and display an error message.
*/
return (
!!this.ngControl &&
(!!this.ngControl.touched || !!this.ngControl.dirty) &&
// eslint-disable-next-line no-null/no-null
(!!this.ngControl.invalid || (this.ngControl.control !== null && !!this.validate(this.ngControl.control)))
);
}
/**
* Variable to define to use MatFormFieldControl
* Whether the control is empty.
*/
public get empty(): boolean {
return !this.value || (this.multiSelect && !this.value.length);
}
/**
* @ignore
* @internal
* Original placeholder translation key to keep in memory to translate again when language changes.
*/
private originalPlaceholder = "";
/**
* @ignore
* @internal
*/
private translateOnLangChangeSubscription?: Subscription;
/**
* Class constructor
* @param logger - The `StarkLoggingService` instance of the application.
* @param renderer - Angular `Renderer2` wrapper for DOM manipulations.
* @param elementRef - Reference to the DOM element where this component is attached to.
* @param fm - The Focus Monitor Service.
* @param injector - The Injector of the application.
* @param translateService - The `TranslateService` instance of the application.
* @param formField - The MatFormField of this component (in case the component is used inside a `mat-form-field`).
*/
public constructor(
@Inject(STARK_LOGGING_SERVICE) public logger: StarkLoggingService,
renderer: Renderer2,
elementRef: ElementRef,
private fm: FocusMonitor,
private injector: Injector,
private translateService: TranslateService,
@Optional() private formField?: MatFormField
) {
super(renderer, elementRef);
fm.monitor(elementRef, true).subscribe((origin: FocusOrigin) => {
this.focused = !!origin && !this.disabled;
this.stateChanges.next();
});
}
/**
* Component lifecycle hook
*/
public override ngOnInit(): void {
super.ngOnInit();
this.logger.debug(componentName + ": component initialized");
this.optionsAreSimpleTypes = this.areSimpleTypes();
// eslint-disable-next-line no-null/no-null
this.ngControl = this.injector.get<NgControl>(NgControl, <any>null);
// eslint-disable-next-line no-null/no-null
if (this.ngControl !== null) {
this.ngControl.valueAccessor = this;
}
if (this.formField) {
/**
* Add class on the parent mat-form-field tag
* `mat-form-field-type-mat-select` - class to apply the elliptic effect and other styles for mat-select from mat-form-field
* `stark-dropdown-mat-form-field` - class to control the parent mat-form-field
* `stark-color` - class to set a the input color
*/
this.formField._elementRef.nativeElement.classList.add("mat-form-field-type-mat-select", "stark-dropdown-mat-form-field");
if (this.color) {
this.formField._elementRef.nativeElement.classList.add(`stark-${this.color}`);
}
}
this.translateOnLangChangeSubscription = this.translateService.onLangChange.subscribe(() => {
// Handle translation internally because mat-form-field uses the value of `@Input public placeholder` to display the label / placeholder
this.placeholder = this.originalPlaceholder
? this.translateService.instant(this.originalPlaceholder)
: this.originalPlaceholder;
this.stateChanges.next();
});
this.setDefaultBlank();
}
/**
* Component lifecycle hook
* @param changes - Contains the changed properties
*/
public ngOnChanges(changes: SimpleChanges): void {
if (changes["optionIdProperty"] || changes["optionLabelProperty"]) {
this.optionsAreSimpleTypes = this.areSimpleTypes();
}
if (changes["multiSelect"] && !changes["multiSelect"].isFirstChange()) {
// This avoids an error when switching between "simple select" and "multiple select"
this.onInternalValueChange(undefined);
}
if (changes["placeholder"]) {
this.originalPlaceholder = changes["placeholder"].currentValue || "";
// Handle translation internally because mat-form-field uses the value of `@Input public placeholder` to display the label / placeholder
this.placeholder = this.originalPlaceholder
? this.translateService.instant(this.originalPlaceholder)
: this.originalPlaceholder;
this.stateChanges.next();
}
}
/**
* Component lifecycle hook
*/
public ngOnDestroy(): void {
this.stateChanges.complete();
this.fm.stopMonitoring(this.elementRef.nativeElement);
if (this.translateOnLangChangeSubscription) {
this.translateOnLangChangeSubscription.unsubscribe();
}
}
/**
* Handled when a `change` event is fired on the `<mat-select>` element, then emits the value on `selectionChanged`.
* @param change - The MatSelectChange to to handle
*/
public onSelectionChange(change: MatSelectChange): void {
this.onInternalValueChange(change.value);
}
/**
* Trigger required methods `selectionChanged.emit`, `_onChange`, `_onValidatorChange` when `value´ is changed internally.
* @param value - New value to assign
*/
public onInternalValueChange(value: any | any[]): void {
this.value = value;
this.selectionChanged.emit(this.value);
this._onChange(this.value);
this._onValidatorChange();
}
/**
* Whether the type of the option elements is a primitive type (string, number, boolean...)
* In case none of the optionId and optionLabel parameters are provided, then a simple data types array is assumed
*/
public areSimpleTypes(): boolean {
return !(
typeof this.optionIdProperty !== "undefined" &&
this.optionIdProperty !== "" &&
typeof this.optionLabelProperty !== "undefined" &&
this.optionLabelProperty !== ""
);
}
/**
* set blank by default in the dropdown component
*/
public setDefaultBlank(): void {
// defaultBlank is by default false, you should explicitly set to true to add it
if (!this.defaultBlank) {
this.defaultBlank = false;
}
// if dropdown is required you should not be able to select blank
if (this.required) {
this.defaultBlank = false;
}
}
/**
* Return the id of the option or the option itself if it is a simple type.
* @param option - The option which id we want to retrieve
* @returns The value to return
*/
public getOptionValue(option: any): any {
return this.optionsAreSimpleTypes ? option : option[<string>this.optionIdProperty];
}
/**
* Return the label of the option or the option itself if it is a simple type.
* @param option - The option which value we want to retrieve
* @returns The value to return
*/
public getOptionLabel(option: any): any {
const optionLabel: string = this.optionsAreSimpleTypes ? option : option[<string>this.optionLabelProperty];
return optionLabel.toString(); // IMPORTANT: the label should be a STRING otherwise the translate directive fails
}
/**
* @ignore
*/
public trackItemFn(_index: number, item: any): string {
// FIXME: cannot call areSimpleTypes() from the component since this track function gets no context
return item;
}
/**
* @ignore
* The registered callback function called when an input event occurs on the input element.
*/
private _onChange: (_: any) => void = (_: any) => {
/* noop*/
};
/**
* @ignore
* The registered callback function called when a blur event occurs on the input element.
*/
private _onTouched: () => void = () => {
/* noop*/
};
/**
* The registered callback function called when the validator inputs change.
*/
private _onValidatorChange: () => void = () => {
/* noop*/
};
/**
* Registers a function called when the control value changes.
*
* @param fn - The callback function
*/
public registerOnChange(fn: (_: any) => void): void {
this._onChange = fn;
}
/**
* Registers a function called when the control is touched.
*
* @param fn - The callback function
*/
public registerOnTouched(fn: () => void): void {
this._onTouched = fn;
}
/**
* Sets the "disabled" property on the input element.
*
* @param isDisabled - The disabled value
*/
public setDisabledState(isDisabled: boolean): void {
this.disabled = isDisabled;
this.stateChanges.next();
}
/**
* Sets the "value" property on the input element.
*
* @param obj - The checked value
*/
public writeValue(obj: any): void {
this.value = obj;
}
/**
* Method implemented to use MatFormFieldControl
* Sets the list of element IDs that currently describe this control.
* @param ids - Ids describing the MatFormFieldControl
*/
public setDescribedByIds(ids: string[]): void {
this.describedBy = ids.join(" ");
}
/**
* Method implemented to use MatFormFieldControl
* It handles a click on the control's container.
*/
public onContainerClick(): void {
// Mimic implementation of MatSelect: https://github.com/angular/components/blob/master/src/material/select/select.ts
if (this.singleSelectElement) {
this.singleSelectElement.focus();
this.singleSelectElement.open();
} else if (this.multiSelectElement) {
this.multiSelectElement.focus();
this.multiSelectElement.open();
}
}
/**
* Registers a callback function to call when the validator inputs change.
*
* @param fn - The callback function
*/
public registerOnValidatorChange(fn: () => void): void {
this._onValidatorChange = fn;
}
/**
* Method that performs synchronous validation against the provided control.
*
* @param control - The control to validate against.
*
* @returns A map of validation errors if validation fails, otherwise null.
*/
public validate(control: AbstractControl): ValidationErrors | null {
// eslint-disable-next-line no-null/no-null
return this.required ? Validators.required(control) : null;
}
/**
* Method which set the MatFormField as touched once the mat-select has been clicked.
*/
public onOpenedChange(): void {
this._onTouched();
}
}
<!-- Support for changing multiSelect after init. https://material.angular.io/components/select/overview#error-cannot-change-code-multiple-code-mode-of-select-after-initialization -->
<!-- single-selection view -->
<mat-select
#singleSelectEl
*ngIf="!multiSelect"
[value]="value"
[id]="dropdownId"
[placeholder]="placeholder"
[attr.aria-label]="dropdownName | translate"
[required]="required"
[disabled]="disabled"
(selectionChange)="onSelectionChange($event)"
(openedChange)="onOpenedChange()"
>
<!-- FIXME find a replacement for md-select-header as mat-select-header does not exist yet https://github.com/angular/material2/pull/7835 -->
<!--<mat-select-header-->
<!--*ngIf="header !== undefined && header !== ''"-->
<!--class="stark-dropdown-header"-->
<!--translate>-->
<!--{{header}}-->
<!--</mat-select-header>-->
<mat-option *ngIf="defaultBlank" [value]=""></mat-option>
<mat-option *ngFor="let option of options; trackBy: trackItemFn" [value]="getOptionValue(option)">
{{ getOptionLabel(option) | translate }}
</mat-option>
</mat-select>
<!-- multiple-selection view -->
<mat-select
#multiSelectEl
*ngIf="multiSelect"
[value]="value"
[id]="dropdownId"
[placeholder]="placeholder"
[attr.aria-label]="dropdownName | translate"
[required]="required"
[disabled]="disabled"
(selectionChange)="onSelectionChange($event)"
(openedChange)="onOpenedChange()"
multiple
>
<!-- FIXME find a replacement for md-select-header as mat-select-header does not exist yet https://github.com/angular/material2/pull/7835 -->
<!--<mat-select-header-->
<!--ng-if="$ctrl.header !== undefined && $ctrl.header !== ''"-->
<!--class="stark-dropdown-header"-->
<!--translate>-->
<!--{{header}}-->
<!--</mat-select-header>-->
<mat-option *ngIf="defaultBlank" [value]=""></mat-option>
<mat-option *ngFor="let option of options; trackBy: trackItemFn" [value]="getOptionValue(option)">
<!-- the translate filter should be used in the options instead of the directive, otherwise the text is not translated :( -->
{{ getOptionLabel(option) | translate }}
</mat-option>
</mat-select>