@versatiledatakit/shared
Version:
Versatile Data Kit Shared library enables reusability of shared features like: NgRx Redux, Error Handlers, Utils, Generic Components, etc.
314 lines • 32.3 kB
JavaScript
/*
* Copyright 2023-2025 Broadcom
* SPDX-License-Identifier: Apache-2.0
*/
import { CollectionsUtil } from '../../../utils';
/**
* ** Route Segments Class.
*/
export class RouteSegments {
/**
* ** Constructor.
*/
constructor(routePath, data, params, queryParams, parent, configPath) {
this.routePath = routePath ?? '';
this.data = data || {};
this.params = params || {};
this.queryParams = queryParams || {};
this.parent = parent;
this.configPath = configPath;
}
/**
* ** Factory method.
*/
static of(routePath, data, params, queryParams, parent, configPath) {
return new RouteSegments(routePath, data, params, queryParams, parent, configPath);
}
/**
* ** Factory method for empty RouteSegments.
*/
static empty() {
return RouteSegments.of(null, null, null, null, null, null);
}
/**
* ** Get RoutePath Segments.
*/
get routePathSegments() {
if (this.parent) {
return [].concat(this.parent.routePathSegments, this.routePath).filter((path) => path);
}
return [this.routePath];
}
/**
* ** Get ConfigPath Segments.
*/
get configPathSegments() {
if (this.parent) {
return [].concat(this.parent.configPathSegments, this.configPath).filter((path) => path);
}
return [this.configPath];
}
/**
* ** Get Data from Route configuration by key.
*
* - Return first (closest) found key starting from the current one.
*/
getData(key) {
if (this.data[key]) {
return this.data[key];
}
if (this.parent) {
return this.parent.getData(key);
}
return undefined;
}
/**
* ** Get url param by key.
*
* - Return first (closest) found key starting from the current one.
*/
getParam(key) {
if (this.params[key]) {
return this.params[key];
}
if (this.parent) {
return this.parent.getParam(key);
}
return undefined;
}
/**
* ** Get query param by key.
*/
getQueryParam(key) {
if (this.queryParams[key]) {
return this.queryParams[key];
}
if (this.parent) {
return this.parent.getQueryParam(key);
}
return undefined;
}
}
/**
* ** Route State Class.
*/
export class RouteState {
/**
* ** Constructor.
*/
constructor(routeSegments, url) {
this.routeSegments = routeSegments ?? RouteSegments.empty();
this.url = url ?? '';
}
/**
* ** Factory method.
*/
static of(routeSegments, url) {
return new RouteState(routeSegments, url);
}
/**
* ** Factory method for empty State.
*/
static empty() {
return RouteState.of(null, null);
}
/**
* ** Get serialized queryString.
*/
static serializeQueryParams(queryParams) {
const paramsKeys = Object.keys(queryParams);
if (!paramsKeys.length) {
return '';
}
return paramsKeys.map((key) => `${encodeURIComponent(key)}=${encodeURIComponent(queryParams[key])}`).join('&');
}
/**
* ** Returns current RoutePath.
*/
get routePath() {
return this.routeSegments.routePath;
}
/**
* ** Returns current Absolute RoutePath.
*/
get absoluteRoutePath() {
return RouteState._resolveAbsolutePath(this.routePathSegments);
}
/**
* ** Returns the route paths for each route segment starting from the root.
*/
get routePathSegments() {
return this.routeSegments.routePathSegments;
}
/**
* ** Returns current ConfigPath.
*/
get configPath() {
return this.routeSegments.configPath;
}
/**
* ** Returns current Absolute ConfigPath.
*/
get absoluteConfigPath() {
return RouteState._resolveAbsolutePath(this.configPathSegments);
}
/**
* ** Returns the config paths for each route segment starting from the root.
*/
get configPathSegments() {
return this.routeSegments.configPathSegments;
}
/**
* ** Get all query params.
*/
get queryParams() {
return this.routeSegments.queryParams;
}
/**
* ** Get serialized queryString.
*/
serializeQueryParams() {
return RouteState.serializeQueryParams(this.queryParams);
}
/**
* ** Get url including QueryParams.
*/
getUrl() {
return `${this.absoluteRoutePath}?${this.serializeQueryParams()}`;
}
/**
* ** Get Data from Route configuration by key.
*
* - Return first (closest) found key starting from first RouteSegment.
*/
getData(key) {
return this.routeSegments.getData(key);
}
/**
* ** Get url param by key.
*
* - Return first (closest) found key starting from first RouteSegment.
*/
getParam(key) {
return this.routeSegments.getParam(key);
}
/**
* ** Get query param by key.
*/
getQueryParam(key) {
return this.routeSegments.getQueryParam(key);
}
/**
* ** Get Absolute ConfigPath.
*/
getAbsoluteConfigPath() {
return this.absoluteConfigPath;
}
/**
* ** Get parent of current Absolute ConfigPath.
*/
getParentAbsoluteConfigPath() {
const configPathSegments = this.configPathSegments;
configPathSegments.pop();
return RouteState._resolveAbsolutePath(configPathSegments);
}
/**
* ** Get Absolute RoutePath.
*/
getAbsoluteRoutePath() {
return RouteState._resolveAbsolutePath(this.routePathSegments);
}
/**
* ** Get parent of current Absolute RoutePath.
*/
getParentAbsoluteRoutePath() {
const routePathSegments = this.routePathSegments;
routePathSegments.pop();
return RouteState._resolveAbsolutePath(routePathSegments);
}
/**
* @inheritDoc
*/
toJSON() {
return {
url: this.url,
routePath: this.routePath,
absoluteRoutePath: this.absoluteRoutePath,
routePathSegments: this.routePathSegments,
configPath: this.configPath,
absoluteConfigPath: this.absoluteConfigPath,
configPathSegments: this.configPathSegments,
queryParams: this.queryParams,
routeSegments: this.routeSegments
};
}
/**
* ** Resolve Absolute RoutePath from given routePathSegments.
*/
static _resolveAbsolutePath(routePathSegments) {
const path = routePathSegments.join('/').replace(/^\/+/, '');
if (path === '') {
return '/';
}
return `/${path}`;
}
}
/**
* ** Router state.
*/
export class RouterState {
/**
* ** Constructor.
*/
constructor(state, navigationId) {
this.state = state ?? RouteState.empty();
this.navigationId = navigationId ?? null;
this.previousStates = [];
}
/**
* ** Factory method.
*/
static of(state, navigationId) {
return new RouterState(state, navigationId);
}
/**
* ** Factory method for empty State.
*/
static empty() {
return RouterState.of(null, null);
}
/**
* ** Returns previous RouterState if exist otherwise null.
*
* - Optional parameter could be provided to instruct which previous RouterState to return, default one is 0.
* - 0 means the first before current.
* - 1 means the second before current.
* - 2 means the third before current.
* - 3 ... etc...
*/
getPrevious(index = 0) {
const lookupIndex = CollectionsUtil.isNumber(index) ? index : 0;
if (lookupIndex >= 0 && lookupIndex < this.previousStates.length) {
return this.previousStates[lookupIndex];
}
return null;
}
/**
* ** Append previous RouterState[] to current One.
*
* - Internal API used in reducer, not for public use.
*/
appendPrevious(routerState) {
const previousStoredStates = [...routerState.previousStates];
const cleanedPreviousState = RouterState.of(routerState.state, routerState.navigationId);
if (this.navigationId !== cleanedPreviousState.navigationId) {
if (previousStoredStates.length >= 10) {
previousStoredStates.pop();
}
previousStoredStates.unshift(cleanedPreviousState);
}
this.previousStates.length = 0;
this.previousStates.push(...previousStoredStates);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"route.model.js","sourceRoot":"","sources":["../../../../../../../projects/shared/src/lib/core/router/model/route.model.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2023-2025 Broadcom\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/* eslint-disable @typescript-eslint/member-ordering */\n\nimport { Params } from '@angular/router';\n\nimport { BaseRouterStoreState, RouterReducerState } from '@ngrx/router-store';\n\nimport { CollectionsUtil, PrimitivesNil } from '../../../utils';\n\nimport { Serializable, TaurusRouteData } from '../../../common';\n\n/**\n * ** Route Segments Class.\n */\nexport class RouteSegments {\n    public readonly routePath: string;\n    public readonly data: TaurusRouteData;\n    public readonly params: Params;\n    public readonly queryParams: Params;\n    public readonly parent?: RouteSegments;\n    public readonly configPath?: string;\n\n    /**\n     * ** Constructor.\n     */\n    constructor(\n        routePath: string,\n        data: TaurusRouteData,\n        params: Params,\n        queryParams: Params,\n        parent?: RouteSegments,\n        configPath?: string\n    ) {\n        this.routePath = routePath ?? '';\n        this.data = data || {};\n        this.params = params || {};\n        this.queryParams = queryParams || {};\n        this.parent = parent;\n        this.configPath = configPath;\n    }\n\n    /**\n     * ** Factory method.\n     */\n    static of(\n        routePath?: string,\n        data?: TaurusRouteData,\n        params?: Params,\n        queryParams?: Params,\n        parent?: RouteSegments,\n        configPath?: string\n    ): RouteSegments {\n        return new RouteSegments(routePath, data, params, queryParams, parent, configPath);\n    }\n\n    /**\n     * ** Factory method for empty RouteSegments.\n     */\n    static empty(): RouteSegments {\n        return RouteSegments.of(null, null, null, null, null, null);\n    }\n\n    /**\n     * ** Get RoutePath Segments.\n     */\n    get routePathSegments(): string[] {\n        if (this.parent) {\n            return ([] as string[]).concat(this.parent.routePathSegments, this.routePath).filter((path) => path);\n        }\n\n        return [this.routePath];\n    }\n\n    /**\n     * ** Get ConfigPath Segments.\n     */\n    get configPathSegments(): string[] {\n        if (this.parent) {\n            return ([] as string[]).concat(this.parent.configPathSegments, this.configPath).filter((path) => path);\n        }\n\n        return [this.configPath];\n    }\n\n    /**\n     * ** Get Data from Route configuration by key.\n     *\n     *      - Return first (closest) found key starting from the current one.\n     */\n    getData<T>(key: string): T {\n        if (this.data[key]) {\n            return this.data[key] as T;\n        }\n\n        if (this.parent) {\n            return this.parent.getData<T>(key);\n        }\n\n        return undefined;\n    }\n\n    /**\n     * ** Get url param by key.\n     *\n     *      - Return first (closest) found key starting from the current one.\n     */\n    getParam(key: string): string {\n        if (this.params[key]) {\n            return this.params[key] as string;\n        }\n\n        if (this.parent) {\n            return this.parent.getParam(key);\n        }\n\n        return undefined;\n    }\n\n    /**\n     * ** Get query param by key.\n     */\n    getQueryParam(key: string): string {\n        if (this.queryParams[key]) {\n            return this.queryParams[key] as string;\n        }\n\n        if (this.parent) {\n            return this.parent.getQueryParam(key);\n        }\n\n        return undefined;\n    }\n}\n\n/**\n * ** Route State Class.\n */\nexport class RouteState implements BaseRouterStoreState, Serializable<SerializedRouteState> {\n    public readonly routeSegments: RouteSegments;\n    public readonly url: string;\n\n    /**\n     * ** Constructor.\n     */\n    constructor(routeSegments: RouteSegments, url: string) {\n        this.routeSegments = routeSegments ?? RouteSegments.empty();\n        this.url = url ?? '';\n    }\n\n    /**\n     * ** Factory method.\n     */\n    static of(routeSegments: RouteSegments, url: string): RouteState {\n        return new RouteState(routeSegments, url);\n    }\n\n    /**\n     * ** Factory method for empty State.\n     */\n    static empty(): RouteState {\n        return RouteState.of(null, null);\n    }\n\n    /**\n     * ** Get serialized queryString.\n     */\n    static serializeQueryParams(queryParams: unknown): string {\n        const paramsKeys = Object.keys(queryParams);\n\n        if (!paramsKeys.length) {\n            return '';\n        }\n\n        return paramsKeys.map((key) => `${encodeURIComponent(key)}=${encodeURIComponent(queryParams[key] as string)}`).join('&');\n    }\n\n    /**\n     * ** Returns current RoutePath.\n     */\n    get routePath(): string {\n        return this.routeSegments.routePath;\n    }\n\n    /**\n     * ** Returns current Absolute RoutePath.\n     */\n    get absoluteRoutePath(): string {\n        return RouteState._resolveAbsolutePath(this.routePathSegments);\n    }\n\n    /**\n     * ** Returns the route paths for each route segment starting from the root.\n     */\n    get routePathSegments(): string[] {\n        return this.routeSegments.routePathSegments;\n    }\n\n    /**\n     * ** Returns current ConfigPath.\n     */\n    get configPath(): string {\n        return this.routeSegments.configPath;\n    }\n\n    /**\n     * ** Returns current Absolute ConfigPath.\n     */\n    get absoluteConfigPath(): string {\n        return RouteState._resolveAbsolutePath(this.configPathSegments);\n    }\n\n    /**\n     * ** Returns the config paths for each route segment starting from the root.\n     */\n    get configPathSegments(): string[] {\n        return this.routeSegments.configPathSegments;\n    }\n\n    /**\n     * ** Get all query params.\n     */\n    get queryParams(): Params {\n        return this.routeSegments.queryParams;\n    }\n\n    /**\n     * ** Get serialized queryString.\n     */\n    serializeQueryParams(): string {\n        return RouteState.serializeQueryParams(this.queryParams);\n    }\n\n    /**\n     * ** Get url including QueryParams.\n     */\n    getUrl(): string {\n        return `${this.absoluteRoutePath}?${this.serializeQueryParams()}`;\n    }\n\n    /**\n     * ** Get Data from Route configuration by key.\n     *\n     *      - Return first (closest) found key starting from first RouteSegment.\n     */\n    getData<T>(key: string): T {\n        return this.routeSegments.getData<T>(key);\n    }\n\n    /**\n     * ** Get url param by key.\n     *\n     *      - Return first (closest) found key starting from first RouteSegment.\n     */\n    getParam(key: string): string {\n        return this.routeSegments.getParam(key);\n    }\n\n    /**\n     * ** Get query param by key.\n     */\n    getQueryParam(key: string): string {\n        return this.routeSegments.getQueryParam(key);\n    }\n\n    /**\n     * ** Get Absolute ConfigPath.\n     */\n    getAbsoluteConfigPath(): string {\n        return this.absoluteConfigPath;\n    }\n\n    /**\n     * ** Get parent of current Absolute ConfigPath.\n     */\n    getParentAbsoluteConfigPath(): string {\n        const configPathSegments = this.configPathSegments;\n        configPathSegments.pop();\n\n        return RouteState._resolveAbsolutePath(configPathSegments);\n    }\n\n    /**\n     * ** Get Absolute RoutePath.\n     */\n    getAbsoluteRoutePath(): string {\n        return RouteState._resolveAbsolutePath(this.routePathSegments);\n    }\n\n    /**\n     * ** Get parent of current Absolute RoutePath.\n     */\n    getParentAbsoluteRoutePath(): string {\n        const routePathSegments = this.routePathSegments;\n        routePathSegments.pop();\n\n        return RouteState._resolveAbsolutePath(routePathSegments);\n    }\n\n    /**\n     * @inheritDoc\n     */\n    toJSON(): SerializedRouteState {\n        return {\n            url: this.url,\n            routePath: this.routePath,\n            absoluteRoutePath: this.absoluteRoutePath,\n            routePathSegments: this.routePathSegments,\n            configPath: this.configPath,\n            absoluteConfigPath: this.absoluteConfigPath,\n            configPathSegments: this.configPathSegments,\n            queryParams: this.queryParams,\n            routeSegments: this.routeSegments\n        };\n    }\n\n    /**\n     * ** Resolve Absolute RoutePath from given routePathSegments.\n     */\n    private static _resolveAbsolutePath(routePathSegments: string[]): string {\n        const path = routePathSegments.join('/').replace(/^\\/+/, '');\n\n        if (path === '') {\n            return '/';\n        }\n\n        return `/${path}`;\n    }\n}\n\n/**\n * ** Router state.\n */\nexport class RouterState implements RouterReducerState<RouteState> {\n    readonly state: RouteState;\n    readonly navigationId: number;\n    readonly previousStates: RouterState[];\n\n    /**\n     * ** Constructor.\n     */\n    constructor(state: RouteState, navigationId: number) {\n        this.state = state ?? RouteState.empty();\n        this.navigationId = navigationId ?? null;\n        this.previousStates = [];\n    }\n\n    /**\n     * ** Factory method.\n     */\n    static of(state: RouteState, navigationId: number): RouterState {\n        return new RouterState(state, navigationId);\n    }\n\n    /**\n     * ** Factory method for empty State.\n     */\n    static empty(): RouterState {\n        return RouterState.of(null, null);\n    }\n\n    /**\n     * ** Returns previous RouterState if exist otherwise null.\n     *\n     *      - Optional parameter could be provided to instruct which previous RouterState to return, default one is 0.\n     *          - 0 means the first before current.\n     *          - 1 means the second before current.\n     *          - 2 means the third before current.\n     *          - 3 ... etc...\n     */\n    getPrevious(index = 0): RouterState | null {\n        const lookupIndex = CollectionsUtil.isNumber(index) ? index : 0;\n\n        if (lookupIndex >= 0 && lookupIndex < this.previousStates.length) {\n            return this.previousStates[lookupIndex];\n        }\n\n        return null;\n    }\n\n    /**\n     * ** Append previous RouterState[] to current One.\n     *\n     *      - Internal API used in reducer, not for public use.\n     */\n    appendPrevious(routerState: RouterState): void {\n        const previousStoredStates: RouterState[] = [...routerState.previousStates];\n        const cleanedPreviousState = RouterState.of(routerState.state, routerState.navigationId);\n\n        if (this.navigationId !== cleanedPreviousState.navigationId) {\n            if (previousStoredStates.length >= 10) {\n                previousStoredStates.pop();\n            }\n\n            previousStoredStates.unshift(cleanedPreviousState);\n        }\n\n        this.previousStates.length = 0;\n        this.previousStates.push(...previousStoredStates);\n    }\n}\n\n/**\n * ** Route state serialized.\n */\ninterface SerializedRouteState {\n    url: string;\n    routePath: string;\n    absoluteRoutePath: string;\n    routePathSegments: string[];\n    configPath: string;\n    absoluteConfigPath: string;\n    configPathSegments: string[];\n    queryParams: { [key: string]: PrimitivesNil };\n    routeSegments: RouteSegments;\n}\n"]}