You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					149 lines
				
				5.2 KiB
			
		
		
			
		
	
	
					149 lines
				
				5.2 KiB
			| 
											3 years ago
										 | 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`.
 | ||
|  |   color?: string | ||
|  | 
 | ||
|  |   /// 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 | ||
|  |   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 || "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!), rect | ||
|  |     if (!$pos.parent.inlineContent) { | ||
|  |       let before = $pos.nodeBefore, after = $pos.nodeAfter | ||
|  |       if (before || after) { | ||
|  |         let nodeRect = (this.editorView.nodeDOM(this.cursorPos! - (before ?  before.nodeSize : 0)) 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; background-color: " + this.color | ||
|  |     } | ||
|  |     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) : disableDropCursor | ||
|  | 
 | ||
|  |     if (pos && !disabled) { | ||
|  |       let target: number | null = pos.pos | ||
|  |       if (this.editorView.dragging && this.editorView.dragging.slice) { | ||
|  |         target = dropPoint(this.editorView.state.doc, target, this.editorView.dragging.slice) | ||
|  |         if (target == null) return this.setCursor(null) | ||
|  |       } | ||
|  |       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) | ||
|  |   } | ||
|  | } |