UNPKG

ng2-ckeditor

Version:
251 lines 27 kB
// Imports import { Component, Input, Output, ViewChild, EventEmitter, forwardRef, ContentChildren } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { CKButtonDirective } from './ckbutton.directive'; import { CKGroupDirective } from './ckgroup.directive'; import * as i0 from "@angular/core"; /** * CKEditor component * Usage : * <ckeditor [(ngModel)]="data" [config]="{...}" debounce="500"></ckeditor> */ export class CKEditorComponent { /** * Constructor */ constructor(zone) { this.zone = zone; this.change = new EventEmitter(); this.editorChange = new EventEmitter(); this.ready = new EventEmitter(); this.blur = new EventEmitter(); this.focus = new EventEmitter(); this.contentDom = new EventEmitter(); this.fileUploadRequest = new EventEmitter(); this.fileUploadResponse = new EventEmitter(); this.paste = new EventEmitter(); this.drop = new EventEmitter(); this._value = ''; this.destroyed = false; } get value() { return this._value; } set value(v) { if (v !== this._value) { this._value = v; this.onChange(v); } } ngOnChanges(changes) { if (changes.readonly && this.instance) { this.instance.setReadOnly(changes.readonly.currentValue); } } /** * On component destroy */ ngOnDestroy() { this.destroyed = true; this.zone.runOutsideAngular(() => { if (this.instance) { CKEDITOR.removeAllListeners(); this.instance.destroy(); this.instance = null; } }); } /** * On component view init */ ngAfterViewInit() { if (this.destroyed) { return; } this.ckeditorInit(this.config || {}); } /** * On component view checked */ ngAfterViewChecked() { this.ckeditorInit(this.config || {}); } /** * Value update process */ updateValue(value) { this.zone.run(() => { this.value = value; this.onChange(value); this.onTouched(); this.change.emit(value); }); } /** * CKEditor init */ ckeditorInit(config) { if (typeof CKEDITOR === 'undefined') { console.warn('CKEditor 4.x is missing (http://ckeditor.com/)'); } else { // Check textarea exists if (this.instance || !this.documentContains(this.host.nativeElement)) { return; } if (this.readonly) { config.readOnly = this.readonly; } // CKEditor replace textarea this.instance = CKEDITOR.replace(this.host.nativeElement, config); // Set initial value this.instance.setData(this.value); // listen for instanceReady event this.instance.on('instanceReady', (evt) => { // if value has changed while instance loading // update instance with current component value if (this.instance.getData() !== this.value) { this.instance.setData(this.value); } // send the evt to the EventEmitter this.ready.emit(evt); }); // CKEditor change event this.instance.on('change', (evt) => { this.onTouched(); const value = this.instance.getData(); if (this.value !== value) { // Debounce update if (this.debounce) { if (this.debounceTimeout) { clearTimeout(this.debounceTimeout); } this.debounceTimeout = window.setTimeout(() => { this.updateValue(value); this.debounceTimeout = null; }, parseInt(this.debounce)); // Live update } else { this.updateValue(value); } } // Original ckeditor event dispatch this.editorChange.emit(evt); }); // CKEditor blur event this.instance.on('blur', (evt) => { this.blur.emit(evt); }); // CKEditor focus event this.instance.on('focus', (evt) => { this.focus.emit(evt); }); // CKEditor contentDom event this.instance.on('contentDom', (evt) => { this.contentDom.emit(evt); }); // CKEditor fileUploadRequest event this.instance.on('fileUploadRequest', (evt) => { this.fileUploadRequest.emit(evt); }); // CKEditor fileUploadResponse event this.instance.on('fileUploadResponse', (evt) => { this.fileUploadResponse.emit(evt); }); // CKEditor paste event this.instance.on('paste', (evt) => { this.paste.emit(evt); }); // CKEditor drop event this.instance.on('drop', (evt) => { this.drop.emit(evt); }); // Add Toolbar Groups to Editor. This will also add Buttons within groups. this.toolbarGroups.forEach((group) => { group.initialize(this); }); // Add Toolbar Buttons to Editor. this.toolbarButtons.forEach((button) => { button.initialize(this); }); } } /** * Implements ControlValueAccessor */ writeValue(value) { this._value = value; if (this.instance) this.instance.setData(value); } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } documentContains(node) { return document.contains ? document.contains(node) : document.body.contains(node); } } CKEditorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.2", ngImport: i0, type: CKEditorComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); CKEditorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.2", type: CKEditorComponent, selector: "ckeditor", inputs: { config: "config", readonly: "readonly", debounce: "debounce", value: "value" }, outputs: { change: "change", editorChange: "editorChange", ready: "ready", blur: "blur", focus: "focus", contentDom: "contentDom", fileUploadRequest: "fileUploadRequest", fileUploadResponse: "fileUploadResponse", paste: "paste", drop: "drop" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => CKEditorComponent), multi: true, }, ], queries: [{ propertyName: "toolbarButtons", predicate: CKButtonDirective }, { propertyName: "toolbarGroups", predicate: CKGroupDirective }], viewQueries: [{ propertyName: "host", first: true, predicate: ["host"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `<textarea #host></textarea>`, isInline: true }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.2", ngImport: i0, type: CKEditorComponent, decorators: [{ type: Component, args: [{ selector: 'ckeditor', providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => CKEditorComponent), multi: true, }, ], template: `<textarea #host></textarea>`, }] }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { config: [{ type: Input }], readonly: [{ type: Input }], debounce: [{ type: Input }], change: [{ type: Output }], editorChange: [{ type: Output }], ready: [{ type: Output }], blur: [{ type: Output }], focus: [{ type: Output }], contentDom: [{ type: Output }], fileUploadRequest: [{ type: Output }], fileUploadResponse: [{ type: Output }], paste: [{ type: Output }], drop: [{ type: Output }], host: [{ type: ViewChild, args: ['host', { static: false }] }], toolbarButtons: [{ type: ContentChildren, args: [CKButtonDirective] }], toolbarGroups: [{ type: ContentChildren, args: [CKGroupDirective] }], value: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ckeditor.component.js","sourceRoot":"","sources":["../../src/ckeditor.component.ts"],"names":[],"mappings":"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","sourcesContent":["// Imports\nimport {\n  Component,\n  Input,\n  Output,\n  ViewChild,\n  EventEmitter,\n  NgZone,\n  forwardRef,\n  QueryList,\n  AfterViewInit,\n  ContentChildren,\n  SimpleChanges,\n  OnChanges,\n  OnDestroy,\n  ElementRef\n} from '@angular/core';\nimport { NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { CKButtonDirective } from './ckbutton.directive';\nimport { CKGroupDirective } from './ckgroup.directive';\n\n/**\n * CKEditor component\n * Usage :\n *  <ckeditor [(ngModel)]=\"data\" [config]=\"{...}\" debounce=\"500\"></ckeditor>\n */\n@Component({\n  selector: 'ckeditor',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CKEditorComponent),\n      multi: true,\n    },\n  ],\n  template: `<textarea #host></textarea>`,\n})\nexport class CKEditorComponent implements OnChanges, AfterViewInit, OnDestroy {\n  @Input() config: CKEDITOR.config;\n  @Input() readonly: boolean;\n  @Input() debounce: string;\n\n  @Output() change = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() editorChange = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() ready = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() blur = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() focus = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() contentDom = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() fileUploadRequest = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() fileUploadResponse = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() paste = new EventEmitter<CKEDITOR.eventInfo>();\n  @Output() drop = new EventEmitter<CKEDITOR.eventInfo>();\n\n  @ViewChild('host', { static: false }) host: ElementRef<HTMLTextAreaElement>;\n\n  @ContentChildren(CKButtonDirective) toolbarButtons: QueryList<CKButtonDirective>;\n  @ContentChildren(CKGroupDirective) toolbarGroups: QueryList<CKGroupDirective>;\n\n  _value = '';\n  instance: CKEDITOR.editor;\n  debounceTimeout: number;\n  private destroyed = false;\n\n  /**\n   * Constructor\n   */\n  constructor(private zone: NgZone) {}\n\n  get value(): string {\n    return this._value;\n  }\n\n  @Input()\n  set value(v: string) {\n    if (v !== this._value) {\n      this._value = v;\n      this.onChange(v);\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.readonly && this.instance) {\n      this.instance.setReadOnly(changes.readonly.currentValue);\n    }\n  }\n\n  /**\n   * On component destroy\n   */\n  ngOnDestroy(): void {\n    this.destroyed = true;\n    this.zone.runOutsideAngular(() => {\n      if (this.instance) {\n        CKEDITOR.removeAllListeners();\n        this.instance.destroy();\n        this.instance = null;\n      }\n    });\n  }\n\n  /**\n   * On component view init\n   */\n  ngAfterViewInit(): void {\n    if (this.destroyed) {\n      return;\n    }\n    this.ckeditorInit(this.config || {});\n  }\n\n  /**\n   * On component view checked\n   */\n  ngAfterViewChecked(): void {\n    this.ckeditorInit(this.config || {});\n  }\n\n  /**\n   * Value update process\n   */\n  updateValue(value: any): void {\n    this.zone.run(() => {\n      this.value = value;\n\n      this.onChange(value);\n\n      this.onTouched();\n      this.change.emit(value);\n    });\n  }\n\n  /**\n   * CKEditor init\n   */\n  ckeditorInit(config: CKEDITOR.config): void {\n    if (typeof CKEDITOR === 'undefined') {\n      console.warn('CKEditor 4.x is missing (http://ckeditor.com/)');\n    } else {\n      // Check textarea exists\n      if (this.instance || !this.documentContains(this.host.nativeElement)) {\n        return;\n      }\n\n      if (this.readonly) {\n        config.readOnly = this.readonly;\n      }\n      // CKEditor replace textarea\n      this.instance = CKEDITOR.replace(this.host.nativeElement, config);\n\n      // Set initial value\n      this.instance.setData(this.value);\n\n      // listen for instanceReady event\n      this.instance.on('instanceReady', (evt: CKEDITOR.eventInfo) => {\n        // if value has changed while instance loading\n        // update instance with current component value\n        if (this.instance.getData() !== this.value) {\n          this.instance.setData(this.value);\n        }\n\n        // send the evt to the EventEmitter\n        this.ready.emit(evt);\n      });\n\n      // CKEditor change event\n      this.instance.on('change', (evt: CKEDITOR.eventInfo) => {\n        this.onTouched();\n        const value = this.instance.getData();\n\n        if (this.value !== value) {\n          // Debounce update\n          if (this.debounce) {\n            if (this.debounceTimeout) {\n              clearTimeout(this.debounceTimeout);\n            }\n            this.debounceTimeout = window.setTimeout(() => {\n              this.updateValue(value);\n              this.debounceTimeout = null;\n            }, parseInt(this.debounce));\n\n            // Live update\n          } else {\n            this.updateValue(value);\n          }\n        }\n\n        // Original ckeditor event dispatch\n        this.editorChange.emit(evt);\n      });\n\n      // CKEditor blur event\n      this.instance.on('blur', (evt: CKEDITOR.eventInfo) => {\n        this.blur.emit(evt);\n      });\n\n      // CKEditor focus event\n      this.instance.on('focus', (evt: CKEDITOR.eventInfo) => {\n        this.focus.emit(evt);\n      });\n\n      // CKEditor contentDom event\n      this.instance.on('contentDom', (evt: CKEDITOR.eventInfo) => {\n        this.contentDom.emit(evt);\n      });\n\n      // CKEditor fileUploadRequest event\n      this.instance.on('fileUploadRequest', (evt: CKEDITOR.eventInfo) => {\n        this.fileUploadRequest.emit(evt);\n      });\n\n      // CKEditor fileUploadResponse event\n      this.instance.on('fileUploadResponse', (evt: CKEDITOR.eventInfo) => {\n        this.fileUploadResponse.emit(evt);\n      });\n\n      // CKEditor paste event\n      this.instance.on('paste', (evt: CKEDITOR.eventInfo) => {\n        this.paste.emit(evt);\n      });\n\n      // CKEditor drop event\n      this.instance.on('drop', (evt: CKEDITOR.eventInfo) => {\n        this.drop.emit(evt);\n      });\n\n      // Add Toolbar Groups to Editor. This will also add Buttons within groups.\n      this.toolbarGroups.forEach((group) => {\n        group.initialize(this);\n      });\n      // Add Toolbar Buttons to Editor.\n      this.toolbarButtons.forEach((button) => {\n        button.initialize(this);\n      });\n    }\n  }\n\n  /**\n   * Implements ControlValueAccessor\n   */\n  writeValue(value: string): void {\n    this._value = value;\n    if (this.instance) this.instance.setData(value);\n  }\n\n  onChange: (_: string) => void;\n\n  onTouched: () => void;\n\n  registerOnChange(fn: () => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  private documentContains(node: Node) {\n    return document.contains ? document.contains(node) : document.body.contains(node);\n  }\n}\n"]}