File

src/modules/dropdown/components/dropdown.component.ts

Description

Component to display dropdown list based on the options passed as parameters. The dropdown component is based on the Angular Material MatSelect.

Extends

AbstractStarkUiComponent

Implements

OnInit OnChanges OnInit OnDestroy ControlValueAccessor MatFormFieldControl Validator

Metadata

Index

Properties
Methods
Inputs
Outputs
HostBindings
Accessors

Constructor

Public constructor(logger: StarkLoggingService, renderer: Renderer2, elementRef: ElementRef, fm: FocusMonitor, injector: Injector, translateService: TranslateService, formField?: MatFormField)

Class constructor

Parameters :
Name Type Optional Description
logger StarkLoggingService No
  • The StarkLoggingService instance of the application.
renderer Renderer2 No
  • Angular Renderer2 wrapper for DOM manipulations.
elementRef ElementRef No
  • Reference to the DOM element where this component is attached to.
fm FocusMonitor No
  • The Focus Monitor Service.
injector Injector No
  • The Injector of the application.
translateService TranslateService No
  • The TranslateService instance of the application.
formField MatFormField Yes
  • The MatFormField of this component (in case the component is used inside a mat-form-field).

Inputs

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

Color theme

Outputs

selectionChanged
Type : EventEmitter

This will emit the newly selected value.

HostBindings

attr.aria-describedby
Type : string
Default value : ""

ARIA describedBy attribute to be set to the DOM element where this component is attached to

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

Methods

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 :
Name Type Optional Description
option any No
  • The option which value we want to retrieve
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 :
Name Type Optional Description
option any No
  • The option which id we want to retrieve
Returns : any

The value to return

Public ngOnChanges
ngOnChanges(changes: SimpleChanges)

Component lifecycle hook

Parameters :
Name Type Optional Description
changes SimpleChanges No
  • Contains the changed properties
Returns : void
Public ngOnDestroy
ngOnDestroy()

Component lifecycle hook

Returns : void
Public ngOnInit
ngOnInit()
Inherited from AbstractStarkUiComponent

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 selectionChanged.emit, _onChange, _onValidatorChange when `value´ is changed internally.

Parameters :
Name Type Optional Description
value any | any[] No
  • New value to assign
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 change event is fired on the <mat-select> element, then emits the value on selectionChanged.

Parameters :
Name Type Optional Description
change MatSelectChange No
  • The MatSelectChange to to handle
Returns : void
Public registerOnChange
registerOnChange(fn: (_: any) => void)

Registers a function called when the control value changes.

Parameters :
Name Type Optional Description
fn function No
  • The callback function
Returns : void
Public registerOnTouched
registerOnTouched(fn: () => void)

Registers a function called when the control is touched.

Parameters :
Name Type Optional Description
fn function No
  • The callback function
Returns : void
Public registerOnValidatorChange
registerOnValidatorChange(fn: () => void)

Registers a callback function to call when the validator inputs change.

Parameters :
Name Type Optional Description
fn function No
  • The callback function
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 :
Name Type Optional Description
ids string[] No
  • Ids describing the MatFormFieldControl
Returns : void
Public setDisabledState
setDisabledState(isDisabled: boolean)

Sets the "disabled" property on the input element.

Parameters :
Name Type Optional Description
isDisabled boolean No
  • The disabled value
Returns : void
Public validate
validate(control: AbstractControl)

Method that performs synchronous validation against the provided control.

Parameters :
Name Type Optional Description
control AbstractControl No
  • The control to validate against.
Returns : ValidationErrors | null

A map of validation errors if validation fails, otherwise null.

Public writeValue
writeValue(obj: any)

Sets the "value" property on the input element.

Parameters :
Name Type Optional Description
obj any No
  • The checked value
Returns : void

Properties

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 describedBy attribute to be set to the DOM element where this component is attached to

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 MatFormField needs to run change detection.

Accessors

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 :
Name Type Optional
value boolean No
Returns : void
required
getrequired()

If the dropdown is required or not. by default, the dropdown is not required

Returns : boolean
setrequired(value: boolean)
Parameters :
Name Type Optional
value boolean No
Returns : void
value
getvalue()

Source object to be bound to the dropdown ngModel.

Returns : any | []
setvalue(value: any | any[])
Parameters :
Name Type Optional
value any | any[] No
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>
Legend
Html element
Component
Html element with directive

results matching ""

    No results matching ""