@spartacus/smartedit
Version:
Smart Edit feature library for Spartacus
162 lines • 20.7 kB
JavaScript
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"]}