diff --git a/openmetadata-ui/src/main/resources/ui/src/assets/svg/ic-duplicate.svg b/openmetadata-ui/src/main/resources/ui/src/assets/svg/ic-duplicate.svg new file mode 100644 index 00000000000..2ade592c087 --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/assets/svg/ic-duplicate.svg @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockEditor.tsx b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockEditor.tsx index 187a8dee74d..f469ff49ad7 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockEditor.tsx +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockEditor.tsx @@ -49,6 +49,11 @@ const BlockEditor = forwardRef( onChange?.(backendFormat); }, + editorProps: { + attributes: { + class: 'om-block-editor', + }, + }, }); useImperativeHandle(ref, () => ({ diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/BlockMenu.tsx b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/BlockMenu.tsx new file mode 100644 index 00000000000..fbc5b2b1297 --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/BlockMenu.tsx @@ -0,0 +1,184 @@ +/* + * Copyright 2023 Collate. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import React, { useCallback, useEffect, useRef } from 'react'; +import tippy, { Instance } from 'tippy.js'; + +import { Editor } from '@tiptap/react'; + +import { NodeSelection } from '@tiptap/pm/state'; +import { isUndefined } from 'lodash'; +import { useTranslation } from 'react-i18next'; +import { ReactComponent as DeleteIcon } from '../../../assets/svg/ic-delete.svg'; +import { ReactComponent as DuplicateIcon } from '../../../assets/svg/ic-duplicate.svg'; +import { + nodeDOMAtCoords, + nodePosAtDOM, +} from '../Extensions/BlockAndDragDrop/helpers'; +import './block-menu.less'; + +interface BlockMenuProps { + editor: Editor; +} + +export const BlockMenu = (props: BlockMenuProps) => { + const { t } = useTranslation(); + const { editor } = props; + const { view } = editor; + const menuRef = useRef(null); + const popup = useRef(null); + + const handleClickBlockHandle = useCallback( + (event: MouseEvent) => { + const { view: editorView } = editor; + + const node = nodeDOMAtCoords({ + x: event.clientX + 24 * 4 + 24, + y: event.clientY, + }); + + if (!(node instanceof Element)) { + return; + } + + const nodePos = nodePosAtDOM(node, editorView); + if (isUndefined(nodePos)) { + return; + } + const nodeSelection = NodeSelection.create(editorView.state.doc, nodePos); + + editor + .chain() + .insertContentAt( + nodeSelection.to, + { type: 'paragraph' }, + { + updateSelection: true, + } + ) + .focus(nodeSelection.to) + .run(); + }, + [editor] + ); + + const handleClickDragHandle = useCallback( + (event: MouseEvent) => { + const target = event.target as HTMLElement; + + if (target.matches('[data-block-handle]')) { + handleClickBlockHandle(event); + } + + if (!target.matches('[data-drag-handle]')) { + popup.current?.hide(); + + return; + } + + event.preventDefault(); + event.stopPropagation(); + + popup.current?.setProps({ + getReferenceClientRect: () => target.getBoundingClientRect(), + }); + + popup.current?.show(); + }, + [view] + ); + + const handleKeyDown = () => { + popup.current?.hide(); + }; + + const handleDuplicate = useCallback(() => { + const { view } = editor; + const { state } = view; + const { selection } = state; + + editor + .chain() + .insertContentAt( + selection.to, + selection.content().content.firstChild?.toJSON(), + { + updateSelection: true, + } + ) + .focus(selection.to) + .run(); + + popup.current?.hide(); + }, [editor]); + + const handleDelete = useCallback(() => { + editor.commands.deleteSelection(); + popup.current?.hide(); + }, [editor]); + + useEffect(() => { + if (menuRef.current) { + menuRef.current.remove(); + menuRef.current.style.visibility = 'visible'; + + popup.current = tippy(view.dom, { + getReferenceClientRect: null, + content: menuRef.current, + appendTo: 'parent', + trigger: 'manual', + interactive: true, + arrow: false, + placement: 'top', + hideOnClick: true, + onShown: () => { + menuRef.current?.focus(); + }, + }); + } + + return () => { + popup.current?.destroy(); + popup.current = null; + }; + }, []); + + useEffect(() => { + document.addEventListener('click', handleClickDragHandle); + document.addEventListener('keydown', handleKeyDown); + + return () => { + document.removeEventListener('click', handleClickDragHandle); + document.addEventListener('keydown', handleKeyDown); + }; + }, [handleClickDragHandle, handleKeyDown]); + + return ( +
+ + + +
+ ); +}; + +export default BlockMenu; diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/block-menu.less b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/block-menu.less new file mode 100644 index 00000000000..21faa55180f --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/BlockMenu/block-menu.less @@ -0,0 +1,59 @@ +/* + * Copyright 2023 Collate. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@import (reference) url('../../../styles/variables.less'); + +.tippy-box { + max-width: 400px !important; +} + +.block-menu { + max-height: 300px; + padding: 12px 0px; + background-color: @white; + border: @global-border; + border-radius: @border-radius-base; + overflow-y: auto; + display: flex; + flex-direction: column; + gap: 0.25rem; + + .action { + background: none; + display: flex; + align-items: center; + gap: 0.5rem; + border: none; + padding: 4px 12px; + cursor: pointer; + transition: all 0.2s; + + &:disabled { + cursor: not-allowed; + } + + &:hover { + background-color: @grey-2; + } + + .action-icon-container { + display: flex; + align-items: center; + justify-content: center; + } + + .action-name { + font-family: inherit; + font-size: 0.95rem; + } + } +} diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/EditorSlots.tsx b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/EditorSlots.tsx index b38e9b30991..8f568cdc4cc 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/EditorSlots.tsx +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/EditorSlots.tsx @@ -15,6 +15,7 @@ import { isEmpty, isNil } from 'lodash'; import React, { forwardRef, useImperativeHandle, useState } from 'react'; import tippy, { Instance, Props } from 'tippy.js'; import { EditorSlotsRef } from './BlockEditor.interface'; +import BlockMenu from './BlockMenu/BlockMenu'; import BubbleMenu from './BubbleMenu/BubbleMenu'; import LinkModal, { LinkData } from './LinkModal/LinkModal'; import LinkPopup from './LinkPopup/LinkPopup'; @@ -164,6 +165,7 @@ const EditorSlots = forwardRef( /> )} {menus} + {!isNil(editor) && } ); } diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragDrop.tsx b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragDrop.tsx new file mode 100644 index 00000000000..08b76097ff8 --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragDrop.tsx @@ -0,0 +1,43 @@ +/* + * Copyright 2023 Collate. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import { Extension } from '@tiptap/core'; +import { BlockAndDragHandle } from './BlockAndDragHandle'; + +export interface BlockAndDragHandleOptions { + /** + * The width of the drag handle + */ + dragHandleWidth: number; + /** + * The width of the drag handle + */ + blockHandleWidth: number; +} + +// eslint-disable-next-line @typescript-eslint/no-empty-interface +interface DragAndDropOptions {} + +const DragAndDrop = Extension.create({ + name: 'dragAndDrop', + + addProseMirrorPlugins() { + return [ + BlockAndDragHandle({ + dragHandleWidth: 24, + blockHandleWidth: 24, + }), + ]; + }, +}); + +export default DragAndDrop; diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/drag-and-drop.tsx b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragHandle.ts similarity index 52% rename from openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/drag-and-drop.tsx rename to openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragHandle.ts index de4fbb32b51..0de14946f7a 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/drag-and-drop.tsx +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/BlockAndDragHandle.ts @@ -10,57 +10,20 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -import { Extension } from '@tiptap/core'; - import { NodeSelection, Plugin } from '@tiptap/pm/state'; // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore import { EditorView, __serializeForClipboard } from '@tiptap/pm/view'; +import { isUndefined } from 'lodash'; +import { BlockAndDragHandleOptions } from './BlockAndDragDrop'; +import { absoluteRect, nodeDOMAtCoords, nodePosAtDOM } from './helpers'; -export interface DragHandleOptions { - /** - * The width of the drag handle - */ - dragHandleWidth: number; -} -const absoluteRect = (node: Element) => { - const data = node.getBoundingClientRect(); +export const BlockAndDragHandle = (options: BlockAndDragHandleOptions) => { + let dragHandleElement: HTMLElement | null = null; + let blockHandleElement: HTMLElement | null = null; - return { - top: data.top, - left: data.left, - width: data.width, - }; -}; + // Drag Handle handlers -const nodeDOMAtCoords = (coords: { x: number; y: number }) => { - return document - .elementsFromPoint(coords.x, coords.y) - .find( - (elem: Element) => - elem.parentElement?.matches?.('.ProseMirror') || - elem.matches( - [ - 'li', - 'p:not(:first-child)', - 'pre', - 'blockquote', - 'h1, h2, h3, h4, h5, h6', - ].join(', ') - ) - ); -}; - -const nodePosAtDOM = (node: Element, view: EditorView) => { - const boundingRect = node.getBoundingClientRect(); - - return view.posAtCoords({ - left: boundingRect.left + 1, - top: boundingRect.top + 1, - })?.inside; -}; - -const DragHandle = (options: DragHandleOptions) => { const handleDragStart = (event: DragEvent, view: EditorView) => { view.focus(); @@ -78,7 +41,7 @@ const DragHandle = (options: DragHandleOptions) => { } const nodePos = nodePosAtDOM(node, view); - if (nodePos == null || nodePos < 0) { + if (isUndefined(nodePos)) { return; } @@ -99,7 +62,7 @@ const DragHandle = (options: DragHandleOptions) => { view.dragging = { slice, move: event.ctrlKey }; }; - const handleClick = (event: MouseEvent, view: EditorView) => { + const handleDragClick = (event: MouseEvent, view: EditorView) => { view.focus(); view.dom.classList.remove('om-node-dragging'); @@ -114,7 +77,7 @@ const DragHandle = (options: DragHandleOptions) => { } const nodePos = nodePosAtDOM(node, view); - if (!nodePos) { + if (isUndefined(nodePos)) { return; } @@ -123,8 +86,6 @@ const DragHandle = (options: DragHandleOptions) => { ); }; - let dragHandleElement: HTMLElement | null = null; - const hideDragHandle = () => { if (dragHandleElement) { dragHandleElement.classList.add('hidden'); @@ -137,27 +98,128 @@ const DragHandle = (options: DragHandleOptions) => { } }; + const handleMouseMoveForDragHandle = (event: MouseEvent) => { + const node = nodeDOMAtCoords({ + x: event.clientX + 50 + options.dragHandleWidth, + y: event.clientY, + }); + + if (!(node instanceof Element) || node.matches('ul, ol')) { + hideDragHandle(); + + return; + } + + const compStyle = window.getComputedStyle(node); + const lineHeight = parseInt(compStyle.lineHeight, 10); + const paddingTop = parseInt(compStyle.paddingTop, 10); + + const rect = absoluteRect(node); + + rect.top += (lineHeight - 24) / 2; + rect.top += paddingTop; + // Li markers + if (node.matches('ul:not([data-type=taskList]) li, ol li')) { + rect.left -= options.dragHandleWidth; + } + rect.width = options.dragHandleWidth; + + if (!dragHandleElement) { + return; + } + + dragHandleElement.style.left = `${rect.left - rect.width}px`; + dragHandleElement.style.top = `${rect.top}px`; + showDragHandle(); + }; + + // Block Handle handlers + + const hideBlockHandle = () => { + if (blockHandleElement) { + blockHandleElement.classList.add('hidden'); + } + }; + + const showBlockHandle = () => { + if (blockHandleElement) { + blockHandleElement.classList.remove('hidden'); + } + }; + + const handleMouseMoveForBlockHandle = (event: MouseEvent) => { + const node = nodeDOMAtCoords({ + x: event.clientX + options.dragHandleWidth * 4 + options.blockHandleWidth, + y: event.clientY, + }); + + if (!(node instanceof Element) || node.matches('ul, ol')) { + hideBlockHandle(); + + return; + } + + const compStyle = window.getComputedStyle(node); + const lineHeight = parseInt(compStyle.lineHeight, 10); + const paddingTop = parseInt(compStyle.paddingTop, 10); + + const rect = absoluteRect(node); + + rect.top += (lineHeight - 24) / 2; + rect.top += paddingTop; + // Li markers + if (node.matches('ul:not([data-type=taskList]) li, ol li')) { + rect.left -= options.blockHandleWidth; + } + rect.width = options.blockHandleWidth; + + if (!blockHandleElement) { + return; + } + + blockHandleElement.style.left = `${ + rect.left - rect.width - options.blockHandleWidth + }px`; + blockHandleElement.style.top = `${rect.top}px`; + showBlockHandle(); + }; + return new Plugin({ view: (view) => { + // drag handle initialization dragHandleElement = document.createElement('div'); dragHandleElement.draggable = true; dragHandleElement.dataset.dragHandle = ''; + dragHandleElement.title = 'Drag to move\nClick to open menu'; dragHandleElement.classList.add('om-drag-handle'); dragHandleElement.addEventListener('dragstart', (e) => { handleDragStart(e, view); }); dragHandleElement.addEventListener('click', (e) => { - handleClick(e, view); + handleDragClick(e, view); }); hideDragHandle(); + // block handle initialization + blockHandleElement = document.createElement('div'); + blockHandleElement.draggable = false; + blockHandleElement.dataset.blockHandle = ''; + blockHandleElement.title = 'Add new node'; + blockHandleElement.classList.add('om-block-handle'); + + hideBlockHandle(); + view?.dom?.parentElement?.appendChild(dragHandleElement); + view?.dom?.parentElement?.appendChild(blockHandleElement); return { destroy: () => { dragHandleElement?.remove?.(); dragHandleElement = null; + + blockHandleElement?.remove?.(); + blockHandleElement = null; }, }; }, @@ -167,45 +229,16 @@ const DragHandle = (options: DragHandleOptions) => { if (!view.editable) { return; } - - const node = nodeDOMAtCoords({ - x: event.clientX + 50 + options.dragHandleWidth, - y: event.clientY, - }); - - if (!(node instanceof Element) || node.matches('ul, ol')) { - hideDragHandle(); - - return; - } - - const compStyle = window.getComputedStyle(node); - const lineHeight = parseInt(compStyle.lineHeight, 10); - const paddingTop = parseInt(compStyle.paddingTop, 10); - - const rect = absoluteRect(node); - - rect.top += (lineHeight - 24) / 2; - rect.top += paddingTop; - // Li markers - if (node.matches('ul:not([data-type=taskList]) li, ol li')) { - rect.left -= options.dragHandleWidth; - } - rect.width = options.dragHandleWidth; - - if (!dragHandleElement) { - return; - } - - dragHandleElement.style.left = `${rect.left - rect.width}px`; - dragHandleElement.style.top = `${rect.top}px`; - showDragHandle(); + handleMouseMoveForDragHandle(event); + handleMouseMoveForBlockHandle(event); }, keydown: () => { hideDragHandle(); + hideBlockHandle(); }, mousewheel: () => { hideDragHandle(); + hideBlockHandle(); }, // dragging class is used for CSS dragstart: (view) => { @@ -221,20 +254,3 @@ const DragHandle = (options: DragHandleOptions) => { }, }); }; - -// eslint-disable-next-line @typescript-eslint/no-empty-interface -interface DragAndDropOptions {} - -const DragAndDrop = Extension.create({ - name: 'dragAndDrop', - - addProseMirrorPlugins() { - return [ - DragHandle({ - dragHandleWidth: 24, - }), - ]; - }, -}); - -export default DragAndDrop; diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/helpers.ts b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/helpers.ts new file mode 100644 index 00000000000..27a65511cc7 --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/BlockAndDragDrop/helpers.ts @@ -0,0 +1,50 @@ +/* + * Copyright 2023 Collate. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import { EditorView } from '@tiptap/pm/view'; + +export const absoluteRect = (node: Element) => { + const data = node.getBoundingClientRect(); + + return { + top: data.top, + left: data.left, + width: data.width, + }; +}; + +export const nodeDOMAtCoords = (coords: { x: number; y: number }) => { + return document + .elementsFromPoint(coords.x, coords.y) + .find( + (elem: Element) => + elem.parentElement?.matches?.('.ProseMirror') || + elem.matches( + [ + 'li', + 'p:not(:first-child)', + 'pre', + 'blockquote', + 'h1, h2, h3, h4, h5, h6', + ].join(', ') + ) + ); +}; + +export const nodePosAtDOM = (node: Element, view: EditorView) => { + const boundingRect = node.getBoundingClientRect(); + + return view.posAtCoords({ + left: boundingRect.left + 1, + top: boundingRect.top + 1, + })?.inside; +}; diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/focus.ts b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/focus.ts new file mode 100644 index 00000000000..13911f19046 --- /dev/null +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/focus.ts @@ -0,0 +1,109 @@ +/* + * Copyright 2023 Collate. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import { Extension } from '@tiptap/core'; +import { Plugin, PluginKey } from '@tiptap/pm/state'; +import { Decoration, DecorationSet } from '@tiptap/pm/view'; + +export interface FocusOptions { + className: string; + mode: 'all' | 'deepest' | 'shallowest'; +} + +export const Focus = Extension.create({ + name: 'focus', + + addOptions() { + return { + className: 'has-focus', + mode: 'all', + }; + }, + + addProseMirrorPlugins() { + return [ + new Plugin({ + key: new PluginKey('focus'), + props: { + decorations: ({ doc, selection }) => { + const { isEditable, isFocused } = this.editor; + const { anchor } = selection; + const decorations: Decoration[] = []; + + if (!isEditable || !isFocused) { + return DecorationSet.create(doc, []); + } + + // Maximum Levels + let maxLevels = 0; + + if (this.options.mode === 'deepest') { + doc.descendants((node, pos) => { + if (node.isText) { + return; + } + + const isCurrent = + anchor >= pos && anchor <= pos + node.nodeSize - 1; + + if (!isCurrent) { + return false; + } + + maxLevels += 1; + + return; + }); + } + + // Loop through current + let currentLevel = 0; + + doc.descendants((node, pos) => { + if (node.isText) { + return false; + } + + const isCurrent = + anchor >= pos && anchor <= pos + node.nodeSize - 1; + + if (!isCurrent) { + return false; + } + + currentLevel += 1; + + const outOfScope = + (this.options.mode === 'deepest' && + maxLevels - currentLevel > 0) || + (this.options.mode === 'shallowest' && currentLevel > 1); + + if (outOfScope) { + return this.options.mode === 'deepest'; + } + + decorations.push( + Decoration.node(pos, pos + node.nodeSize, { + class: this.options.className, + }) + ); + + return; + }); + + return DecorationSet.create(doc, decorations); + }, + }, + }), + ]; + }, +}); diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/index.ts b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/index.ts index 8b1e3213428..72db4a0bafa 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/index.ts +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/index.ts @@ -15,8 +15,9 @@ import Placeholder from '@tiptap/extension-placeholder'; import TaskItem from '@tiptap/extension-task-item'; import TaskList from '@tiptap/extension-task-list'; import StarterKit from '@tiptap/starter-kit'; +import BlockAndDragDrop from './BlockAndDragDrop/BlockAndDragDrop'; import DiffView from './diff-view'; -import DragAndDrop from './drag-and-drop'; +import { Focus } from './focus'; import { Hashtag } from './hashtag'; import { hashtagSuggestion } from './hashtag/hashtagSuggestion'; import { Image } from './image/image'; @@ -102,5 +103,8 @@ export const extensions = [ allowBase64: true, inline: true, }), - DragAndDrop, + BlockAndDragDrop, + Focus.configure({ + mode: 'deepest', + }), ]; diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/slash-command/index.ts b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/slash-command/index.ts index 0c531c0a476..faa754c9f6b 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/slash-command/index.ts +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/Extensions/slash-command/index.ts @@ -14,6 +14,8 @@ import { Extension } from '@tiptap/core'; import { PluginKey } from '@tiptap/pm/state'; import Suggestion, { SuggestionOptions } from '@tiptap/suggestion'; +export const slashMenuPluginKey = new PluginKey('slashSuggestion'); + export default Extension.create({ name: 'slashCommand', @@ -33,7 +35,7 @@ export default Extension.create({ addProseMirrorPlugins() { return [ Suggestion({ - pluginKey: new PluginKey('slashSuggestion'), + pluginKey: slashMenuPluginKey, ...this.options.slashSuggestion, editor: this.editor, }), diff --git a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/block-editor.less b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/block-editor.less index 11ef58c563e..a93a883fa4b 100644 --- a/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/block-editor.less +++ b/openmetadata-ui/src/main/resources/ui/src/components/BlockEditor/block-editor.less @@ -17,8 +17,12 @@ @markdown-bg-color: #f8f8fa; .block-editor-wrapper { + .om-block-editor > p:last-child { + // this is to have enough space after last node, referred from the reference editor + padding-bottom: 30vh; + } // show placeholder when editor is in focused mode - .tiptap .is-node-empty:last-child::before { + .tiptap.ProseMirror-focused .is-node-empty.has-focus::before { color: @grey-3; content: attr(data-placeholder); float: left; @@ -250,7 +254,8 @@ box-shadow: none; } -.om-drag-handle { +.om-drag-handle, +.om-block-handle { position: fixed; opacity: 1; transition: opacity ease-in 0.2s; @@ -282,12 +287,18 @@ pointer-events: none; } } +.om-block-handle { + cursor: pointer; + background-image: url(''); +} .om-list-decimal { list-style-type: decimal !important; + padding-left: 16px; } .om-list-disc { list-style-type: disc !important; + padding-left: 16px; } .om-leading-normal { line-height: 1.5; diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/de-de.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/de-de.json index d567c5943e9..05de10ee103 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/de-de.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/de-de.json @@ -331,6 +331,7 @@ "domain-plural": "Bereiche", "domain-type": "Bereichstyp", "downstream-depth": "Nachgelagerte Tiefe", + "duplicate": "Duplicate", "duration": "Dauer", "edge": "Kante", "edge-information": "Kanteninformationen", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/en-us.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/en-us.json index a667480afd9..349d7e6b903 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/en-us.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/en-us.json @@ -331,6 +331,7 @@ "domain-plural": "Domains", "domain-type": "Domain Type", "downstream-depth": "Downstream Depth", + "duplicate": "Duplicate", "duration": "Duration", "edge": "Edge", "edge-information": "Edge Information", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/es-es.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/es-es.json index 5622a7b071e..64a2adb3f94 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/es-es.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/es-es.json @@ -331,6 +331,7 @@ "domain-plural": "Domains", "domain-type": "Domain Type", "downstream-depth": "Profundidad del flujo", + "duplicate": "Duplicate", "duration": "Duration", "edge": "Edge", "edge-information": "Información de la arista", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/fr-fr.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/fr-fr.json index f1922d8b5df..bc05a0c1c90 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/fr-fr.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/fr-fr.json @@ -331,6 +331,7 @@ "domain-plural": "Domaines", "domain-type": "Type de Domaine", "downstream-depth": "Profondeur en Aval", + "duplicate": "Duplicate", "duration": "Durée", "edge": "Bord", "edge-information": "Informations du Bord", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/ja-jp.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/ja-jp.json index fe1eada4e11..967870a53af 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/ja-jp.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/ja-jp.json @@ -331,6 +331,7 @@ "domain-plural": "Domains", "domain-type": "Domain Type", "downstream-depth": "Downstream Depth", + "duplicate": "Duplicate", "duration": "Duration", "edge": "Edge", "edge-information": "エッジの情報", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/pt-br.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/pt-br.json index 48487453ddc..3b2c4ad1e3f 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/pt-br.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/pt-br.json @@ -331,6 +331,7 @@ "domain-plural": "Domains", "domain-type": "Domain Type", "downstream-depth": "Profundidade abaixo", + "duplicate": "Duplicate", "duration": "Duration", "edge": "Edge", "edge-information": "Informação sobre o limite", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/ru-ru.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/ru-ru.json index 5f60cb95298..16ae2c8d9ca 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/ru-ru.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/ru-ru.json @@ -331,6 +331,7 @@ "domain-plural": "Domains", "domain-type": "Domain Type", "downstream-depth": "Нисходящая линия", + "duplicate": "Duplicate", "duration": "Длительность", "edge": "Edge", "edge-information": "Граница информации", diff --git a/openmetadata-ui/src/main/resources/ui/src/locale/languages/zh-cn.json b/openmetadata-ui/src/main/resources/ui/src/locale/languages/zh-cn.json index c7f44c3d045..0bc520e3ace 100644 --- a/openmetadata-ui/src/main/resources/ui/src/locale/languages/zh-cn.json +++ b/openmetadata-ui/src/main/resources/ui/src/locale/languages/zh-cn.json @@ -331,6 +331,7 @@ "domain-plural": "域", "domain-type": "域类型", "downstream-depth": "下游深度", + "duplicate": "Duplicate", "duration": "持续时间", "edge": "Edge", "edge-information": "连线信息",