@spartacus/core
Version:
Spartacus - the core framework
173 lines • 23.1 kB
JavaScript
import { Injectable, InjectionToken } from '@angular/core';
import * as fromNgrxRouter from '@ngrx/router-store';
import { PageType } from '../../../model/cms.model';
import { HOME_PAGE_CONTEXT, SMART_EDIT_CONTEXT, } from '../../models/page-context.model';
import { CHANGE_NEXT_PAGE_CONTEXT } from '../actions/router.action';
import * as i0 from "@angular/core";
import * as i1 from "../../configurable-routes/routing-config.service";
export const initialState = {
navigationId: 0,
state: {
url: '',
queryParams: {},
params: {},
context: {
id: '',
},
cmsRequired: false,
semanticRoute: undefined,
},
nextState: undefined,
};
export function getReducers() {
return {
router: reducer,
};
}
export function reducer(state = initialState, action) {
var _a, _b;
switch (action.type) {
case fromNgrxRouter.ROUTER_NAVIGATION: {
return Object.assign(Object.assign({}, state), { nextState: action.payload.routerState, navigationId: action.payload.event.id });
}
case fromNgrxRouter.ROUTER_ERROR:
case fromNgrxRouter.ROUTER_CANCEL: {
return Object.assign(Object.assign({}, state), { nextState: undefined });
}
case CHANGE_NEXT_PAGE_CONTEXT: {
return state.nextState
? Object.assign(Object.assign({}, state), { nextState: Object.assign(Object.assign({}, state.nextState), { context: action.payload }) }) : state;
}
case fromNgrxRouter.ROUTER_NAVIGATED: {
return {
state: Object.assign(Object.assign({}, action.payload.routerState), { context:
// we want to preserve already resolved context,
// in case it was changed while navigating
(_b = (_a = state.nextState) === null || _a === void 0 ? void 0 : _a.context) !== null && _b !== void 0 ? _b : action.payload.routerState.context }),
navigationId: action.payload.event.id,
nextState: undefined,
};
}
default: {
return state;
}
}
}
export const reducerToken = new InjectionToken('RouterReducers');
export const reducerProvider = {
provide: reducerToken,
useFactory: getReducers,
};
/* The serializer is there to parse the RouterStateSnapshot,
and to reduce the amount of properties to be passed to the reducer.
*/
export class CustomSerializer {
constructor(routingConfig) {
this.routingConfig = routingConfig;
}
serialize(routerState) {
var _a, _b;
let state = routerState.root;
let cmsRequired = false;
let context;
let semanticRoute;
let urlString = '';
while (state.firstChild) {
state = state.firstChild;
urlString +=
'/' + state.url.map((urlSegment) => urlSegment.path).join('/');
// we use semantic route information embedded from any parent route
if ((_a = state.data) === null || _a === void 0 ? void 0 : _a.cxRoute) {
semanticRoute = (_b = state.data) === null || _b === void 0 ? void 0 : _b.cxRoute;
}
// we use context information embedded in Cms driven routes from any parent route
if (state.data && state.data.cxCmsRouteContext) {
context = state.data.cxCmsRouteContext;
}
// we assume, that any route that has CmsPageGuard or it's child
// is cmsRequired
if (!cmsRequired &&
(context ||
(state.routeConfig &&
state.routeConfig.canActivate &&
state.routeConfig.canActivate.find((x) => x && x.guardName === 'CmsPageGuard')))) {
cmsRequired = true;
}
}
// If `semanticRoute` couldn't be already recognized using `data.cxRoute` property
// let's lookup the routing configuration to find the semantic route that has exactly the same configured path as the current URL.
// This will work only for simple URLs without any dynamic routing parameters.
semanticRoute = semanticRoute || this.lookupSemanticRoute(urlString);
const { params } = state;
// we give smartedit preview page a PageContext
if (state.url.length > 0 && state.url[0].path === 'cx-preview') {
context = {
id: SMART_EDIT_CONTEXT,
type: PageType.CONTENT_PAGE,
};
}
else {
if (params['productCode']) {
context = { id: params['productCode'], type: PageType.PRODUCT_PAGE };
}
else if (params['categoryCode']) {
context = { id: params['categoryCode'], type: PageType.CATEGORY_PAGE };
}
else if (params['brandCode']) {
context = { id: params['brandCode'], type: PageType.CATEGORY_PAGE };
}
else if (state.data.pageLabel !== undefined) {
context = { id: state.data.pageLabel, type: PageType.CONTENT_PAGE };
}
else if (!context) {
if (state.url.length > 0) {
const pageLabel = '/' + state.url.map((urlSegment) => urlSegment.path).join('/');
context = {
id: pageLabel,
type: PageType.CONTENT_PAGE,
};
}
else {
context = {
// We like URLs to be driven by the backend, the CMS actually returns the homepage
// if no page label is given. Our logic however requires an id. undefined doesn't work.
id: HOME_PAGE_CONTEXT,
// We currently need to support a hardcoded page type, since the internal store uses the page
// type to store the content.
type: PageType.CONTENT_PAGE,
};
}
}
}
return {
url: routerState.url,
queryParams: routerState.root.queryParams,
params,
context,
cmsRequired,
semanticRoute,
};
}
/**
* Returns the semantic route name for given page label.
*
* *NOTE*: It works only for simple static urls that are equal to the page label
* of cms-driven content page. For example: `/my-account/address-book`.
*
* It doesn't work for URLs with dynamic parameters. But such case can be handled
* by reading the defined `data.cxRoute` from the Angular Routes.
*
* @param path path to be found in the routing config
*/
lookupSemanticRoute(path) {
// Page label is assumed to start with `/`, but Spartacus configured paths
// don't start with slash. So we remove the leading slash:
return this.routingConfig.getRouteName(path.substr(1));
}
}
CustomSerializer.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CustomSerializer, deps: [{ token: i1.RoutingConfigService }], target: i0.ɵɵFactoryTarget.Injectable });
CustomSerializer.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CustomSerializer });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: CustomSerializer, decorators: [{
type: Injectable
}], ctorParameters: function () { return [{ type: i1.RoutingConfigService }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"router.reducer.js","sourceRoot":"","sources":["../../../../../../../projects/core/src/routing/store/reducers/router.reducer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,cAAc,EAAY,MAAM,eAAe,CAAC;AAErE,OAAO,KAAK,cAAc,MAAM,oBAAoB,CAAC;AAErD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AAGpD,OAAO,EACL,iBAAiB,EAEjB,kBAAkB,GACnB,MAAM,iCAAiC,CAAC;AACzC,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAC;;;AAOpE,MAAM,CAAC,MAAM,YAAY,GAAgB;IACvC,YAAY,EAAE,CAAC;IACf,KAAK,EAAE;QACL,GAAG,EAAE,EAAE;QACP,WAAW,EAAE,EAAE;QACf,MAAM,EAAE,EAAE;QACV,OAAO,EAAE;YACP,EAAE,EAAE,EAAE;SACP;QACD,WAAW,EAAE,KAAK;QAClB,aAAa,EAAE,SAAS;KACzB;IACD,SAAS,EAAE,SAAS;CACrB,CAAC;AAEF,MAAM,UAAU,WAAW;IACzB,OAAO;QACL,MAAM,EAAE,OAAO;KAChB,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,OAAO,CACrB,QAAqB,YAAY,EACjC,MAAW;;IAEX,QAAQ,MAAM,CAAC,IAAI,EAAE;QACnB,KAAK,cAAc,CAAC,iBAAiB,CAAC,CAAC;YACrC,uCACK,KAAK,KACR,SAAS,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW,EACrC,YAAY,EAAE,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,IACrC;SACH;QAED,KAAK,cAAc,CAAC,YAAY,CAAC;QACjC,KAAK,cAAc,CAAC,aAAa,CAAC,CAAC;YACjC,uCACK,KAAK,KACR,SAAS,EAAE,SAAS,IACpB;SACH;QAED,KAAK,wBAAwB,CAAC,CAAC;YAC7B,OAAO,KAAK,CAAC,SAAS;gBACpB,CAAC,iCACM,KAAK,KACR,SAAS,kCAAO,KAAK,CAAC,SAAS,KAAE,OAAO,EAAE,MAAM,CAAC,OAAO,OAE5D,CAAC,CAAC,KAAK,CAAC;SACX;QAED,KAAK,cAAc,CAAC,gBAAgB,CAAC,CAAC;YACpC,OAAO;gBACL,KAAK,kCACA,MAAM,CAAC,OAAO,CAAC,WAAW,KAC7B,OAAO;oBACL,gDAAgD;oBAChD,0CAA0C;oBAC1C,MAAA,MAAA,KAAK,CAAC,SAAS,0CAAE,OAAO,mCAAI,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,OAAO,GACjE;gBACD,YAAY,EAAE,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;gBACrC,SAAS,EAAE,SAAS;aACrB,CAAC;SACH;QAED,OAAO,CAAC,CAAC;YACP,OAAO,KAAK,CAAC;SACd;KACF;AACH,CAAC;AAED,MAAM,CAAC,MAAM,YAAY,GACvB,IAAI,cAAc,CAA0B,gBAAgB,CAAC,CAAC;AAEhE,MAAM,CAAC,MAAM,eAAe,GAAa;IACvC,OAAO,EAAE,YAAY;IACrB,UAAU,EAAE,WAAW;CACxB,CAAC;AAEF;;GAEG;AAEH,MAAM,OAAO,gBAAgB;IA+G3B,YAAoB,aAAmC;QAAnC,kBAAa,GAAb,aAAa,CAAsB;IAAG,CAAC;IA5G3D,SAAS,CAAC,WAAgC;;QACxC,IAAI,KAAK,GACP,WAAW,CAAC,IAAiC,CAAC;QAChD,IAAI,WAAW,GAAG,KAAK,CAAC;QACxB,IAAI,OAAoB,CAAC;QACzB,IAAI,aAAqB,CAAC;QAC1B,IAAI,SAAS,GAAG,EAAE,CAAC;QAEnB,OAAO,KAAK,CAAC,UAAU,EAAE;YACvB,KAAK,GAAG,KAAK,CAAC,UAAuC,CAAC;YACtD,SAAS;gBACP,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAEjE,mEAAmE;YACnE,IAAI,MAAA,KAAK,CAAC,IAAI,0CAAE,OAAO,EAAE;gBACvB,aAAa,GAAG,MAAA,KAAK,CAAC,IAAI,0CAAE,OAAO,CAAC;aACrC;YAED,iFAAiF;YACjF,IAAI,KAAK,CAAC,IAAI,IAAI,KAAK,CAAC,IAAI,CAAC,iBAAiB,EAAE;gBAC9C,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,iBAAiB,CAAC;aACxC;YAED,gEAAgE;YAChE,iBAAiB;YACjB,IACE,CAAC,WAAW;gBACZ,CAAC,OAAO;oBACN,CAAC,KAAK,CAAC,WAAW;wBAChB,KAAK,CAAC,WAAW,CAAC,WAAW;wBAC7B,KAAK,CAAC,WAAW,CAAC,WAAW,CAAC,IAAI,CAChC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,SAAS,KAAK,cAAc,CAC3C,CAAC,CAAC,EACP;gBACA,WAAW,GAAG,IAAI,CAAC;aACpB;SACF;QAED,kFAAkF;QAClF,kIAAkI;QAClI,8EAA8E;QAC9E,aAAa,GAAG,aAAa,IAAI,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;QAErE,MAAM,EAAE,MAAM,EAAE,GAAG,KAAK,CAAC;QACzB,+CAA+C;QAC/C,IAAI,KAAK,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,YAAY,EAAE;YAC9D,OAAO,GAAG;gBACR,EAAE,EAAE,kBAAkB;gBACtB,IAAI,EAAE,QAAQ,CAAC,YAAY;aAC5B,CAAC;SACH;aAAM;YACL,IAAI,MAAM,CAAC,aAAa,CAAC,EAAE;gBACzB,OAAO,GAAG,EAAE,EAAE,EAAE,MAAM,CAAC,aAAa,CAAC,EAAE,IAAI,EAAE,QAAQ,CAAC,YAAY,EAAE,CAAC;aACtE;iBAAM,IAAI,MAAM,CAAC,cAAc,CAAC,EAAE;gBACjC,OAAO,GAAG,EAAE,EAAE,EAAE,MAAM,CAAC,cAAc,CAAC,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,EAAE,CAAC;aACxE;iBAAM,IAAI,MAAM,CAAC,WAAW,CAAC,EAAE;gBAC9B,OAAO,GAAG,EAAE,EAAE,EAAE,MAAM,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,EAAE,CAAC;aACrE;iBAAM,IAAI,KAAK,CAAC,IAAI,CAAC,SAAS,KAAK,SAAS,EAAE;gBAC7C,OAAO,GAAG,EAAE,EAAE,EAAE,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,QAAQ,CAAC,YAAY,EAAE,CAAC;aACrE;iBAAM,IAAI,CAAC,OAAO,EAAE;gBACnB,IAAI,KAAK,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE;oBACxB,MAAM,SAAS,GACb,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBACjE,OAAO,GAAG;wBACR,EAAE,EAAE,SAAS;wBACb,IAAI,EAAE,QAAQ,CAAC,YAAY;qBAC5B,CAAC;iBACH;qBAAM;oBACL,OAAO,GAAG;wBACR,kFAAkF;wBAClF,uFAAuF;wBACvF,EAAE,EAAE,iBAAiB;wBAErB,6FAA6F;wBAC7F,6BAA6B;wBAC7B,IAAI,EAAE,QAAQ,CAAC,YAAY;qBAC5B,CAAC;iBACH;aACF;SACF;QAED,OAAO;YACL,GAAG,EAAE,WAAW,CAAC,GAAG;YACpB,WAAW,EAAE,WAAW,CAAC,IAAI,CAAC,WAAW;YACzC,MAAM;YACN,OAAO;YACP,WAAW;YACX,aAAa;SACd,CAAC;IACJ,CAAC;IAED;;;;;;;;;;OAUG;IACK,mBAAmB,CAAC,IAAY;QACtC,0EAA0E;QAC1E,0DAA0D;QAC1D,OAAO,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,CAAC;;6GA7GU,gBAAgB;iHAAhB,gBAAgB;2FAAhB,gBAAgB;kBAD5B,UAAU","sourcesContent":["import { Injectable, InjectionToken, Provider } from '@angular/core';\nimport { RouterStateSnapshot } from '@angular/router';\nimport * as fromNgrxRouter from '@ngrx/router-store';\nimport { ActionReducerMap } from '@ngrx/store';\nimport { PageType } from '../../../model/cms.model';\nimport { RoutingConfigService } from '../../configurable-routes/routing-config.service';\nimport { CmsActivatedRouteSnapshot } from '../../models/cms-route';\nimport {\n  HOME_PAGE_CONTEXT,\n  PageContext,\n  SMART_EDIT_CONTEXT,\n} from '../../models/page-context.model';\nimport { CHANGE_NEXT_PAGE_CONTEXT } from '../actions/router.action';\nimport {\n  ActivatedRouterStateSnapshot,\n  RouterState,\n  State,\n} from '../routing-state';\n\nexport const initialState: RouterState = {\n  navigationId: 0,\n  state: {\n    url: '',\n    queryParams: {},\n    params: {},\n    context: {\n      id: '',\n    },\n    cmsRequired: false,\n    semanticRoute: undefined,\n  },\n  nextState: undefined,\n};\n\nexport function getReducers(): ActionReducerMap<State> {\n  return {\n    router: reducer,\n  };\n}\n\nexport function reducer(\n  state: RouterState = initialState,\n  action: any\n): RouterState {\n  switch (action.type) {\n    case fromNgrxRouter.ROUTER_NAVIGATION: {\n      return {\n        ...state,\n        nextState: action.payload.routerState,\n        navigationId: action.payload.event.id,\n      };\n    }\n\n    case fromNgrxRouter.ROUTER_ERROR:\n    case fromNgrxRouter.ROUTER_CANCEL: {\n      return {\n        ...state,\n        nextState: undefined,\n      };\n    }\n\n    case CHANGE_NEXT_PAGE_CONTEXT: {\n      return state.nextState\n        ? {\n            ...state,\n            nextState: { ...state.nextState, context: action.payload },\n          }\n        : state;\n    }\n\n    case fromNgrxRouter.ROUTER_NAVIGATED: {\n      return {\n        state: {\n          ...action.payload.routerState,\n          context:\n            // we want to preserve already resolved context,\n            // in case it was changed while navigating\n            state.nextState?.context ?? action.payload.routerState.context,\n        },\n        navigationId: action.payload.event.id,\n        nextState: undefined,\n      };\n    }\n\n    default: {\n      return state;\n    }\n  }\n}\n\nexport const reducerToken: InjectionToken<ActionReducerMap<State>> =\n  new InjectionToken<ActionReducerMap<State>>('RouterReducers');\n\nexport const reducerProvider: Provider = {\n  provide: reducerToken,\n  useFactory: getReducers,\n};\n\n/* The serializer is there to parse the RouterStateSnapshot,\nand to reduce the amount of properties to be passed to the reducer.\n */\n@Injectable()\nexport class CustomSerializer\n  implements fromNgrxRouter.RouterStateSerializer<ActivatedRouterStateSnapshot>\n{\n  serialize(routerState: RouterStateSnapshot): ActivatedRouterStateSnapshot {\n    let state: CmsActivatedRouteSnapshot =\n      routerState.root as CmsActivatedRouteSnapshot;\n    let cmsRequired = false;\n    let context: PageContext;\n    let semanticRoute: string;\n    let urlString = '';\n\n    while (state.firstChild) {\n      state = state.firstChild as CmsActivatedRouteSnapshot;\n      urlString +=\n        '/' + state.url.map((urlSegment) => urlSegment.path).join('/');\n\n      // we use semantic route information embedded from any parent route\n      if (state.data?.cxRoute) {\n        semanticRoute = state.data?.cxRoute;\n      }\n\n      // we use context information embedded in Cms driven routes from any parent route\n      if (state.data && state.data.cxCmsRouteContext) {\n        context = state.data.cxCmsRouteContext;\n      }\n\n      // we assume, that any route that has CmsPageGuard or it's child\n      // is cmsRequired\n      if (\n        !cmsRequired &&\n        (context ||\n          (state.routeConfig &&\n            state.routeConfig.canActivate &&\n            state.routeConfig.canActivate.find(\n              (x) => x && x.guardName === 'CmsPageGuard'\n            )))\n      ) {\n        cmsRequired = true;\n      }\n    }\n\n    // If `semanticRoute` couldn't be already recognized using `data.cxRoute` property\n    // let's lookup the routing configuration to find the semantic route that has exactly the same configured path as the current URL.\n    // This will work only for simple URLs without any dynamic routing parameters.\n    semanticRoute = semanticRoute || this.lookupSemanticRoute(urlString);\n\n    const { params } = state;\n    // we give smartedit preview page a PageContext\n    if (state.url.length > 0 && state.url[0].path === 'cx-preview') {\n      context = {\n        id: SMART_EDIT_CONTEXT,\n        type: PageType.CONTENT_PAGE,\n      };\n    } else {\n      if (params['productCode']) {\n        context = { id: params['productCode'], type: PageType.PRODUCT_PAGE };\n      } else if (params['categoryCode']) {\n        context = { id: params['categoryCode'], type: PageType.CATEGORY_PAGE };\n      } else if (params['brandCode']) {\n        context = { id: params['brandCode'], type: PageType.CATEGORY_PAGE };\n      } else if (state.data.pageLabel !== undefined) {\n        context = { id: state.data.pageLabel, type: PageType.CONTENT_PAGE };\n      } else if (!context) {\n        if (state.url.length > 0) {\n          const pageLabel =\n            '/' + state.url.map((urlSegment) => urlSegment.path).join('/');\n          context = {\n            id: pageLabel,\n            type: PageType.CONTENT_PAGE,\n          };\n        } else {\n          context = {\n            // We like URLs to be driven by the backend, the CMS actually returns the homepage\n            // if no page label is given. Our logic however requires an id. undefined doesn't work.\n            id: HOME_PAGE_CONTEXT,\n\n            // We currently need to support a hardcoded page type, since the internal store uses the page\n            // type to store the content.\n            type: PageType.CONTENT_PAGE,\n          };\n        }\n      }\n    }\n\n    return {\n      url: routerState.url,\n      queryParams: routerState.root.queryParams,\n      params,\n      context,\n      cmsRequired,\n      semanticRoute,\n    };\n  }\n\n  /**\n   * Returns the semantic route name for given page label.\n   *\n   * *NOTE*: It works only for simple static urls that are equal to the page label\n   * of cms-driven content page. For example: `/my-account/address-book`.\n   *\n   * It doesn't work for URLs with dynamic parameters. But such case can be handled\n   * by reading the defined `data.cxRoute` from the Angular Routes.\n   *\n   * @param path path to be found in the routing config\n   */\n  private lookupSemanticRoute(path: string): string {\n    // Page label is assumed to start with `/`, but Spartacus configured paths\n    // don't start with slash. So we remove the leading slash:\n    return this.routingConfig.getRouteName(path.substr(1));\n  }\n\n  constructor(private routingConfig: RoutingConfigService) {}\n}\n"]}