UNPKG

@zodiac-ui/editor

Version:

A rich text editor for Angular based on `@atlaskit/editor-core`.

128 lines 20.2 kB
import { Plugin, PluginKey } from "prosemirror-state"; import { BLOCK_QUOTE, CODE_BLOCK, HEADING_1, HEADING_2, HEADING_3, HEADING_4, HEADING_5, HEADING_6, HEADINGS_BY_LEVEL, NORMAL_TEXT, OTHER, PANEL, TEXT_BLOCK_TYPES, WRAPPER_BLOCK_TYPES, } from "./interfaces"; import { getSelectedWrapperNodes } from "../../lib/utils/get-selected-wrapper-nodes"; import { inputRulePlugin } from "./block-type.inputrule"; import { keymapPlugin } from "./keymap"; export function areBlockTypesDisabled(state) { const nodesTypes = getSelectedWrapperNodes(state); const { panel } = state.schema.nodes; return nodesTypes.filter(type => type !== panel).length > 0; } const blockTypeForNode = (node, schema) => { if (node.type === schema.nodes.heading) { const maybeNode = HEADINGS_BY_LEVEL[node.attrs['level']]; if (maybeNode) { return maybeNode; } } else if (node.type === schema.nodes.paragraph) { return NORMAL_TEXT; } return OTHER; }; const ɵ0 = blockTypeForNode; const isBlockTypeSchemaSupported = (blockType, state) => { switch (blockType) { case NORMAL_TEXT: return !!state.schema.nodes.paragraph; case HEADING_1: case HEADING_2: case HEADING_3: case HEADING_4: case HEADING_5: case HEADING_6: return !!state.schema.nodes.heading; case BLOCK_QUOTE: return !!state.schema.nodes.blockquote; case CODE_BLOCK: return !!state.schema.nodes.codeBlock; case PANEL: return !!state.schema.nodes.panel; } }; const ɵ1 = isBlockTypeSchemaSupported; const detectBlockType = (availableBlockTypes, state) => { // Before a document is loaded, there is no selection. if (!state.selection) { return NORMAL_TEXT; } let blockType; const { $from, $to } = state.selection; state.doc.nodesBetween($from.pos, $to.pos, (node, pos) => { const nodeBlockType = availableBlockTypes.filter( // tslint:disable-next-line:no-shadowed-variable blockType => blockType === blockTypeForNode(node, state.schema)); if (nodeBlockType.length > 0) { if (!blockType) { blockType = nodeBlockType[0]; } else if (blockType !== OTHER && blockType !== nodeBlockType[0]) { blockType = OTHER; } } }); return blockType || OTHER; }; const ɵ2 = detectBlockType; export const pluginKey = new PluginKey('blockTypePlugin'); export const createPlugin = (dispatch) => { return new Plugin({ appendTransaction(transactions, oldState, newState) { // if (appearance === 'comment') { // const pos = newState.doc.resolve(newState.doc.content.size - 1); // const lastNode = pos.node(1); // const { paragraph } = newState.schema.nodes; // if (lastNode && lastNode.isBlock && lastNode.type !== paragraph) { // return newState.tr.insert( // newState.doc.content.size, // newState.schema.nodes.paragraph.create(), // ); // } // } }, state: { init(config, state) { const availableBlockTypes = TEXT_BLOCK_TYPES.filter(blockType => isBlockTypeSchemaSupported(blockType, state)); const availableWrapperBlockTypes = WRAPPER_BLOCK_TYPES.filter(blockType => isBlockTypeSchemaSupported(blockType, state)); return { currentBlockType: detectBlockType(availableBlockTypes, state), blockTypesDisabled: areBlockTypesDisabled(state), availableBlockTypes, availableWrapperBlockTypes, }; }, apply(tr, oldPluginState, oldState, newState) { const newPluginState = Object.assign({}, oldPluginState, { currentBlockType: detectBlockType(oldPluginState.availableBlockTypes, newState), blockTypesDisabled: areBlockTypesDisabled(newState) }); if (newPluginState.currentBlockType !== oldPluginState.currentBlockType || newPluginState.blockTypesDisabled !== oldPluginState.blockTypesDisabled) { dispatch(pluginKey, newPluginState); } return newPluginState; }, }, key: pluginKey, }); }; export const blockTypePlugin = { pmPlugins() { return [ { name: 'blockType', plugin: ({ dispatch }) => createPlugin(dispatch), }, { name: 'blockTypeInputRule', plugin: ({ schema }) => inputRulePlugin(schema), }, // Needs to be lower priority than prosemirror-tables.tableEditing // plugin as it is currently swallowing right/down arrow events inside tables { name: 'blockTypeKeyMap', plugin: ({ schema }) => keymapPlugin(schema), }, ]; }, }; export { ɵ0, ɵ1, ɵ2 }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-type.plugin.js","sourceRoot":"ng://@zodiac-ui/editor/","sources":["plugins/block-type/block-type.plugin.ts"],"names":[],"mappings":"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","sourcesContent":["import { EditorPlugin } from \"../../lib/interfaces/editor-plugin\"\r\n\r\nimport { Node, NodeSpec, NodeType, Schema } from \"prosemirror-model\"\r\nimport { EditorState, Plugin, PluginKey, Transaction } from \"prosemirror-state\"\r\nimport {\r\n    BLOCK_QUOTE,\r\n    BlockType,\r\n    CODE_BLOCK,\r\n    HEADING_1,\r\n    HEADING_2,\r\n    HEADING_3,\r\n    HEADING_4,\r\n    HEADING_5,\r\n    HEADING_6,\r\n    HEADINGS_BY_LEVEL,\r\n    NORMAL_TEXT,\r\n    OTHER,\r\n    PANEL,\r\n    TEXT_BLOCK_TYPES,\r\n    WRAPPER_BLOCK_TYPES,\r\n} from \"./interfaces\"\r\nimport { getSelectedWrapperNodes } from \"../../lib/utils/get-selected-wrapper-nodes\"\r\nimport { inputRulePlugin } from \"./block-type.inputrule\"\r\nimport { keymapPlugin } from \"./keymap\"\r\n\r\n\r\nexport interface BlockTypeState {\r\n    currentBlockType: BlockType;\r\n    blockTypesDisabled: boolean;\r\n    availableBlockTypes: BlockType[];\r\n    availableWrapperBlockTypes: BlockType[];\r\n}\r\n\r\nexport function areBlockTypesDisabled(state: EditorState): boolean {\r\n    const nodesTypes: NodeType[] = getSelectedWrapperNodes(state);\r\n    const { panel } = state.schema.nodes;\r\n    return nodesTypes.filter(type => type !== panel).length > 0;\r\n}\r\n\r\nconst blockTypeForNode = (node: Node, schema: Schema): BlockType => {\r\n    if (node.type === schema.nodes.heading) {\r\n        const maybeNode = HEADINGS_BY_LEVEL[node.attrs['level']];\r\n        if (maybeNode) {\r\n            return maybeNode;\r\n        }\r\n    } else if (node.type === schema.nodes.paragraph) {\r\n        return NORMAL_TEXT;\r\n    }\r\n    return OTHER;\r\n};\r\n\r\nconst isBlockTypeSchemaSupported = (\r\n    blockType: BlockType,\r\n    state: EditorState,\r\n) => {\r\n    switch (blockType) {\r\n        case NORMAL_TEXT:\r\n            return !!state.schema.nodes.paragraph;\r\n        case HEADING_1:\r\n        case HEADING_2:\r\n        case HEADING_3:\r\n        case HEADING_4:\r\n        case HEADING_5:\r\n        case HEADING_6:\r\n            return !!state.schema.nodes.heading;\r\n        case BLOCK_QUOTE:\r\n            return !!state.schema.nodes.blockquote;\r\n        case CODE_BLOCK:\r\n            return !!state.schema.nodes.codeBlock;\r\n        case PANEL:\r\n            return !!state.schema.nodes.panel;\r\n    }\r\n};\r\n\r\nconst detectBlockType = (\r\n    availableBlockTypes: BlockType[],\r\n    state: EditorState,\r\n): BlockType => {\r\n    // Before a document is loaded, there is no selection.\r\n    if (!state.selection) {\r\n        return NORMAL_TEXT;\r\n    }\r\n    let blockType;\r\n    const { $from, $to } = state.selection;\r\n    state.doc.nodesBetween($from.pos, $to.pos, (node, pos) => {\r\n        const nodeBlockType = availableBlockTypes.filter(\r\n            // tslint:disable-next-line:no-shadowed-variable\r\n            blockType => blockType === blockTypeForNode(node, state.schema),\r\n        );\r\n        if (nodeBlockType.length > 0) {\r\n            if (!blockType) {\r\n                blockType = nodeBlockType[0];\r\n            } else if (blockType !== OTHER && blockType !== nodeBlockType[0]) {\r\n                blockType = OTHER;\r\n            }\r\n        }\r\n    });\r\n    return blockType || OTHER;\r\n};\r\n\r\nexport const pluginKey = new PluginKey('blockTypePlugin');\r\n\r\nexport const createPlugin = (\r\n    dispatch: (eventName: string | PluginKey, data: any) => void,\r\n    // appearance?,\r\n) => {\r\n    return new Plugin({\r\n        appendTransaction(\r\n            transactions: Transaction[],\r\n            oldState: EditorState,\r\n            newState: EditorState,\r\n        ): Transaction | void {\r\n            // if (appearance === 'comment') {\r\n            //     const pos = newState.doc.resolve(newState.doc.content.size - 1);\r\n            //     const lastNode = pos.node(1);\r\n            //     const { paragraph } = newState.schema.nodes;\r\n            //     if (lastNode && lastNode.isBlock && lastNode.type !== paragraph) {\r\n            //         return newState.tr.insert(\r\n            //             newState.doc.content.size,\r\n            //             newState.schema.nodes.paragraph.create(),\r\n            //         );\r\n            //     }\r\n            // }\r\n        },\r\n\r\n        state: {\r\n            init(config, state: EditorState) {\r\n                const availableBlockTypes = TEXT_BLOCK_TYPES.filter(blockType =>\r\n                    isBlockTypeSchemaSupported(blockType, state),\r\n                );\r\n                const availableWrapperBlockTypes = WRAPPER_BLOCK_TYPES.filter(\r\n                    blockType => isBlockTypeSchemaSupported(blockType, state),\r\n                );\r\n\r\n                return {\r\n                    currentBlockType: detectBlockType(availableBlockTypes, state),\r\n                    blockTypesDisabled: areBlockTypesDisabled(state),\r\n                    availableBlockTypes,\r\n                    availableWrapperBlockTypes,\r\n                };\r\n            },\r\n\r\n            apply(\r\n                tr,\r\n                oldPluginState: BlockTypeState,\r\n                oldState: EditorState,\r\n                newState: EditorState,\r\n            ) {\r\n                const newPluginState = {\r\n                    ...oldPluginState,\r\n                    currentBlockType: detectBlockType(\r\n                        oldPluginState.availableBlockTypes,\r\n                        newState,\r\n                    ),\r\n                    blockTypesDisabled: areBlockTypesDisabled(newState),\r\n                };\r\n\r\n                if (\r\n                    newPluginState.currentBlockType !== oldPluginState.currentBlockType ||\r\n                    newPluginState.blockTypesDisabled !==\r\n                    oldPluginState.blockTypesDisabled\r\n                ) {\r\n                    dispatch(pluginKey, newPluginState);\r\n                }\r\n\r\n                return newPluginState;\r\n            },\r\n        },\r\n\r\n        key: pluginKey,\r\n    });\r\n};\r\n\r\nexport type AllowedBlockTypes =\r\n    | 'heading'\r\n    | 'blockquote'\r\n    | 'hardBreak'\r\n    | 'codeBlock';\r\n\r\nexport interface BlockTypeNode {\r\n    name: AllowedBlockTypes;\r\n    node: NodeSpec;\r\n}\r\n\r\nexport const blockTypePlugin: EditorPlugin = {\r\n    pmPlugins() {\r\n        return [\r\n            {\r\n                name: 'blockType',\r\n                plugin: ({ dispatch }) => createPlugin(dispatch),\r\n            },\r\n            {\r\n                name: 'blockTypeInputRule',\r\n                plugin: ({ schema }) => inputRulePlugin(schema),\r\n            },\r\n            // Needs to be lower priority than prosemirror-tables.tableEditing\r\n            // plugin as it is currently swallowing right/down arrow events inside tables\r\n            {\r\n                name: 'blockTypeKeyMap',\r\n                plugin: ({ schema }) => keymapPlugin(schema),\r\n            },\r\n        ];\r\n    },\r\n}\r\n"]}