import {Plugin, EditorState} from "prosemirror-state" import {EditorView} from "prosemirror-view" import {dropPoint} from "prosemirror-transform" interface DropCursorOptions { /// The color of the cursor. Defaults to `black`. Use `false` to apply no color and rely only on class. color?: string | false /// The precise width of the cursor in pixels. Defaults to 1. width?: number /// A CSS class name to add to the cursor element. class?: string } /// Create a plugin that, when added to a ProseMirror instance, /// causes a decoration to show up at the drop position when something /// is dragged over the editor. /// /// Nodes may add a `disableDropCursor` property to their spec to /// control the showing of a drop cursor inside them. This may be a /// boolean or a function, which will be called with a view and a /// position, and should return a boolean. export function dropCursor(options: DropCursorOptions = {}): Plugin { return new Plugin({ view(editorView) { return new DropCursorView(editorView, options) } }) } class DropCursorView { width: number color: string | undefined class: string | undefined cursorPos: number | null = null element: HTMLElement | null = null timeout: number = -1 handlers: {name: string, handler: (event: Event) => void}[] constructor(readonly editorView: EditorView, options: DropCursorOptions) { this.width = options.width ?? 1 this.color = options.color === false ? undefined : (options.color || "black") this.class = options.class this.handlers = ["dragover", "dragend", "drop", "dragleave"].map(name => { let handler = (e: Event) => { (this as any)[name](e) } editorView.dom.addEventListener(name, handler) return {name, handler} }) } destroy() { this.handlers.forEach(({name, handler}) => this.editorView.dom.removeEventListener(name, handler)) } update(editorView: EditorView, prevState: EditorState) { if (this.cursorPos != null && prevState.doc != editorView.state.doc) { if (this.cursorPos > editorView.state.doc.content.size) this.setCursor(null) else this.updateOverlay() } } setCursor(pos: number | null) { if (pos == this.cursorPos) return this.cursorPos = pos if (pos == null) { this.element!.parentNode!.removeChild(this.element!) this.element = null } else { this.updateOverlay() } } updateOverlay() { let $pos = this.editorView.state.doc.resolve(this.cursorPos!) let isBlock = !$pos.parent.inlineContent, rect if (isBlock) { let before = $pos.nodeBefore, after = $pos.nodeAfter if (before || after) { let node = this.editorView.nodeDOM(this.cursorPos! - (before ? before.nodeSize : 0)) if (node) { let nodeRect = (node as HTMLElement).getBoundingClientRect() let top = before ? nodeRect.bottom : nodeRect.top if (before && after) top = (top + (this.editorView.nodeDOM(this.cursorPos!) as HTMLElement).getBoundingClientRect().top) / 2 rect = {left: nodeRect.left, right: nodeRect.right, top: top - this.width / 2, bottom: top + this.width / 2} } } } if (!rect) { let coords = this.editorView.coordsAtPos(this.cursorPos!) rect = {left: coords.left - this.width / 2, right: coords.left + this.width / 2, top: coords.top, bottom: coords.bottom} } let parent = this.editorView.dom.offsetParent! if (!this.element) { this.element = parent.appendChild(document.createElement("div")) if (this.class) this.element.className = this.class this.element.style.cssText = "position: absolute; z-index: 50; pointer-events: none;" if (this.color) { this.element.style.backgroundColor = this.color } } this.element.classList.toggle("prosemirror-dropcursor-block", isBlock) this.element.classList.toggle("prosemirror-dropcursor-inline", !isBlock) let parentLeft, parentTop if (!parent || parent == document.body && getComputedStyle(parent).position == "static") { parentLeft = -pageXOffset parentTop = -pageYOffset } else { let rect = parent.getBoundingClientRect() parentLeft = rect.left - parent.scrollLeft parentTop = rect.top - parent.scrollTop } this.element.style.left = (rect.left - parentLeft) + "px" this.element.style.top = (rect.top - parentTop) + "px" this.element.style.width = (rect.right - rect.left) + "px" this.element.style.height = (rect.bottom - rect.top) + "px" } scheduleRemoval(timeout: number) { clearTimeout(this.timeout) this.timeout = setTimeout(() => this.setCursor(null), timeout) } dragover(event: DragEvent) { if (!this.editorView.editable) return let pos = this.editorView.posAtCoords({left: event.clientX, top: event.clientY}) let node = pos && pos.inside >= 0 && this.editorView.state.doc.nodeAt(pos.inside) let disableDropCursor = node && node.type.spec.disableDropCursor let disabled = typeof disableDropCursor == "function" ? disableDropCursor(this.editorView, pos, event) : disableDropCursor if (pos && !disabled) { let target: number | null = pos.pos if (this.editorView.dragging && this.editorView.dragging.slice) { let point = dropPoint(this.editorView.state.doc, target, this.editorView.dragging.slice) if (point != null) target = point } this.setCursor(target) this.scheduleRemoval(5000) } } dragend() { this.scheduleRemoval(20) } drop() { this.scheduleRemoval(20) } dragleave(event: DragEvent) { if (event.target == this.editorView.dom || !this.editorView.dom.contains((event as any).relatedTarget)) this.setCursor(null) } }