@adobe/cq-angular-editable-components
Version:
* [API](#api) * [Documentation](#documentation) * [Changelog](#changelog)
194 lines • 21.4 kB
JavaScript
/*
* 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"]}