UNPKG

ngx-editor

Version:

Rich Text Editor for angular using ProseMirror

161 lines 20.3 kB
import { Component, HostBinding, HostListener, Input } from '@angular/core'; import { NodeSelection } from 'prosemirror-state'; import { asyncScheduler, fromEvent } from 'rxjs'; import { throttleTime } from 'rxjs/operators'; import * as i0 from "@angular/core"; import * as i1 from "../../../pipes/sanitize/sanitize-html.pipe"; import * as i2 from "../bubble/bubble.component"; import * as i3 from "@angular/common"; export class FloatingMenuComponent { constructor(el, sanitizeHTML) { this.el = el; this.sanitizeHTML = sanitizeHTML; this.posLeft = 0; this.posTop = 0; this.showMenu = false; this.dragging = false; this.execulableItems = []; this.activeItems = []; } get display() { return { visibility: this.showMenu ? 'visible' : 'hidden', opacity: this.showMenu ? '1' : '0', top: this.posTop + 'px', left: this.posLeft + 'px', }; } get view() { return this.editor.view; } onMouseDown(e) { const target = e.target; if (this.el.nativeElement.contains(target) && target.nodeName !== 'INPUT') { e.preventDefault(); return; } this.dragging = true; } onKeyDown(e) { const target = e.target; if (target.nodeName === 'INPUT') { return; } this.dragging = true; this.hide(); } onMouseUp(e) { const target = e.target; if (this.el.nativeElement.contains(target) || target.nodeName === 'INPUT') { e.preventDefault(); return; } this.dragging = false; this.useUpdate(); } onKeyUp(e) { const target = e.target; if (target.nodeName === 'INPUT') { return; } this.dragging = false; this.useUpdate(); } useUpdate() { if (!this.view) { return; } this.update(this.view); } hide() { this.showMenu = false; } show() { this.showMenu = true; } calculateBubblePosition(view) { const { state: { selection } } = view; const { from } = selection; // the floating bubble itself const bubbleEl = this.el.nativeElement; const bubble = bubbleEl.getBoundingClientRect(); // The box in which the tooltip is positioned, to use as base const box = bubbleEl.parentElement.getBoundingClientRect(); const start = view.coordsAtPos(from); let left = start.left - box.left; const overflowsRight = (box.right < (start.left + bubble.width) || bubble.right > box.right); if (overflowsRight) { left = box.width - bubble.width; } if (left < 0) { left = 0; } const bubbleHeight = bubble.height + parseInt(getComputedStyle(bubbleEl).marginBottom, 10); const top = (start.top - box.top) - bubbleHeight; return { left, top }; } update(view) { const { state } = view; const { selection } = state; const { empty } = selection; if (selection instanceof NodeSelection) { if (selection.node.type.name === 'image') { this.hide(); return; } } const hasFocus = this.view.hasFocus(); if (!hasFocus || empty || this.dragging) { this.hide(); return; } const { top, left } = this.calculateBubblePosition(this.view); this.posLeft = left; this.posTop = top; this.show(); } ngOnInit() { if (!this.editor) { throw new Error('NgxEditor: Required editor instance'); } this.updateSubscription = this.editor.update .subscribe((view) => { this.update(view); }); this.resizeSubscription = fromEvent(window, 'resize').pipe(throttleTime(500, asyncScheduler, { leading: true, trailing: true })).subscribe(() => { this.useUpdate(); }); } ngOnDestroy() { this.updateSubscription.unsubscribe(); this.resizeSubscription.unsubscribe(); } } FloatingMenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.0", ngImport: i0, type: FloatingMenuComponent, deps: [{ token: i0.ElementRef }, { token: i1.SanitizeHtmlPipe }], target: i0.ɵɵFactoryTarget.Component }); FloatingMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.2.0", type: FloatingMenuComponent, selector: "ngx-editor-floating-menu", inputs: { editor: "editor" }, host: { listeners: { "document:mousedown": "onMouseDown($event)", "document:keydown": "onKeyDown($event)", "document:mouseup": "onMouseUp($event)", "document:keyup": "onKeyUp($event)" }, properties: { "style": "this.display" } }, ngImport: i0, template: "<div #ref>\n <ng-content></ng-content>\n</div>\n<ng-container *ngIf=\"ref.children.length === 0\">\n <ngx-bubble [editor]=\"editor\"></ngx-bubble>\n</ng-container>\n", styles: ["*,*:before,*:after{box-sizing:border-box}:host{position:absolute;z-index:20;margin-bottom:5px;visibility:hidden;opacity:0}\n"], components: [{ type: i2.BubbleComponent, selector: "ngx-bubble", inputs: ["editor"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.0", ngImport: i0, type: FloatingMenuComponent, decorators: [{ type: Component, args: [{ selector: 'ngx-editor-floating-menu', template: "<div #ref>\n <ng-content></ng-content>\n</div>\n<ng-container *ngIf=\"ref.children.length === 0\">\n <ngx-bubble [editor]=\"editor\"></ngx-bubble>\n</ng-container>\n", styles: ["*,*:before,*:after{box-sizing:border-box}:host{position:absolute;z-index:20;margin-bottom:5px;visibility:hidden;opacity:0}\n"] }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.SanitizeHtmlPipe }]; }, propDecorators: { display: [{ type: HostBinding, args: ['style'] }], editor: [{ type: Input }], onMouseDown: [{ type: HostListener, args: ['document:mousedown', ['$event']] }], onKeyDown: [{ type: HostListener, args: ['document:keydown', ['$event']] }], onMouseUp: [{ type: HostListener, args: ['document:mouseup', ['$event']] }], onKeyUp: [{ type: HostListener, args: ['document:keyup', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"floating-menu.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-editor/src/lib/modules/menu/floating-menu/floating-menu.component.ts","../../../../../../../projects/ngx-editor/src/lib/modules/menu/floating-menu/floating-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, ElementRef, HostBinding,\n  HostListener, Input, OnDestroy, OnInit\n} from '@angular/core';\nimport { SafeHtml } from '@angular/platform-browser';\nimport { NodeSelection } from 'prosemirror-state';\nimport { EditorView } from 'prosemirror-view';\nimport { asyncScheduler, fromEvent, Subscription } from 'rxjs';\nimport { throttleTime } from 'rxjs/operators';\n\nimport Editor from '../../../Editor';\nimport Icon from '../../../icons';\nimport { TBItems } from '../../../types';\nimport { SanitizeHtmlPipe } from '../../../pipes/sanitize/sanitize-html.pipe';\n\ninterface BubblePosition {\n  top: number;\n  left: number;\n}\n\n@Component({\n  selector: 'ngx-editor-floating-menu',\n  templateUrl: './floating-menu.component.html',\n  styleUrls: ['./floating-menu.component.scss']\n})\nexport class FloatingMenuComponent implements OnInit, OnDestroy {\n\n  constructor(public el: ElementRef<HTMLElement>, private sanitizeHTML: SanitizeHtmlPipe) { }\n\n  @HostBinding('style') get display(): Partial<CSSStyleDeclaration> {\n    return {\n      visibility: this.showMenu ? 'visible' : 'hidden',\n      opacity: this.showMenu ? '1' : '0',\n      top: this.posTop + 'px',\n      left: this.posLeft + 'px',\n    };\n  }\n\n  private get view(): EditorView {\n    return this.editor.view;\n  }\n\n  @Input() editor: Editor;\n\n  private posLeft = 0;\n  private posTop = 0;\n  private showMenu = false;\n  private updateSubscription: Subscription;\n  private dragging = false;\n  private resizeSubscription: Subscription;\n  execulableItems: TBItems[] = [];\n  activeItems: TBItems[] = [];\n\n  @HostListener('document:mousedown', ['$event']) onMouseDown(e: MouseEvent): void {\n    const target = e.target as Node\n\n    if (this.el.nativeElement.contains(target) && target.nodeName !== 'INPUT') {\n      e.preventDefault();\n      return;\n    }\n\n    this.dragging = true;\n  }\n\n  @HostListener('document:keydown', ['$event']) onKeyDown(e: KeyboardEvent): void {\n    const target = e.target as Node\n\n    if (target.nodeName === 'INPUT') {\n      return;\n    }\n\n    this.dragging = true;\n    this.hide();\n  }\n\n  @HostListener('document:mouseup', ['$event']) onMouseUp(e: MouseEvent): void {\n    const target = e.target as Node\n\n    if (this.el.nativeElement.contains(target) || target.nodeName === 'INPUT') {\n      e.preventDefault();\n      return\n    }\n\n    this.dragging = false;\n    this.useUpdate();\n  }\n\n  @HostListener('document:keyup', ['$event']) onKeyUp(e: KeyboardEvent): void {\n    const target = e.target as Node\n\n    if (target.nodeName === 'INPUT') {\n      return;\n    }\n\n    this.dragging = false;\n    this.useUpdate();\n  }\n\n  private useUpdate(): void {\n    if (!this.view) {\n      return;\n    }\n\n    this.update(this.view);\n  }\n\n  private hide(): void {\n    this.showMenu = false;\n  }\n\n  private show(): void {\n    this.showMenu = true;\n  }\n\n  private calculateBubblePosition(view: EditorView): BubblePosition {\n    const { state: { selection } } = view;\n    const { from } = selection;\n\n    // the floating bubble itself\n    const bubbleEl = this.el.nativeElement;\n    const bubble = bubbleEl.getBoundingClientRect();\n\n    // The box in which the tooltip is positioned, to use as base\n    const box = bubbleEl.parentElement.getBoundingClientRect();\n\n    const start = view.coordsAtPos(from);\n\n    let left = start.left - box.left;\n\n    const overflowsRight = (\n      box.right < (start.left + bubble.width) ||\n      bubble.right > box.right\n    );\n\n    if (overflowsRight) {\n      left = box.width - bubble.width;\n    }\n\n    if (left < 0) {\n      left = 0;\n    }\n\n    const bubbleHeight = bubble.height + parseInt(getComputedStyle(bubbleEl).marginBottom, 10);\n    const top = (start.top - box.top) - bubbleHeight;\n\n    return {\n      left,\n      top\n    };\n  }\n\n  private update(view: EditorView): void {\n    const { state } = view;\n    const { selection } = state;\n    const { empty } = selection;\n\n    if (selection instanceof NodeSelection) {\n      if (selection.node.type.name === 'image') {\n        this.hide();\n        return;\n      }\n    }\n\n    const hasFocus = this.view.hasFocus();\n\n    if (!hasFocus || empty || this.dragging) {\n      this.hide();\n      return;\n    }\n\n    const { top, left } = this.calculateBubblePosition(this.view);\n\n    this.posLeft = left;\n    this.posTop = top;\n\n    this.show();\n  }\n\n  ngOnInit(): void {\n    if (!this.editor) {\n      throw new Error('NgxEditor: Required editor instance');\n    }\n\n    this.updateSubscription = this.editor.update\n      .subscribe((view) => {\n        this.update(view);\n      });\n\n    this.resizeSubscription = fromEvent(window, 'resize').pipe(\n      throttleTime(500, asyncScheduler, { leading: true, trailing: true })\n    ).subscribe(() => {\n      this.useUpdate();\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.updateSubscription.unsubscribe();\n    this.resizeSubscription.unsubscribe();\n  }\n}\n","<div #ref>\n  <ng-content></ng-content>\n</div>\n<ng-container *ngIf=\"ref.children.length === 0\">\n  <ngx-bubble [editor]=\"editor\"></ngx-bubble>\n</ng-container>\n"]}