Files
payload/packages/richtext-lexical/src/index.ts
Alessio Gravili e6fea1d132 fix: localized fields within block references were not handled properly if any parent is localized (#11207)
The `localized` properly was not stripped out of referenced block fields, if any parent was localized. For normal fields, this is done in sanitizeConfig. As the same referenced block config can be used in both a localized and non-localized config, we are not able to strip it out inside sanitizeConfig by modifying the block config.

Instead, this PR had to bring back tedious logic to handle it everywhere the `field.localized` property is accessed. For backwards-compatibility, we need to keep the existing sanitizeConfig logic. In 4.0, we should remove it to benefit from better test coverage of runtime field.localized handling - for now, this is done for our test suite using the `PAYLOAD_DO_NOT_SANITIZE_LOCALIZED_PROPERTY` flag.
2025-02-17 19:50:32 +00:00

1067 lines
38 KiB
TypeScript

import type { JSONSchema4 } from 'json-schema'
import type { SerializedEditorState, SerializedLexicalNode } from 'lexical'
import {
afterChangeTraverseFields,
afterReadTraverseFields,
beforeChangeTraverseFields,
beforeValidateTraverseFields,
checkDependencies,
withNullableJSONSchemaType,
} from 'payload'
import type { FeatureProviderServer, ResolvedServerFeatureMap } from './features/typesServer.js'
import type { SanitizedServerEditorConfig } from './lexical/config/types.js'
import type {
AdapterProps,
LexicalEditorProps,
LexicalRichTextAdapter,
LexicalRichTextAdapterProvider,
} from './types.js'
import { getDefaultSanitizedEditorConfig } from './getDefaultSanitizedEditorConfig.js'
import { i18n } from './i18n.js'
import { defaultEditorConfig, defaultEditorFeatures } from './lexical/config/server/default.js'
import { loadFeatures } from './lexical/config/server/loader.js'
import { sanitizeServerFeatures } from './lexical/config/server/sanitize.js'
import { populateLexicalPopulationPromises } from './populateGraphQL/populateLexicalPopulationPromises.js'
import { getGenerateImportMap } from './utilities/generateImportMap.js'
import { getGenerateSchemaMap } from './utilities/generateSchemaMap.js'
import { recurseNodeTree } from './utilities/recurseNodeTree.js'
import { richTextValidateHOC } from './validate/index.js'
let checkedDependencies = false
export const lexicalTargetVersion = '0.21.0'
export function lexicalEditor(props?: LexicalEditorProps): LexicalRichTextAdapterProvider {
if (
process.env.NODE_ENV !== 'production' &&
process.env.PAYLOAD_DISABLE_DEPENDENCY_CHECKER !== 'true' &&
!checkedDependencies
) {
checkedDependencies = true
void checkDependencies({
dependencyGroups: [
{
name: 'lexical',
dependencies: [
'lexical',
'@lexical/headless',
'@lexical/link',
'@lexical/list',
'@lexical/mark',
'@lexical/react',
'@lexical/rich-text',
'@lexical/selection',
'@lexical/utils',
],
targetVersion: lexicalTargetVersion,
},
],
})
}
return async ({ config, isRoot, parentIsLocalized }) => {
let features: FeatureProviderServer<unknown, unknown, unknown>[] = []
let resolvedFeatureMap: ResolvedServerFeatureMap
let finalSanitizedEditorConfig: SanitizedServerEditorConfig // For server only
if (!props || (!props.features && !props.lexical)) {
finalSanitizedEditorConfig = await getDefaultSanitizedEditorConfig({
config,
parentIsLocalized,
})
features = defaultEditorFeatures
resolvedFeatureMap = finalSanitizedEditorConfig.resolvedFeatureMap
} else {
if (props.features && typeof props.features === 'function') {
const rootEditor = config.editor
let rootEditorFeatures: FeatureProviderServer<unknown, unknown, unknown>[] = []
if (typeof rootEditor === 'object' && 'features' in rootEditor) {
rootEditorFeatures = (rootEditor as LexicalRichTextAdapter).features
}
features = props.features({
defaultFeatures: defaultEditorFeatures,
rootFeatures: rootEditorFeatures,
})
} else {
features = props.features as FeatureProviderServer<unknown, unknown, unknown>[]
}
if (!features) {
features = defaultEditorFeatures
}
const lexical = props.lexical ?? defaultEditorConfig.lexical
resolvedFeatureMap = await loadFeatures({
config,
isRoot,
parentIsLocalized,
unSanitizedEditorConfig: {
features,
lexical,
},
})
finalSanitizedEditorConfig = {
features: sanitizeServerFeatures(resolvedFeatureMap),
lexical: props.lexical,
resolvedFeatureMap,
}
}
const featureI18n = finalSanitizedEditorConfig.features.i18n
for (const lang in i18n) {
if (!featureI18n[lang as keyof typeof featureI18n]) {
featureI18n[lang as keyof typeof featureI18n] = {
lexical: {},
}
}
// @ts-expect-error - vestiges of when tsconfig was not strict. Feel free to improve
featureI18n[lang].lexical.general = i18n[lang]
}
return {
CellComponent: {
path: '@payloadcms/richtext-lexical/rsc#RscEntryLexicalCell',
serverProps: {
admin: props?.admin,
sanitizedEditorConfig: finalSanitizedEditorConfig,
},
},
editorConfig: finalSanitizedEditorConfig,
features,
FieldComponent: {
path: '@payloadcms/richtext-lexical/rsc#RscEntryLexicalField',
serverProps: {
admin: props?.admin,
sanitizedEditorConfig: finalSanitizedEditorConfig,
},
},
generateImportMap: getGenerateImportMap({
resolvedFeatureMap,
}),
generateSchemaMap: getGenerateSchemaMap({
resolvedFeatureMap,
}),
graphQLPopulationPromises({
context,
currentDepth,
depth,
draft,
field,
fieldPromises,
findMany,
flattenLocales,
overrideAccess,
parentIsLocalized,
populationPromises,
req,
showHiddenFields,
siblingDoc,
}) {
// check if there are any features with nodes which have populationPromises for this field
if (finalSanitizedEditorConfig?.features?.graphQLPopulationPromises?.size) {
populateLexicalPopulationPromises({
context,
currentDepth: currentDepth ?? 0,
depth,
draft,
editorPopulationPromises: finalSanitizedEditorConfig.features.graphQLPopulationPromises,
field,
fieldPromises,
findMany,
flattenLocales,
overrideAccess,
parentIsLocalized,
populationPromises,
req,
showHiddenFields,
siblingDoc,
})
}
},
hooks: {
afterChange: [
async (args) => {
const {
collection,
context: _context,
data,
field,
global,
indexPath,
operation,
originalDoc,
parentIsLocalized,
path,
previousDoc,
previousValue,
req,
schemaPath,
} = args
let { value } = args
if (finalSanitizedEditorConfig?.features?.hooks?.afterChange?.length) {
for (const hook of finalSanitizedEditorConfig.features.hooks.afterChange) {
value = await hook(args)
}
}
if (
!finalSanitizedEditorConfig.features.nodeHooks?.afterChange?.size &&
!finalSanitizedEditorConfig.features.getSubFields?.size
) {
return value
}
// TO-DO: We should not use context, as it is intended for external use only
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const context: any = _context
const nodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
const previousNodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
/**
* Get the originalNodeIDMap from the beforeValidate hook, which is always run before this hook.
*/
const originalNodeIDMap: {
[key: string]: SerializedLexicalNode
} = context?.internal?.richText?.[path.join('.')]?.originalNodeIDMap
if (!originalNodeIDMap || !Object.keys(originalNodeIDMap).length || !value) {
return value
}
recurseNodeTree({
nodeIDMap,
nodes: (value as SerializedEditorState)?.root?.children ?? [],
})
recurseNodeTree({
nodeIDMap: previousNodeIDMap,
nodes: (previousValue as SerializedEditorState)?.root?.children ?? [],
})
// eslint-disable-next-line prefer-const
for (let [id, node] of Object.entries(nodeIDMap)) {
const afterChangeHooks = finalSanitizedEditorConfig.features.nodeHooks?.afterChange
const afterChangeHooksForNode = afterChangeHooks?.get(node.type)
if (afterChangeHooksForNode) {
for (const hook of afterChangeHooksForNode) {
if (!originalNodeIDMap[id]) {
console.warn(
'(afterChange) No original node found for node with id',
id,
'node:',
node,
'path',
path.join('.'),
)
continue
}
node = await hook({
context,
node,
operation,
originalNode: originalNodeIDMap[id],
parentRichTextFieldPath: path,
parentRichTextFieldSchemaPath: schemaPath,
previousNode: previousNodeIDMap[id]!,
req,
})
}
}
const subFieldFn = finalSanitizedEditorConfig.features.getSubFields?.get(node.type)
const subFieldDataFn = finalSanitizedEditorConfig.features.getSubFieldsData?.get(
node.type,
)
if (subFieldFn && subFieldDataFn) {
const subFields = subFieldFn({ node, req })
const nodeSiblingData = subFieldDataFn({ node, req }) ?? {}
const nodeSiblingDoc = subFieldDataFn({ node: originalNodeIDMap[id]!, req }) ?? {}
const nodePreviousSiblingDoc =
subFieldDataFn({ node: previousNodeIDMap[id]!, req }) ?? {}
if (subFields?.length) {
await afterChangeTraverseFields({
blockData: nodeSiblingData,
collection,
context,
data: data ?? {},
doc: originalDoc,
fields: subFields,
global,
operation,
parentIndexPath: indexPath.join('-'),
parentIsLocalized: parentIsLocalized || field.localized || false,
parentPath: path.join('.'),
parentSchemaPath: schemaPath.join('.'),
previousDoc,
previousSiblingDoc: { ...nodePreviousSiblingDoc },
req,
siblingData: nodeSiblingData || {},
siblingDoc: { ...nodeSiblingDoc },
})
}
}
}
return value
},
],
afterRead: [
/**
* afterRead hooks do not receive the originalNode. Thus, they can run on all nodes, not just nodes with an ID.
*/
async (args) => {
const {
collection,
context: context,
currentDepth,
depth,
draft,
fallbackLocale,
field,
fieldPromises,
findMany,
flattenLocales,
global,
indexPath,
locale,
originalDoc,
overrideAccess,
parentIsLocalized,
path,
populate,
populationPromises,
req,
schemaPath,
showHiddenFields,
triggerAccessControl,
triggerHooks,
} = args
let { value } = args
if (finalSanitizedEditorConfig?.features?.hooks?.afterRead?.length) {
for (const hook of finalSanitizedEditorConfig.features.hooks.afterRead) {
value = await hook(args)
}
}
if (
!finalSanitizedEditorConfig.features.nodeHooks?.afterRead?.size &&
!finalSanitizedEditorConfig.features.getSubFields?.size
) {
return value
}
const flattenedNodes: SerializedLexicalNode[] = []
recurseNodeTree({
flattenedNodes,
nodes: (value as SerializedEditorState)?.root?.children ?? [],
})
for (let node of flattenedNodes) {
const afterReadHooks = finalSanitizedEditorConfig.features.nodeHooks?.afterRead
const afterReadHooksForNode = afterReadHooks?.get(node.type)
if (afterReadHooksForNode) {
for (const hook of afterReadHooksForNode) {
node = await hook({
context,
currentDepth: currentDepth!,
depth: depth!,
draft: draft!,
fallbackLocale: fallbackLocale!,
fieldPromises: fieldPromises!,
findMany: findMany!,
flattenLocales: flattenLocales!,
locale: locale!,
node,
overrideAccess: overrideAccess!,
parentRichTextFieldPath: path,
parentRichTextFieldSchemaPath: schemaPath,
populateArg: populate,
populationPromises: populationPromises!,
req,
showHiddenFields: showHiddenFields!,
triggerAccessControl: triggerAccessControl!,
triggerHooks: triggerHooks!,
})
}
}
const subFieldFn = finalSanitizedEditorConfig.features.getSubFields?.get(node.type)
const subFieldDataFn = finalSanitizedEditorConfig.features.getSubFieldsData?.get(
node.type,
)
if (subFieldFn && subFieldDataFn) {
const subFields = subFieldFn({ node, req })
const nodeSiblingData = subFieldDataFn({ node, req }) ?? {}
if (subFields?.length) {
afterReadTraverseFields({
blockData: nodeSiblingData,
collection,
context,
currentDepth: currentDepth!,
depth: depth!,
doc: originalDoc,
draft: draft!,
fallbackLocale: fallbackLocale!,
fieldPromises: fieldPromises!,
fields: subFields,
findMany: findMany!,
flattenLocales: flattenLocales!,
global,
locale: locale!,
overrideAccess: overrideAccess!,
parentIndexPath: indexPath.join('-'),
parentIsLocalized: parentIsLocalized || field.localized || false,
parentPath: path.join('.'),
parentSchemaPath: schemaPath.join('.'),
populate,
populationPromises: populationPromises!,
req,
showHiddenFields: showHiddenFields!,
siblingDoc: nodeSiblingData,
triggerAccessControl,
triggerHooks,
})
}
}
}
return value
},
],
beforeChange: [
async (args) => {
const {
collection,
context: _context,
data,
docWithLocales,
errors,
field,
global,
indexPath,
mergeLocaleActions,
operation,
originalDoc,
parentIsLocalized,
path,
previousValue,
req,
schemaPath,
siblingData,
siblingDocWithLocales,
skipValidation,
} = args
let { value } = args
if (finalSanitizedEditorConfig?.features?.hooks?.beforeChange?.length) {
for (const hook of finalSanitizedEditorConfig.features.hooks.beforeChange) {
value = await hook(args)
}
}
if (
!finalSanitizedEditorConfig.features.nodeHooks?.beforeChange?.size &&
!finalSanitizedEditorConfig.features.getSubFields?.size
) {
return value
}
// TO-DO: We should not use context, as it is intended for external use only
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const context: any = _context
const nodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
/**
* Get the originalNodeIDMap from the beforeValidate hook, which is always run before this hook.
*/
const originalNodeIDMap: {
[key: string]: SerializedLexicalNode
} = context?.internal?.richText?.[path.join('.')]?.originalNodeIDMap
if (!originalNodeIDMap || !Object.keys(originalNodeIDMap).length || !value) {
return value
}
const previousNodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
const originalNodeWithLocalesIDMap: {
[key: string]: SerializedLexicalNode
} = {}
recurseNodeTree({
nodeIDMap,
nodes: (value as SerializedEditorState)?.root?.children ?? [],
})
recurseNodeTree({
nodeIDMap: previousNodeIDMap,
nodes: (previousValue as SerializedEditorState)?.root?.children ?? [],
})
if (field.name && siblingDocWithLocales?.[field.name]) {
recurseNodeTree({
nodeIDMap: originalNodeWithLocalesIDMap,
nodes:
(siblingDocWithLocales[field.name] as SerializedEditorState)?.root?.children ??
[],
})
}
// eslint-disable-next-line prefer-const
for (let [id, node] of Object.entries(nodeIDMap)) {
const beforeChangeHooks = finalSanitizedEditorConfig.features.nodeHooks?.beforeChange
const beforeChangeHooksForNode = beforeChangeHooks?.get(node.type)
if (beforeChangeHooksForNode) {
for (const hook of beforeChangeHooksForNode) {
if (!originalNodeIDMap[id]) {
console.warn(
'(beforeChange) No original node found for node with id',
id,
'node:',
node,
'path',
path.join('.'),
)
continue
}
node = await hook({
context,
errors: errors!,
mergeLocaleActions: mergeLocaleActions!,
node,
operation: operation!,
originalNode: originalNodeIDMap[id],
originalNodeWithLocales: originalNodeWithLocalesIDMap[id],
parentRichTextFieldPath: path,
parentRichTextFieldSchemaPath: schemaPath,
previousNode: previousNodeIDMap[id]!,
req,
skipValidation: skipValidation!,
})
}
}
const subFieldFn = finalSanitizedEditorConfig.features.getSubFields?.get(node.type)
const subFieldDataFn = finalSanitizedEditorConfig.features.getSubFieldsData?.get(
node.type,
)
if (subFieldFn && subFieldDataFn) {
const subFields = subFieldFn({ node, req })
const nodeSiblingData = subFieldDataFn({ node, req }) ?? {}
const nodeSiblingDocWithLocales =
subFieldDataFn({
node: originalNodeWithLocalesIDMap[id]!,
req,
}) ?? {}
const nodePreviousSiblingDoc =
subFieldDataFn({ node: previousNodeIDMap[id]!, req }) ?? {}
if (subFields?.length) {
await beforeChangeTraverseFields({
id,
blockData: nodeSiblingData,
collection,
context,
data: data ?? {},
doc: originalDoc ?? {},
docWithLocales: docWithLocales ?? {},
errors: errors!,
fields: subFields,
global,
mergeLocaleActions: mergeLocaleActions!,
operation: operation!,
parentIndexPath: indexPath.join('-'),
parentIsLocalized: parentIsLocalized || field.localized || false,
parentPath: path.join('.'),
parentSchemaPath: schemaPath.join('.'),
req,
siblingData: nodeSiblingData,
siblingDoc: nodePreviousSiblingDoc,
siblingDocWithLocales: nodeSiblingDocWithLocales ?? {},
skipValidation,
})
}
}
}
/**
* within the beforeChange hook, id's may be re-generated.
* Example:
* 1. Seed data contains IDs for block feature blocks.
* 2. Those are used in beforeValidate
* 3. in beforeChange, those IDs are regenerated, because you cannot provide IDs during document creation. See baseIDField beforeChange hook for reasoning
* 4. Thus, in order for all post-beforeChange hooks to receive the correct ID, we need to update the originalNodeIDMap with the new ID's, by regenerating the nodeIDMap.
* The reason this is not generated for every hook, is to save on performance. We know we only really have to generate it in beforeValidate, which is the first hook,
* and in beforeChange, which is where modifications to the provided IDs can occur.
*/
const newOriginalNodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
const previousOriginalValue = siblingData[field.name!]
recurseNodeTree({
nodeIDMap: newOriginalNodeIDMap,
nodes: (previousOriginalValue as SerializedEditorState)?.root?.children ?? [],
})
if (!context.internal) {
// Add to context, for other hooks to use
context.internal = {}
}
if (!context.internal.richText) {
context.internal.richText = {}
}
context.internal.richText[path.join('.')] = {
originalNodeIDMap: newOriginalNodeIDMap,
}
return value
},
],
beforeValidate: [
async (args) => {
const {
collection,
context,
data,
field,
global,
indexPath,
operation,
originalDoc,
overrideAccess,
parentIsLocalized,
path,
previousValue,
req,
schemaPath,
} = args
let { value } = args
if (finalSanitizedEditorConfig?.features?.hooks?.beforeValidate?.length) {
for (const hook of finalSanitizedEditorConfig.features.hooks.beforeValidate) {
value = await hook(args)
}
}
// return value if there are NO hooks
if (
!finalSanitizedEditorConfig.features.nodeHooks?.beforeValidate?.size &&
!finalSanitizedEditorConfig.features.nodeHooks?.afterChange?.size &&
!finalSanitizedEditorConfig.features.nodeHooks?.beforeChange?.size &&
!finalSanitizedEditorConfig.features.getSubFields?.size
) {
return value
}
/**
* beforeValidate is the first field hook which runs. This is where we can create the node map, which can then be used in the other hooks.
*
*/
/**
* flattenedNodes contains all nodes in the editor, in the order they appear in the editor. They will be used for the following hooks:
* - afterRead
*
* The other hooks require nodes to have IDs, which is why those are ran only from the nodeIDMap. They require IDs because they have both doc/siblingDoc and data/siblingData, and
* thus require a reliable way to match new node data to old node data. Given that node positions can change in between hooks, this is only reliably possible for nodes which are saved with
* an ID.
*/
//const flattenedNodes: SerializedLexicalNode[] = []
/**
* Only nodes with id's (so, nodes with hooks added to them) will be added to the nodeIDMap. They will be used for the following hooks:
* - afterChange
* - beforeChange
* - beforeValidate
*
* Other hooks are handled by the flattenedNodes. All nodes in the nodeIDMap are part of flattenedNodes.
*/
const originalNodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
recurseNodeTree({
nodeIDMap: originalNodeIDMap,
nodes: (previousValue as SerializedEditorState)?.root?.children ?? [],
})
if (!context.internal) {
// Add to context, for other hooks to use
context.internal = {}
}
if (!(context as any).internal.richText) {
;(context as any).internal.richText = {}
}
;(context as any).internal.richText[path.join('.')] = {
originalNodeIDMap,
}
/**
* Now that the maps for all hooks are set up, we can run the validate hook
*/
if (!finalSanitizedEditorConfig.features.nodeHooks?.beforeValidate?.size) {
return value
}
const nodeIDMap: {
[key: string]: SerializedLexicalNode
} = {}
recurseNodeTree({
//flattenedNodes,
nodeIDMap,
nodes: (value as SerializedEditorState)?.root?.children ?? [],
})
// eslint-disable-next-line prefer-const
for (let [id, node] of Object.entries(nodeIDMap)) {
const beforeValidateHooks =
finalSanitizedEditorConfig.features.nodeHooks.beforeValidate
const beforeValidateHooksForNode = beforeValidateHooks?.get(node.type)
if (beforeValidateHooksForNode) {
for (const hook of beforeValidateHooksForNode) {
if (!originalNodeIDMap[id]) {
console.warn(
'(beforeValidate) No original node found for node with id',
id,
'node:',
node,
'path',
path.join('.'),
)
continue
}
node = await hook({
context,
node,
operation,
originalNode: originalNodeIDMap[id],
overrideAccess: overrideAccess!,
parentRichTextFieldPath: path,
parentRichTextFieldSchemaPath: schemaPath,
req,
})
}
}
const subFieldFn = finalSanitizedEditorConfig.features.getSubFields?.get(node.type)
const subFieldDataFn = finalSanitizedEditorConfig.features.getSubFieldsData?.get(
node.type,
)
if (subFieldFn && subFieldDataFn) {
const subFields = subFieldFn({ node, req })
const nodeSiblingData = subFieldDataFn({ node, req }) ?? {}
const nodeSiblingDoc = subFieldDataFn({ node: originalNodeIDMap[id]!, req }) ?? {}
if (subFields?.length) {
await beforeValidateTraverseFields({
id,
blockData: nodeSiblingData,
collection,
context,
data,
doc: originalDoc,
fields: subFields,
global,
operation,
overrideAccess: overrideAccess!,
parentIndexPath: indexPath.join('-'),
parentIsLocalized: parentIsLocalized || field.localized || false,
parentPath: path.join('.'),
parentSchemaPath: schemaPath.join('.'),
req,
siblingData: nodeSiblingData,
siblingDoc: nodeSiblingDoc,
})
}
}
}
return value
},
],
},
i18n: featureI18n,
outputSchema: ({
collectionIDFieldTypes,
config,
field,
i18n,
interfaceNameDefinitions,
isRequired,
}) => {
let outputSchema: JSONSchema4 = {
// This schema matches the SerializedEditorState type so far, that it's possible to cast SerializedEditorState to this schema without any errors.
// In the future, we should
// 1) allow recursive children
// 2) Pass in all the different types for every node added to the editorconfig. This can be done with refs in the schema.
type: withNullableJSONSchemaType('object', isRequired),
properties: {
root: {
type: 'object',
additionalProperties: false,
properties: {
type: {
type: 'string',
},
children: {
type: 'array',
items: {
type: 'object',
additionalProperties: true,
properties: {
type: {
type: 'string',
},
version: {
type: 'integer',
},
},
required: ['type', 'version'],
},
},
direction: {
oneOf: [
{
enum: ['ltr', 'rtl'],
},
{
type: 'null',
},
],
},
format: {
type: 'string',
enum: ['left', 'start', 'center', 'right', 'end', 'justify', ''], // ElementFormatType, since the root node is an element
},
indent: {
type: 'integer',
},
version: {
type: 'integer',
},
},
required: ['children', 'direction', 'format', 'indent', 'type', 'version'],
},
},
required: ['root'],
}
for (const modifyOutputSchema of finalSanitizedEditorConfig.features.generatedTypes
.modifyOutputSchemas) {
outputSchema = modifyOutputSchema({
collectionIDFieldTypes,
config,
currentSchema: outputSchema,
field,
i18n,
interfaceNameDefinitions,
isRequired,
})
}
return outputSchema
},
validate: richTextValidateHOC({
editorConfig: finalSanitizedEditorConfig,
}),
}
}
}
export { AlignFeature } from './features/align/server/index.js'
export { BlockquoteFeature } from './features/blockquote/server/index.js'
export { BlocksFeature, type BlocksFeatureProps } from './features/blocks/server/index.js'
export {
$createServerBlockNode,
$isServerBlockNode,
type BlockFields,
ServerBlockNode,
} from './features/blocks/server/nodes/BlocksNode.js'
export { LinebreakHTMLConverter } from './features/converters/html/converter/converters/linebreak.js'
export { ParagraphHTMLConverter } from './features/converters/html/converter/converters/paragraph.js'
export { TabHTMLConverter } from './features/converters/html/converter/converters/tab.js'
export { TextHTMLConverter } from './features/converters/html/converter/converters/text.js'
export { defaultHTMLConverters } from './features/converters/html/converter/defaultConverters.js'
export {
convertLexicalNodesToHTML,
convertLexicalToHTML,
} from './features/converters/html/converter/index.js'
export type { HTMLConverter } from './features/converters/html/converter/types.js'
export { consolidateHTMLConverters, lexicalHTML } from './features/converters/html/field/index.js'
export {
HTMLConverterFeature,
type HTMLConverterFeatureProps,
} from './features/converters/html/index.js'
export { TestRecorderFeature } from './features/debug/testRecorder/server/index.js'
export { TreeViewFeature } from './features/debug/treeView/server/index.js'
export { EXPERIMENTAL_TableFeature } from './features/experimental_table/server/index.js'
export { BoldFeature } from './features/format/bold/feature.server.js'
export { InlineCodeFeature } from './features/format/inlineCode/feature.server.js'
export { ItalicFeature } from './features/format/italic/feature.server.js'
export { StrikethroughFeature } from './features/format/strikethrough/feature.server.js'
export { SubscriptFeature } from './features/format/subscript/feature.server.js'
export { SuperscriptFeature } from './features/format/superscript/feature.server.js'
export { UnderlineFeature } from './features/format/underline/feature.server.js'
export { HeadingFeature, type HeadingFeatureProps } from './features/heading/server/index.js'
export { HorizontalRuleFeature } from './features/horizontalRule/server/index.js'
export { IndentFeature } from './features/indent/server/index.js'
export { AutoLinkNode } from './features/link/nodes/AutoLinkNode.js'
export { LinkNode } from './features/link/nodes/LinkNode.js'
export type { LinkFields } from './features/link/nodes/types.js'
export { LinkFeature, type LinkFeatureServerProps } from './features/link/server/index.js'
export { ChecklistFeature } from './features/lists/checklist/server/index.js'
export { OrderedListFeature } from './features/lists/orderedList/server/index.js'
export { UnorderedListFeature } from './features/lists/unorderedList/server/index.js'
export type {
SlateNode,
SlateNodeConverter,
} from './features/migrations/slateToLexical/converter/types.js'
export { ParagraphFeature } from './features/paragraph/server/index.js'
export {
RelationshipFeature,
type RelationshipFeatureProps,
} from './features/relationship/server/index.js'
export {
type RelationshipData,
RelationshipServerNode,
} from './features/relationship/server/nodes/RelationshipNode.js'
export { FixedToolbarFeature } from './features/toolbars/fixed/server/index.js'
export { InlineToolbarFeature } from './features/toolbars/inline/server/index.js'
export type { ToolbarGroup, ToolbarGroupItem } from './features/toolbars/types.js'
export type {
BaseClientFeatureProps,
ClientFeature,
ClientFeatureProviderMap,
FeatureProviderClient,
FeatureProviderProviderClient,
PluginComponent,
PluginComponentWithAnchor,
ResolvedClientFeature,
ResolvedClientFeatureMap,
SanitizedClientFeatures,
SanitizedPlugin,
} from './features/typesClient.js'
export type {
AfterChangeNodeHook,
AfterChangeNodeHookArgs,
AfterReadNodeHook,
AfterReadNodeHookArgs,
BaseNodeHookArgs,
BeforeChangeNodeHook,
BeforeChangeNodeHookArgs,
BeforeValidateNodeHook,
BeforeValidateNodeHookArgs,
FeatureProviderProviderServer,
FeatureProviderServer,
NodeValidation,
NodeWithHooks,
PopulationPromise,
ResolvedServerFeature,
ResolvedServerFeatureMap,
SanitizedServerFeatures,
ServerFeature,
ServerFeatureProviderMap,
} from './features/typesServer.js'
export { createNode } from './features/typeUtilities.js' // Only useful in feature.server.ts
export { UploadFeature } from './features/upload/server/feature.server.js'
export type { UploadFeatureProps } from './features/upload/server/feature.server.js'
export { type UploadData, UploadServerNode } from './features/upload/server/nodes/UploadNode.js'
export type { EditorConfigContextType } from './lexical/config/client/EditorConfigProvider.js'
export {
defaultEditorConfig,
defaultEditorFeatures,
defaultEditorLexicalConfig,
} from './lexical/config/server/default.js'
export { loadFeatures, sortFeaturesForOptimalLoading } from './lexical/config/server/loader.js'
export {
sanitizeServerEditorConfig,
sanitizeServerFeatures,
} from './lexical/config/server/sanitize.js'
export type {
ClientEditorConfig,
SanitizedClientEditorConfig,
SanitizedServerEditorConfig,
ServerEditorConfig,
} from './lexical/config/types.js'
export { getEnabledNodes, getEnabledNodesFromServerNodes } from './lexical/nodes/index.js'
export type { AdapterProps }
export type {
SlashMenuGroup,
SlashMenuItem,
} from './lexical/plugins/SlashMenu/LexicalTypeaheadMenuPlugin/types.js'
export {
DETAIL_TYPE_TO_DETAIL,
DOUBLE_LINE_BREAK,
ELEMENT_FORMAT_TO_TYPE,
ELEMENT_TYPE_TO_FORMAT,
IS_ALL_FORMATTING,
LTR_REGEX,
NodeFormat,
NON_BREAKING_SPACE,
RTL_REGEX,
TEXT_MODE_TO_TYPE,
TEXT_TYPE_TO_FORMAT,
TEXT_TYPE_TO_MODE,
} from './lexical/utils/nodeFormat.js'
export { sanitizeUrl, validateUrl } from './lexical/utils/url.js'
export type * from './nodeTypes.js'
export { $convertFromMarkdownString } from './packages/@lexical/markdown/index.js'
export { defaultRichTextValue } from './populateGraphQL/defaultValue.js'
export { populate } from './populateGraphQL/populate.js'
export type { LexicalEditorProps, LexicalRichTextAdapter } from './types.js'
export { createServerFeature } from './utilities/createServerFeature.js'
export type { FieldsDrawerProps } from './utilities/fieldsDrawer/Drawer.js'
export { extractPropsFromJSXPropsString } from './utilities/jsx/extractPropsFromJSXPropsString.js'
export {
extractFrontmatter,
frontmatterToObject,
objectToFrontmatter,
propsToJSXString,
} from './utilities/jsx/jsx.js'
export { upgradeLexicalData } from './utilities/upgradeLexicalData/index.js'