ngx-jodit
Version:
Angular wrapper for jodit WYSIWYG editor
224 lines • 28.6 kB
JavaScript
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"]}