UNPKG

@adobe/cq-angular-editable-components

Version:

* [API](#api) * [Documentation](#documentation) * [Changelog](#changelog)

194 lines 21.4 kB
/* * ADOBE CONFIDENTIAL * * Copyright 2018 Adobe Systems Incorporated * All Rights Reserved. * * NOTICE: All information contained herein is, and remains * the property of Adobe Systems Incorporated and its suppliers, * if any. The intellectual and technical concepts contained * herein are proprietary to Adobe Systems Incorporated and its * suppliers and may be covered by U.S. and Foreign Patents, * patents in process, and are protected by trade secret or copyright law. * Dissemination of this information or reproduction of this material * is strictly forbidden unless prior written permission is obtained * from Adobe Systems Incorporated. */ import { __decorate, __metadata } from "tslib"; import { Directive, Input, Renderer2, ViewContainerRef, ComponentFactoryResolver, ComponentRef, AfterViewInit, OnInit, OnDestroy, ChangeDetectorRef, OnChanges } from '@angular/core'; import { ComponentMapping } from './component-mapping'; import { Constants } from './constants'; import { Utils } from './utils'; const PLACEHOLDER_CLASS_NAME = 'cq-placeholder'; let AEMComponentDirective = /** * The current directive provides advanced capabilities among which are * * - The management of the component placeholder in the Page Editor * - The dynamic instantiation of components based on a component definition * - The conversion from model fields to properties and injection in the component instance * - The management of HTMLElement attributes and class names on the native element */ class AEMComponentDirective { constructor(renderer, viewContainer, factoryResolver, _changeDetectorRef) { this.renderer = renderer; this.viewContainer = viewContainer; this.factoryResolver = factoryResolver; this._changeDetectorRef = _changeDetectorRef; } get cqItem() { return this._cqItem; } set cqItem(value) { this._cqItem = value; } get changeDetectorRef() { return this._changeDetectorRef; } ngOnInit() { this.renderComponent(ComponentMapping.get(this.type)); } ngOnChanges(changes) { this.updateComponentData(); } /** * Returns the type of the cqItem if exists. */ get type() { return this.cqItem && this.cqItem[Constants.TYPE_PROP]; } /** * Renders a component dynamically based on the component definition * * @param componentDefinition The component definition to render */ renderComponent(componentDefinition) { if (componentDefinition) { const factory = this.factoryResolver.resolveComponentFactory(componentDefinition); this.viewContainer.clear(); this._component = this.viewContainer.createComponent(factory); this.updateComponentData(); } } /** * Updates the data of the component based the data of the directive */ updateComponentData() { if (!this._component || !this._component.instance) { return; } const keys = Object.getOwnPropertyNames(this.cqItem); keys.forEach((key) => { let propKey = key; if (propKey.startsWith(':')) { // Transformation of internal properties namespaced with [:] to [cq] // :myProperty => cqMyProperty const tempKey = propKey.substr(1); propKey = 'cq' + tempKey.substr(0, 1).toUpperCase() + tempKey.substr(1); } this._component.instance[propKey] = this.cqItem[key]; }); this._component.instance.cqPath = this.cqPath; this._component.instance.itemName = this.itemName; const editConfig = ComponentMapping.getEditConfig(this.type); if (editConfig && Utils.isInEditor) { this.setupPlaceholder(editConfig); } this._changeDetectorRef.detectChanges(); } /** * Adds the specified item attributes to the element */ setupItemAttrs() { if (this.itemAttrs) { const keys = Object.getOwnPropertyNames(this.itemAttrs); keys.forEach((key) => { if (key === 'class') { const classes = this.itemAttrs[key].split(' '); classes.forEach((itemClass) => { this.renderer.addClass(this._component.location.nativeElement, itemClass); }); } else { this.renderer.setAttribute(this._component.location.nativeElement, key, this.itemAttrs[key]); } }); } } /** * Determines if the placeholder should e displayed. * * @param editConfig - the edit config of the directive */ usePlaceholder(editConfig) { return editConfig.isEmpty && typeof editConfig.isEmpty === 'function' && editConfig.isEmpty(this.cqItem); } /** * Setups the placeholder of needed for the AEM editor * * @param editConfig - the editConfig, which will dictate the classes to be added on. */ setupPlaceholder(editConfig) { if (this.usePlaceholder(editConfig)) { this.renderer.addClass(this._component.location.nativeElement, PLACEHOLDER_CLASS_NAME); this.renderer.setAttribute(this._component.location.nativeElement, 'data-emptytext', editConfig.emptyLabel); } else { this.renderer.removeClass(this._component.location.nativeElement, PLACEHOLDER_CLASS_NAME); this.renderer.removeAttribute(this._component.location.nativeElement, 'data-emptytext'); } } ngAfterViewInit() { this.setupItemAttrs(); } ngOnDestroy() { this._component && this._component.destroy(); } }; AEMComponentDirective.ctorParameters = () => [ { type: Renderer2 }, { type: ViewContainerRef }, { type: ComponentFactoryResolver }, { type: ChangeDetectorRef } ]; __decorate([ Input(), __metadata("design:type", Object), __metadata("design:paramtypes", [Object]) ], AEMComponentDirective.prototype, "cqItem", null); __decorate([ Input(), __metadata("design:type", String) ], AEMComponentDirective.prototype, "cqPath", void 0); __decorate([ Input(), __metadata("design:type", String) ], AEMComponentDirective.prototype, "itemName", void 0); __decorate([ Input(), __metadata("design:type", Object) ], AEMComponentDirective.prototype, "itemAttrs", void 0); __decorate([ Input(), __metadata("design:type", Object) ], AEMComponentDirective.prototype, "aemComponent", void 0); AEMComponentDirective = __decorate([ Directive({ selector: '[aemComponent]' }) /** * The current directive provides advanced capabilities among which are * * - The management of the component placeholder in the Page Editor * - The dynamic instantiation of components based on a component definition * - The conversion from model fields to properties and injection in the component instance * - The management of HTMLElement attributes and class names on the native element */ , __metadata("design:paramtypes", [Renderer2, ViewContainerRef, ComponentFactoryResolver, ChangeDetectorRef]) ], AEMComponentDirective); export { AEMComponentDirective }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aem-component.directive.js","sourceRoot":"ng://@adobe/cq-angular-editable-components/","sources":["lib/layout/aem-component.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * ADOBE CONFIDENTIAL\n *\n * Copyright 2018 Adobe Systems Incorporated\n * All Rights Reserved.\n *\n * NOTICE:  All information contained herein is, and remains\n * the property of Adobe Systems Incorporated and its suppliers,\n * if any.  The intellectual and technical concepts contained\n * herein are proprietary to Adobe Systems Incorporated and its\n * suppliers and may be covered by U.S. and Foreign Patents,\n * patents in process, and are protected by trade secret or copyright law.\n * Dissemination of this information or reproduction of this material\n * is strictly forbidden unless prior written permission is obtained\n * from Adobe Systems Incorporated.\n */\n\nimport {\n  Directive,\n  Input,\n  Renderer2,\n  ViewContainerRef,\n  ComponentFactoryResolver,\n  ComponentRef,\n  AfterViewInit,\n  OnInit,\n  OnDestroy, ChangeDetectorRef, OnChanges\n} from '@angular/core';\n\nimport { ComponentMapping } from './component-mapping';\nimport { Constants } from './constants';\nimport { Utils } from './utils';\n\n\nconst PLACEHOLDER_CLASS_NAME = 'cq-placeholder';\n\n@Directive({\n  selector: '[aemComponent]'\n})\n\n/**\n * The current directive provides advanced capabilities among which are\n *\n * - The management of the component placeholder in the Page Editor\n * - The dynamic instantiation of components based on a component definition\n * - The conversion from model fields to properties and injection in the component instance\n * - The management of HTMLElement attributes and class names on the native element\n */\nexport class AEMComponentDirective implements AfterViewInit, OnInit, OnDestroy, OnChanges {\n\n  /**\n   * Dynamically created component\n   */\n  private _component: ComponentRef<any>;\n  /**\n   * Model item that corresponds to the current component\n   */\n  private _cqItem: object;\n\n  get cqItem(): object {\n    return this._cqItem;\n  }\n\n  @Input()\n  set cqItem(value: object) {\n    this._cqItem = value;\n  }\n\n  get changeDetectorRef(): ChangeDetectorRef {\n    return this._changeDetectorRef;\n  }\n\n  /**\n   * Path to the model structure associated with the current component\n   */\n  @Input() cqPath: string;\n  /**\n   * Name of the current instance of the component\n   */\n  @Input() itemName: string;\n  /**\n   * HtmlElement attributes for the current instance of the component\n   */\n  @Input() itemAttrs: object;\n\n  @Input() aemComponent;\n\n  constructor(\n    private renderer: Renderer2,\n    private viewContainer: ViewContainerRef,\n    private factoryResolver: ComponentFactoryResolver,\n    private _changeDetectorRef: ChangeDetectorRef) {\n  }\n\n  ngOnInit() {\n    this.renderComponent(ComponentMapping.get(this.type));\n  }\n\n  ngOnChanges(changes: import(\"@angular/core\").SimpleChanges): void {\n    this.updateComponentData();\n  }\n\n  /**\n   * Returns the type of the cqItem if exists.\n   */\n  get type() {\n    return this.cqItem && this.cqItem[Constants.TYPE_PROP];\n  }\n\n  /**\n   * Renders a component dynamically based on the component definition\n   *\n   * @param componentDefinition The component definition to render\n   */\n  private renderComponent(componentDefinition: any) {\n    if (componentDefinition) {\n      const factory = this.factoryResolver.resolveComponentFactory(componentDefinition);\n      this.viewContainer.clear();\n      this._component = this.viewContainer.createComponent(factory);\n      this.updateComponentData();\n    }\n  }\n\n  /**\n   * Updates the data of the component based the data of the directive\n   */\n  private updateComponentData() {\n    if (!this._component || !this._component.instance) {\n      return;\n    }\n\n    const keys = Object.getOwnPropertyNames(this.cqItem);\n\n    keys.forEach((key) => {\n      let propKey = key;\n\n      if (propKey.startsWith(':')) {\n        // Transformation of internal properties namespaced with [:] to [cq]\n        // :myProperty => cqMyProperty\n        const tempKey = propKey.substr(1);\n        propKey = 'cq' + tempKey.substr(0, 1).toUpperCase() + tempKey.substr(1);\n      }\n\n      this._component.instance[propKey] = this.cqItem[key];\n    });\n\n    this._component.instance.cqPath = this.cqPath;\n    this._component.instance.itemName = this.itemName;\n    const editConfig = ComponentMapping.getEditConfig(this.type);\n    if (editConfig && Utils.isInEditor) {\n      this.setupPlaceholder(editConfig);\n    }\n\n    this._changeDetectorRef.detectChanges();\n  }\n\n  /**\n   * Adds the specified item attributes to the element\n   */\n  private setupItemAttrs() {\n    if (this.itemAttrs) {\n      const keys = Object.getOwnPropertyNames(this.itemAttrs);\n\n      keys.forEach((key) => {\n        if (key === 'class') {\n          const classes = this.itemAttrs[key].split(' ');\n          classes.forEach((itemClass) => {\n            this.renderer.addClass(this._component.location.nativeElement, itemClass);\n          });\n        } else {\n          this.renderer.setAttribute(this._component.location.nativeElement, key, this.itemAttrs[key]);\n        }\n      });\n    }\n  }\n\n  /**\n   * Determines if the placeholder should e displayed.\n   *\n   * @param editConfig - the edit config of the directive\n   */\n  private usePlaceholder(editConfig) {\n    return editConfig.isEmpty && typeof editConfig.isEmpty === 'function' && editConfig.isEmpty(this.cqItem);\n  }\n\n  /**\n   * Setups the placeholder of needed for the AEM editor\n   *\n   * @param editConfig - the editConfig, which will dictate the classes to be added on.\n   */\n  private setupPlaceholder(editConfig) {\n    if (this.usePlaceholder(editConfig)) {\n      this.renderer.addClass(this._component.location.nativeElement, PLACEHOLDER_CLASS_NAME);\n      this.renderer.setAttribute(this._component.location.nativeElement, 'data-emptytext', editConfig.emptyLabel);\n    } else {\n      this.renderer.removeClass(this._component.location.nativeElement, PLACEHOLDER_CLASS_NAME);\n      this.renderer.removeAttribute(this._component.location.nativeElement, 'data-emptytext');\n    }\n  }\n\n  ngAfterViewInit() {\n    this.setupItemAttrs();\n  }\n\n  ngOnDestroy() {\n    this._component && this._component.destroy();\n  }\n\n}\n"]}