415 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
		
		
			
		
	
	
			415 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
|  | /** | ||
|  |  * Copyright (c) Meta Platforms, Inc. and affiliates. | ||
|  |  * | ||
|  |  * This source code is licensed under the MIT license found in the | ||
|  |  * LICENSE file in the root directory of this source tree. | ||
|  |  * | ||
|  |  */ | ||
|  | 
 | ||
|  | import type {LexicalEditor, NodeKey, TextFormatType} from 'lexical'; | ||
|  | 
 | ||
|  | import { | ||
|  |   addClassNamesToElement, | ||
|  |   removeClassNamesFromElement, | ||
|  | } from '@lexical/utils'; | ||
|  | import { | ||
|  |   $createParagraphNode, | ||
|  |   $createRangeSelection, | ||
|  |   $createTextNode, | ||
|  |   $getNearestNodeFromDOMNode, | ||
|  |   $getNodeByKey, | ||
|  |   $getRoot, | ||
|  |   $getSelection, | ||
|  |   $isElementNode, | ||
|  |   $setSelection, | ||
|  |   SELECTION_CHANGE_COMMAND, | ||
|  | } from 'lexical'; | ||
|  | import invariant from 'lexical/shared/invariant'; | ||
|  | 
 | ||
|  | import {$isTableCellNode} from './LexicalTableCellNode'; | ||
|  | import {$isTableNode} from './LexicalTableNode'; | ||
|  | import { | ||
|  |   $createTableSelection, | ||
|  |   $isTableSelection, | ||
|  |   type TableSelection, | ||
|  | } from './LexicalTableSelection'; | ||
|  | import { | ||
|  |   $findTableNode, | ||
|  |   $updateDOMForSelection, | ||
|  |   getDOMSelection, | ||
|  |   getTable, | ||
|  | } from './LexicalTableSelectionHelpers'; | ||
|  | 
 | ||
|  | export type TableDOMCell = { | ||
|  |   elem: HTMLElement; | ||
|  |   highlighted: boolean; | ||
|  |   hasBackgroundColor: boolean; | ||
|  |   x: number; | ||
|  |   y: number; | ||
|  | }; | ||
|  | 
 | ||
|  | export type TableDOMRows = Array<Array<TableDOMCell | undefined> | undefined>; | ||
|  | 
 | ||
|  | export type TableDOMTable = { | ||
|  |   domRows: TableDOMRows; | ||
|  |   columns: number; | ||
|  |   rows: number; | ||
|  | }; | ||
|  | 
 | ||
|  | export class TableObserver { | ||
|  |   focusX: number; | ||
|  |   focusY: number; | ||
|  |   listenersToRemove: Set<() => void>; | ||
|  |   table: TableDOMTable; | ||
|  |   isHighlightingCells: boolean; | ||
|  |   anchorX: number; | ||
|  |   anchorY: number; | ||
|  |   tableNodeKey: NodeKey; | ||
|  |   anchorCell: TableDOMCell | null; | ||
|  |   focusCell: TableDOMCell | null; | ||
|  |   anchorCellNodeKey: NodeKey | null; | ||
|  |   focusCellNodeKey: NodeKey | null; | ||
|  |   editor: LexicalEditor; | ||
|  |   tableSelection: TableSelection | null; | ||
|  |   hasHijackedSelectionStyles: boolean; | ||
|  |   isSelecting: boolean; | ||
|  | 
 | ||
|  |   constructor(editor: LexicalEditor, tableNodeKey: string) { | ||
|  |     this.isHighlightingCells = false; | ||
|  |     this.anchorX = -1; | ||
|  |     this.anchorY = -1; | ||
|  |     this.focusX = -1; | ||
|  |     this.focusY = -1; | ||
|  |     this.listenersToRemove = new Set(); | ||
|  |     this.tableNodeKey = tableNodeKey; | ||
|  |     this.editor = editor; | ||
|  |     this.table = { | ||
|  |       columns: 0, | ||
|  |       domRows: [], | ||
|  |       rows: 0, | ||
|  |     }; | ||
|  |     this.tableSelection = null; | ||
|  |     this.anchorCellNodeKey = null; | ||
|  |     this.focusCellNodeKey = null; | ||
|  |     this.anchorCell = null; | ||
|  |     this.focusCell = null; | ||
|  |     this.hasHijackedSelectionStyles = false; | ||
|  |     this.trackTable(); | ||
|  |     this.isSelecting = false; | ||
|  |   } | ||
|  | 
 | ||
|  |   getTable(): TableDOMTable { | ||
|  |     return this.table; | ||
|  |   } | ||
|  | 
 | ||
|  |   removeListeners() { | ||
|  |     Array.from(this.listenersToRemove).forEach((removeListener) => | ||
|  |       removeListener(), | ||
|  |     ); | ||
|  |   } | ||
|  | 
 | ||
|  |   trackTable() { | ||
|  |     const observer = new MutationObserver((records) => { | ||
|  |       this.editor.update(() => { | ||
|  |         let gridNeedsRedraw = false; | ||
|  | 
 | ||
|  |         for (let i = 0; i < records.length; i++) { | ||
|  |           const record = records[i]; | ||
|  |           const target = record.target; | ||
|  |           const nodeName = target.nodeName; | ||
|  | 
 | ||
|  |           if ( | ||
|  |             nodeName === 'TABLE' || | ||
|  |             nodeName === 'TBODY' || | ||
|  |             nodeName === 'THEAD' || | ||
|  |             nodeName === 'TR' | ||
|  |           ) { | ||
|  |             gridNeedsRedraw = true; | ||
|  |             break; | ||
|  |           } | ||
|  |         } | ||
|  | 
 | ||
|  |         if (!gridNeedsRedraw) { | ||
|  |           return; | ||
|  |         } | ||
|  | 
 | ||
|  |         const tableElement = this.editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |         if (!tableElement) { | ||
|  |           throw new Error('Expected to find TableElement in DOM'); | ||
|  |         } | ||
|  | 
 | ||
|  |         this.table = getTable(tableElement); | ||
|  |       }); | ||
|  |     }); | ||
|  |     this.editor.update(() => { | ||
|  |       const tableElement = this.editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!tableElement) { | ||
|  |         throw new Error('Expected to find TableElement in DOM'); | ||
|  |       } | ||
|  | 
 | ||
|  |       this.table = getTable(tableElement); | ||
|  |       observer.observe(tableElement, { | ||
|  |         attributes: true, | ||
|  |         childList: true, | ||
|  |         subtree: true, | ||
|  |       }); | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   clearHighlight() { | ||
|  |     const editor = this.editor; | ||
|  |     this.isHighlightingCells = false; | ||
|  |     this.anchorX = -1; | ||
|  |     this.anchorY = -1; | ||
|  |     this.focusX = -1; | ||
|  |     this.focusY = -1; | ||
|  |     this.tableSelection = null; | ||
|  |     this.anchorCellNodeKey = null; | ||
|  |     this.focusCellNodeKey = null; | ||
|  |     this.anchorCell = null; | ||
|  |     this.focusCell = null; | ||
|  |     this.hasHijackedSelectionStyles = false; | ||
|  | 
 | ||
|  |     this.enableHighlightStyle(); | ||
|  | 
 | ||
|  |     editor.update(() => { | ||
|  |       const tableNode = $getNodeByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!$isTableNode(tableNode)) { | ||
|  |         throw new Error('Expected TableNode.'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const tableElement = editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!tableElement) { | ||
|  |         throw new Error('Expected to find TableElement in DOM'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const grid = getTable(tableElement); | ||
|  |       $updateDOMForSelection(editor, grid, null); | ||
|  |       $setSelection(null); | ||
|  |       editor.dispatchCommand(SELECTION_CHANGE_COMMAND, undefined); | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   enableHighlightStyle() { | ||
|  |     const editor = this.editor; | ||
|  |     editor.update(() => { | ||
|  |       const tableElement = editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!tableElement) { | ||
|  |         throw new Error('Expected to find TableElement in DOM'); | ||
|  |       } | ||
|  | 
 | ||
|  |       removeClassNamesFromElement( | ||
|  |         tableElement, | ||
|  |         editor._config.theme.tableSelection, | ||
|  |       ); | ||
|  |       tableElement.classList.remove('disable-selection'); | ||
|  |       this.hasHijackedSelectionStyles = false; | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   disableHighlightStyle() { | ||
|  |     const editor = this.editor; | ||
|  |     editor.update(() => { | ||
|  |       const tableElement = editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!tableElement) { | ||
|  |         throw new Error('Expected to find TableElement in DOM'); | ||
|  |       } | ||
|  | 
 | ||
|  |       addClassNamesToElement(tableElement, editor._config.theme.tableSelection); | ||
|  |       this.hasHijackedSelectionStyles = true; | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   updateTableTableSelection(selection: TableSelection | null): void { | ||
|  |     if (selection !== null && selection.tableKey === this.tableNodeKey) { | ||
|  |       const editor = this.editor; | ||
|  |       this.tableSelection = selection; | ||
|  |       this.isHighlightingCells = true; | ||
|  |       this.disableHighlightStyle(); | ||
|  |       $updateDOMForSelection(editor, this.table, this.tableSelection); | ||
|  |     } else if (selection == null) { | ||
|  |       this.clearHighlight(); | ||
|  |     } else { | ||
|  |       this.tableNodeKey = selection.tableKey; | ||
|  |       this.updateTableTableSelection(selection); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   setFocusCellForSelection(cell: TableDOMCell, ignoreStart = false) { | ||
|  |     const editor = this.editor; | ||
|  |     editor.update(() => { | ||
|  |       const tableNode = $getNodeByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!$isTableNode(tableNode)) { | ||
|  |         throw new Error('Expected TableNode.'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const tableElement = editor.getElementByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!tableElement) { | ||
|  |         throw new Error('Expected to find TableElement in DOM'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const cellX = cell.x; | ||
|  |       const cellY = cell.y; | ||
|  |       this.focusCell = cell; | ||
|  | 
 | ||
|  |       if (this.anchorCell !== null) { | ||
|  |         const domSelection = getDOMSelection(editor._window); | ||
|  |         // Collapse the selection
 | ||
|  |         if (domSelection) { | ||
|  |           domSelection.setBaseAndExtent( | ||
|  |             this.anchorCell.elem, | ||
|  |             0, | ||
|  |             this.focusCell.elem, | ||
|  |             0, | ||
|  |           ); | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       if ( | ||
|  |         !this.isHighlightingCells && | ||
|  |         (this.anchorX !== cellX || this.anchorY !== cellY || ignoreStart) | ||
|  |       ) { | ||
|  |         this.isHighlightingCells = true; | ||
|  |         this.disableHighlightStyle(); | ||
|  |       } else if (cellX === this.focusX && cellY === this.focusY) { | ||
|  |         return; | ||
|  |       } | ||
|  | 
 | ||
|  |       this.focusX = cellX; | ||
|  |       this.focusY = cellY; | ||
|  | 
 | ||
|  |       if (this.isHighlightingCells) { | ||
|  |         const focusTableCellNode = $getNearestNodeFromDOMNode(cell.elem); | ||
|  | 
 | ||
|  |         if ( | ||
|  |           this.tableSelection != null && | ||
|  |           this.anchorCellNodeKey != null && | ||
|  |           $isTableCellNode(focusTableCellNode) && | ||
|  |           tableNode.is($findTableNode(focusTableCellNode)) | ||
|  |         ) { | ||
|  |           const focusNodeKey = focusTableCellNode.getKey(); | ||
|  | 
 | ||
|  |           this.tableSelection = | ||
|  |             this.tableSelection.clone() || $createTableSelection(); | ||
|  | 
 | ||
|  |           this.focusCellNodeKey = focusNodeKey; | ||
|  |           this.tableSelection.set( | ||
|  |             this.tableNodeKey, | ||
|  |             this.anchorCellNodeKey, | ||
|  |             this.focusCellNodeKey, | ||
|  |           ); | ||
|  | 
 | ||
|  |           $setSelection(this.tableSelection); | ||
|  | 
 | ||
|  |           editor.dispatchCommand(SELECTION_CHANGE_COMMAND, undefined); | ||
|  | 
 | ||
|  |           $updateDOMForSelection(editor, this.table, this.tableSelection); | ||
|  |         } | ||
|  |       } | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   setAnchorCellForSelection(cell: TableDOMCell) { | ||
|  |     this.isHighlightingCells = false; | ||
|  |     this.anchorCell = cell; | ||
|  |     this.anchorX = cell.x; | ||
|  |     this.anchorY = cell.y; | ||
|  | 
 | ||
|  |     this.editor.update(() => { | ||
|  |       const anchorTableCellNode = $getNearestNodeFromDOMNode(cell.elem); | ||
|  | 
 | ||
|  |       if ($isTableCellNode(anchorTableCellNode)) { | ||
|  |         const anchorNodeKey = anchorTableCellNode.getKey(); | ||
|  |         this.tableSelection = | ||
|  |           this.tableSelection != null | ||
|  |             ? this.tableSelection.clone() | ||
|  |             : $createTableSelection(); | ||
|  |         this.anchorCellNodeKey = anchorNodeKey; | ||
|  |       } | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   formatCells(type: TextFormatType) { | ||
|  |     this.editor.update(() => { | ||
|  |       const selection = $getSelection(); | ||
|  | 
 | ||
|  |       if (!$isTableSelection(selection)) { | ||
|  |         invariant(false, 'Expected grid selection'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const formatSelection = $createRangeSelection(); | ||
|  | 
 | ||
|  |       const anchor = formatSelection.anchor; | ||
|  |       const focus = formatSelection.focus; | ||
|  | 
 | ||
|  |       selection.getNodes().forEach((cellNode) => { | ||
|  |         if ($isTableCellNode(cellNode) && cellNode.getTextContentSize() !== 0) { | ||
|  |           anchor.set(cellNode.getKey(), 0, 'element'); | ||
|  |           focus.set(cellNode.getKey(), cellNode.getChildrenSize(), 'element'); | ||
|  |           formatSelection.formatText(type); | ||
|  |         } | ||
|  |       }); | ||
|  | 
 | ||
|  |       $setSelection(selection); | ||
|  | 
 | ||
|  |       this.editor.dispatchCommand(SELECTION_CHANGE_COMMAND, undefined); | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   clearText() { | ||
|  |     const editor = this.editor; | ||
|  |     editor.update(() => { | ||
|  |       const tableNode = $getNodeByKey(this.tableNodeKey); | ||
|  | 
 | ||
|  |       if (!$isTableNode(tableNode)) { | ||
|  |         throw new Error('Expected TableNode.'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const selection = $getSelection(); | ||
|  | 
 | ||
|  |       if (!$isTableSelection(selection)) { | ||
|  |         invariant(false, 'Expected grid selection'); | ||
|  |       } | ||
|  | 
 | ||
|  |       const selectedNodes = selection.getNodes().filter($isTableCellNode); | ||
|  | 
 | ||
|  |       if (selectedNodes.length === this.table.columns * this.table.rows) { | ||
|  |         tableNode.selectPrevious(); | ||
|  |         // Delete entire table
 | ||
|  |         tableNode.remove(); | ||
|  |         const rootNode = $getRoot(); | ||
|  |         rootNode.selectStart(); | ||
|  |         return; | ||
|  |       } | ||
|  | 
 | ||
|  |       selectedNodes.forEach((cellNode) => { | ||
|  |         if ($isElementNode(cellNode)) { | ||
|  |           const paragraphNode = $createParagraphNode(); | ||
|  |           const textNode = $createTextNode(); | ||
|  |           paragraphNode.append(textNode); | ||
|  |           cellNode.append(paragraphNode); | ||
|  |           cellNode.getChildren().forEach((child) => { | ||
|  |             if (child !== paragraphNode) { | ||
|  |               child.remove(); | ||
|  |             } | ||
|  |           }); | ||
|  |         } | ||
|  |       }); | ||
|  | 
 | ||
|  |       $updateDOMForSelection(editor, this.table, null); | ||
|  | 
 | ||
|  |       $setSelection(null); | ||
|  | 
 | ||
|  |       editor.dispatchCommand(SELECTION_CHANGE_COMMAND, undefined); | ||
|  |     }); | ||
|  |   } | ||
|  | } |