@versatiledatakit/shared
Version:
Versatile Data Kit Shared library enables reusability of shared features like: NgRx Redux, Error Handlers, Utils, Generic Components, etc.
193 lines • 33.4 kB
JavaScript
/*
* Copyright 2023-2025 Broadcom
* SPDX-License-Identifier: Apache-2.0
*/
/* eslint-disable @typescript-eslint/unified-signatures,ngrx/avoid-mapping-selectors */
import { Injectable } from '@angular/core';
import { filter, map, switchMap, take, withLatestFrom } from 'rxjs/operators';
import { CollectionsUtil } from '../../../utils';
import { GenericAction, STORE_COMPONENTS } from '../../ngrx';
import { ComponentIdle, ComponentInit, ComponentLoading, ComponentUpdate } from '../state';
import { ComponentModel, ComponentsStateHelper, ComponentStateImpl, FAILED, IDLE, INITIALIZED, LOADED, LOADING } from '../model';
import * as i0 from "@angular/core";
import * as i1 from "@ngrx/store";
import * as i2 from "../../router";
/**
* ** Service that manage Components State.
*/
export class ComponentService {
}
/**
* @inheritDoc
*/
export class ComponentServiceImpl extends ComponentService {
/**
* ** Constructor.
*/
constructor(store$, routerService) {
super();
this.store$ = store$;
this.routerService = routerService;
this.componentsStateHelper = new ComponentsStateHelper();
}
/**
* @inheritDoc
*/
init(id, routeState) {
this.store$
.select((store) => store)
.pipe(take(1), map((store) => this._getComponentState(id, routeState.routePathSegments, store.router, store.components)))
.subscribe((componentState) => {
if (componentState.status === INITIALIZED) {
this.store$.dispatch(ComponentInit.of(componentState));
}
});
return this.onInit(id, routeState.routePathSegments);
}
/**
* @inheritDoc
*/
idle(componentState) {
this.store$.dispatch(ComponentIdle.of(componentState));
}
/**
* @inheritDoc
*/
load(componentState) {
this.routerService
.get()
.pipe(take(1))
.subscribe((routerState) => {
if (componentState.status === INITIALIZED) {
this.store$.dispatch(ComponentIdle.of(componentState.copy({
status: IDLE,
navigationId: routerState.navigationId
})));
}
this.store$.dispatch(ComponentLoading.of(componentState.copy({
status: LOADING,
navigationId: routerState.navigationId
})));
});
return this.onLoaded(componentState.id, componentState.routePathSegments);
}
/**
* @inheritDoc
*/
update(componentState) {
this.routerService
.get()
.pipe(take(1))
.subscribe((routerState) => {
this.store$.dispatch(ComponentUpdate.of(componentState.copy({
navigationId: routerState.navigationId
})));
});
}
/**
* @inheritDoc
*/
hasInSegment(id, routePathSegments) {
return this.store$.select(STORE_COMPONENTS).pipe(withLatestFrom(this.routerService.getState()), map(([literalComponentsState, routeState]) => this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, ['*'])), take(1));
}
/**
* @inheritDoc
*/
onInit(id, routePathSegments) {
return this.store$.select(STORE_COMPONENTS).pipe(withLatestFrom(this.routerService.getState()), map(([literalComponentsState, routeState]) => this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, ['*'])), filter((isInitialized) => isInitialized), switchMap(() => this.getModel(id, routePathSegments, ['*'])), take(1));
}
/**
* @inheritDoc
*/
onLoaded(id, routePathSegments) {
return this.store$.select(STORE_COMPONENTS).pipe(withLatestFrom(this.routerService.getState()), map(([literalComponentsState, routeState]) => this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, [LOADED, FAILED])), filter((isLoaded) => isLoaded), switchMap(() => this.getModel(id, routePathSegments)), take(1));
}
/**
* @inheritDoc
*/
getModel(id, routePathSegments, statusWatch) {
const _statusWatch = statusWatch ?? [LOADED, FAILED];
return this.store$.select(STORE_COMPONENTS).pipe(switchMap((literalComponentsState) => this.routerService.get().pipe(map((routerState) => [literalComponentsState, routerState]))), filter(([literalComponentsState, routerState]) => this._isComponentInStatus(id, routePathSegments, literalComponentsState, routerState.state, _statusWatch)), map(([literalComponentsState, routerState]) => this._createModel(id, routePathSegments, literalComponentsState, routerState)));
}
/**
* @inheritDoc
*/
dispatchAction(type, componentState, task) {
this.getModel(componentState.id, componentState.routePathSegments, ['*'])
.pipe(take(1))
.subscribe((model) => this.store$.dispatch(GenericAction.of(type, model.getComponentState(), task)));
}
/**
* @inheritDoc
*/
initialize() {
// No-op.
}
// Get Component State from Store if exist, otherwise create new State.
_getComponentState(id, routePathSegments, routerState, literalComponentsState) {
let _navigationId = null;
let _routePath = null;
let _routePathSegments = [];
if (routerState) {
_navigationId = routerState.navigationId;
if (routerState.state && !routePathSegments) {
_routePath = routerState.state.routePath;
_routePathSegments = routerState.state.routePathSegments;
}
}
if (routePathSegments) {
_routePath = routePathSegments.slice().pop();
_routePathSegments = routePathSegments;
}
let componentState = this.componentsStateHelper.setState(literalComponentsState).getComponentState(id, _routePathSegments);
if (componentState) {
return componentState;
}
componentState = ComponentStateImpl.of({
id,
status: INITIALIZED,
routePath: _routePath,
routePathSegments: _routePathSegments,
navigationId: _navigationId
});
return componentState;
}
// Utility method that filter if provided state is in desired status.
_isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, statusWatch) {
let _routePathSegments = [];
if (CollectionsUtil.isArray(routePathSegments)) {
_routePathSegments = routePathSegments;
}
else if (routeState) {
_routePathSegments = routeState.routePathSegments;
}
const componentLiteralState = this.componentsStateHelper
.setState(literalComponentsState)
.getLiteralComponentState(id, _routePathSegments);
if (!componentLiteralState) {
return false;
}
if (statusWatch.indexOf('*') !== -1) {
return true;
}
return statusWatch.indexOf(componentLiteralState.status) !== -1;
}
// Creates Model from provided data.
_createModel(id, routePathSegments, literalComponentsState, routerState) {
let _routePathSegments = [];
if (CollectionsUtil.isArray(routePathSegments)) {
_routePathSegments = routePathSegments;
}
else if (routerState && routerState.state) {
_routePathSegments = routerState.state.routePathSegments;
}
const componentState = this.componentsStateHelper.setState(literalComponentsState).getComponentState(id, _routePathSegments);
return ComponentModel.of(componentState, CollectionsUtil.cloneDeep(routerState));
}
}
ComponentServiceImpl.ɵfac = function ComponentServiceImpl_Factory(t) { return new (t || ComponentServiceImpl)(i0.ɵɵinject(i1.Store), i0.ɵɵinject(i2.RouterService)); };
ComponentServiceImpl.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: ComponentServiceImpl, factory: ComponentServiceImpl.ɵfac });
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ComponentServiceImpl, [{
type: Injectable
}], function () { return [{ type: i1.Store }, { type: i2.RouterService }]; }, null); })();
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component.service.js","sourceRoot":"","sources":["../../../../../../../projects/shared/src/lib/core/component/services/component.service.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/unified-signatures,ngrx/avoid-mapping-selectors */\n\nimport { Injectable } from '@angular/core';\n\nimport { Observable } from 'rxjs';\nimport { filter, map, switchMap, take, withLatestFrom } from 'rxjs/operators';\n\nimport { Store } from '@ngrx/store';\n\nimport { CollectionsUtil } from '../../../utils';\n\nimport { RouterService, RouterState, RouteState } from '../../router';\nimport { GenericAction, STORE_COMPONENTS, StoreState } from '../../ngrx';\n\nimport { ComponentIdle, ComponentInit, ComponentLoading, ComponentUpdate } from '../state';\nimport {\n    ComponentModel,\n    ComponentsStateHelper,\n    ComponentState,\n    ComponentStateImpl,\n    FAILED,\n    IDLE,\n    INITIALIZED,\n    LiteralComponentsState,\n    LOADED,\n    LOADING,\n    StatusType\n} from '../model';\n\n/**\n * ** Service that manage Components State.\n */\nexport abstract class ComponentService {\n    /**\n     * ** Initialize Component State and return Model.\n     */\n    abstract init(id: string, routeState: RouteState): Observable<ComponentModel>;\n\n    /**\n     * ** Set Component status to IDLE.\n     */\n    abstract idle(componentState: ComponentState): void;\n\n    /**\n     * ** Load Component State and return Model.\n     */\n    abstract load(componentState: ComponentState): Observable<ComponentModel>;\n\n    /**\n     * ** Update Component State.\n     */\n    abstract update(componentState: ComponentState): void;\n\n    /**\n     * ** Acknowledge if has ComponentState in segment.\n     *\n     *      - true - has ComponentState.\n     *      - false - doesn't have ComponentState.\n     */\n    abstract hasInSegment(id: string, routePathSegments: string[]): Observable<boolean>;\n\n    /**\n     * ** Listener that fires once after successful Component State initialization and returns Model.\n     */\n    abstract onInit(id: string, routePathSegments: string[]): Observable<ComponentModel>;\n\n    /**\n     * ** Listener that fires once after successful Component State load and returns Model.\n     */\n    abstract onLoaded(id: string, routePathSegments: string[]): Observable<ComponentModel>;\n\n    /**\n     * ** Returns stream with value Component Model and fires whenever Component State changes in Store.\n     *\n     *      - If no statusWatch provided by default will listen for statuses {@link LOADED} and {@link FAILED}.\n     */\n    abstract getModel(id: string, routePathSegments: string[], statusWatch?: Array<StatusType | '*'>): Observable<ComponentModel>;\n\n    /**\n     * ** Dispatch GenericAction with provided Type, ComponentState and optionally task.\n     */\n    abstract dispatchAction(type: string, componentState: ComponentState, task?: string): void;\n\n    /**\n     * ** Initialize Service.\n     */\n    abstract initialize(): void;\n}\n\n/**\n * @inheritDoc\n */\n@Injectable()\nexport class ComponentServiceImpl extends ComponentService {\n    private readonly componentsStateHelper: ComponentsStateHelper;\n\n    /**\n     * ** Constructor.\n     */\n    constructor(\n        private readonly store$: Store<StoreState>,\n        private readonly routerService: RouterService\n    ) {\n        super();\n\n        this.componentsStateHelper = new ComponentsStateHelper();\n    }\n\n    /**\n     * @inheritDoc\n     */\n    init(id: string, routeState: RouteState): Observable<ComponentModel> {\n        this.store$\n            .select((store) => store)\n            .pipe(\n                take(1),\n                map((store) => this._getComponentState(id, routeState.routePathSegments, store.router, store.components))\n            )\n            .subscribe((componentState) => {\n                if (componentState.status === INITIALIZED) {\n                    this.store$.dispatch(ComponentInit.of(componentState));\n                }\n            });\n\n        return this.onInit(id, routeState.routePathSegments);\n    }\n\n    /**\n     * @inheritDoc\n     */\n    idle(componentState: ComponentState): void {\n        this.store$.dispatch(ComponentIdle.of(componentState));\n    }\n\n    /**\n     * @inheritDoc\n     */\n    load(componentState: ComponentState): Observable<ComponentModel> {\n        this.routerService\n            .get()\n            .pipe(take(1))\n            .subscribe((routerState: RouterState) => {\n                if (componentState.status === INITIALIZED) {\n                    this.store$.dispatch(\n                        ComponentIdle.of(\n                            componentState.copy({\n                                status: IDLE,\n                                navigationId: routerState.navigationId\n                            })\n                        )\n                    );\n                }\n\n                this.store$.dispatch(\n                    ComponentLoading.of(\n                        componentState.copy({\n                            status: LOADING,\n                            navigationId: routerState.navigationId\n                        })\n                    )\n                );\n            });\n\n        return this.onLoaded(componentState.id, componentState.routePathSegments);\n    }\n\n    /**\n     * @inheritDoc\n     */\n    update(componentState: ComponentState): void {\n        this.routerService\n            .get()\n            .pipe(take(1))\n            .subscribe((routerState) => {\n                this.store$.dispatch(\n                    ComponentUpdate.of(\n                        componentState.copy({\n                            navigationId: routerState.navigationId\n                        })\n                    )\n                );\n            });\n    }\n\n    /**\n     * @inheritDoc\n     */\n    hasInSegment(id: string, routePathSegments: string[]): Observable<boolean> {\n        return this.store$.select(STORE_COMPONENTS).pipe(\n            withLatestFrom(this.routerService.getState()),\n            map(([literalComponentsState, routeState]) =>\n                this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, ['*'])\n            ),\n            take(1)\n        );\n    }\n\n    /**\n     * @inheritDoc\n     */\n    onInit(id: string, routePathSegments: string[]): Observable<ComponentModel> {\n        return this.store$.select(STORE_COMPONENTS).pipe(\n            withLatestFrom(this.routerService.getState()),\n            map(([literalComponentsState, routeState]) =>\n                this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, ['*'])\n            ),\n            filter((isInitialized) => isInitialized),\n            switchMap(() => this.getModel(id, routePathSegments, ['*'])),\n            take(1)\n        );\n    }\n\n    /**\n     * @inheritDoc\n     */\n    onLoaded(id: string, routePathSegments: string[]): Observable<ComponentModel> {\n        return this.store$.select(STORE_COMPONENTS).pipe(\n            withLatestFrom(this.routerService.getState()),\n            map(([literalComponentsState, routeState]) =>\n                this._isComponentInStatus(id, routePathSegments, literalComponentsState, routeState, [LOADED, FAILED])\n            ),\n            filter((isLoaded) => isLoaded),\n            switchMap(() => this.getModel(id, routePathSegments)),\n            take(1)\n        );\n    }\n\n    /**\n     * @inheritDoc\n     */\n    getModel(id: string, routePathSegments: string[], statusWatch?: Array<StatusType | '*'>): Observable<ComponentModel> {\n        const _statusWatch = statusWatch ?? [LOADED, FAILED];\n\n        return this.store$.select(STORE_COMPONENTS).pipe(\n            switchMap((literalComponentsState) =>\n                this.routerService.get().pipe(map((routerState) => [literalComponentsState, routerState]))\n            ),\n            filter(([literalComponentsState, routerState]: [LiteralComponentsState, RouterState]) =>\n                this._isComponentInStatus(id, routePathSegments, literalComponentsState, routerState.state, _statusWatch)\n            ),\n            map(([literalComponentsState, routerState]) => this._createModel(id, routePathSegments, literalComponentsState, routerState))\n        );\n    }\n\n    /**\n     * @inheritDoc\n     */\n    dispatchAction(type: string, componentState: ComponentState, task?: string): void {\n        this.getModel(componentState.id, componentState.routePathSegments, ['*'])\n            .pipe(take(1))\n            .subscribe((model) => this.store$.dispatch(GenericAction.of(type, model.getComponentState(), task)));\n    }\n\n    /**\n     * @inheritDoc\n     */\n    initialize() {\n        // No-op.\n    }\n\n    // Get Component State from Store if exist, otherwise create new State.\n    private _getComponentState(\n        id: string,\n        routePathSegments: string[],\n        routerState: RouterState,\n        literalComponentsState: LiteralComponentsState\n    ): ComponentState {\n        let _navigationId: number = null;\n        let _routePath: string = null;\n        let _routePathSegments: string[] = [];\n\n        if (routerState) {\n            _navigationId = routerState.navigationId;\n\n            if (routerState.state && !routePathSegments) {\n                _routePath = routerState.state.routePath;\n                _routePathSegments = routerState.state.routePathSegments;\n            }\n        }\n\n        if (routePathSegments) {\n            _routePath = routePathSegments.slice().pop();\n            _routePathSegments = routePathSegments;\n        }\n\n        let componentState = this.componentsStateHelper.setState(literalComponentsState).getComponentState(id, _routePathSegments);\n\n        if (componentState) {\n            return componentState;\n        }\n\n        componentState = ComponentStateImpl.of({\n            id,\n            status: INITIALIZED,\n            routePath: _routePath,\n            routePathSegments: _routePathSegments,\n            navigationId: _navigationId\n        });\n\n        return componentState;\n    }\n\n    // Utility method that filter if provided state is in desired status.\n    private _isComponentInStatus(\n        id: string,\n        routePathSegments: string[],\n        literalComponentsState: LiteralComponentsState,\n        routeState: RouteState,\n        statusWatch: Array<StatusType | '*'>\n    ): boolean {\n        let _routePathSegments: string[] = [];\n\n        if (CollectionsUtil.isArray(routePathSegments)) {\n            _routePathSegments = routePathSegments;\n        } else if (routeState) {\n            _routePathSegments = routeState.routePathSegments;\n        }\n\n        const componentLiteralState = this.componentsStateHelper\n            .setState(literalComponentsState)\n            .getLiteralComponentState(id, _routePathSegments);\n\n        if (!componentLiteralState) {\n            return false;\n        }\n\n        if (statusWatch.indexOf('*') !== -1) {\n            return true;\n        }\n\n        return statusWatch.indexOf(componentLiteralState.status) !== -1;\n    }\n\n    // Creates Model from provided data.\n    private _createModel(\n        id: string,\n        routePathSegments: string[],\n        literalComponentsState: LiteralComponentsState,\n        routerState: RouterState\n    ): ComponentModel {\n        let _routePathSegments: string[] = [];\n\n        if (CollectionsUtil.isArray(routePathSegments)) {\n            _routePathSegments = routePathSegments;\n        } else if (routerState && routerState.state) {\n            _routePathSegments = routerState.state.routePathSegments;\n        }\n\n        const componentState = this.componentsStateHelper.setState(literalComponentsState).getComponentState(id, _routePathSegments);\n\n        return ComponentModel.of(componentState, CollectionsUtil.cloneDeep(routerState));\n    }\n}\n"]}