Merge PR: upgrade lexical, add table feature converter (#7521)
This PR - upgrades lexical and ports all bug fixes from the playground over - adds table action buttons. When hovering the edges of the table, buttons pop up to easily add a new table column or row - adds an html converter for the table feature - makes the placeholder shown in the editor when no text is present accessible **BREAKING:** This upgrades lexical from 0.16.1 to 0.17.0. If you have any lexical packages installed in your project, please update them accordingly. Additionally, if you depend on the lexical APIs, please consult their changelog, as lexical may introduce breaking changes: https://github.com/facebook/lexical/releases/tag/v0.17.0
This commit is contained in:
@@ -41,24 +41,24 @@
|
|||||||
"translateNewKeys": "tsx scripts/translateNewKeys.ts"
|
"translateNewKeys": "tsx scripts/translateNewKeys.ts"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@lexical/headless": "0.16.1",
|
"@lexical/headless": "0.17.0",
|
||||||
"@lexical/link": "0.16.1",
|
"@lexical/link": "0.17.0",
|
||||||
"@lexical/list": "0.16.1",
|
"@lexical/list": "0.17.0",
|
||||||
"@lexical/mark": "0.16.1",
|
"@lexical/mark": "0.17.0",
|
||||||
"@lexical/markdown": "0.16.1",
|
"@lexical/markdown": "0.17.0",
|
||||||
"@lexical/react": "0.16.1",
|
"@lexical/react": "0.17.0",
|
||||||
"@lexical/rich-text": "0.16.1",
|
"@lexical/rich-text": "0.17.0",
|
||||||
"@lexical/selection": "0.16.1",
|
"@lexical/selection": "0.17.0",
|
||||||
"@lexical/utils": "0.16.1",
|
"@lexical/utils": "0.17.0",
|
||||||
"@types/uuid": "10.0.0",
|
"@types/uuid": "10.0.0",
|
||||||
"bson-objectid": "2.0.4",
|
"bson-objectid": "2.0.4",
|
||||||
"dequal": "2.0.3",
|
"dequal": "2.0.3",
|
||||||
"lexical": "0.16.1",
|
"lexical": "0.17.0",
|
||||||
"react-error-boundary": "4.0.13",
|
"react-error-boundary": "4.0.13",
|
||||||
"uuid": "10.0.0"
|
"uuid": "10.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@lexical/eslint-plugin": " 0.16.1",
|
"@lexical/eslint-plugin": "0.17.0",
|
||||||
"@payloadcms/eslint-config": "workspace:*",
|
"@payloadcms/eslint-config": "workspace:*",
|
||||||
"@payloadcms/next": "workspace:*",
|
"@payloadcms/next": "workspace:*",
|
||||||
"@payloadcms/translations": "workspace:*",
|
"@payloadcms/translations": "workspace:*",
|
||||||
@@ -75,20 +75,20 @@
|
|||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@faceless-ui/modal": "3.0.0-beta.2",
|
"@faceless-ui/modal": "3.0.0-beta.2",
|
||||||
"@faceless-ui/scroll-info": "2.0.0-beta.0",
|
"@faceless-ui/scroll-info": "2.0.0-beta.0",
|
||||||
"@lexical/headless": "0.16.1",
|
"@lexical/headless": "0.17.0",
|
||||||
"@lexical/link": "0.16.1",
|
"@lexical/link": "0.17.0",
|
||||||
"@lexical/list": "0.16.1",
|
"@lexical/list": "0.17.0",
|
||||||
"@lexical/mark": "0.16.1",
|
"@lexical/mark": "0.17.0",
|
||||||
"@lexical/markdown": "0.16.1",
|
"@lexical/markdown": "0.17.0",
|
||||||
"@lexical/react": "0.16.1",
|
"@lexical/react": "0.17.0",
|
||||||
"@lexical/rich-text": "0.16.1",
|
"@lexical/rich-text": "0.17.0",
|
||||||
"@lexical/selection": "0.16.1",
|
"@lexical/selection": "0.17.0",
|
||||||
"@lexical/table": "0.16.1",
|
"@lexical/table": "0.17.0",
|
||||||
"@lexical/utils": "0.16.1",
|
"@lexical/utils": "0.17.0",
|
||||||
"@payloadcms/next": "workspace:*",
|
"@payloadcms/next": "workspace:*",
|
||||||
"@payloadcms/translations": "workspace:*",
|
"@payloadcms/translations": "workspace:*",
|
||||||
"@payloadcms/ui": "workspace:*",
|
"@payloadcms/ui": "workspace:*",
|
||||||
"lexical": "0.16.1",
|
"lexical": "0.17.0",
|
||||||
"payload": "workspace:*",
|
"payload": "workspace:*",
|
||||||
"react": "^19.0.0 || ^19.0.0-rc-6230622a1a-20240610",
|
"react": "^19.0.0 || ^19.0.0-rc-6230622a1a-20240610",
|
||||||
"react-dom": "^19.0.0 || ^19.0.0-rc-6230622a1a-20240610"
|
"react-dom": "^19.0.0 || ^19.0.0-rc-6230622a1a-20240610"
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import { slashMenuBasicGroupWithItems } from '../shared/slashMenu/basicGroup.js'
|
|||||||
import { toolbarAddDropdownGroupWithItems } from '../shared/toolbar/addDropdownGroup.js'
|
import { toolbarAddDropdownGroupWithItems } from '../shared/toolbar/addDropdownGroup.js'
|
||||||
import { TableActionMenuPlugin } from './plugins/TableActionMenuPlugin/index.js'
|
import { TableActionMenuPlugin } from './plugins/TableActionMenuPlugin/index.js'
|
||||||
import { TableCellResizerPlugin } from './plugins/TableCellResizerPlugin/index.js'
|
import { TableCellResizerPlugin } from './plugins/TableCellResizerPlugin/index.js'
|
||||||
|
import { TableHoverActionsPlugin } from './plugins/TableHoverActionsPlugin/index.js'
|
||||||
import {
|
import {
|
||||||
OPEN_TABLE_DRAWER_COMMAND,
|
OPEN_TABLE_DRAWER_COMMAND,
|
||||||
TableContext,
|
TableContext,
|
||||||
@@ -29,6 +30,10 @@ export const TableFeatureClient = createClientFeature({
|
|||||||
Component: TableActionMenuPlugin,
|
Component: TableActionMenuPlugin,
|
||||||
position: 'floatingAnchorElem',
|
position: 'floatingAnchorElem',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
Component: TableHoverActionsPlugin,
|
||||||
|
position: 'floatingAnchorElem',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
providers: [TableContext],
|
providers: [TableContext],
|
||||||
slashMenu: {
|
slashMenu: {
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import { sanitizeFields } from 'payload'
|
|||||||
// eslint-disable-next-line payload/no-imports-from-exports-dir
|
// eslint-disable-next-line payload/no-imports-from-exports-dir
|
||||||
import { TableFeatureClient } from '../../exports/client/index.js'
|
import { TableFeatureClient } from '../../exports/client/index.js'
|
||||||
import { createServerFeature } from '../../utilities/createServerFeature.js'
|
import { createServerFeature } from '../../utilities/createServerFeature.js'
|
||||||
|
import { convertLexicalNodesToHTML } from '../converters/html/converter/index.js'
|
||||||
|
import { createNode } from '../typeUtilities.js'
|
||||||
|
|
||||||
const fields: Field[] = [
|
const fields: Field[] = [
|
||||||
{
|
{
|
||||||
@@ -41,15 +43,75 @@ export const EXPERIMENTAL_TableFeature = createServerFeature({
|
|||||||
return schemaMap
|
return schemaMap
|
||||||
},
|
},
|
||||||
nodes: [
|
nodes: [
|
||||||
{
|
createNode({
|
||||||
|
converters: {
|
||||||
|
html: {
|
||||||
|
converter: async ({ converters, node, parent, req }) => {
|
||||||
|
const childrenText = await convertLexicalNodesToHTML({
|
||||||
|
converters,
|
||||||
|
lexicalNodes: node.children,
|
||||||
|
parent: {
|
||||||
|
...node,
|
||||||
|
parent,
|
||||||
|
},
|
||||||
|
req,
|
||||||
|
})
|
||||||
|
return `<table class="lexical-table" style="border-collapse: collapse;">${childrenText}</table>`
|
||||||
|
},
|
||||||
|
nodeTypes: [TableNode.getType()],
|
||||||
|
},
|
||||||
|
},
|
||||||
node: TableNode,
|
node: TableNode,
|
||||||
},
|
}),
|
||||||
{
|
createNode({
|
||||||
|
converters: {
|
||||||
|
html: {
|
||||||
|
converter: async ({ converters, node, parent, req }) => {
|
||||||
|
const childrenText = await convertLexicalNodesToHTML({
|
||||||
|
converters,
|
||||||
|
lexicalNodes: node.children,
|
||||||
|
parent: {
|
||||||
|
...node,
|
||||||
|
parent,
|
||||||
|
},
|
||||||
|
req,
|
||||||
|
})
|
||||||
|
|
||||||
|
const tagName = node.headerState > 0 ? 'th' : 'td'
|
||||||
|
const headerStateClass = `lexical-table-cell-header-${node.headerState}`
|
||||||
|
const backgroundColor = node.backgroundColor
|
||||||
|
? `background-color: ${node.backgroundColor};`
|
||||||
|
: ''
|
||||||
|
const colSpan = node.colSpan > 1 ? `colspan="${node.colSpan}"` : ''
|
||||||
|
const rowSpan = node.rowSpan > 1 ? `rowspan="${node.rowSpan}"` : ''
|
||||||
|
|
||||||
|
return `<${tagName} class="lexical-table-cell ${headerStateClass}" style="border: 1px solid #ccc; padding: 8px; ${backgroundColor}" ${colSpan} ${rowSpan}>${childrenText}</${tagName}>`
|
||||||
|
},
|
||||||
|
nodeTypes: [TableCellNode.getType()],
|
||||||
|
},
|
||||||
|
},
|
||||||
node: TableCellNode,
|
node: TableCellNode,
|
||||||
},
|
}),
|
||||||
{
|
createNode({
|
||||||
|
converters: {
|
||||||
|
html: {
|
||||||
|
converter: async ({ converters, node, parent, req }) => {
|
||||||
|
const childrenText = await convertLexicalNodesToHTML({
|
||||||
|
converters,
|
||||||
|
lexicalNodes: node.children,
|
||||||
|
parent: {
|
||||||
|
...node,
|
||||||
|
parent,
|
||||||
|
},
|
||||||
|
req,
|
||||||
|
})
|
||||||
|
return `<tr class="lexical-table-row">${childrenText}</tr>`
|
||||||
|
},
|
||||||
|
nodeTypes: [TableRowNode.getType()],
|
||||||
|
},
|
||||||
|
},
|
||||||
node: TableRowNode,
|
node: TableRowNode,
|
||||||
},
|
}),
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -160,15 +160,19 @@ function TableActionMenu({
|
|||||||
const { y } = useScrollInfo()
|
const { y } = useScrollInfo()
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
return editor.registerMutationListener(TableCellNode, (nodeMutations) => {
|
return editor.registerMutationListener(
|
||||||
const nodeUpdated = nodeMutations.get(tableCellNode.getKey()) === 'updated'
|
TableCellNode,
|
||||||
|
(nodeMutations) => {
|
||||||
|
const nodeUpdated = nodeMutations.get(tableCellNode.getKey()) === 'updated'
|
||||||
|
|
||||||
if (nodeUpdated) {
|
if (nodeUpdated) {
|
||||||
editor.getEditorState().read(() => {
|
editor.getEditorState().read(() => {
|
||||||
updateTableCellNode(tableCellNode.getLatest())
|
updateTableCellNode(tableCellNode.getLatest())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
})
|
},
|
||||||
|
{ skipInitialization: true },
|
||||||
|
)
|
||||||
}, [editor, tableCellNode])
|
}, [editor, tableCellNode])
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
.TableCellResizer__resizer {
|
|
||||||
position: absolute;
|
|
||||||
}
|
|
||||||
@@ -20,9 +20,9 @@ import * as React from 'react'
|
|||||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
||||||
import { createPortal } from 'react-dom'
|
import { createPortal } from 'react-dom'
|
||||||
|
|
||||||
import type { PluginComponent, PluginComponentWithAnchor } from '../../../typesClient.js'
|
import type { PluginComponent } from '../../../typesClient.js'
|
||||||
|
|
||||||
import './index.scss'
|
import { useEditorConfigContext } from '../../../../lexical/config/client/EditorConfigProvider.js'
|
||||||
|
|
||||||
type MousePosition = {
|
type MousePosition = {
|
||||||
x: number
|
x: number
|
||||||
@@ -38,6 +38,7 @@ function TableCellResizer({ editor }: { editor: LexicalEditor }): JSX.Element {
|
|||||||
const targetRef = useRef<HTMLElement | null>(null)
|
const targetRef = useRef<HTMLElement | null>(null)
|
||||||
const resizerRef = useRef<HTMLDivElement | null>(null)
|
const resizerRef = useRef<HTMLDivElement | null>(null)
|
||||||
const tableRectRef = useRef<ClientRect | null>(null)
|
const tableRectRef = useRef<ClientRect | null>(null)
|
||||||
|
const editorConfig = useEditorConfigContext()
|
||||||
|
|
||||||
const mouseStartPosRef = useRef<MousePosition | null>(null)
|
const mouseStartPosRef = useRef<MousePosition | null>(null)
|
||||||
const [mouseCurrentPos, updateMouseCurrentPos] = useState<MousePosition | null>(null)
|
const [mouseCurrentPos, updateMouseCurrentPos] = useState<MousePosition | null>(null)
|
||||||
@@ -117,14 +118,18 @@ function TableCellResizer({ editor }: { editor: LexicalEditor }): JSX.Element {
|
|||||||
}, 0)
|
}, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener('mousemove', onMouseMove)
|
const removeRootListener = editor.registerRootListener((rootElement, prevRootElement) => {
|
||||||
document.addEventListener('mousedown', onMouseDown)
|
rootElement?.addEventListener('mousemove', onMouseMove)
|
||||||
document.addEventListener('mouseup', onMouseUp)
|
rootElement?.addEventListener('mousedown', onMouseDown)
|
||||||
|
rootElement?.addEventListener('mouseup', onMouseUp)
|
||||||
|
|
||||||
|
prevRootElement?.removeEventListener('mousemove', onMouseMove)
|
||||||
|
prevRootElement?.removeEventListener('mousedown', onMouseDown)
|
||||||
|
prevRootElement?.removeEventListener('mouseup', onMouseUp)
|
||||||
|
})
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
document.removeEventListener('mousemove', onMouseMove)
|
removeRootListener()
|
||||||
document.removeEventListener('mousedown', onMouseDown)
|
|
||||||
document.removeEventListener('mouseup', onMouseUp)
|
|
||||||
}
|
}
|
||||||
}, [activeCell, draggingDirection, editor, resetState])
|
}, [activeCell, draggingDirection, editor, resetState])
|
||||||
|
|
||||||
@@ -378,12 +383,12 @@ function TableCellResizer({ editor }: { editor: LexicalEditor }): JSX.Element {
|
|||||||
{activeCell != null && !isMouseDown && (
|
{activeCell != null && !isMouseDown && (
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<div
|
<div
|
||||||
className="TableCellResizer__resizer TableCellResizer__ui"
|
className={`${editorConfig.editorConfig.lexical.theme.tableCellResizer} TableCellResizer__ui`}
|
||||||
onMouseDown={toggleResize('right')}
|
onMouseDown={toggleResize('right')}
|
||||||
style={resizerStyles.right || undefined}
|
style={resizerStyles.right || undefined}
|
||||||
/>
|
/>
|
||||||
<div
|
<div
|
||||||
className="TableCellResizer__resizer TableCellResizer__ui"
|
className={`${editorConfig.editorConfig.lexical.theme.tableCellResizer} TableCellResizer__ui`}
|
||||||
onMouseDown={toggleResize('bottom')}
|
onMouseDown={toggleResize('bottom')}
|
||||||
style={resizerStyles.bottom || undefined}
|
style={resizerStyles.bottom || undefined}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -0,0 +1,243 @@
|
|||||||
|
/**
|
||||||
|
* 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 { TableCellNode, TableRowNode } from '@lexical/table'
|
||||||
|
import type { EditorConfig, NodeKey } from 'lexical'
|
||||||
|
import type { JSX } from 'react'
|
||||||
|
|
||||||
|
import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext'
|
||||||
|
import {
|
||||||
|
$getTableColumnIndexFromTableCellNode,
|
||||||
|
$getTableRowIndexFromTableCellNode,
|
||||||
|
$insertTableColumn__EXPERIMENTAL,
|
||||||
|
$insertTableRow__EXPERIMENTAL,
|
||||||
|
$isTableCellNode,
|
||||||
|
$isTableNode,
|
||||||
|
TableNode,
|
||||||
|
} from '@lexical/table'
|
||||||
|
import { $findMatchingParent, mergeRegister } from '@lexical/utils'
|
||||||
|
import { $getNearestNodeFromDOMNode } from 'lexical'
|
||||||
|
import { useEffect, useRef, useState } from 'react'
|
||||||
|
import * as React from 'react'
|
||||||
|
import { createPortal } from 'react-dom'
|
||||||
|
|
||||||
|
import { useEditorConfigContext } from '../../../../lexical/config/client/EditorConfigProvider.js'
|
||||||
|
import { useDebounce } from '../../utils/useDebounce.js'
|
||||||
|
|
||||||
|
const BUTTON_WIDTH_PX = 20
|
||||||
|
|
||||||
|
function TableHoverActionsContainer({ anchorElem }: { anchorElem: HTMLElement }): JSX.Element {
|
||||||
|
const [editor] = useLexicalComposerContext()
|
||||||
|
const editorConfig = useEditorConfigContext()
|
||||||
|
const [isShownRow, setShownRow] = useState<boolean>(false)
|
||||||
|
const [isShownColumn, setShownColumn] = useState<boolean>(false)
|
||||||
|
const [shouldListenMouseMove, setShouldListenMouseMove] = useState<boolean>(false)
|
||||||
|
const [position, setPosition] = useState({})
|
||||||
|
const codeSetRef = useRef<Set<NodeKey>>(new Set())
|
||||||
|
const tableDOMNodeRef = useRef<HTMLElement | null>(null)
|
||||||
|
|
||||||
|
const debouncedOnMouseMove = useDebounce(
|
||||||
|
(event: MouseEvent) => {
|
||||||
|
const { isOutside, tableDOMNode } = getMouseInfo(event, editorConfig.editorConfig?.lexical)
|
||||||
|
|
||||||
|
if (isOutside) {
|
||||||
|
setShownRow(false)
|
||||||
|
setShownColumn(false)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tableDOMNode) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tableDOMNodeRef.current = tableDOMNode
|
||||||
|
|
||||||
|
let hoveredRowNode: TableCellNode | null = null
|
||||||
|
let hoveredColumnNode: TableCellNode | null = null
|
||||||
|
let tableDOMElement: HTMLElement | null = null
|
||||||
|
|
||||||
|
editor.update(() => {
|
||||||
|
const maybeTableCell = $getNearestNodeFromDOMNode(tableDOMNode)
|
||||||
|
|
||||||
|
if ($isTableCellNode(maybeTableCell)) {
|
||||||
|
const table = $findMatchingParent(maybeTableCell, (node) => $isTableNode(node))
|
||||||
|
if (!$isTableNode(table)) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tableDOMElement = editor.getElementByKey(table?.getKey())
|
||||||
|
|
||||||
|
if (tableDOMElement) {
|
||||||
|
const rowCount = table.getChildrenSize()
|
||||||
|
const colCount =
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
|
||||||
|
((table as TableNode).getChildAtIndex(0) as TableRowNode)?.getChildrenSize()
|
||||||
|
|
||||||
|
const rowIndex = $getTableRowIndexFromTableCellNode(maybeTableCell)
|
||||||
|
const colIndex = $getTableColumnIndexFromTableCellNode(maybeTableCell)
|
||||||
|
|
||||||
|
if (rowIndex === rowCount - 1) {
|
||||||
|
hoveredRowNode = maybeTableCell
|
||||||
|
} else if (colIndex === colCount - 1) {
|
||||||
|
hoveredColumnNode = maybeTableCell
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (tableDOMElement) {
|
||||||
|
const {
|
||||||
|
bottom: tableElemBottom,
|
||||||
|
height: tableElemHeight,
|
||||||
|
right: tableElemRight,
|
||||||
|
width: tableElemWidth,
|
||||||
|
x: tableElemX,
|
||||||
|
y: tableElemY,
|
||||||
|
} = (tableDOMElement as HTMLTableElement).getBoundingClientRect()
|
||||||
|
|
||||||
|
const { left: editorElemLeft, y: editorElemY } = anchorElem.getBoundingClientRect()
|
||||||
|
|
||||||
|
if (hoveredRowNode) {
|
||||||
|
setShownColumn(false)
|
||||||
|
setShownRow(true)
|
||||||
|
setPosition({
|
||||||
|
height: BUTTON_WIDTH_PX,
|
||||||
|
left: tableElemX - editorElemLeft,
|
||||||
|
top: tableElemBottom - editorElemY + 5,
|
||||||
|
width: tableElemWidth,
|
||||||
|
})
|
||||||
|
} else if (hoveredColumnNode) {
|
||||||
|
setShownColumn(true)
|
||||||
|
setShownRow(false)
|
||||||
|
setPosition({
|
||||||
|
height: tableElemHeight,
|
||||||
|
left: tableElemRight - editorElemLeft + 5,
|
||||||
|
top: tableElemY - editorElemY,
|
||||||
|
width: BUTTON_WIDTH_PX,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
50,
|
||||||
|
250,
|
||||||
|
)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!shouldListenMouseMove) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('mousemove', debouncedOnMouseMove)
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
setShownRow(false)
|
||||||
|
setShownColumn(false)
|
||||||
|
|
||||||
|
document.removeEventListener('mousemove', debouncedOnMouseMove)
|
||||||
|
}
|
||||||
|
}, [shouldListenMouseMove, debouncedOnMouseMove])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
return mergeRegister(
|
||||||
|
editor.registerMutationListener(
|
||||||
|
TableNode,
|
||||||
|
(mutations) => {
|
||||||
|
editor.getEditorState().read(() => {
|
||||||
|
for (const [key, type] of mutations) {
|
||||||
|
switch (type) {
|
||||||
|
case 'created':
|
||||||
|
codeSetRef.current.add(key)
|
||||||
|
setShouldListenMouseMove(codeSetRef.current.size > 0)
|
||||||
|
break
|
||||||
|
|
||||||
|
case 'destroyed':
|
||||||
|
codeSetRef.current.delete(key)
|
||||||
|
setShouldListenMouseMove(codeSetRef.current.size > 0)
|
||||||
|
break
|
||||||
|
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
{ skipInitialization: false },
|
||||||
|
),
|
||||||
|
)
|
||||||
|
}, [editor])
|
||||||
|
|
||||||
|
const insertAction = (insertRow: boolean) => {
|
||||||
|
editor.update(() => {
|
||||||
|
if (tableDOMNodeRef.current) {
|
||||||
|
const maybeTableNode = $getNearestNodeFromDOMNode(tableDOMNodeRef.current)
|
||||||
|
maybeTableNode?.selectEnd()
|
||||||
|
if (insertRow) {
|
||||||
|
$insertTableRow__EXPERIMENTAL()
|
||||||
|
setShownRow(false)
|
||||||
|
} else {
|
||||||
|
$insertTableColumn__EXPERIMENTAL()
|
||||||
|
setShownColumn(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{isShownRow && (
|
||||||
|
<button
|
||||||
|
className={editorConfig.editorConfig.lexical.theme.tableAddRows}
|
||||||
|
onClick={() => insertAction(true)}
|
||||||
|
style={{ ...position }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{isShownColumn && (
|
||||||
|
<button
|
||||||
|
className={editorConfig.editorConfig.lexical.theme.tableAddColumns}
|
||||||
|
onClick={() => insertAction(false)}
|
||||||
|
style={{ ...position }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getMouseInfo(
|
||||||
|
event: MouseEvent,
|
||||||
|
editorConfig: EditorConfig,
|
||||||
|
): {
|
||||||
|
isOutside: boolean
|
||||||
|
tableDOMNode: HTMLElement | null
|
||||||
|
} {
|
||||||
|
const target = event.target
|
||||||
|
|
||||||
|
if (target && target instanceof HTMLElement) {
|
||||||
|
const tableDOMNode = target.closest<HTMLElement>(
|
||||||
|
`td.${editorConfig.theme.tableCell}, th.${editorConfig.theme.tableCell}`,
|
||||||
|
)
|
||||||
|
|
||||||
|
const isOutside = !(
|
||||||
|
tableDOMNode ||
|
||||||
|
target.closest<HTMLElement>(`button.${editorConfig.theme.tableAddRows}`) ||
|
||||||
|
target.closest<HTMLElement>(`button.${editorConfig.theme.tableAddColumns}`) ||
|
||||||
|
target.closest<HTMLElement>(`div.${editorConfig.theme.tableCellResizer}`)
|
||||||
|
)
|
||||||
|
|
||||||
|
return { isOutside, tableDOMNode }
|
||||||
|
} else {
|
||||||
|
return { isOutside: true, tableDOMNode: null }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function TableHoverActionsPlugin({
|
||||||
|
anchorElem = document.body,
|
||||||
|
}: {
|
||||||
|
anchorElem?: HTMLElement
|
||||||
|
}): React.ReactPortal | null {
|
||||||
|
return createPortal(<TableHoverActionsContainer anchorElem={anchorElem} />, anchorElem)
|
||||||
|
}
|
||||||
@@ -4,11 +4,11 @@
|
|||||||
&__table {
|
&__table {
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
border-spacing: 0;
|
border-spacing: 0;
|
||||||
max-width: 100%;
|
|
||||||
overflow-y: scroll;
|
overflow-y: scroll;
|
||||||
|
overflow-x: scroll;
|
||||||
table-layout: fixed;
|
table-layout: fixed;
|
||||||
width: calc(100% - 25px);
|
width: max-content;
|
||||||
margin: 30px 0;
|
margin: 0 25px 30px 0;
|
||||||
|
|
||||||
::selection {
|
::selection {
|
||||||
background: rgba(172, 206, 247);
|
background: rgba(172, 206, 247);
|
||||||
@@ -82,36 +82,41 @@
|
|||||||
|
|
||||||
&__tableAddColumns {
|
&__tableAddColumns {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
background-color: var(--theme-elevation-100);
|
||||||
width: 20px;
|
|
||||||
background-color: #eee;
|
|
||||||
height: 100%;
|
height: 100%;
|
||||||
right: 0;
|
|
||||||
animation: table-controls 0.2s ease;
|
animation: table-controls 0.2s ease;
|
||||||
border: 0;
|
border: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__tableAddColumns:hover {
|
&__tableAddColumns:after, &__tableAddRows:after {
|
||||||
background-color: #c9dbf0;
|
background-image: url(../../../../lexical/ui/icons/Add/index.svg);
|
||||||
|
background-size: contain;
|
||||||
|
background-position: center;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
display: block;
|
||||||
|
content: ' ';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
opacity: 0.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__tableAddColumns:hover, &__tableAddRows:hover {
|
||||||
|
background-color: var(--theme-elevation-200);
|
||||||
}
|
}
|
||||||
|
|
||||||
&__tableAddRows {
|
&__tableAddRows {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
bottom: -25px;
|
|
||||||
width: calc(100% - 25px);
|
width: calc(100% - 25px);
|
||||||
background-color: #eee;
|
background-color: var(--theme-elevation-100);
|
||||||
height: 20px;
|
|
||||||
left: 0;
|
|
||||||
animation: table-controls 0.2s ease;
|
animation: table-controls 0.2s ease;
|
||||||
border: 0;
|
border: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__tableAddRows:hover {
|
|
||||||
background-color: #c9dbf0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes table-controls {
|
@keyframes table-controls {
|
||||||
0% {
|
0% {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
@@ -162,5 +167,9 @@ html[data-theme='dark'] {
|
|||||||
&__tableCellHeader {
|
&__tableCellHeader {
|
||||||
background-color: var(--theme-elevation-50);
|
background-color: var(--theme-elevation-50);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__tableAddColumns:after, &__tableAddRows:after {
|
||||||
|
background-image: url(../../../../lexical/ui/icons/Add/light.svg);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,244 @@
|
|||||||
|
// Copied & modified from https://github.com/lodash/lodash/blob/main/src/debounce.ts
|
||||||
|
/*
|
||||||
|
The MIT License
|
||||||
|
|
||||||
|
Copyright JS Foundation and other contributors <https://js.foundation/>
|
||||||
|
|
||||||
|
Based on Underscore.js, copyright Jeremy Ashkenas,
|
||||||
|
DocumentCloud and Investigative Reporters & Editors <http://underscorejs.org/>
|
||||||
|
|
||||||
|
This software consists of voluntary contributions made by many
|
||||||
|
individuals. For exact contribution history, see the revision history
|
||||||
|
available at https://github.com/lodash/lodash
|
||||||
|
|
||||||
|
The following license applies to all parts of this software except as
|
||||||
|
documented below:
|
||||||
|
|
||||||
|
====
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
====
|
||||||
|
|
||||||
|
Copyright and related rights for sample code are waived via CC0. Sample
|
||||||
|
code is defined as all source code displayed within the prose of the
|
||||||
|
documentation.
|
||||||
|
|
||||||
|
CC0: http://creativecommons.org/publicdomain/zero/1.0/
|
||||||
|
|
||||||
|
====
|
||||||
|
|
||||||
|
Files located in the node_modules and vendor directories are externally
|
||||||
|
maintained libraries used by this software which have their own
|
||||||
|
licenses; we recommend you read them, as their terms may differ from the
|
||||||
|
terms above.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** Error message constants. */
|
||||||
|
const FUNC_ERROR_TEXT = 'Expected a function'
|
||||||
|
|
||||||
|
/* Built-in method references for those with the same name as other `lodash` methods. */
|
||||||
|
const nativeMax = Math.max,
|
||||||
|
nativeMin = Math.min
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a debounced function that delays invoking `func` until after `wait`
|
||||||
|
* milliseconds have elapsed since the last time the debounced function was
|
||||||
|
* invoked. The debounced function comes with a `cancel` method to cancel
|
||||||
|
* delayed `func` invocations and a `flush` method to immediately invoke them.
|
||||||
|
* Provide `options` to indicate whether `func` should be invoked on the
|
||||||
|
* leading and/or trailing edge of the `wait` timeout. The `func` is invoked
|
||||||
|
* with the last arguments provided to the debounced function. Subsequent
|
||||||
|
* calls to the debounced function return the result of the last `func`
|
||||||
|
* invocation.
|
||||||
|
*
|
||||||
|
* **Note:** If `leading` and `trailing` options are `true`, `func` is
|
||||||
|
* invoked on the trailing edge of the timeout only if the debounced function
|
||||||
|
* is invoked more than once during the `wait` timeout.
|
||||||
|
*
|
||||||
|
* If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
|
||||||
|
* until to the next tick, similar to `setTimeout` with a timeout of `0`.
|
||||||
|
*
|
||||||
|
* See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
|
||||||
|
* for details over the differences between `_.debounce` and `_.throttle`.
|
||||||
|
*
|
||||||
|
* @static
|
||||||
|
* @memberOf _
|
||||||
|
* @since 0.1.0
|
||||||
|
* @category Function
|
||||||
|
* @param {Function} func The function to debounce.
|
||||||
|
* @param {number} [wait=0] The number of milliseconds to delay.
|
||||||
|
* @param {Object} [options={}] The options object.
|
||||||
|
* @param {boolean} [options.leading=false]
|
||||||
|
* Specify invoking on the leading edge of the timeout.
|
||||||
|
* @param {number} [options.maxWait]
|
||||||
|
* The maximum time `func` is allowed to be delayed before it's invoked.
|
||||||
|
* @param {boolean} [options.trailing=true]
|
||||||
|
* Specify invoking on the trailing edge of the timeout.
|
||||||
|
* @returns {Function} Returns the new debounced function.
|
||||||
|
* @example
|
||||||
|
*
|
||||||
|
* // Avoid costly calculations while the window size is in flux.
|
||||||
|
* jQuery(window).on('resize', _.debounce(calculateLayout, 150));
|
||||||
|
*
|
||||||
|
* // Invoke `sendMail` when clicked, debouncing subsequent calls.
|
||||||
|
* jQuery(element).on('click', _.debounce(sendMail, 300, {
|
||||||
|
* 'leading': true,
|
||||||
|
* 'trailing': false
|
||||||
|
* }));
|
||||||
|
*
|
||||||
|
* // Ensure `batchLog` is invoked once after 1 second of debounced calls.
|
||||||
|
* var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
|
||||||
|
* var source = new EventSource('/stream');
|
||||||
|
* jQuery(source).on('message', debounced);
|
||||||
|
*
|
||||||
|
* // Cancel the trailing debounced invocation.
|
||||||
|
* jQuery(window).on('popstate', debounced.cancel);
|
||||||
|
*/
|
||||||
|
function debounce(func, wait, options) {
|
||||||
|
let lastArgs,
|
||||||
|
lastThis,
|
||||||
|
maxWait,
|
||||||
|
result,
|
||||||
|
timerId,
|
||||||
|
lastCallTime,
|
||||||
|
lastInvokeTime = 0,
|
||||||
|
leading = false,
|
||||||
|
maxing = false,
|
||||||
|
trailing = true
|
||||||
|
|
||||||
|
if (typeof func != 'function') {
|
||||||
|
throw new TypeError(FUNC_ERROR_TEXT)
|
||||||
|
}
|
||||||
|
wait = wait || 0
|
||||||
|
if (typeof options === 'object') {
|
||||||
|
leading = !!options.leading
|
||||||
|
maxing = 'maxWait' in options
|
||||||
|
maxWait = maxing ? nativeMax(options.maxWait || 0, wait) : maxWait
|
||||||
|
trailing = 'trailing' in options ? !!options.trailing : trailing
|
||||||
|
}
|
||||||
|
|
||||||
|
function invokeFunc(time) {
|
||||||
|
const args = lastArgs,
|
||||||
|
thisArg = lastThis
|
||||||
|
|
||||||
|
lastArgs = lastThis = undefined
|
||||||
|
lastInvokeTime = time
|
||||||
|
result = func.apply(thisArg, args)
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
function leadingEdge(time) {
|
||||||
|
// Reset any `maxWait` timer.
|
||||||
|
lastInvokeTime = time
|
||||||
|
// Start the timer for the trailing edge.
|
||||||
|
timerId = setTimeout(timerExpired, wait)
|
||||||
|
// Invoke the leading edge.
|
||||||
|
return leading ? invokeFunc(time) : result
|
||||||
|
}
|
||||||
|
|
||||||
|
function remainingWait(time) {
|
||||||
|
const timeSinceLastCall = time - lastCallTime,
|
||||||
|
timeSinceLastInvoke = time - lastInvokeTime,
|
||||||
|
timeWaiting = wait - timeSinceLastCall
|
||||||
|
|
||||||
|
return maxing ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldInvoke(time) {
|
||||||
|
const timeSinceLastCall = time - lastCallTime,
|
||||||
|
timeSinceLastInvoke = time - lastInvokeTime
|
||||||
|
|
||||||
|
// Either this is the first call, activity has stopped and we're at the
|
||||||
|
// trailing edge, the system time has gone backwards and we're treating
|
||||||
|
// it as the trailing edge, or we've hit the `maxWait` limit.
|
||||||
|
return (
|
||||||
|
lastCallTime === undefined ||
|
||||||
|
timeSinceLastCall >= wait ||
|
||||||
|
timeSinceLastCall < 0 ||
|
||||||
|
(maxing && timeSinceLastInvoke >= maxWait)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function timerExpired() {
|
||||||
|
const time = Date.now()
|
||||||
|
if (shouldInvoke(time)) {
|
||||||
|
return trailingEdge(time)
|
||||||
|
}
|
||||||
|
// Restart the timer.
|
||||||
|
timerId = setTimeout(timerExpired, remainingWait(time))
|
||||||
|
}
|
||||||
|
|
||||||
|
function trailingEdge(time) {
|
||||||
|
timerId = undefined
|
||||||
|
|
||||||
|
// Only invoke if we have `lastArgs` which means `func` has been
|
||||||
|
// debounced at least once.
|
||||||
|
if (trailing && lastArgs) {
|
||||||
|
return invokeFunc(time)
|
||||||
|
}
|
||||||
|
lastArgs = lastThis = undefined
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
function cancel() {
|
||||||
|
if (timerId !== undefined) {
|
||||||
|
clearTimeout(timerId)
|
||||||
|
}
|
||||||
|
lastInvokeTime = 0
|
||||||
|
lastArgs = lastCallTime = lastThis = timerId = undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
function flush() {
|
||||||
|
return timerId === undefined ? result : trailingEdge(Date.now())
|
||||||
|
}
|
||||||
|
|
||||||
|
function debounced() {
|
||||||
|
const time = Date.now(),
|
||||||
|
isInvoking = shouldInvoke(time)
|
||||||
|
|
||||||
|
// eslint-disable-next-line prefer-rest-params
|
||||||
|
lastArgs = arguments
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
||||||
|
lastThis = this
|
||||||
|
lastCallTime = time
|
||||||
|
|
||||||
|
if (isInvoking) {
|
||||||
|
if (timerId === undefined) {
|
||||||
|
return leadingEdge(lastCallTime)
|
||||||
|
}
|
||||||
|
if (maxing) {
|
||||||
|
// Handle invocations in a tight loop.
|
||||||
|
clearTimeout(timerId)
|
||||||
|
timerId = setTimeout(timerExpired, wait)
|
||||||
|
return invokeFunc(lastCallTime)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (timerId === undefined) {
|
||||||
|
timerId = setTimeout(timerExpired, wait)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
debounced.cancel = cancel
|
||||||
|
debounced.flush = flush
|
||||||
|
return debounced
|
||||||
|
}
|
||||||
|
|
||||||
|
export default debounce
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import { useMemo, useRef } from 'react'
|
||||||
|
|
||||||
|
import debounce from './debounce.js'
|
||||||
|
|
||||||
|
export function useDebounce<T extends (...args: never[]) => void>(
|
||||||
|
fn: T,
|
||||||
|
ms: number,
|
||||||
|
maxWait?: number,
|
||||||
|
) {
|
||||||
|
const funcRef = useRef<T | null>(null)
|
||||||
|
funcRef.current = fn
|
||||||
|
|
||||||
|
return useMemo(
|
||||||
|
() =>
|
||||||
|
debounce(
|
||||||
|
(...args: Parameters<T>) => {
|
||||||
|
if (funcRef.current) {
|
||||||
|
funcRef.current(...args)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
ms,
|
||||||
|
{ maxWait },
|
||||||
|
),
|
||||||
|
[ms, maxWait],
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -77,8 +77,19 @@ function startsWithSeparator(textContent: string): boolean {
|
|||||||
return isSeparator(textContent[0])
|
return isSeparator(textContent[0])
|
||||||
}
|
}
|
||||||
|
|
||||||
function startsWithFullStop(textContent: string): boolean {
|
/**
|
||||||
return /^\.[a-z\d]+/i.test(textContent)
|
* Check if the text content starts with a fullstop followed by a top-level domain.
|
||||||
|
* Meaning if the text content can be a beginning of a top level domain.
|
||||||
|
* @param textContent
|
||||||
|
* @param isEmail
|
||||||
|
* @returns boolean
|
||||||
|
*/
|
||||||
|
function startsWithTLD(textContent: string, isEmail: boolean): boolean {
|
||||||
|
if (isEmail) {
|
||||||
|
return /^\.[a-z]{2,}/i.test(textContent)
|
||||||
|
} else {
|
||||||
|
return /^\.[a-z0-9]+/i.test(textContent)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function isPreviousNodeValid(node: LexicalNode): boolean {
|
function isPreviousNodeValid(node: LexicalNode): boolean {
|
||||||
@@ -343,13 +354,15 @@ function handleBadNeighbors(
|
|||||||
const nextSibling = textNode.getNextSibling()
|
const nextSibling = textNode.getNextSibling()
|
||||||
const text = textNode.getTextContent()
|
const text = textNode.getTextContent()
|
||||||
|
|
||||||
if (
|
if ($isAutoLinkNode(previousSibling)) {
|
||||||
$isAutoLinkNode(previousSibling) &&
|
const isEmailURI = previousSibling.getFields()?.url
|
||||||
(!startsWithSeparator(text) || startsWithFullStop(text))
|
? previousSibling.getFields()?.url?.startsWith('mailto:')
|
||||||
) {
|
: false
|
||||||
previousSibling.append(textNode)
|
if (!startsWithSeparator(text) || startsWithTLD(text, isEmailURI)) {
|
||||||
handleLinkEdit(previousSibling, matchers, onChange)
|
previousSibling.append(textNode)
|
||||||
onChange(null, previousSibling.getFields()?.url ?? null)
|
handleLinkEdit(previousSibling, matchers, onChange)
|
||||||
|
onChange(null, previousSibling.getFields()?.url ?? null)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($isAutoLinkNode(nextSibling) && !endsWithSeparator(text)) {
|
if ($isAutoLinkNode(nextSibling) && !endsWithSeparator(text)) {
|
||||||
|
|||||||
@@ -99,6 +99,7 @@ export function convertParagraphNode(
|
|||||||
format: '',
|
format: '',
|
||||||
indent: 0,
|
indent: 0,
|
||||||
textFormat: 0,
|
textFormat: 0,
|
||||||
|
textStyle: '',
|
||||||
version: 1,
|
version: 1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -101,7 +101,7 @@ html[data-theme='dark'] {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
> .editor-placeholder {
|
> .editor-scroller > .editor > div > .editor-placeholder {
|
||||||
top: calc(var(--base) * 1.25);
|
top: calc(var(--base) * 1.25);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,12 +25,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&--show-gutter {
|
&--show-gutter {
|
||||||
> .rich-text-lexical__wrap > .editor-container > .editor-placeholder {
|
> .rich-text-lexical__wrap > .editor-container > .editor-scroller > .editor > div > .editor-placeholder {
|
||||||
left: 3rem;
|
left: 3rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&:not(&--show-gutter) > .rich-text-lexical__wrap > .editor-container > .editor-placeholder {
|
&:not(&--show-gutter) > .rich-text-lexical__wrap > .editor-container > .editor-scroller > .editor > div > .editor-placeholder {
|
||||||
left: 0;
|
left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import { HistoryPlugin } from '@lexical/react/LexicalHistoryPlugin.js'
|
|||||||
import { OnChangePlugin } from '@lexical/react/LexicalOnChangePlugin.js'
|
import { OnChangePlugin } from '@lexical/react/LexicalOnChangePlugin.js'
|
||||||
import { RichTextPlugin } from '@lexical/react/LexicalRichTextPlugin.js'
|
import { RichTextPlugin } from '@lexical/react/LexicalRichTextPlugin.js'
|
||||||
import { TabIndentationPlugin } from '@lexical/react/LexicalTabIndentationPlugin.js'
|
import { TabIndentationPlugin } from '@lexical/react/LexicalTabIndentationPlugin.js'
|
||||||
import { useTranslation } from '@payloadcms/ui'
|
|
||||||
import { BLUR_COMMAND, COMMAND_PRIORITY_LOW, FOCUS_COMMAND } from 'lexical'
|
import { BLUR_COMMAND, COMMAND_PRIORITY_LOW, FOCUS_COMMAND } from 'lexical'
|
||||||
import * as React from 'react'
|
import * as React from 'react'
|
||||||
import { useEffect, useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
@@ -29,7 +28,6 @@ export const LexicalEditor: React.FC<
|
|||||||
const { editorConfig, editorContainerRef, onChange } = props
|
const { editorConfig, editorContainerRef, onChange } = props
|
||||||
const editorConfigContext = useEditorConfigContext()
|
const editorConfigContext = useEditorConfigContext()
|
||||||
const [editor] = useLexicalComposerContext()
|
const [editor] = useLexicalComposerContext()
|
||||||
const { t } = useTranslation<{}, string>()
|
|
||||||
|
|
||||||
const [floatingAnchorElem, setFloatingAnchorElem] = useState<HTMLDivElement | null>(null)
|
const [floatingAnchorElem, setFloatingAnchorElem] = useState<HTMLDivElement | null>(null)
|
||||||
const onRef = (_floatingAnchorElem: HTMLDivElement) => {
|
const onRef = (_floatingAnchorElem: HTMLDivElement) => {
|
||||||
@@ -122,7 +120,6 @@ export const LexicalEditor: React.FC<
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
placeholder={<p className="editor-placeholder">{t('lexical:general:placeholder')}</p>}
|
|
||||||
/>
|
/>
|
||||||
<OnChangePlugin
|
<OnChangePlugin
|
||||||
// Selection changes can be ignored here, reducing the
|
// Selection changes can be ignored here, reducing the
|
||||||
|
|||||||
@@ -1,10 +1,19 @@
|
|||||||
import type { JSX } from 'react'
|
import type { JSX } from 'react'
|
||||||
|
|
||||||
import { ContentEditable } from '@lexical/react/LexicalContentEditable.js'
|
import { ContentEditable } from '@lexical/react/LexicalContentEditable.js'
|
||||||
|
import { useTranslation } from '@payloadcms/ui'
|
||||||
import * as React from 'react'
|
import * as React from 'react'
|
||||||
|
|
||||||
import './ContentEditable.scss'
|
import './ContentEditable.scss'
|
||||||
|
|
||||||
export function LexicalContentEditable({ className }: { className?: string }): JSX.Element {
|
export function LexicalContentEditable({ className }: { className?: string }): JSX.Element {
|
||||||
return <ContentEditable className={className ?? 'ContentEditable__root'} />
|
const { t } = useTranslation<{}, string>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ContentEditable
|
||||||
|
aria-placeholder={t('lexical:general:placeholder')}
|
||||||
|
className={className ?? 'ContentEditable__root'}
|
||||||
|
placeholder={<p className="editor-placeholder">{t('lexical:general:placeholder')}</p>}
|
||||||
|
/>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ export const defaultRichTextValue: SerializedEditorState = {
|
|||||||
format: '',
|
format: '',
|
||||||
indent: 0,
|
indent: 0,
|
||||||
textFormat: 0,
|
textFormat: 0,
|
||||||
|
textStyle: '',
|
||||||
version: 1,
|
version: 1,
|
||||||
} as SerializedParagraphNode,
|
} as SerializedParagraphNode,
|
||||||
],
|
],
|
||||||
|
|||||||
304
pnpm-lock.yaml
generated
304
pnpm-lock.yaml
generated
@@ -1177,35 +1177,35 @@ importers:
|
|||||||
specifier: 2.0.0-beta.0
|
specifier: 2.0.0-beta.0
|
||||||
version: 2.0.0-beta.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)
|
version: 2.0.0-beta.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)
|
||||||
'@lexical/headless':
|
'@lexical/headless':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/link':
|
'@lexical/link':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/list':
|
'@lexical/list':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/mark':
|
'@lexical/mark':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/markdown':
|
'@lexical/markdown':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/react':
|
'@lexical/react':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)(yjs@13.6.18)
|
version: 0.17.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)(yjs@13.6.18)
|
||||||
'@lexical/rich-text':
|
'@lexical/rich-text':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/selection':
|
'@lexical/selection':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/table':
|
'@lexical/table':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/utils':
|
'@lexical/utils':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@types/uuid':
|
'@types/uuid':
|
||||||
specifier: 10.0.0
|
specifier: 10.0.0
|
||||||
version: 10.0.0
|
version: 10.0.0
|
||||||
@@ -1216,8 +1216,8 @@ importers:
|
|||||||
specifier: 2.0.3
|
specifier: 2.0.3
|
||||||
version: 2.0.3
|
version: 2.0.3
|
||||||
lexical:
|
lexical:
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
react:
|
react:
|
||||||
specifier: ^19.0.0-rc-6230622a1a-20240610
|
specifier: ^19.0.0-rc-6230622a1a-20240610
|
||||||
version: 19.0.0-rc-fb9a90fa48-20240614
|
version: 19.0.0-rc-fb9a90fa48-20240614
|
||||||
@@ -1232,8 +1232,8 @@ importers:
|
|||||||
version: 10.0.0
|
version: 10.0.0
|
||||||
devDependencies:
|
devDependencies:
|
||||||
'@lexical/eslint-plugin':
|
'@lexical/eslint-plugin':
|
||||||
specifier: ' 0.16.1'
|
specifier: 0.17.0
|
||||||
version: 0.16.1(eslint@9.6.0)
|
version: 0.17.0(eslint@9.6.0)
|
||||||
'@payloadcms/eslint-config':
|
'@payloadcms/eslint-config':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../eslint-config
|
version: link:../eslint-config
|
||||||
@@ -1555,11 +1555,11 @@ importers:
|
|||||||
specifier: ^3.525.0
|
specifier: ^3.525.0
|
||||||
version: 3.614.0
|
version: 3.614.0
|
||||||
'@lexical/headless':
|
'@lexical/headless':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@lexical/markdown':
|
'@lexical/markdown':
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
'@payloadcms/db-mongodb':
|
'@payloadcms/db-mongodb':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../packages/db-mongodb
|
version: link:../packages/db-mongodb
|
||||||
@@ -1690,8 +1690,8 @@ importers:
|
|||||||
specifier: 4.0.0
|
specifier: 4.0.0
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
lexical:
|
lexical:
|
||||||
specifier: 0.16.1
|
specifier: 0.17.0
|
||||||
version: 0.16.1
|
version: 0.17.0
|
||||||
payload:
|
payload:
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../packages/payload
|
version: link:../packages/payload
|
||||||
@@ -5693,154 +5693,154 @@ packages:
|
|||||||
resolution: {integrity: sha512-dfLbk+PwWvFzSxwk3n5ySL0hfBog779o8h68wK/7/APo/7cgyWp5jcXockbxdk5kFRkbeXWm4Fbi9FrdN381sA==}
|
resolution: {integrity: sha512-dfLbk+PwWvFzSxwk3n5ySL0hfBog779o8h68wK/7/APo/7cgyWp5jcXockbxdk5kFRkbeXWm4Fbi9FrdN381sA==}
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/clipboard@0.16.1:
|
/@lexical/clipboard@0.17.0:
|
||||||
resolution: {integrity: sha512-0dWs/SwKS5KPpuf6fUVVt9vSCl6HAqcDGhSITw/okv0rrIlXTUT6WhVsMJtXfFxTyVvwMeOecJHvQH3i/jRQtA==}
|
resolution: {integrity: sha512-wYtC6VJhuSxUZc69VTU+vBgzB4HQqhve2hLrr3v+3tR2aimx3KnKphCCP1TexCntxpEnOTPXafEgpOW/EVQE+Q==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/html': 0.16.1
|
'@lexical/html': 0.17.0
|
||||||
'@lexical/list': 0.16.1
|
'@lexical/list': 0.17.0
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/code@0.16.1:
|
/@lexical/code@0.17.0:
|
||||||
resolution: {integrity: sha512-pOC28rRZ2XkmI2nIJm50DbKaCJtk5D0o7r6nORYp4i0z+lxt5Sf2m82DL9ksUHJRqKy87pwJDpoWvJ2SAI0ohw==}
|
resolution: {integrity: sha512-8zrgHzf27aYySfUVeSKw8YP/LkRlXHSwD03BKlkSZAb4HX/WC60SGmdXUhtyTIBucqe0pnuGsRYfR9euD0/tfw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
prismjs: 1.29.0
|
prismjs: 1.29.0
|
||||||
|
|
||||||
/@lexical/devtools-core@0.16.1(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614):
|
/@lexical/devtools-core@0.17.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614):
|
||||||
resolution: {integrity: sha512-8CvGERGL7ySDVGLU+YPeq+JupIXsOFlXa3EuJ88koLKqXxYenwMleZgGqayFp6lCP78xqPKnATVeoOZUt/NabQ==}
|
resolution: {integrity: sha512-0ftqWsoCb96oTc8Ok+uvjGAXZpsN9oc6ml3d46BdufdZyxHXC4qU3YVoPfLkgAHzH+4fQlNypu7u3Ym3dZ2rJg==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ^19.0.0-rc-6230622a1a-20240610
|
react: ^19.0.0-rc-6230622a1a-20240610
|
||||||
react-dom: ^19.0.0-rc-6230622a1a-20240610
|
react-dom: ^19.0.0-rc-6230622a1a-20240610
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/html': 0.16.1
|
'@lexical/html': 0.17.0
|
||||||
'@lexical/link': 0.16.1
|
'@lexical/link': 0.17.0
|
||||||
'@lexical/mark': 0.16.1
|
'@lexical/mark': 0.17.0
|
||||||
'@lexical/table': 0.16.1
|
'@lexical/table': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
react: 19.0.0-rc-fb9a90fa48-20240614
|
react: 19.0.0-rc-fb9a90fa48-20240614
|
||||||
react-dom: 19.0.0-rc-fb9a90fa48-20240614(react@19.0.0-rc-fb9a90fa48-20240614)
|
react-dom: 19.0.0-rc-fb9a90fa48-20240614(react@19.0.0-rc-fb9a90fa48-20240614)
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/dragon@0.16.1:
|
/@lexical/dragon@0.17.0:
|
||||||
resolution: {integrity: sha512-Rvd60GIYN5kpjjBumS34EnNbBaNsoseI0AlzOdtIV302jiHPCLH0noe9kxzu9nZy+MZmjZy8Dx2zTbQT2mueRw==}
|
resolution: {integrity: sha512-XSsrHVwhjBIVF9VN9MFm6Go8fquj5H/jlYuyNzemHq0tOli8NaoSovGc5q0LwXr88RPsuIt1jluazR7Q1+kxTQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/eslint-plugin@0.16.1(eslint@9.6.0):
|
/@lexical/eslint-plugin@0.17.0(eslint@9.6.0):
|
||||||
resolution: {integrity: sha512-C68eSFBAJ5H8vDae46l9iPUYYw6btC4ZAOr2vMdri8tuN4Aid5c2skDv/Ruiyk12SWsgzz9jHiyo5Fsa1ESLdg==}
|
resolution: {integrity: sha512-O6RyQBXAdi90jlthWwfOuxYG4zqzWkpNwsX1V6N8t5iH80Te04LsnfG+hIB/5V8rxm8WPkTjMrqAX3UEZy5Shg==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
eslint: '>=7.31.0 || ^8.0.0'
|
eslint: '>=7.31.0 || ^8.0.0'
|
||||||
dependencies:
|
dependencies:
|
||||||
eslint: 9.6.0
|
eslint: 9.6.0
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/@lexical/hashtag@0.16.1:
|
/@lexical/hashtag@0.17.0:
|
||||||
resolution: {integrity: sha512-G+YOxStAKs3q1utqm9KR4D4lCkwIH52Rctm4RgaVTI+4lvTaybeDRGFV75P/pI/qlF7/FvAYHTYEzCjtC3GNMQ==}
|
resolution: {integrity: sha512-E6nSoz9haB6JypQtYxG5OYr36AHgam/FBMu77OWNl1KsJbkP8nInm+P22QFsNnEvs4Hk6/0FJ5g42+lTEnGmIg==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/headless@0.16.1:
|
/@lexical/headless@0.17.0:
|
||||||
resolution: {integrity: sha512-L00TQk9vD1o7c25QMNdwD7MvkmQYP/jebG2M8GbX/3KSjHag2QB8MwcFlccSGXBhmbVm1X1Zo7z7urxY//3atw==}
|
resolution: {integrity: sha512-yKvXcq2F6S1lwDLcwv+bHht/al1LcFmidPT3rjISRxLX+/YjUcUT8MmvV773Du4piV4rFPbVlBPFBZfHJkDxXw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/history@0.16.1:
|
/@lexical/history@0.17.0:
|
||||||
resolution: {integrity: sha512-WQhScx0TJeKSQAnEkRpIaWdUXqirrNrom2MxbBUc/32zEUMm9FzV7nRGknvUabEFUo7vZq6xTZpOExQJqHInQA==}
|
resolution: {integrity: sha512-SfeUKAXf9pZpqee9rMOTt33V0J0p/AS9TZLT9Un9dU6wAaHfv6NFax1ND0JoG1a9YkTc539mufxVLNjsNRc0ag==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/html@0.16.1:
|
/@lexical/html@0.17.0:
|
||||||
resolution: {integrity: sha512-vbtAdCvQ3PaAqa5mFmtmrvbiAvjCu1iXBAJ0bsHqFXCF2Sba5LwHVe8dUAOTpfEZEMbiHfjul6b5fj4vNPGF2A==}
|
resolution: {integrity: sha512-sI458CEP/j+Gd2YEo1+vTax31ZAjdq5jmRJMgSKxzKlkVYAUY9eH5u3Y3awPLwLVXJHiIopMX02GeZytibuTiw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/link@0.16.1:
|
/@lexical/link@0.17.0:
|
||||||
resolution: {integrity: sha512-zG36gEnEqbIe6tK/MhXi7wn/XMY/zdivnPcOY5WyC3derkEezeLSSIFsC1u5UNeK5pbpNMSy4LDpLhi1Ww4Y5w==}
|
resolution: {integrity: sha512-Kux6yvPit6y0ksPpwimv3seVrXAsggkqB6oT6oAVBaDpYuygVEwNDqg/rCTtB3mHQ4eeuU33mdK7MSXZ34bZRQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/list@0.16.1:
|
/@lexical/list@0.17.0:
|
||||||
resolution: {integrity: sha512-i9YhLAh5N6YO9dP+R1SIL9WEdCKeTiQQYVUzj84vDvX5DIBxMPUjTmMn3LXu9T+QO3h1s2L/vJusZASrl45eAw==}
|
resolution: {integrity: sha512-anDuSUykTv+lqyCwl1m+sThrB15OKCa00Eo68/d2HQSHDD3KNWgSx709dcR17bD9oT204yOhMJbQGywuzcEyGQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/mark@0.16.1:
|
/@lexical/mark@0.17.0:
|
||||||
resolution: {integrity: sha512-CZRGMLcxn5D+jzf1XnH+Z+uUugmpg1mBwTbGybCPm8UWpBrKDHkrscfMgWz62iRWz0cdVjM5+0zWpNElxFTRjQ==}
|
resolution: {integrity: sha512-Ynqh9KHXUcB9qLOTGC9s+bbWtawOwRStkeIeAugTqrwckyYWeDaePpyJ6IhBBJy1E1CfpiZn71NDeP+FuRjnXQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/markdown@0.16.1:
|
/@lexical/markdown@0.17.0:
|
||||||
resolution: {integrity: sha512-0sBLttMvfQO/hVaIqpHdvDowpgV2CoRuWo2CNwvRLZPPWvPVjL4Nkb73wmi8zAZsAOTbX2aw+g4m/+k5oJqNig==}
|
resolution: {integrity: sha512-6IuJ2l5p/Ma+VBUIStIRXwTC01GEzx21gvqqywuqBUzAOiMr1oRM+DGsQgrzZrcjX+LzUlZ5ZgjuWtK8XKVAZw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/code': 0.16.1
|
'@lexical/code': 0.17.0
|
||||||
'@lexical/link': 0.16.1
|
'@lexical/link': 0.17.0
|
||||||
'@lexical/list': 0.16.1
|
'@lexical/list': 0.17.0
|
||||||
'@lexical/rich-text': 0.16.1
|
'@lexical/rich-text': 0.17.0
|
||||||
'@lexical/text': 0.16.1
|
'@lexical/text': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/offset@0.16.1:
|
/@lexical/offset@0.17.0:
|
||||||
resolution: {integrity: sha512-/i2J04lQmFeydUZIF8tKXLQTXiJDTQ6GRnkfv1OpxU4amc0rwGa7+qAz/PuF1n58rP6InpLmSHxgY5JztXa2jw==}
|
resolution: {integrity: sha512-onE6SD2mIAwBLTT5v5fVBVtRg/NpQj+o10vTWJ1ImvEUERpSoCyHMTy3IMoSMuCRwuOG9C0cFEret2u+QS8Icw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/overflow@0.16.1:
|
/@lexical/overflow@0.17.0:
|
||||||
resolution: {integrity: sha512-xh5YpoxwA7K4wgMQF/Sjl8sdjaxqesLCtH5ZrcMsaPlmucDIEEs+i8xxk+kDUTEY7y+3FvRxs4lGNgX8RVWkvQ==}
|
resolution: {integrity: sha512-dh+nQAmeobKvZFodWyzNh1ZjX043Patk/1Lwct9XmtAGMUdXL+tB0bbguWVcDfY8OYu1CTQGfbdq2oMEJYzwsg==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/plain-text@0.16.1:
|
/@lexical/plain-text@0.17.0:
|
||||||
resolution: {integrity: sha512-GjY4ylrBZIaAVIF8IFnmW0XGyHAuRmWA6gKB8iTTlsjgFrCHFIYC74EeJSp309O0Hflg9rRBnKoX1TYruFHVwA==}
|
resolution: {integrity: sha512-AEk+3ttbRyRi7m9UbU1CdLUtGsXh4FFZkBC12twV3U82lZHOdHocLlTutP+lcbYlGjeq6UF43NxOSGzsYEunsA==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/clipboard': 0.16.1
|
'@lexical/clipboard': 0.17.0
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/react@0.16.1(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)(yjs@13.6.18):
|
/@lexical/react@0.17.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)(yjs@13.6.18):
|
||||||
resolution: {integrity: sha512-SsGgLt9iKfrrMRy9lFb6ROVPUYOgv6b+mCn9Al+TLqs/gBReDBi3msA7m526nrtBUKYUnjHdQ1QXIJzuKgOxcg==}
|
resolution: {integrity: sha512-HZ3joq+5g2++2vo/6scTd60Y2bsu8ya8EUdopyudnmGZGKAcAPue9pLOlBaEpsYZ7vqTuGjiPgtEBfFzDy9rlg==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ^19.0.0-rc-6230622a1a-20240610
|
react: ^19.0.0-rc-6230622a1a-20240610
|
||||||
react-dom: ^19.0.0-rc-6230622a1a-20240610
|
react-dom: ^19.0.0-rc-6230622a1a-20240610
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/clipboard': 0.16.1
|
'@lexical/clipboard': 0.17.0
|
||||||
'@lexical/code': 0.16.1
|
'@lexical/code': 0.17.0
|
||||||
'@lexical/devtools-core': 0.16.1(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)
|
'@lexical/devtools-core': 0.17.0(react-dom@19.0.0-rc-fb9a90fa48-20240614)(react@19.0.0-rc-fb9a90fa48-20240614)
|
||||||
'@lexical/dragon': 0.16.1
|
'@lexical/dragon': 0.17.0
|
||||||
'@lexical/hashtag': 0.16.1
|
'@lexical/hashtag': 0.17.0
|
||||||
'@lexical/history': 0.16.1
|
'@lexical/history': 0.17.0
|
||||||
'@lexical/link': 0.16.1
|
'@lexical/link': 0.17.0
|
||||||
'@lexical/list': 0.16.1
|
'@lexical/list': 0.17.0
|
||||||
'@lexical/mark': 0.16.1
|
'@lexical/mark': 0.17.0
|
||||||
'@lexical/markdown': 0.16.1
|
'@lexical/markdown': 0.17.0
|
||||||
'@lexical/overflow': 0.16.1
|
'@lexical/overflow': 0.17.0
|
||||||
'@lexical/plain-text': 0.16.1
|
'@lexical/plain-text': 0.17.0
|
||||||
'@lexical/rich-text': 0.16.1
|
'@lexical/rich-text': 0.17.0
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/table': 0.16.1
|
'@lexical/table': 0.17.0
|
||||||
'@lexical/text': 0.16.1
|
'@lexical/text': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
'@lexical/yjs': 0.16.1(yjs@13.6.18)
|
'@lexical/yjs': 0.17.0(yjs@13.6.18)
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
react: 19.0.0-rc-fb9a90fa48-20240614
|
react: 19.0.0-rc-fb9a90fa48-20240614
|
||||||
react-dom: 19.0.0-rc-fb9a90fa48-20240614(react@19.0.0-rc-fb9a90fa48-20240614)
|
react-dom: 19.0.0-rc-fb9a90fa48-20240614(react@19.0.0-rc-fb9a90fa48-20240614)
|
||||||
react-error-boundary: 3.1.4(react@19.0.0-rc-fb9a90fa48-20240614)
|
react-error-boundary: 3.1.4(react@19.0.0-rc-fb9a90fa48-20240614)
|
||||||
@@ -5848,45 +5848,45 @@ packages:
|
|||||||
- yjs
|
- yjs
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
/@lexical/rich-text@0.16.1:
|
/@lexical/rich-text@0.17.0:
|
||||||
resolution: {integrity: sha512-4uEVXJur7tdSbqbmsToCW4YVm0AMh4y9LK077Yq2O9hSuA5dqpI8UbTDnxZN2D7RfahNvwlqp8eZKFB1yeiJGQ==}
|
resolution: {integrity: sha512-XJc8gQBSwppCkESQaNcGtyTaPXZaeCQDcUVpnDjDK0vM/ZZN8TErxbujwbSqA3kO2dBds9N8WxNboSwuncMBcQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/clipboard': 0.16.1
|
'@lexical/clipboard': 0.17.0
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/selection@0.16.1:
|
/@lexical/selection@0.17.0:
|
||||||
resolution: {integrity: sha512-+nK3RvXtyQvQDq7AZ46JpphmM33pwuulwiRfeXR5T9iFQTtgWOEjsAi/KKX7vGm70BxACfiSxy5QCOgBWFwVJg==}
|
resolution: {integrity: sha512-UTjlvyhFY/lmHtBaIaVRwYnRfO9gR4I32+PT7vHQr4v3VfcgS63YEGSgEZy3Gh1pfeJqaZATN58+jCuMAQXlWQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/table@0.16.1:
|
/@lexical/table@0.17.0:
|
||||||
resolution: {integrity: sha512-GWb0/MM1sVXpi1p2HWWOBldZXASMQ4c6WRNYnRmq7J/aB5N66HqQgJGKp3m66Kz4k1JjhmZfPs7F018qIBhnFQ==}
|
resolution: {integrity: sha512-RQF7IG0rGL2/bPaPFUIMgDA3QMdDflvXSnE7Udgbj9yMqSKhYkaERVfNyoLckDUSuusGJd6XV+qum6JWn0nSNA==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/utils': 0.16.1
|
'@lexical/utils': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/text@0.16.1:
|
/@lexical/text@0.17.0:
|
||||||
resolution: {integrity: sha512-Os/nKQegORTrKKN6vL3/FMVszyzyqaotlisPynvTaHTUC+yY4uyjM2hlF93i5a2ixxyiPLF9bDroxUP96TMPXg==}
|
resolution: {integrity: sha512-kFH0V6yjW8YswmoY7vHT4zHFDflGfamuUxTPHROpdnq/JMjHeaVwtmFBdrP0gknaC8XMRXdr3EsemQ7cbOoDPA==}
|
||||||
dependencies:
|
dependencies:
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/utils@0.16.1:
|
/@lexical/utils@0.17.0:
|
||||||
resolution: {integrity: sha512-BVyJxDQi/rIxFTDjf2zE7rMDKSuEaeJ4dybHRa/hRERt85gavGByQawSLeQlTjLaYLVsy+x7wCcqh2fNhlLf0g==}
|
resolution: {integrity: sha512-B/n0rRGDmdMrqi2qnprLt6SntC6jb4JItLmPl8zDDdg7/HxMdLq3F93vogeiXQJn0mlNqgiENWHvLAy5K2C2uQ==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/list': 0.16.1
|
'@lexical/list': 0.17.0
|
||||||
'@lexical/selection': 0.16.1
|
'@lexical/selection': 0.17.0
|
||||||
'@lexical/table': 0.16.1
|
'@lexical/table': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
|
|
||||||
/@lexical/yjs@0.16.1(yjs@13.6.18):
|
/@lexical/yjs@0.17.0(yjs@13.6.18):
|
||||||
resolution: {integrity: sha512-QHw1bmzB/IypIV1tRWMH4hhwE1xX7wV+HxbzBS8oJAkoU5AYXM/kyp/sQicgqiwVfpai1Px7zatOoUDFgbyzHQ==}
|
resolution: {integrity: sha512-xJv3frcK/jskssLbzdY4yfBaM7+LWaZD4YjYkJ/bvRDTey2w+McF+SvsJ/yBA8YF1oaL3rT+0aIQJ7rfH+AxjA==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
yjs: '>=13.5.22'
|
yjs: '>=13.5.22'
|
||||||
dependencies:
|
dependencies:
|
||||||
'@lexical/offset': 0.16.1
|
'@lexical/offset': 0.17.0
|
||||||
lexical: 0.16.1
|
lexical: 0.17.0
|
||||||
yjs: 13.6.18
|
yjs: 13.6.18
|
||||||
dev: false
|
dev: false
|
||||||
|
|
||||||
@@ -12284,8 +12284,8 @@ packages:
|
|||||||
prelude-ls: 1.2.1
|
prelude-ls: 1.2.1
|
||||||
type-check: 0.4.0
|
type-check: 0.4.0
|
||||||
|
|
||||||
/lexical@0.16.1:
|
/lexical@0.17.0:
|
||||||
resolution: {integrity: sha512-+R05d3+N945OY8pTUjTqQrWoApjC+ctzvjnmNETtx9WmVAaiW0tQVG+AYLt5pDGY8dQXtd4RPorvnxBTECt9SA==}
|
resolution: {integrity: sha512-cCFmANO5rIf34NF0go/hxp5S3V5Z8G2Rsa1FJy50qF2WM5EJNJ/MqN75TApjfgMkfrbO6gau3X12nCqwsT7aDg==}
|
||||||
|
|
||||||
/lib0@0.2.94:
|
/lib0@0.2.94:
|
||||||
resolution: {integrity: sha512-hZ3p54jL4Wpu7IOg26uC7dnEWiMyNlUrb9KoG7+xYs45WkQwpVvKFndVq2+pqLYKe1u8Fp3+zAfZHVvTK34PvQ==}
|
resolution: {integrity: sha512-hZ3p54jL4Wpu7IOg26uC7dnEWiMyNlUrb9KoG7+xYs45WkQwpVvKFndVq2+pqLYKe1u8Fp3+zAfZHVvTK34PvQ==}
|
||||||
|
|||||||
@@ -594,6 +594,19 @@ export interface BlockField {
|
|||||||
blockType: 'text';
|
blockType: 'text';
|
||||||
}[]
|
}[]
|
||||||
| null;
|
| null;
|
||||||
|
blocksWithLocalizedArray?:
|
||||||
|
| {
|
||||||
|
array?:
|
||||||
|
| {
|
||||||
|
text?: string | null;
|
||||||
|
id?: string | null;
|
||||||
|
}[]
|
||||||
|
| null;
|
||||||
|
id?: string | null;
|
||||||
|
blockName?: string | null;
|
||||||
|
blockType: 'localizedArray';
|
||||||
|
}[]
|
||||||
|
| null;
|
||||||
blocksWithSimilarConfigs?:
|
blocksWithSimilarConfigs?:
|
||||||
| (
|
| (
|
||||||
| {
|
| {
|
||||||
|
|||||||
@@ -23,8 +23,8 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@aws-sdk/client-s3": "^3.525.0",
|
"@aws-sdk/client-s3": "^3.525.0",
|
||||||
"@lexical/headless": "0.16.1",
|
"@lexical/headless": "0.17.0",
|
||||||
"@lexical/markdown": "0.16.1",
|
"@lexical/markdown": "0.17.0",
|
||||||
"@payloadcms/db-mongodb": "workspace:*",
|
"@payloadcms/db-mongodb": "workspace:*",
|
||||||
"@payloadcms/db-postgres": "workspace:*",
|
"@payloadcms/db-postgres": "workspace:*",
|
||||||
"@payloadcms/db-sqlite": "workspace:*",
|
"@payloadcms/db-sqlite": "workspace:*",
|
||||||
@@ -68,7 +68,7 @@
|
|||||||
"file-type": "17.1.6",
|
"file-type": "17.1.6",
|
||||||
"http-status": "1.6.2",
|
"http-status": "1.6.2",
|
||||||
"jwt-decode": "4.0.0",
|
"jwt-decode": "4.0.0",
|
||||||
"lexical": "0.16.1",
|
"lexical": "0.17.0",
|
||||||
"payload": "workspace:*",
|
"payload": "workspace:*",
|
||||||
"qs-esm": "7.0.2",
|
"qs-esm": "7.0.2",
|
||||||
"server-only": "^0.0.1",
|
"server-only": "^0.0.1",
|
||||||
|
|||||||
Reference in New Issue
Block a user