File

src/modules/route-search/components/route-search.component.ts

Description

Component to search a particular route with autocompletion and to redirect to the route chosen by the user.

IMPORTANT: by default, when no menu config is provided, the Route Search component will display only those routes already registered by the Router at the time the component is initialized. In this case the lazy loaded routes that have not been loaded yet will not be displayed.

Setting labels for routes

The component will construct the routes to display by recursively processing all the routes registered by the Router. In this case, the label for each route will be taken from the route's data.translationKey property or from the route name if such property is not found.

Customizing component labels

It is possible to override the following translations used in the different labels of this component:

  • STARK.ROUTE_SEARCH.ABOUT: label shown in the tooltip of the search button
  • STARK.ROUTE_SEARCH.PLACEHOLDER: label shown in the placeholder of the search field

Extends

AbstractStarkUiComponent

Implements

OnInit

Metadata

Index

Properties
Methods
Inputs

Constructor

Public constructor(routingService: StarkRoutingService, logger: StarkLoggingService, translateService: TranslateService, renderer: Renderer2, elementRef: ElementRef)

Class constructor

Parameters :
Name Type Optional Description
routingService StarkRoutingService No
  • The StarkRoutingService instance of the application.
logger StarkLoggingService No
  • The StarkLoggingService instance of the application.
translateService TranslateService No
  • The TranslateService instance of the application.
renderer Renderer2 No
  • Angular Renderer wrapper for DOM manipulations
elementRef ElementRef No
  • Reference to the DOM element where this component is attached to.

Inputs

direction
Type : StarkRouteSearchDirection
Default value : "left"

The direction in which the Route Search field will be displayed.

Default: "left"

icon
Type : "magnify" | string
Default value : "magnify"

Desired icon of the search button.

Default: "magnify"

menuConfig
Type : StarkMenuConfig

The StarkMenuConfig where the routes should be taken from. Normally the same StarkMenuConfig is used for the StarkAppMenuComponent.

IMPORTANT: when this is provided, the component will display all the routes from the config including the lazy loaded routes (if any).

color
Type : string
Inherited from AbstractStarkUiComponent

Color theme

Methods

Public constructRouteEntriesFromMenuConfig
constructRouteEntriesFromMenuConfig(menuConfig: StarkMenuConfig)

Retrieve the list of routes from a StarkMenuConfig object which the user have passed to the component

Parameters :
Name Type Optional Description
menuConfig StarkMenuConfig No
  • StarkMenuConfig object
Public constructRouteEntriesFromRouterStates
constructRouteEntriesFromRouterStates()

Retrieve the list of routes from the Router if the user haven't passed a list to the component

Public extractRoutesFromMenuGroup
extractRoutesFromMenuGroup(group: StarkMenuGroup)

Recursive method to extract routes from a StarkMenuGroup object.

Parameters :
Name Type Optional Description
group StarkMenuGroup No
  • The group which entries need to be extracted

The extracted array of {@link StarkRouteSearchEntry} objects

Public filterRouteEntries
filterRouteEntries(value: string)

Filters the options according to what has been entered by the user in the input field

Parameters :
Name Type Optional Description
value string No
  • Filter value
Public getDirection
getDirection()

Determine the CSS class to use according to the direction chosen by the user

Returns : string
Public ngOnInit
ngOnInit()
Inherited from AbstractStarkUiComponent

Component lifecycle hook

Returns : void
Public redirect
redirect(routeEntry: StarkRouteSearchEntry)

Redirect the user to the chosen path

Parameters :
Name Type Optional Description
routeEntry StarkRouteSearchEntry No
  • Route to navigate to
Returns : void
Public show
show()

Show/hide the input field

Returns : void
Public sortRoutesLabels
sortRoutesLabels(routeEntries: StarkRouteSearchEntry[])

Sort the array of StarkRouteSearchEntry objects by alphabetical order

Parameters :
Name Type Optional Description
routeEntries StarkRouteSearchEntry[] No
  • The list of entries to sort

The sorted array of {@link StarkRouteSearchEntry} objects

Public translateRoutesLabels
translateRoutesLabels(routeEntries: StarkRouteSearchEntry[])

Translate the label of all elements in the StarkRouteSearchEntry object. This is useful as the language can change during runtime.

Parameters :
Name Type Optional Description
routeEntries StarkRouteSearchEntry[] No
  • Route entries to translate

The translate route entries

Properties

Public filteredRouteEntries
Type : Observable<StarkRouteSearchEntry[]>

The list of StarkRouteSearchEntry's objects filtered by the auto-complete field

Public hide
Default value : true

Whether the input field must be hidden or not

Public logger
Type : StarkLoggingService
Decorators :
@Inject(STARK_LOGGING_SERVICE)
- The `StarkLoggingService` instance of the application.
Public routesToDisplay
Type : StarkRouteSearchEntry[]
Default value : []

The list of routes to be displayed

Public routingService
Type : StarkRoutingService
Decorators :
@Inject(STARK_ROUTING_SERVICE)
- The `StarkRoutingService` instance of the application.
Public searchField
Type : UntypedFormControl

The source FormControl object of the search field

Public translateService
Type : TranslateService
Decorators :
@Inject(TranslateService)
- The `TranslateService` instance of the application.
import { ChangeDetectionStrategy, Component, ElementRef, Inject, Input, OnInit, Renderer2, ViewEncapsulation } from "@angular/core";
import { UntypedFormControl } from "@angular/forms";
import { LangChangeEvent, TranslateService } from "@ngx-translate/core";
import { Ng2StateDeclaration } from "@uirouter/angular";
import { Observable } from "rxjs";
import { map, startWith } from "rxjs/operators";
import {
	STARK_LOGGING_SERVICE,
	STARK_ROUTING_SERVICE,
	starkAppExitStateName,
	starkAppInitStateName,
	StarkLoggingService,
	StarkRoutingService
} from "@nationalbankbelgium/stark-core";
import { AbstractStarkUiComponent } from "@nationalbankbelgium/stark-ui/src/internal-common";
import { StarkRouteSearchEntry } from "./route-search-entry.intf";
import { StarkMenuConfig, StarkMenuGroup } from "@nationalbankbelgium/stark-ui/src/modules/app-menu";
import sortBy from "lodash-es/sortBy";

/**
 * @ignore
 */
const componentName = "stark-route-search";

/**
 * Direction in which the Route Search field can be displayed
 */
export type StarkRouteSearchDirection = "left" | "right";

/**
 * Component to search a particular route with autocompletion and to redirect to the route chosen by the user.
 *
 * **`IMPORTANT:`** by default, when no menu config is provided, the Route Search component will display only those routes already registered by the Router at the time the component is initialized.
 * In this case the lazy loaded routes that have not been loaded yet will not be displayed.
 *
 * ### Setting labels for routes
 * The component will construct the routes to display by recursively processing all the routes registered by the Router.
 * In this case, the label for each route will be taken from the route's `data.translationKey` property or from the route name if such property is not found.
 *
 * ### Customizing component labels
 * It is possible to override the following translations used in the different labels of this component:
 *
 * - **STARK.ROUTE_SEARCH.ABOUT:** label shown in the tooltip of the search button
 * - **STARK.ROUTE_SEARCH.PLACEHOLDER:** label shown in the placeholder of the search field
 */
@Component({
	selector: "stark-route-search",
	templateUrl: "./route-search.component.html",
	encapsulation: ViewEncapsulation.None,
	changeDetection: ChangeDetectionStrategy.OnPush,
	// We need to use host instead of @HostBinding: https://github.com/NationalBankBelgium/stark/issues/664
	host: {
		class: componentName
	}
})
export class StarkRouteSearchComponent extends AbstractStarkUiComponent implements OnInit {
	/**
	 * The {@link StarkMenuConfig} where the routes should be taken from.
	 * Normally the same {@link StarkMenuConfig} is used for the {@link StarkAppMenuComponent}.
	 *
	 * **`IMPORTANT:`** when this is provided, the component will display all the routes from the config including the lazy loaded routes (if any).
	 */
	@Input()
	public menuConfig?: StarkMenuConfig;

	/**
	 * The direction in which the Route Search field will be displayed.
	 *
	 * Default: `"left"`
	 */
	@Input()
	public direction: StarkRouteSearchDirection = "left";

	/**
	 * Desired icon of the search button.
	 *
	 * Default: `"magnify"`
	 */
	@Input()
	public icon: "magnify" | string = "magnify";

	/**
	 * The source FormControl object of the search field
	 */
	public searchField: UntypedFormControl;

	/**
	 * The list of {@link StarkRouteSearchEntry}'s objects filtered by the auto-complete field
	 */
	public filteredRouteEntries!: Observable<StarkRouteSearchEntry[]>;

	/**
	 * The list of routes to be displayed
	 */
	public routesToDisplay: StarkRouteSearchEntry[] = [];

	/**
	 * Whether the input field must be hidden or not
	 */
	public hide = true;

	/**
	 * Class constructor
	 * @param routingService - The `StarkRoutingService` instance of the application.
	 * @param logger - The `StarkLoggingService` instance of the application.
	 * @param translateService - The `TranslateService` instance of the application.
	 * @param renderer - Angular Renderer wrapper for DOM manipulations
	 * @param elementRef - Reference to the DOM element where this component is attached to.
	 */
	public constructor(
		@Inject(STARK_ROUTING_SERVICE) public routingService: StarkRoutingService,
		@Inject(STARK_LOGGING_SERVICE) public logger: StarkLoggingService,
		@Inject(TranslateService) public translateService: TranslateService,
		renderer: Renderer2,
		elementRef: ElementRef
	) {
		super(renderer, elementRef);
		this.searchField = new UntypedFormControl();
	}

	/**
	 * Filters the options according to what has been entered by the user in the input field
	 * @param value - Filter value
	 */
	public filterRouteEntries(value: string): StarkRouteSearchEntry[] {
		const filterValue: string = value.toLowerCase();
		return this.routesToDisplay.filter((routeEntry: StarkRouteSearchEntry) =>
			routeEntry.label.toString().toLowerCase().includes(filterValue)
		);
	}

	/**
	 * Component lifecycle hook
	 */
	public override ngOnInit(): void {
		super.ngOnInit();

		this.searchField.setValue("");
		this.filteredRouteEntries = this.searchField.valueChanges.pipe(
			startWith(""),
			map((option: string) => (option ? this.filterRouteEntries(option) : [...this.routesToDisplay]))
		);

		if (typeof this.menuConfig === "undefined") {
			this.routesToDisplay = this.constructRouteEntriesFromRouterStates();
		} else {
			this.routesToDisplay = this.constructRouteEntriesFromMenuConfig(this.menuConfig);
		}

		this.routesToDisplay = this.translateRoutesLabels(this.routesToDisplay);
		this.routesToDisplay = this.sortRoutesLabels(this.routesToDisplay);

		this.translateService.onLangChange.subscribe((_ev: LangChangeEvent) => {
			this.routesToDisplay = this.translateRoutesLabels(this.routesToDisplay);
			this.routesToDisplay = this.sortRoutesLabels(this.routesToDisplay);
		});

		this.logger.debug(componentName + ": component initialized");
	}

	/**
	 * Redirect the user to the chosen path
	 * @param routeEntry - Route to navigate to
	 */
	public redirect(routeEntry: StarkRouteSearchEntry): void {
		this.routingService.navigateTo(routeEntry.targetState, routeEntry.targetStateParams).subscribe(
			() => {
				this.hide = true;
				this.searchField.setValue("");
			},
			(error: any) => {
				this.logger.warn(`${componentName}: navigation failed. Error:${error}`);
			}
		);
	}

	/**
	 * Show/hide the input field
	 */
	public show(): void {
		this.hide = !this.hide;

		if (!this.hide) {
			// set focus on the search field automatically
			setTimeout(() => {
				const inputField: HTMLElement = this.elementRef.nativeElement.querySelector(".search-field-input");
				inputField.focus();
			});
		}
	}

	/**
	 * @ignore
	 */
	public trackPath(_index: number, routeEntry: StarkRouteSearchEntry): string {
		return routeEntry.targetState;
	}

	/**
	 * Retrieve the list of routes from a {@link StarkMenuConfig} object which the user have passed to the component
	 * @param menuConfig - `StarkMenuConfig` object
	 */
	public constructRouteEntriesFromMenuConfig(menuConfig: StarkMenuConfig): StarkRouteSearchEntry[] {
		let routesToDisplay: StarkRouteSearchEntry[] = [];

		if (menuConfig !== undefined && menuConfig.menuSections !== undefined) {
			for (const section of menuConfig.menuSections) {
				for (const group of section.menuGroups) {
					routesToDisplay = [...routesToDisplay, ...this.extractRoutesFromMenuGroup(group)];
				}
			}
		}

		if (menuConfig !== undefined && menuConfig.menuGroups !== undefined) {
			for (const group of menuConfig.menuGroups) {
				routesToDisplay = [...routesToDisplay, ...this.extractRoutesFromMenuGroup(group)];
			}
		}

		return routesToDisplay;
	}

	/**
	 * Recursive method to extract routes from a StarkMenuGroup object.
	 * @param group - The group which entries need to be extracted
	 * @returns The extracted array of {@link StarkRouteSearchEntry} objects
	 */
	public extractRoutesFromMenuGroup(group: StarkMenuGroup): StarkRouteSearchEntry[] {
		let routesToDisplay: StarkRouteSearchEntry[] = [];

		if (group.isVisible && group.isEnabled && group.targetState) {
			routesToDisplay.push({
				label: group.label,
				targetState: group.targetState,
				targetStateParams: group.targetStateParams
			});
		}
		if (group.entries) {
			for (const subGroup of group.entries) {
				routesToDisplay = [...routesToDisplay, ...this.extractRoutesFromMenuGroup(subGroup)];
			}
		}

		return routesToDisplay;
	}

	/**
	 * Retrieve the list of routes from the Router if the user haven't passed a list to the component
	 */
	public constructRouteEntriesFromRouterStates(): StarkRouteSearchEntry[] {
		const routesToDisplay: StarkRouteSearchEntry[] = [];
		for (const state of this.routingService.getStatesConfig()) {
			const ng2State: Ng2StateDeclaration = state;
			// eslint-disable-next-line @typescript-eslint/restrict-plus-operands
			const regexInitExitStateName = new RegExp("(" + starkAppInitStateName + "|" + starkAppExitStateName + ")");
			if (
				ng2State.name !== undefined &&
				ng2State.url !== undefined &&
				ng2State.url.length > 1 &&
				!ng2State.abstract &&
				!ng2State.loadChildren &&
				!ng2State.name.match(regexInitExitStateName)
			) {
				const translationKey: string = this.routingService.getTranslationKeyFromState(ng2State.name);
				routesToDisplay.push({ label: translationKey, targetState: ng2State.name });
			}
		}

		return routesToDisplay;
	}

	/**
	 * Translate the label of all elements in the {@link StarkRouteSearchEntry} object.
	 * This is useful as the language can change during runtime.
	 * @param routeEntries - Route entries to translate
	 * @returns The translate route entries
	 */
	public translateRoutesLabels(routeEntries: StarkRouteSearchEntry[]): StarkRouteSearchEntry[] {
		const routesToDisplay: StarkRouteSearchEntry[] = [];
		for (const route of routeEntries) {
			let translatedLabel: string | object = this.translateService.instant(route.label.toString());
			// in case the translation fails because the key is not found, an object containing all the translations will be returned!
			// in that case the route label is used
			translatedLabel = typeof translatedLabel === "string" ? translatedLabel : route.label.toString();
			routesToDisplay.push({
				label: translatedLabel,
				targetState: route.targetState,
				targetStateParams: route.targetStateParams
			});
		}

		return routesToDisplay;
	}

	/**
	 * Sort the array of {@link StarkRouteSearchEntry} objects by alphabetical order
	 * @param routeEntries - The list of entries to sort
	 * @returns The sorted array of {@link StarkRouteSearchEntry} objects
	 */
	public sortRoutesLabels(routeEntries: StarkRouteSearchEntry[]): StarkRouteSearchEntry[] {
		routeEntries = sortBy(routeEntries, ["label"]);
		return routeEntries;
	}

	/**
	 * Determine the CSS class to use according to the direction chosen by the user
	 */
	public getDirection(): string {
		if (this.direction === "right") {
			return "route-search-right";
		}
		return "route-search-left";
	}
}
<div [ngClass]="getDirection()">
	<button
		mat-mini-fab
		mat-icon-button
		class="search-button"
		color="none"
		(click)="show()"
		[matTooltip]="'STARK.ROUTE_SEARCH.ABOUT' | translate"
	>
		<mat-icon [svgIcon]="icon"></mat-icon>
	</button>
	<div [ngClass]="{ hide: hide }" class="route-search-input-wrapper">
		<mat-form-field floatLabel="never" class="search-field">
			<input
				class="search-field-input"
				matInput
				[matAutocomplete]="autocompleteComp"
				[placeholder]="'STARK.ROUTE_SEARCH.PLACEHOLDER' | translate"
				[formControl]="searchField"
			/>
			<mat-autocomplete #autocompleteComp="matAutocomplete" autoActiveFirstOption class="search-route-autocomplete">
				<mat-option
					(onSelectionChange)="redirect(option)"
					*ngFor="let option of filteredRouteEntries | async; trackBy: trackPath"
					[value]="option.label"
				>
					<p>{{ option.label }}</p>
				</mat-option>
			</mat-autocomplete>
		</mat-form-field>
	</div>
</div>
Legend
Html element
Component
Html element with directive

results matching ""

    No results matching ""