UNPKG

ngx-jodit

Version:

Angular wrapper for jodit WYSIWYG editor

224 lines 28.6 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, Input, Output, ViewChild, } from '@angular/core'; import { FormsModule, NG_VALUE_ACCESSOR, } from '@angular/forms'; import { Jodit } from 'jodit'; import { BehaviorSubject, combineLatest, delay, distinctUntilChanged, filter, withLatestFrom, } from 'rxjs'; import * as i0 from "@angular/core"; export class NgxJoditComponent { set options(value) { this._options = value; if (value) { this.initJoditContainer(); } } set value(value) { const sanitizedText = this.prepareText(value); if (!this.internValueChange) { this.valueSubject.next(sanitizedText); } else { this.internValueChange = false; } this.onChange(sanitizedText); } get value() { return this.valueSubject.getValue(); } constructor(cdr) { this.cdr = cdr; // value property (subject) this.valueSubject = new BehaviorSubject(''); this.valueChange = new EventEmitter(); //events this.joditChange = new EventEmitter(); this.joditKeyDown = new EventEmitter(); this.joditKeyUp = new EventEmitter(); this.joditMousedown = new EventEmitter(); this.joditMouseup = new EventEmitter(); this.joditClick = new EventEmitter(); this.joditFocus = new EventEmitter(); this.joditPaste = new EventEmitter(); this.joditResize = new EventEmitter(); this.joditBeforeEnter = new EventEmitter(); this.joditBeforeCommand = new EventEmitter(); this.joditAfterExec = new EventEmitter(); this.joditAfterPaste = new EventEmitter(); this.joditChangeSelection = new EventEmitter(); // Used for delay value assignment to wait for jodit to be initialized this.joditInitializedSubject = new BehaviorSubject(false); this.internValueChange = false; /* FUNCTIONS RELEVANT FOR ANGULAR FORMS */ this.onChange = (text) => { // implemented by user }; this.onTouched = () => { // implemented by user }; this.valueSubscription = combineLatest([ // Handle value changes ... this.valueSubject.asObservable().pipe(distinctUntilChanged()), // ...additionally ensuring that the value is reapplied if the editor was not initialized when value was set this.joditInitializedSubject.pipe(distinctUntilChanged(), filter((initialized) => initialized)), ]) .pipe( // Pass through the latest value in case of editor initialization withLatestFrom(this.valueSubject), // Prevent ExpressionChangedAfterItHasBeenCheckedError delay(0)) .subscribe(([[_, initialized], text]) => { if (this.jodit && initialized) { this.jodit.value = text; } }); } isHTML(text) { const elem = document.createElement('div'); elem.innerHTML = text; return (text && elem.childNodes.length > 0 && elem.childNodes.item(0).nodeType !== 3); } ngAfterViewInit() { this.initJoditContainer(); } ngOnDestroy() { this.valueSubscription?.unsubscribe(); this.jodit?.destruct(); } initJoditContainer() { if (this.joditContainer) { if (this.jodit) { this.jodit.destruct(); this.joditInitializedSubject.next(false); } this.jodit = Jodit.make(this.joditContainer.nativeElement, this._options); this.jodit.value = this.valueSubject.getValue(); this.jodit.events.on('change', (text) => { this.internValueChange = true; this.changeValue(text); this.joditChange.emit(text); this.onChange(text); }); this.jodit.events.on('keydown', (a) => { this.joditKeyDown.emit(a); }); this.jodit.events.on('keyup', (a) => { this.joditKeyUp.emit(a); }); this.jodit.events.on('mousedown', (a) => { this.joditMousedown.emit(a); }); this.jodit.events.on('mouseup', (a) => { this.joditMouseup.emit(a); }); this.jodit.events.on('click', (a) => { this.joditClick.emit(a); this.onTouched(); }); this.jodit.events.on('focus', (a) => { this.joditFocus.emit(a); }); this.jodit.events.on('paste', (a) => { this.joditPaste.emit(a); }); this.jodit.events.on('resize', () => { this.joditResize.emit(); }); this.jodit.events.on('beforeEnter', (a) => { this.joditBeforeEnter.emit(a); }); this.jodit.events.on('beforeCommand', (a) => { this.joditBeforeCommand.emit(a); }); this.jodit.events.on('afterExec', () => { this.joditAfterExec.emit(); }); this.jodit.events.on('afterPaste', (a) => { this.joditAfterPaste.emit(a); }); this.jodit.events.on('changeSelection', () => { this.joditChangeSelection.emit(); }); this.joditInitializedSubject.next(true); } } changeValue(value) { this.valueChange.emit(value); } writeValue(text) { this.valueSubject.next(this.prepareText(text)); } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.options = { ...this._options, disabled: isDisabled, }; } prepareText(text) { return this.isHTML(text) ? text : `<p>${text}</p>`; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NgxJoditComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: NgxJoditComponent, isStandalone: true, selector: "ngx-jodit", inputs: { options: "options", value: "value" }, outputs: { valueChange: "valueChange", joditChange: "joditChange", joditKeyDown: "joditKeyDown", joditKeyUp: "joditKeyUp", joditMousedown: "joditMousedown", joditMouseup: "joditMouseup", joditClick: "joditClick", joditFocus: "joditFocus", joditPaste: "joditPaste", joditResize: "joditResize", joditBeforeEnter: "joditBeforeEnter", joditBeforeCommand: "joditBeforeCommand", joditAfterExec: "joditAfterExec", joditAfterPaste: "joditAfterPaste", joditChangeSelection: "joditChangeSelection" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxJoditComponent), multi: true, }, ], viewQueries: [{ propertyName: "joditContainer", first: true, predicate: ["joditContainer"], descendants: true }], ngImport: i0, template: "<textarea class=\"ngx-jodit-container\" #joditContainer></textarea>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NgxJoditComponent, decorators: [{ type: Component, args: [{ selector: 'ngx-jodit', standalone: true, imports: [CommonModule, FormsModule], providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxJoditComponent), multi: true, }, ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<textarea class=\"ngx-jodit-container\" #joditContainer></textarea>\n" }] }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { joditContainer: [{ type: ViewChild, args: ['joditContainer'] }], options: [{ type: Input }], value: [{ type: Input }], valueChange: [{ type: Output }], joditChange: [{ type: Output }], joditKeyDown: [{ type: Output }], joditKeyUp: [{ type: Output }], joditMousedown: [{ type: Output }], joditMouseup: [{ type: Output }], joditClick: [{ type: Output }], joditFocus: [{ type: Output }], joditPaste: [{ type: Output }], joditResize: [{ type: Output }], joditBeforeEnter: [{ type: Output }], joditBeforeCommand: [{ type: Output }], joditAfterExec: [{ type: Output }], joditAfterPaste: [{ type: Output }], joditChangeSelection: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-jodit.component.js","sourceRoot":"","sources":["../../../../../libs/ngx-jodit/src/lib/ngx-jodit.component.ts","../../../../../libs/ngx-jodit/src/lib/ngx-jodit.component.html"],"names":[],"mappings":"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","sourcesContent":["import {CommonModule} from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnDestroy,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR,} from '@angular/forms';\nimport {Jodit} from 'jodit';\nimport {BehaviorSubject, combineLatest, delay, distinctUntilChanged, filter, Subscription, withLatestFrom,} from 'rxjs';\n\nimport {JoditConfig} from './types';\n\n@Component({\n  selector: 'ngx-jodit',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  templateUrl: './ngx-jodit.component.html',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => NgxJoditComponent),\n      multi: true,\n    },\n  ],\n  styleUrls: ['./ngx-jodit.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class NgxJoditComponent\n  implements ControlValueAccessor, AfterViewInit, OnDestroy {\n  @ViewChild('joditContainer') joditContainer!: ElementRef;\n  jodit?: Jodit;\n\n  /**\n   * options for jodit.\n   * You can add more supported options even Typescript doesn't suggest the options.\n   */\n  private _options?: JoditConfig;\n\n  @Input() set options(value: JoditConfig) {\n    this._options = value;\n\n    if (value) {\n      this.initJoditContainer();\n    }\n  }\n\n  // value property (subject)\n  private valueSubject: BehaviorSubject<string> = new BehaviorSubject<string>(\n    ''\n  );\n\n  @Input() set value(value: string) {\n    const sanitizedText = this.prepareText(value);\n    if (!this.internValueChange) {\n      this.valueSubject.next(sanitizedText);\n    } else {\n      this.internValueChange = false;\n    }\n    this.onChange(sanitizedText);\n  }\n\n  get value(): string {\n    return this.valueSubject.getValue();\n  }\n\n  @Output() valueChange = new EventEmitter<string>();\n\n  //events\n  @Output() joditChange = new EventEmitter<string>();\n  @Output() joditKeyDown = new EventEmitter<KeyboardEvent>();\n  @Output() joditKeyUp = new EventEmitter<KeyboardEvent>();\n  @Output() joditMousedown = new EventEmitter<MouseEvent>();\n  @Output() joditMouseup = new EventEmitter<MouseEvent>();\n  @Output() joditClick = new EventEmitter<PointerEvent>();\n  @Output() joditFocus = new EventEmitter<FocusEvent>();\n  @Output() joditPaste = new EventEmitter<ClipboardEvent>();\n  @Output() joditResize = new EventEmitter<void>();\n  @Output() joditBeforeEnter = new EventEmitter<KeyboardEvent>();\n  @Output() joditBeforeCommand = new EventEmitter<string>();\n  @Output() joditAfterExec = new EventEmitter<void>();\n  @Output() joditAfterPaste = new EventEmitter<ClipboardEvent>();\n  @Output() joditChangeSelection = new EventEmitter<void>();\n\n  // Used for delay value assignment to wait for jodit to be initialized\n  private joditInitializedSubject: BehaviorSubject<boolean> =\n    new BehaviorSubject(false);\n  private valueSubscription?: Subscription;\n  private internValueChange = false;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {\n    this.valueSubscription = combineLatest([\n      // Handle value changes ...\n      this.valueSubject.asObservable().pipe(distinctUntilChanged()),\n      // ...additionally ensuring that the value is reapplied if the editor was not initialized when value was set\n      this.joditInitializedSubject.pipe(\n        distinctUntilChanged(),\n        filter((initialized) => initialized)\n      ),\n    ])\n      .pipe(\n        // Pass through the latest value in case of editor initialization\n        withLatestFrom(this.valueSubject),\n        // Prevent ExpressionChangedAfterItHasBeenCheckedError\n        delay(0)\n      )\n      .subscribe(([[_, initialized], text]) => {\n        if (this.jodit && initialized) {\n          this.jodit.value = text;\n        }\n      });\n  }\n\n  isHTML(text: string) {\n    const elem = document.createElement('div');\n    elem.innerHTML = text;\n\n    return (\n      text &&\n      elem.childNodes.length > 0 &&\n      elem.childNodes.item(0).nodeType !== 3\n    );\n  }\n\n  ngAfterViewInit() {\n    this.initJoditContainer();\n  }\n\n  ngOnDestroy() {\n    this.valueSubscription?.unsubscribe();\n    this.jodit?.destruct();\n  }\n\n  initJoditContainer() {\n    if (this.joditContainer) {\n      if (this.jodit) {\n        this.jodit.destruct();\n        this.joditInitializedSubject.next(false);\n      }\n      this.jodit = Jodit.make(\n        this.joditContainer.nativeElement,\n        this._options as any\n      );\n      this.jodit.value = this.valueSubject.getValue();\n      this.jodit.events.on('change', (text: string) => {\n        this.internValueChange = true;\n        this.changeValue(text);\n        this.joditChange.emit(text);\n        this.onChange(text);\n      });\n      this.jodit.events.on('keydown', (a: KeyboardEvent) => {\n        this.joditKeyDown.emit(a);\n      });\n      this.jodit.events.on('keyup', (a: KeyboardEvent) => {\n        this.joditKeyUp.emit(a);\n      });\n      this.jodit.events.on('mousedown', (a: MouseEvent) => {\n        this.joditMousedown.emit(a);\n      });\n      this.jodit.events.on('mouseup', (a: MouseEvent) => {\n        this.joditMouseup.emit(a);\n      });\n      this.jodit.events.on('click', (a: PointerEvent) => {\n        this.joditClick.emit(a);\n        this.onTouched();\n      });\n      this.jodit.events.on('focus', (a: FocusEvent) => {\n        this.joditFocus.emit(a);\n      });\n      this.jodit.events.on('paste', (a: ClipboardEvent) => {\n        this.joditPaste.emit(a);\n      });\n      this.jodit.events.on('resize', () => {\n        this.joditResize.emit();\n      });\n      this.jodit.events.on('beforeEnter', (a: KeyboardEvent) => {\n        this.joditBeforeEnter.emit(a);\n      });\n      this.jodit.events.on('beforeCommand', (a: string) => {\n        this.joditBeforeCommand.emit(a);\n      });\n      this.jodit.events.on('afterExec', () => {\n        this.joditAfterExec.emit();\n      });\n      this.jodit.events.on('afterPaste', (a: ClipboardEvent) => {\n        this.joditAfterPaste.emit(a);\n      });\n      this.jodit.events.on('changeSelection', () => {\n        this.joditChangeSelection.emit();\n      });\n\n      this.joditInitializedSubject.next(true);\n    }\n  }\n\n  changeValue(value: string) {\n    this.valueChange.emit(value);\n  }\n\n  /*\n  FUNCTIONS RELEVANT FOR ANGULAR FORMS\n   */\n\n  onChange = (text: string) => {\n    // implemented by user\n  };\n\n  onTouched = () => {\n    // implemented by user\n  };\n\n  writeValue(text: string): void {\n    this.valueSubject.next(this.prepareText(text));\n  }\n\n  registerOnChange(fn: (text: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState?(isDisabled: boolean): void {\n    this.options = {\n      ...this._options,\n      disabled: isDisabled,\n    };\n  }\n\n  private prepareText(text: string) {\n    return this.isHTML(text) ? text : `<p>${text}</p>`;\n  }\n}\n","<textarea class=\"ngx-jodit-container\" #joditContainer></textarea>\n"]}