ngx-editor
Version:
Rich Text Editor for angular using ProseMirror
161 lines • 20.3 kB
JavaScript
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"]}