@festo-ui/angular 5.0.0-dev.156 → 5.0.0-dev.161

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -15,13 +15,11 @@ export class FngTextEditorComponent extends FngValueAccessorBaseDirective {
15
15
  set value(value) {
16
16
  if (this.innerValue !== value) {
17
17
  this.innerValue = value;
18
- if (value) {
19
- this.changed.forEach(f => {
20
- this.fngChange.emit(this.innerValue);
21
- this.change.emit(this.innerValue);
22
- return f(value);
23
- });
24
- }
18
+ this.changed.forEach(f => {
19
+ this.fngChange.emit(this.innerValue);
20
+ this.change.emit(this.innerValue);
21
+ return f(value);
22
+ });
25
23
  }
26
24
  }
27
25
  get value() {
@@ -48,7 +46,8 @@ export class FngTextEditorComponent extends FngValueAccessorBaseDirective {
48
46
  sanitizingFn: undefined,
49
47
  skipDefaultSanitizing: false,
50
48
  skipWhitespaceHandling: false
51
- }
49
+ },
50
+ focusCursorToEndOnWrite: true
52
51
  };
53
52
  this.fngChange = new EventEmitter();
54
53
  this.change = new EventEmitter();
@@ -91,11 +90,9 @@ export class FngTextEditorComponent extends FngValueAccessorBaseDirective {
91
90
  this.setEditorLabel();
92
91
  this.addWhitespaceMatcher();
93
92
  this.addImageMatcher();
94
- if (this.innerValue) {
95
- const content = this.valueSetter(this.quillEditor, this.innerValue);
96
- this.quillEditor.setContents(content, 'silent');
97
- this.quillEditor.getModule('history').clear();
98
- }
93
+ const content = this.valueSetter(this.quillEditor, this.innerValue || '');
94
+ this.quillEditor.setContents(content, 'silent');
95
+ this.quillEditor.getModule('history').clear();
99
96
  this.handle();
100
97
  this.quillEditor.on('text-change', () => {
101
98
  this.value = this.valueGetter(this.editorElem);
@@ -114,15 +111,16 @@ export class FngTextEditorComponent extends FngValueAccessorBaseDirective {
114
111
  this.handle();
115
112
  }
116
113
  writeValue(value) {
117
- if (value === null || value === '') {
118
- return;
119
- }
120
114
  this.innerValue = value;
121
115
  if (!this.quillEditor) {
122
116
  return;
123
117
  }
124
- const content = this.valueSetter(this.quillEditor, value);
118
+ const content = this.valueSetter(this.quillEditor, value || '');
125
119
  this.quillEditor.setContents(content);
120
+ if (this.config?.focusCursorToEndOnWrite) {
121
+ // when new content gets written we always set cursor focus to end of text
122
+ this.quillEditor.setSelection(Number.MAX_SAFE_INTEGER, 0);
123
+ }
126
124
  }
127
125
  setDisabledState(isDisabled = this.disabled) {
128
126
  this.disabled = isDisabled;
@@ -307,4 +305,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.1.1", ngImpor
307
305
  }], value: [{
308
306
  type: Input
309
307
  }] } });
310
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"text-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/angular/src/lib/forms/text-editor/text-editor.component.ts","../../../../../../projects/angular/src/lib/forms/text-editor/text-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ViewEncapsulation,\n  Input,\n  AfterViewInit,\n  SecurityContext,\n  ElementRef,\n  Renderer2,\n  EventEmitter,\n  Output,\n  forwardRef,\n  OnChanges,\n  ViewChild,\n  SimpleChanges,\n  OnInit\n} from '@angular/core';\nimport QuillType from 'quill';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\n\nimport { FngValueAccessorBaseDirective } from '../value-accessor-base';\nimport { CommonModule } from '@angular/common';\nimport { DragDropModule } from '@angular/cdk/drag-drop';\nimport { ObserversModule } from '@angular/cdk/observers';\nimport { OverlayModule } from '@angular/cdk/overlay';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\n\nexport interface FngTextEditorConfiguration {\n  toolbar?: {\n    bold?: boolean;\n    italic?: boolean;\n    underline?: boolean;\n    alignCenter?: boolean;\n    alignRight?: boolean;\n    bulletList?: boolean;\n    orderedList?: boolean;\n    image?: boolean;\n    link?: boolean;\n  };\n  sanitizing?: {\n    skipDefaultSanitizing?: boolean;\n    skipWhitespaceHandling?: boolean;\n    sanitizingFn?: (ds: DomSanitizer, value: string) => string;\n  };\n}\n\n@Component({\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, FormsModule, CommonModule, OverlayModule, ScrollingModule, DragDropModule, ObserversModule],\n  selector: 'fng-text-editor',\n  templateUrl: './text-editor.component.html',\n  styleUrls: ['./text-editor.component.scss'],\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FngTextEditorComponent),\n      multi: true\n    }\n  ]\n})\nexport class FngTextEditorComponent extends FngValueAccessorBaseDirective<string> implements OnInit, AfterViewInit, OnChanges {\n  id: string | undefined;\n  quillEditor!: QuillType;\n  editorElem!: HTMLElement;\n  defaultConfig: FngTextEditorConfiguration = {\n    toolbar: {\n      bold: true,\n      italic: true,\n      underline: true,\n      alignCenter: true,\n      alignRight: true,\n      bulletList: true,\n      orderedList: true,\n      image: true,\n      link: true\n    },\n    sanitizing: {\n      sanitizingFn: undefined,\n      skipDefaultSanitizing: false,\n      skipWhitespaceHandling: false\n    }\n  };\n\n  @ViewChild('btnBold') btnBold: ElementRef | undefined;\n  @ViewChild('btnItalic') btnItalic: ElementRef | undefined;\n  @ViewChild('btnUnderline') btnUnderline: ElementRef | undefined;\n  @ViewChild('btnAlignCenter') btnAlignCenter: ElementRef | undefined;\n  @ViewChild('btnAlignRight') btnAlignRight: ElementRef | undefined;\n  @ViewChild('btnLink') btnLink: ElementRef | undefined;\n  @ViewChild('btnUl') btnUl: ElementRef | undefined;\n  @ViewChild('btnOl') btnOl: ElementRef | undefined;\n  @ViewChild('btnImage') btnImage: ElementRef | undefined;\n\n  @Output() fngChange = new EventEmitter<any>();\n  @Output() change = new EventEmitter<any>();\n\n  @Input() config: FngTextEditorConfiguration = this.defaultConfig;\n  @Input() modules: any;\n  @Input() label: string | undefined;\n  @Input() readOnly: boolean = false;\n  @Input() maxLength: number = 0;\n  @Input() error: string | undefined;\n  @Input() hint: string | undefined;\n  @Input()\n  override set value(value: string | undefined) {\n    if (this.innerValue !== value) {\n      this.innerValue = value;\n      if (value) {\n        this.changed.forEach(f => {\n          this.fngChange.emit(this.innerValue);\n          this.change.emit(this.innerValue);\n          return f(value);\n        });\n      }\n    }\n  }\n\n  override get value() {\n    return this.innerValue;\n  }\n\n  // needed to access editor content\n  valueGetter = (editorElement: HTMLElement): string | any => {\n    let html: string | undefined = editorElement?.querySelector('.ql-editor')?.innerHTML;\n    if (html === '<p><br></p>' || html === '<div><br></div>') {\n      html = undefined;\n    }\n    return html;\n  };\n\n  // needed to access editor content\n  valueSetter = (quillEditor: QuillType, value: any): any => {\n    const sanitizeConfig = this.config?.sanitizing || this.defaultConfig.sanitizing;\n\n    if (sanitizeConfig?.sanitizingFn) {\n      value = sanitizeConfig.sanitizingFn(this.domSanitizer, value);\n    }\n    if (sanitizeConfig && !sanitizeConfig.skipDefaultSanitizing) {\n      value = this.domSanitizer.sanitize(SecurityContext.HTML, value);\n    }\n    if (sanitizeConfig && !sanitizeConfig.skipWhitespaceHandling) {\n      value = this.handleWhitespace(value);\n    }\n\n    return quillEditor.clipboard.convert(value);\n  };\n\n  constructor(public elementRef: ElementRef, private renderer: Renderer2, private domSanitizer: DomSanitizer) {\n    super();\n  }\n\n  ngOnInit(): void {\n    this.id = (Math.random() * Date.now()).toString().replace('.', '-');\n  }\n\n  ngAfterViewInit(): void {\n    this.editorElem = this.elementRef.nativeElement.querySelector(`#editor-${this.id}`);\n\n    this.quillEditor = new QuillType(this.editorElem, {\n      modules: { toolbar: `#editor-toolbar-${this.id}`, ...this.modules },\n      theme: 'snow',\n      scrollingContainer: `#editor-container-${this.id}`\n    });\n    this.setEditorLabel();\n    this.addWhitespaceMatcher();\n    this.addImageMatcher();\n\n    if (this.innerValue) {\n      const content = this.valueSetter(this.quillEditor, this.innerValue);\n      this.quillEditor.setContents(content, 'silent');\n      this.quillEditor.getModule('history').clear();\n    }\n\n    this.handle();\n\n    this.quillEditor.on('text-change', () => {\n      this.value = this.valueGetter(this.editorElem);\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes?.['config']?.currentValue != null) {\n      this.config = {\n        toolbar: { ...this.defaultConfig.toolbar, ...changes['config'].currentValue?.toolbar },\n        sanitizing: { ...this.defaultConfig.sanitizing, ...changes['config'].currentValue?.sanitizing }\n      };\n    }\n\n    if (!this.quillEditor) {\n      return;\n    }\n\n    this.handle();\n  }\n\n  override writeValue(value: string) {\n    if (value === null || value === '') {\n      return;\n    }\n\n    this.innerValue = value;\n\n    if (!this.quillEditor) {\n      return;\n    }\n\n    const content = this.valueSetter(this.quillEditor, value);\n    this.quillEditor.setContents(content);\n  }\n\n  override setDisabledState(isDisabled: boolean = this.disabled): void {\n    this.disabled = isDisabled;\n    if (this.quillEditor) {\n      if (isDisabled) {\n        this.quillEditor.disable();\n        this.renderer.setAttribute(this.elementRef.nativeElement, 'disabled', 'disabled');\n      } else {\n        if (!this.readOnly) {\n          this.quillEditor.enable();\n        }\n        this.renderer.removeAttribute(this.elementRef.nativeElement, 'disabled');\n      }\n    }\n  }\n\n  setReadOnlyState() {\n    this.quillEditor.enable(!this.readOnly);\n  }\n\n  delegate(event: any, action: string) {\n    let btn: HTMLElement | undefined = undefined;\n    switch (action) {\n      case 'bold':\n        btn = this.btnBold?.nativeElement;\n        btn?.click();\n        break;\n      case 'italic':\n        btn = this.btnItalic?.nativeElement;\n        btn?.click();\n        break;\n      case 'underline':\n        btn = this.btnUnderline?.nativeElement;\n        btn?.click();\n        break;\n      case 'align-center':\n        btn = this.btnAlignCenter?.nativeElement;\n        btn?.click();\n        break;\n      case 'align-right':\n        btn = this.btnAlignRight?.nativeElement;\n        btn?.click();\n        break;\n      case 'link':\n        btn = this.btnLink?.nativeElement;\n        btn?.click();\n        break;\n      case 'ul':\n        btn = this.btnUl?.nativeElement;\n        btn?.click();\n        break;\n      case 'ol':\n        btn = this.btnOl?.nativeElement;\n        btn?.click();\n        break;\n      case 'image':\n        btn = this.btnImage?.nativeElement;\n        btn?.click();\n        break;\n      default:\n        break;\n    }\n  }\n\n  currentLength(): number {\n    return this.value?.length || 0;\n  }\n\n  hideDivider(name: string) {\n    const linkOrImage = this.config.toolbar?.image || this.config.toolbar?.link;\n    const lists = this.config.toolbar?.bulletList || this.config.toolbar?.orderedList;\n    const typos = this.config.toolbar?.bold || this.config.toolbar?.italic || this.config.toolbar?.underline;\n    const textAlign = this.config.toolbar?.alignCenter || this.config.toolbar?.alignRight;\n    switch (name) {\n      case 'typo':\n        return !typos || (!textAlign && !linkOrImage && !lists);\n      case 'text-align':\n        return !textAlign || (!linkOrImage && !lists);\n      case 'lists':\n        return !lists || !linkOrImage;\n      case 'image':\n        return !this.config.toolbar?.image || !this.config.toolbar.link;\n      default:\n        return false;\n    }\n  }\n\n  private setEditorLabel() {\n    const editorDiv = this.elementRef.nativeElement.querySelector('.ql-editor');\n    this.renderer.setAttribute(editorDiv, 'role', 'textbox');\n    this.renderer.setAttribute(editorDiv, 'aria-labelledby', `editor-label-${this.id}`);\n    this.renderer.setAttribute(editorDiv, 'aria-multiline', 'true');\n  }\n\n  private handle() {\n    this.setReadOnlyState();\n    this.setDisabledState();\n  }\n\n  private addWhitespaceMatcher() {\n    // set up a matcher that replaces placeholder ###tab### with \\t in delta again\n    this.quillEditor.clipboard.addMatcher('p', (_node, delta) => {\n      let str = JSON.stringify(delta);\n      str = str.replace(/###tab###/g, '\\\\t');\n      delta = JSON.parse(str);\n      return delta;\n    });\n  }\n\n  private addImageMatcher() {\n    // set up a matcher that does allow/disallow to paste images corresponding to config settings\n    this.quillEditor.clipboard.addMatcher('img', (_node, delta) => {\n      // allowed, passthru delta\n      if (this.config.toolbar?.image === true) {\n        return delta;\n      }\n\n      // not allowed, remove image by setting up a new delta\n      const Delta = QuillType.import('quill-delta');\n      return new Delta().insert('');\n    });\n  }\n\n  private handleWhitespace(value: string): string {\n    // set a placeholder for tab entity to avoid that quill removes it\n    return value.replace(/&#9;/gi, '###tab###');\n  }\n}\n","<label class=\"fwe-input-text\" [class.fwe-disabled]=\"disabled\">\n  <div class=\"fng-editor-toolbar\" [id]=\"'editor-toolbar-' + id\">\n    <span class=\"ql-formats fwe-d-none\">\n      <select class=\"ql-size\" aria-hidden=\"true\">\n        <option value=\"small\"></option>\n        <option selected></option>\n        <option value=\"large\"></option>\n        <option value=\"huge\"></option>\n      </select>\n    </span>\n    <span class=\"ql-formats fng-editor-toolbar-buttons-container fwe-mr-3\">\n      <ng-container *ngIf=\"config?.toolbar?.bold\">\n        <button type=\"button\" #btnBold class=\"ql-bold fwe-d-none action-bold\" aria-hidden=\"true\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.underline || config.toolbar?.italic\"\n          (click)=\"delegate($event, 'bold')\"\n          [disabled]=\"disabled\"\n        >\n          <div class=\"fng-button-text fng-text-bold\">B</div>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.italic\">\n        <button type=\"button\" #btnItalic class=\"ql-italic fwe-d-none action-italic\" aria-hidden=\"true\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.underline\"\n          (click)=\"delegate($event, 'italic')\"\n          [disabled]=\"disabled\"\n        >\n          <div class=\"fng-button-text fng-text-italic\">I</div>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.underline\">\n        <button type=\"button\" #btnUnderline class=\"ql-underline fwe-d-none action-underline\" aria-hidden=\"true\"></button>\n        <button type=\"button\" class=\"fwe-btn fng-button\" (click)=\"delegate($event, 'underline')\" [disabled]=\"disabled\">\n          <div class=\"fng-button-text fng-text-underline\">U</div>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('typo')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.alignCenter\">\n        <button type=\"button\" #btnAlignCenter class=\"ql-align fwe-d-none action-align-center\" aria-hidden=\"true\" value=\"center\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.alignRight\"\n          (click)=\"delegate($event, 'align-center')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fng-icon-text-align-center fwe-pr-0\" [class.fng-gray]=\"disabled\"></i>\n        </button>\n      </ng-container>\n\n      <ng-container *ngIf=\"config?.toolbar?.alignRight\">\n        <button type=\"button\" #btnAlignRight class=\"ql-align fwe-d-none action-align-right\" aria-hidden=\"true\" value=\"right\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          (click)=\"delegate($event, 'align-right')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fng-icon-text-align-right fwe-pr-0\" [class.fng-gray]=\"disabled\"></i>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('text-align')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.bulletList\">\n        <button type=\"button\" #btnUl class=\"ql-list fwe-d-none action-ul\" aria-hidden=\"true\" value=\"bullet\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.orderedList\"\n          (click)=\"delegate($event, 'ul')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-list-list-view fwe-pr-0\"></i>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.orderedList\">\n        <button type=\"button\" #btnOl class=\"ql-list fwe-d-none action-ol\" aria-hidden=\"true\" value=\"ordered\"></button>\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'ol')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-enumeration fwe-pr-0\"></i>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('lists')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.image\">\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'image')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-image-image fwe-pr-0\"></i>\n        </button>\n        <button type=\"button\" #btnImage class=\"ql-image fwe-d-none\" aria-hidden=\"true\"></button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('image')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.link\">\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'link')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-file-link fwe-pr-0\"></i>\n        </button>\n        <button type=\"button\" #btnLink class=\"ql-link fwe-d-none\" aria-hidden=\"true\"></button>\n      </ng-container>\n    </span>\n  </div>\n  <div class=\"fng-editor-container\" [id]=\"'editor-container-' + id\">\n    <div class=\"fng-editor\" [id]=\"'editor-' + id\"></div>\n  </div>\n  <span class=\"fwe-input-text-label\" [id]=\"'editor-label-' + id\">{{ label }}</span>\n  <span *ngIf=\"hint\" class=\"fng-text-editor-info\">{{ hint }}</span>\n  <span *ngIf=\"error\" class=\"fng-text-editor-invalid\">{{ error }}</span>\n  <span *ngIf=\"maxLength > 0 && value != null\" class=\"fwe-input-text-count\">{{ currentLength() }} / {{ maxLength }}</span>\n</label>\n"]}
308
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"text-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/angular/src/lib/forms/text-editor/text-editor.component.ts","../../../../../../projects/angular/src/lib/forms/text-editor/text-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ViewEncapsulation,\n  Input,\n  AfterViewInit,\n  SecurityContext,\n  ElementRef,\n  Renderer2,\n  EventEmitter,\n  Output,\n  forwardRef,\n  OnChanges,\n  ViewChild,\n  SimpleChanges,\n  OnInit\n} from '@angular/core';\nimport QuillType from 'quill';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { FormsModule, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\n\nimport { FngValueAccessorBaseDirective } from '../value-accessor-base';\nimport { CommonModule } from '@angular/common';\nimport { DragDropModule } from '@angular/cdk/drag-drop';\nimport { ObserversModule } from '@angular/cdk/observers';\nimport { OverlayModule } from '@angular/cdk/overlay';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\n\nexport interface FngTextEditorConfiguration {\n  toolbar?: {\n    bold?: boolean;\n    italic?: boolean;\n    underline?: boolean;\n    alignCenter?: boolean;\n    alignRight?: boolean;\n    bulletList?: boolean;\n    orderedList?: boolean;\n    image?: boolean;\n    link?: boolean;\n  };\n  sanitizing?: {\n    skipDefaultSanitizing?: boolean;\n    skipWhitespaceHandling?: boolean;\n    sanitizingFn?: (ds: DomSanitizer, value: string) => string;\n  };\n  focusCursorToEndOnWrite?: boolean;\n}\n\n@Component({\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, FormsModule, CommonModule, OverlayModule, ScrollingModule, DragDropModule, ObserversModule],\n  selector: 'fng-text-editor',\n  templateUrl: './text-editor.component.html',\n  styleUrls: ['./text-editor.component.scss'],\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FngTextEditorComponent),\n      multi: true\n    }\n  ]\n})\nexport class FngTextEditorComponent extends FngValueAccessorBaseDirective<string | null | undefined> implements OnInit, AfterViewInit, OnChanges {\n  id: string | undefined;\n  quillEditor!: QuillType;\n  editorElem!: HTMLElement;\n  defaultConfig: FngTextEditorConfiguration = {\n    toolbar: {\n      bold: true,\n      italic: true,\n      underline: true,\n      alignCenter: true,\n      alignRight: true,\n      bulletList: true,\n      orderedList: true,\n      image: true,\n      link: true\n    },\n    sanitizing: {\n      sanitizingFn: undefined,\n      skipDefaultSanitizing: false,\n      skipWhitespaceHandling: false\n    },\n    focusCursorToEndOnWrite: true\n  };\n\n  @ViewChild('btnBold') btnBold: ElementRef | undefined;\n  @ViewChild('btnItalic') btnItalic: ElementRef | undefined;\n  @ViewChild('btnUnderline') btnUnderline: ElementRef | undefined;\n  @ViewChild('btnAlignCenter') btnAlignCenter: ElementRef | undefined;\n  @ViewChild('btnAlignRight') btnAlignRight: ElementRef | undefined;\n  @ViewChild('btnLink') btnLink: ElementRef | undefined;\n  @ViewChild('btnUl') btnUl: ElementRef | undefined;\n  @ViewChild('btnOl') btnOl: ElementRef | undefined;\n  @ViewChild('btnImage') btnImage: ElementRef | undefined;\n\n  @Output() fngChange = new EventEmitter<any>();\n  @Output() change = new EventEmitter<any>();\n\n  @Input() config: FngTextEditorConfiguration = this.defaultConfig;\n  @Input() modules: any;\n  @Input() label: string | undefined;\n  @Input() readOnly: boolean = false;\n  @Input() maxLength: number = 0;\n  @Input() error: string | undefined;\n  @Input() hint: string | undefined;\n  @Input()\n  override set value(value: string | null | undefined) {\n    if (this.innerValue !== value) {\n      this.innerValue = value;\n      this.changed.forEach(f => {\n        this.fngChange.emit(this.innerValue);\n        this.change.emit(this.innerValue);\n        return f(value);\n      });\n    }\n  }\n\n  override get value() {\n    return this.innerValue;\n  }\n\n  // needed to access editor content\n  valueGetter = (editorElement: HTMLElement): string | any => {\n    let html: string | undefined = editorElement?.querySelector('.ql-editor')?.innerHTML;\n    if (html === '<p><br></p>' || html === '<div><br></div>') {\n      html = undefined;\n    }\n    return html;\n  };\n\n  // needed to access editor content\n  valueSetter = (quillEditor: QuillType, value: any): any => {\n    const sanitizeConfig = this.config?.sanitizing || this.defaultConfig.sanitizing;\n\n    if (sanitizeConfig?.sanitizingFn) {\n      value = sanitizeConfig.sanitizingFn(this.domSanitizer, value);\n    }\n    if (sanitizeConfig && !sanitizeConfig.skipDefaultSanitizing) {\n      value = this.domSanitizer.sanitize(SecurityContext.HTML, value);\n    }\n    if (sanitizeConfig && !sanitizeConfig.skipWhitespaceHandling) {\n      value = this.handleWhitespace(value);\n    }\n\n    return quillEditor.clipboard.convert(value);\n  };\n\n  constructor(public elementRef: ElementRef, private renderer: Renderer2, private domSanitizer: DomSanitizer) {\n    super();\n  }\n\n  ngOnInit(): void {\n    this.id = (Math.random() * Date.now()).toString().replace('.', '-');\n  }\n\n  ngAfterViewInit(): void {\n    this.editorElem = this.elementRef.nativeElement.querySelector(`#editor-${this.id}`);\n\n    this.quillEditor = new QuillType(this.editorElem, {\n      modules: { toolbar: `#editor-toolbar-${this.id}`, ...this.modules },\n      theme: 'snow',\n      scrollingContainer: `#editor-container-${this.id}`\n    });\n    this.setEditorLabel();\n    this.addWhitespaceMatcher();\n    this.addImageMatcher();\n\n    const content = this.valueSetter(this.quillEditor, this.innerValue || '');\n    this.quillEditor.setContents(content, 'silent');\n    this.quillEditor.getModule('history').clear();\n\n    this.handle();\n\n    this.quillEditor.on('text-change', () => {\n      this.value = this.valueGetter(this.editorElem);\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes?.['config']?.currentValue != null) {\n      this.config = {\n        toolbar: { ...this.defaultConfig.toolbar, ...changes['config'].currentValue?.toolbar },\n        sanitizing: { ...this.defaultConfig.sanitizing, ...changes['config'].currentValue?.sanitizing }\n      };\n    }\n\n    if (!this.quillEditor) {\n      return;\n    }\n\n    this.handle();\n  }\n\n  override writeValue(value: string) {\n    this.innerValue = value;\n\n    if (!this.quillEditor) {\n      return;\n    }\n\n    const content = this.valueSetter(this.quillEditor, value || '');\n    this.quillEditor.setContents(content);\n\n    if (this.config?.focusCursorToEndOnWrite) {\n      // when new content gets written we always set cursor focus to end of text\n      this.quillEditor.setSelection(Number.MAX_SAFE_INTEGER, 0);\n    }\n  }\n\n  override setDisabledState(isDisabled: boolean = this.disabled): void {\n    this.disabled = isDisabled;\n    if (this.quillEditor) {\n      if (isDisabled) {\n        this.quillEditor.disable();\n        this.renderer.setAttribute(this.elementRef.nativeElement, 'disabled', 'disabled');\n      } else {\n        if (!this.readOnly) {\n          this.quillEditor.enable();\n        }\n        this.renderer.removeAttribute(this.elementRef.nativeElement, 'disabled');\n      }\n    }\n  }\n\n  setReadOnlyState() {\n    this.quillEditor.enable(!this.readOnly);\n  }\n\n  delegate(event: any, action: string) {\n    let btn: HTMLElement | undefined = undefined;\n    switch (action) {\n      case 'bold':\n        btn = this.btnBold?.nativeElement;\n        btn?.click();\n        break;\n      case 'italic':\n        btn = this.btnItalic?.nativeElement;\n        btn?.click();\n        break;\n      case 'underline':\n        btn = this.btnUnderline?.nativeElement;\n        btn?.click();\n        break;\n      case 'align-center':\n        btn = this.btnAlignCenter?.nativeElement;\n        btn?.click();\n        break;\n      case 'align-right':\n        btn = this.btnAlignRight?.nativeElement;\n        btn?.click();\n        break;\n      case 'link':\n        btn = this.btnLink?.nativeElement;\n        btn?.click();\n        break;\n      case 'ul':\n        btn = this.btnUl?.nativeElement;\n        btn?.click();\n        break;\n      case 'ol':\n        btn = this.btnOl?.nativeElement;\n        btn?.click();\n        break;\n      case 'image':\n        btn = this.btnImage?.nativeElement;\n        btn?.click();\n        break;\n      default:\n        break;\n    }\n  }\n\n  currentLength(): number {\n    return this.value?.length || 0;\n  }\n\n  hideDivider(name: string) {\n    const linkOrImage = this.config.toolbar?.image || this.config.toolbar?.link;\n    const lists = this.config.toolbar?.bulletList || this.config.toolbar?.orderedList;\n    const typos = this.config.toolbar?.bold || this.config.toolbar?.italic || this.config.toolbar?.underline;\n    const textAlign = this.config.toolbar?.alignCenter || this.config.toolbar?.alignRight;\n    switch (name) {\n      case 'typo':\n        return !typos || (!textAlign && !linkOrImage && !lists);\n      case 'text-align':\n        return !textAlign || (!linkOrImage && !lists);\n      case 'lists':\n        return !lists || !linkOrImage;\n      case 'image':\n        return !this.config.toolbar?.image || !this.config.toolbar.link;\n      default:\n        return false;\n    }\n  }\n\n  private setEditorLabel() {\n    const editorDiv = this.elementRef.nativeElement.querySelector('.ql-editor');\n    this.renderer.setAttribute(editorDiv, 'role', 'textbox');\n    this.renderer.setAttribute(editorDiv, 'aria-labelledby', `editor-label-${this.id}`);\n    this.renderer.setAttribute(editorDiv, 'aria-multiline', 'true');\n  }\n\n  private handle() {\n    this.setReadOnlyState();\n    this.setDisabledState();\n  }\n\n  private addWhitespaceMatcher() {\n    // set up a matcher that replaces placeholder ###tab### with \\t in delta again\n    this.quillEditor.clipboard.addMatcher('p', (_node, delta) => {\n      let str = JSON.stringify(delta);\n      str = str.replace(/###tab###/g, '\\\\t');\n      delta = JSON.parse(str);\n      return delta;\n    });\n  }\n\n  private addImageMatcher() {\n    // set up a matcher that does allow/disallow to paste images corresponding to config settings\n    this.quillEditor.clipboard.addMatcher('img', (_node, delta) => {\n      // allowed, passthru delta\n      if (this.config.toolbar?.image === true) {\n        return delta;\n      }\n\n      // not allowed, remove image by setting up a new delta\n      const Delta = QuillType.import('quill-delta');\n      return new Delta().insert('');\n    });\n  }\n\n  private handleWhitespace(value: string): string {\n    // set a placeholder for tab entity to avoid that quill removes it\n    return value.replace(/&#9;/gi, '###tab###');\n  }\n}\n","<label class=\"fwe-input-text\" [class.fwe-disabled]=\"disabled\">\n  <div class=\"fng-editor-toolbar\" [id]=\"'editor-toolbar-' + id\">\n    <span class=\"ql-formats fwe-d-none\">\n      <select class=\"ql-size\" aria-hidden=\"true\">\n        <option value=\"small\"></option>\n        <option selected></option>\n        <option value=\"large\"></option>\n        <option value=\"huge\"></option>\n      </select>\n    </span>\n    <span class=\"ql-formats fng-editor-toolbar-buttons-container fwe-mr-3\">\n      <ng-container *ngIf=\"config?.toolbar?.bold\">\n        <button type=\"button\" #btnBold class=\"ql-bold fwe-d-none action-bold\" aria-hidden=\"true\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.underline || config.toolbar?.italic\"\n          (click)=\"delegate($event, 'bold')\"\n          [disabled]=\"disabled\"\n        >\n          <div class=\"fng-button-text fng-text-bold\">B</div>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.italic\">\n        <button type=\"button\" #btnItalic class=\"ql-italic fwe-d-none action-italic\" aria-hidden=\"true\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.underline\"\n          (click)=\"delegate($event, 'italic')\"\n          [disabled]=\"disabled\"\n        >\n          <div class=\"fng-button-text fng-text-italic\">I</div>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.underline\">\n        <button type=\"button\" #btnUnderline class=\"ql-underline fwe-d-none action-underline\" aria-hidden=\"true\"></button>\n        <button type=\"button\" class=\"fwe-btn fng-button\" (click)=\"delegate($event, 'underline')\" [disabled]=\"disabled\">\n          <div class=\"fng-button-text fng-text-underline\">U</div>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('typo')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.alignCenter\">\n        <button type=\"button\" #btnAlignCenter class=\"ql-align fwe-d-none action-align-center\" aria-hidden=\"true\" value=\"center\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.alignRight\"\n          (click)=\"delegate($event, 'align-center')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fng-icon-text-align-center fwe-pr-0\" [class.fng-gray]=\"disabled\"></i>\n        </button>\n      </ng-container>\n\n      <ng-container *ngIf=\"config?.toolbar?.alignRight\">\n        <button type=\"button\" #btnAlignRight class=\"ql-align fwe-d-none action-align-right\" aria-hidden=\"true\" value=\"right\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          (click)=\"delegate($event, 'align-right')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fng-icon-text-align-right fwe-pr-0\" [class.fng-gray]=\"disabled\"></i>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('text-align')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.bulletList\">\n        <button type=\"button\" #btnUl class=\"ql-list fwe-d-none action-ul\" aria-hidden=\"true\" value=\"bullet\"></button>\n        <button\n          type=\"button\"\n          class=\"fwe-btn fng-btn-toolbar-list fng-button\"\n          [class.fwe-mr-3]=\"config.toolbar?.orderedList\"\n          (click)=\"delegate($event, 'ul')\"\n          [disabled]=\"disabled\"\n        >\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-list-list-view fwe-pr-0\"></i>\n        </button>\n      </ng-container>\n      <ng-container *ngIf=\"config?.toolbar?.orderedList\">\n        <button type=\"button\" #btnOl class=\"ql-list fwe-d-none action-ol\" aria-hidden=\"true\" value=\"ordered\"></button>\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'ol')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-enumeration fwe-pr-0\"></i>\n        </button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('lists')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.image\">\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'image')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-image-image fwe-pr-0\"></i>\n        </button>\n        <button type=\"button\" #btnImage class=\"ql-image fwe-d-none\" aria-hidden=\"true\"></button>\n      </ng-container>\n\n      <div *ngIf=\"!hideDivider('image')\" class=\"fng-divider-y fwe-mx-4\"></div>\n\n      <ng-container *ngIf=\"config?.toolbar?.link\">\n        <button type=\"button\" class=\"fwe-btn fng-btn-toolbar-list fng-button\" (click)=\"delegate($event, 'link')\" [disabled]=\"disabled\">\n          <i class=\"fwe-icon fwe-icon-toolbar-list fwe-icon-file-link fwe-pr-0\"></i>\n        </button>\n        <button type=\"button\" #btnLink class=\"ql-link fwe-d-none\" aria-hidden=\"true\"></button>\n      </ng-container>\n    </span>\n  </div>\n  <div class=\"fng-editor-container\" [id]=\"'editor-container-' + id\">\n    <div class=\"fng-editor\" [id]=\"'editor-' + id\"></div>\n  </div>\n  <span class=\"fwe-input-text-label\" [id]=\"'editor-label-' + id\">{{ label }}</span>\n  <span *ngIf=\"hint\" class=\"fng-text-editor-info\">{{ hint }}</span>\n  <span *ngIf=\"error\" class=\"fng-text-editor-invalid\">{{ error }}</span>\n  <span *ngIf=\"maxLength > 0 && value != null\" class=\"fwe-input-text-count\">{{ currentLength() }} / {{ maxLength }}</span>\n</label>\n"]}
@@ -6067,13 +6067,11 @@ class FngTextEditorComponent extends FngValueAccessorBaseDirective {
6067
6067
  set value(value) {
6068
6068
  if (this.innerValue !== value) {
6069
6069
  this.innerValue = value;
6070
- if (value) {
6071
- this.changed.forEach(f => {
6072
- this.fngChange.emit(this.innerValue);
6073
- this.change.emit(this.innerValue);
6074
- return f(value);
6075
- });
6076
- }
6070
+ this.changed.forEach(f => {
6071
+ this.fngChange.emit(this.innerValue);
6072
+ this.change.emit(this.innerValue);
6073
+ return f(value);
6074
+ });
6077
6075
  }
6078
6076
  }
6079
6077
  get value() {
@@ -6100,7 +6098,8 @@ class FngTextEditorComponent extends FngValueAccessorBaseDirective {
6100
6098
  sanitizingFn: undefined,
6101
6099
  skipDefaultSanitizing: false,
6102
6100
  skipWhitespaceHandling: false
6103
- }
6101
+ },
6102
+ focusCursorToEndOnWrite: true
6104
6103
  };
6105
6104
  this.fngChange = new EventEmitter();
6106
6105
  this.change = new EventEmitter();
@@ -6145,11 +6144,9 @@ class FngTextEditorComponent extends FngValueAccessorBaseDirective {
6145
6144
  this.setEditorLabel();
6146
6145
  this.addWhitespaceMatcher();
6147
6146
  this.addImageMatcher();
6148
- if (this.innerValue) {
6149
- const content = this.valueSetter(this.quillEditor, this.innerValue);
6150
- this.quillEditor.setContents(content, 'silent');
6151
- this.quillEditor.getModule('history').clear();
6152
- }
6147
+ const content = this.valueSetter(this.quillEditor, this.innerValue || '');
6148
+ this.quillEditor.setContents(content, 'silent');
6149
+ this.quillEditor.getModule('history').clear();
6153
6150
  this.handle();
6154
6151
  this.quillEditor.on('text-change', () => {
6155
6152
  this.value = this.valueGetter(this.editorElem);
@@ -6169,15 +6166,17 @@ class FngTextEditorComponent extends FngValueAccessorBaseDirective {
6169
6166
  this.handle();
6170
6167
  }
6171
6168
  writeValue(value) {
6172
- if (value === null || value === '') {
6173
- return;
6174
- }
6169
+ var _a;
6175
6170
  this.innerValue = value;
6176
6171
  if (!this.quillEditor) {
6177
6172
  return;
6178
6173
  }
6179
- const content = this.valueSetter(this.quillEditor, value);
6174
+ const content = this.valueSetter(this.quillEditor, value || '');
6180
6175
  this.quillEditor.setContents(content);
6176
+ if ((_a = this.config) === null || _a === void 0 ? void 0 : _a.focusCursorToEndOnWrite) {
6177
+ // when new content gets written we always set cursor focus to end of text
6178
+ this.quillEditor.setSelection(Number.MAX_SAFE_INTEGER, 0);
6179
+ }
6181
6180
  }
6182
6181
  setDisabledState(isDisabled = this.disabled) {
6183
6182
  this.disabled = isDisabled;