UNPKG

@spartacus/smartedit

Version:

Smart Edit feature library for Spartacus

162 lines 20.7 kB
import { Injectable } from '@angular/core'; import { PageType, } from '@spartacus/core'; import { filter, take } from 'rxjs/operators'; import * as i0 from "@angular/core"; import * as i1 from "@spartacus/core"; import * as i2 from "@spartacus/smartedit/root"; export class SmartEditService { constructor(cmsService, routingService, baseSiteService, zone, winRef, rendererFactory, config, scriptLoader) { this.cmsService = cmsService; this.routingService = routingService; this.baseSiteService = baseSiteService; this.zone = zone; this.winRef = winRef; this.rendererFactory = rendererFactory; this.config = config; this.scriptLoader = scriptLoader; this.isPreviewPage = false; // load webApplicationInjector.js first this.loadScript(); if (winRef.nativeWindow) { const window = winRef.nativeWindow; // rerender components and slots after editing window.smartedit = window.smartedit || {}; window.smartedit.renderComponent = (componentId, componentType, parentId) => { return this.renderComponent(componentId, componentType, parentId); }; // reprocess page window.smartedit.reprocessPage = this.reprocessPage; } } processCmsPage() { this.baseSiteService .get() .pipe(filter((site) => Boolean(site)), take(1)) .subscribe((site) => { this.defaultPreviewCategoryCode = site.defaultPreviewCategoryCode; this.defaultPreviewProductCode = site.defaultPreviewProductCode; this.cmsService .getCurrentPage() .pipe(filter(Boolean)) .subscribe((cmsPage) => { this._currentPageId = cmsPage.pageId; // before adding contract to page, we need redirect to that page this.goToPreviewPage(cmsPage); this.addPageContract(cmsPage); }); }); } /** * load webApplicationInjector.js */ loadScript() { var _a; this.scriptLoader.embedScript({ src: 'assets/webApplicationInjector.js', params: undefined, attributes: { id: 'text/smartedit-injector', 'data-smartedit-allow-origin': (_a = this.config.smartEdit) === null || _a === void 0 ? void 0 : _a.allowOrigin, }, }); } /** * add CSS classes in a body tag */ addPageContract(cmsPage) { const renderer = this.rendererFactory.createRenderer('body', null); const element = this.winRef.document.body; // remove old page contract const previousContract = []; Array.from(element.classList).forEach((attr) => previousContract.push(attr)); previousContract.forEach((attr) => renderer.removeClass(element, attr)); // add new page contract this.addSmartEditContract(element, renderer, cmsPage.properties); } /** * go to the default preview page */ goToPreviewPage(cmsPage) { // only the first page is the smartedit preview page if (!this.isPreviewPage) { this.isPreviewPage = true; if (cmsPage.type === PageType.PRODUCT_PAGE && this.defaultPreviewProductCode) { this.routingService.go({ cxRoute: 'product', params: { code: this.defaultPreviewProductCode, name: '' }, }); } else if (cmsPage.type === PageType.CATEGORY_PAGE && this.defaultPreviewCategoryCode) { this.routingService.go({ cxRoute: 'category', params: { code: this.defaultPreviewCategoryCode }, }); } } } /** * re-render CMS components and slots */ renderComponent(componentId, componentType, parentId) { if (componentId) { this.zone.run(() => { // without parentId, it is slot if (!parentId) { if (this._currentPageId) { this.cmsService.refreshPageById(this._currentPageId); } else { this.cmsService.refreshLatestPage(); } } else if (componentType) { this.cmsService.refreshComponent(componentId); } }); } return true; } reprocessPage() { // TODO: reprocess page API } /** * add smartedit HTML markup contract */ addSmartEditContract(element, renderer, properties) { if (properties) { // check each group of properties, e.g. smartedit Object.keys(properties).forEach((group) => { const name = 'data-' + group + '-'; const groupProps = properties[group]; // check each property in the group Object.keys(groupProps).forEach((propName) => { const propValue = groupProps[propName]; if (propName === 'classes') { const classes = propValue.split(' '); classes.forEach((classItem) => { renderer.addClass(element, classItem); }); } else { renderer.setAttribute(element, name + propName .split(/(?=[A-Z])/) .join('-') .toLowerCase(), propValue); } }); }); } } } SmartEditService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SmartEditService, deps: [{ token: i1.CmsService }, { token: i1.RoutingService }, { token: i1.BaseSiteService }, { token: i0.NgZone }, { token: i1.WindowRef }, { token: i0.RendererFactory2 }, { token: i2.SmartEditConfig }, { token: i1.ScriptLoader }], target: i0.ɵɵFactoryTarget.Injectable }); SmartEditService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SmartEditService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SmartEditService, decorators: [{ type: Injectable, args: [{ providedIn: 'root', }] }], ctorParameters: function () { return [{ type: i1.CmsService }, { type: i1.RoutingService }, { type: i1.BaseSiteService }, { type: i0.NgZone }, { type: i1.WindowRef }, { type: i0.RendererFactory2 }, { type: i2.SmartEditConfig }, { type: i1.ScriptLoader }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"smart-edit.service.js","sourceRoot":"","sources":["../../../../../feature-libs/smartedit/core/services/smart-edit.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, Renderer2, RendererFactory2 } from '@angular/core';\nimport {\n  BaseSiteService,\n  CmsService,\n  Page,\n  PageType,\n  RoutingService,\n  ScriptLoader,\n  WindowRef,\n} from '@spartacus/core';\nimport { SmartEditConfig } from '@spartacus/smartedit/root';\nimport { filter, take } from 'rxjs/operators';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class SmartEditService {\n  private isPreviewPage = false;\n  private _currentPageId: string | undefined;\n\n  private defaultPreviewProductCode: string | undefined;\n  private defaultPreviewCategoryCode: string | undefined;\n\n  constructor(\n    protected cmsService: CmsService,\n    protected routingService: RoutingService,\n    protected baseSiteService: BaseSiteService,\n    protected zone: NgZone,\n    protected winRef: WindowRef,\n    protected rendererFactory: RendererFactory2,\n    protected config: SmartEditConfig,\n    protected scriptLoader: ScriptLoader\n  ) {\n    // load webApplicationInjector.js first\n    this.loadScript();\n\n    if (winRef.nativeWindow) {\n      const window = winRef.nativeWindow as any;\n      // rerender components and slots after editing\n      window.smartedit = window.smartedit || {};\n      window.smartedit.renderComponent = (\n        componentId: string,\n        componentType: string,\n        parentId: string\n      ) => {\n        return this.renderComponent(componentId, componentType, parentId);\n      };\n\n      // reprocess page\n      window.smartedit.reprocessPage = this.reprocessPage;\n    }\n  }\n\n  public processCmsPage(): void {\n    this.baseSiteService\n      .get()\n      .pipe(\n        filter((site: any) => Boolean(site)),\n        take(1)\n      )\n      .subscribe((site) => {\n        this.defaultPreviewCategoryCode = site.defaultPreviewCategoryCode;\n        this.defaultPreviewProductCode = site.defaultPreviewProductCode;\n\n        this.cmsService\n          .getCurrentPage()\n          .pipe(filter<Page>(Boolean))\n          .subscribe((cmsPage) => {\n            this._currentPageId = cmsPage.pageId;\n            // before adding contract to page, we need redirect to that page\n            this.goToPreviewPage(cmsPage);\n            this.addPageContract(cmsPage);\n          });\n      });\n  }\n\n  /**\n   * load webApplicationInjector.js\n   */\n  protected loadScript(): void {\n    this.scriptLoader.embedScript({\n      src: 'assets/webApplicationInjector.js',\n      params: undefined,\n      attributes: {\n        id: 'text/smartedit-injector',\n        'data-smartedit-allow-origin': this.config.smartEdit?.allowOrigin,\n      },\n    });\n  }\n\n  /**\n   * add CSS classes in a body tag\n   */\n  protected addPageContract(cmsPage: Page) {\n    const renderer = this.rendererFactory.createRenderer('body', null);\n    const element = this.winRef.document.body;\n\n    // remove old page contract\n    const previousContract: string[] = [];\n    Array.from(element.classList).forEach((attr) =>\n      previousContract.push(attr)\n    );\n    previousContract.forEach((attr) => renderer.removeClass(element, attr));\n\n    // add new page contract\n    this.addSmartEditContract(element, renderer, cmsPage.properties);\n  }\n\n  /**\n   * go to the default preview page\n   */\n  protected goToPreviewPage(cmsPage: Page) {\n    // only the first page is the smartedit preview page\n    if (!this.isPreviewPage) {\n      this.isPreviewPage = true;\n      if (\n        cmsPage.type === PageType.PRODUCT_PAGE &&\n        this.defaultPreviewProductCode\n      ) {\n        this.routingService.go({\n          cxRoute: 'product',\n          params: { code: this.defaultPreviewProductCode, name: '' },\n        });\n      } else if (\n        cmsPage.type === PageType.CATEGORY_PAGE &&\n        this.defaultPreviewCategoryCode\n      ) {\n        this.routingService.go({\n          cxRoute: 'category',\n          params: { code: this.defaultPreviewCategoryCode },\n        });\n      }\n    }\n  }\n\n  /**\n   * re-render CMS components and slots\n   */\n  protected renderComponent(\n    componentId: string,\n    componentType?: string,\n    parentId?: string\n  ): boolean {\n    if (componentId) {\n      this.zone.run(() => {\n        // without parentId, it is slot\n        if (!parentId) {\n          if (this._currentPageId) {\n            this.cmsService.refreshPageById(this._currentPageId);\n          } else {\n            this.cmsService.refreshLatestPage();\n          }\n        } else if (componentType) {\n          this.cmsService.refreshComponent(componentId);\n        }\n      });\n    }\n\n    return true;\n  }\n\n  protected reprocessPage() {\n    // TODO: reprocess page API\n  }\n\n  /**\n   * add smartedit HTML markup contract\n   */\n  public addSmartEditContract(\n    element: Element,\n    renderer: Renderer2,\n    properties: any\n  ): void {\n    if (properties) {\n      // check each group of properties, e.g. smartedit\n      Object.keys(properties).forEach((group) => {\n        const name = 'data-' + group + '-';\n        const groupProps = properties[group];\n\n        // check each property in the group\n        Object.keys(groupProps).forEach((propName) => {\n          const propValue = groupProps[propName];\n          if (propName === 'classes') {\n            const classes = propValue.split(' ');\n            classes.forEach((classItem: string) => {\n              renderer.addClass(element, classItem);\n            });\n          } else {\n            renderer.setAttribute(\n              element,\n              name +\n                propName\n                  .split(/(?=[A-Z])/)\n                  .join('-')\n                  .toLowerCase(),\n              propValue\n            );\n          }\n        });\n      });\n    }\n  }\n}\n"]}