mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-01-07 17:23:49 +08:00
2add15bd72
Also removed duplicated dir functionality that remained in core.
139 lines
4.0 KiB
TypeScript
139 lines
4.0 KiB
TypeScript
import {
|
|
DOMConversionFn,
|
|
DOMConversionMap, EditorConfig,
|
|
LexicalNode,
|
|
Spread
|
|
} from "lexical";
|
|
import {$isListItemNode, ListItemNode, ListNode, ListType, SerializedListNode} from "@lexical/list";
|
|
import {$createCustomListItemNode} from "./custom-list-item";
|
|
import {extractDirectionFromElement} from "./_common";
|
|
|
|
|
|
export type SerializedCustomListNode = Spread<{
|
|
id: string;
|
|
}, SerializedListNode>
|
|
|
|
export class CustomListNode extends ListNode {
|
|
__id: string = '';
|
|
|
|
static getType() {
|
|
return 'custom-list';
|
|
}
|
|
|
|
setId(id: string) {
|
|
const self = this.getWritable();
|
|
self.__id = id;
|
|
}
|
|
|
|
getId(): string {
|
|
const self = this.getLatest();
|
|
return self.__id;
|
|
}
|
|
|
|
static clone(node: CustomListNode) {
|
|
const newNode = new CustomListNode(node.__listType, node.__start, node.__key);
|
|
newNode.__id = node.__id;
|
|
newNode.__dir = node.__dir;
|
|
return newNode;
|
|
}
|
|
|
|
createDOM(config: EditorConfig): HTMLElement {
|
|
const dom = super.createDOM(config);
|
|
if (this.__id) {
|
|
dom.setAttribute('id', this.__id);
|
|
}
|
|
|
|
if (this.__dir) {
|
|
dom.setAttribute('dir', this.__dir);
|
|
}
|
|
|
|
return dom;
|
|
}
|
|
|
|
updateDOM(prevNode: ListNode, dom: HTMLElement, config: EditorConfig): boolean {
|
|
return super.updateDOM(prevNode, dom, config) ||
|
|
prevNode.__dir !== this.__dir;
|
|
}
|
|
|
|
exportJSON(): SerializedCustomListNode {
|
|
return {
|
|
...super.exportJSON(),
|
|
type: 'custom-list',
|
|
version: 1,
|
|
id: this.__id,
|
|
};
|
|
}
|
|
|
|
static importJSON(serializedNode: SerializedCustomListNode): CustomListNode {
|
|
const node = $createCustomListNode(serializedNode.listType);
|
|
node.setId(serializedNode.id);
|
|
node.setDirection(serializedNode.direction);
|
|
return node;
|
|
}
|
|
|
|
static importDOM(): DOMConversionMap | null {
|
|
// @ts-ignore
|
|
const converter = super.importDOM().ol().conversion as DOMConversionFn<HTMLElement>;
|
|
const customConvertFunction = (element: HTMLElement) => {
|
|
const baseResult = converter(element);
|
|
if (element.id && baseResult?.node) {
|
|
(baseResult.node as CustomListNode).setId(element.id);
|
|
}
|
|
|
|
if (element.dir && baseResult?.node) {
|
|
(baseResult.node as CustomListNode).setDirection(extractDirectionFromElement(element));
|
|
}
|
|
|
|
if (baseResult) {
|
|
baseResult.after = $normalizeChildren;
|
|
}
|
|
|
|
return baseResult;
|
|
};
|
|
|
|
return {
|
|
ol: () => ({
|
|
conversion: customConvertFunction,
|
|
priority: 0,
|
|
}),
|
|
ul: () => ({
|
|
conversion: customConvertFunction,
|
|
priority: 0,
|
|
}),
|
|
};
|
|
}
|
|
}
|
|
|
|
/*
|
|
* This function is a custom normalization function to allow nested lists within list item elements.
|
|
* Original taken from https://github.com/facebook/lexical/blob/6e10210fd1e113ccfafdc999b1d896733c5c5bea/packages/lexical-list/src/LexicalListNode.ts#L284-L303
|
|
* With modifications made.
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
* MIT license
|
|
*/
|
|
function $normalizeChildren(nodes: Array<LexicalNode>): Array<ListItemNode> {
|
|
const normalizedListItems: Array<ListItemNode> = [];
|
|
|
|
for (const node of nodes) {
|
|
if ($isListItemNode(node)) {
|
|
normalizedListItems.push(node);
|
|
} else {
|
|
normalizedListItems.push($wrapInListItem(node));
|
|
}
|
|
}
|
|
|
|
return normalizedListItems;
|
|
}
|
|
|
|
function $wrapInListItem(node: LexicalNode): ListItemNode {
|
|
const listItemWrapper = $createCustomListItemNode();
|
|
return listItemWrapper.append(node);
|
|
}
|
|
|
|
export function $createCustomListNode(type: ListType): CustomListNode {
|
|
return new CustomListNode(type, 1);
|
|
}
|
|
|
|
export function $isCustomListNode(node: LexicalNode | null | undefined): node is CustomListNode {
|
|
return node instanceof CustomListNode;
|
|
} |