UNPKG

@zodiac-ui/editor

Version:

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

107 lines 15.5 kB
import { Plugin, PluginKey } from "prosemirror-state"; import { findParentDomRefOfType, findParentNodeOfType } from "prosemirror-utils"; import { codeBlock } from "./code.node"; import { arrowHandlers, codeBlockFactory } from "./code.nodeview"; import { code } from "./code.mark"; export const getPluginState = (state) => pluginKey.getState(state); export const setPluginState = (stateProps) => (state, dispatch) => { const pluginState = getPluginState(state); dispatch(state.tr.setMeta(pluginKey, Object.assign({}, pluginState, stateProps))); return true; }; export const pluginKey = new PluginKey("codeBlockPlugin"); export const createPlugin = ({ dispatch }) => new Plugin({ state: { init(config, state) { return { toolbarVisible: false, }; }, apply(tr, pluginState, oldState, newState) { const nextPluginState = tr.getMeta(pluginKey); if (nextPluginState) { dispatch(pluginKey, nextPluginState); return nextPluginState; } return pluginState; }, }, key: pluginKey, view: () => { return { update: (view, prevState) => { const { state: { selection, schema: { // tslint:disable-next-line:no-shadowed-variable nodes: { codeBlock }, }, }, } = view; const pluginState = getPluginState(view.state); const parentDOM = findParentDomRefOfType(codeBlock, view.domAtPos.bind(view))(selection); const parent = findParentNodeOfType(codeBlock)(selection); let language; if (parent) { language = parent.node.attrs.language; } if (parentDOM !== pluginState.element || pluginState.language !== language) { const newState = { element: parentDOM, language, toolbarVisible: !!parent, }; setPluginState(newState)(view.state, view.dispatch); return true; } /** Plugin dispatch needed to reposition the toolbar */ dispatch(pluginKey, Object.assign({}, pluginState, { language })); }, }; }, props: { nodeViews: { codeBlock: codeBlockFactory, }, handleDOMEvents: { blur(view, event) { const pluginState = getPluginState(view.state); if (pluginState.toolbarVisible) { setPluginState({ toolbarVisible: false, element: null, language: null })(view.state, view.dispatch); return true; } return false; }, }, }, }); export function codeBlockPlugin(options = {}) { return { nodes() { return [{ name: "codeBlock", node: codeBlock }]; }, marks() { return [{ name: 'code', mark: code }]; }, pmPlugins() { return [ { name: "codeBlock", plugin: createPlugin }, { name: "codeBlockKeyHandlers", plugin: () => arrowHandlers } // { // name: "codeBlockIDEKeyBindings", // plugin: () => (options.enableKeybindingsForIDE ? ideUX : undefined), // }, // { // name: "codeBlockKeyMap", // plugin: ({ schema }) => keymap(schema), // }, ]; }, }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code.plugin.js","sourceRoot":"ng://@zodiac-ui/editor/","sources":["plugins/code/code.plugin.ts"],"names":[],"mappings":"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","sourcesContent":["import { EditorState, Plugin, PluginKey } from \"prosemirror-state\"\r\nimport { EditorView } from \"prosemirror-view\"\r\nimport { findParentDomRefOfType, findParentNodeOfType } from \"prosemirror-utils\"\r\nimport { EditorPlugin } from \"../../lib/interfaces/editor-plugin\"\r\nimport { codeBlock } from \"./code.node\"\r\nimport { arrowHandlers, codeBlockFactory } from \"./code.nodeview\"\r\nimport { code } from \"./code.mark\"\r\n\r\nexport interface CodeBlockState {\r\n    element?: HTMLElement\r\n    toolbarVisible?: boolean | undefined\r\n    language?: string\r\n}\r\n\r\nexport const getPluginState = (state: EditorState): CodeBlockState => pluginKey.getState(state)\r\n\r\nexport const setPluginState = (stateProps: Object) => (\r\n    state: EditorState,\r\n    dispatch: (tr) => void,\r\n): boolean => {\r\n    const pluginState = getPluginState(state)\r\n    dispatch(\r\n        state.tr.setMeta(pluginKey, {\r\n            ...pluginState,\r\n            ...stateProps,\r\n        }),\r\n    )\r\n    return true\r\n}\r\n\r\nexport type CodeBlockStateSubscriber = (state: CodeBlockState) => any\r\n\r\nexport const pluginKey = new PluginKey(\"codeBlockPlugin\")\r\n\r\nexport const createPlugin = ({ dispatch }) =>\r\n    new Plugin({\r\n        state: {\r\n            init(config, state: EditorState): CodeBlockState {\r\n                return {\r\n                    toolbarVisible: false,\r\n                }\r\n            },\r\n            apply(tr, pluginState: CodeBlockState, oldState, newState) {\r\n                const nextPluginState = tr.getMeta(pluginKey)\r\n                if (nextPluginState) {\r\n                    dispatch(pluginKey, nextPluginState)\r\n                    return nextPluginState\r\n                }\r\n                return pluginState\r\n            },\r\n        },\r\n        key: pluginKey,\r\n        view: () => {\r\n            return {\r\n                update: (view: EditorView, prevState: EditorState) => {\r\n                    const {\r\n                        state: {\r\n                            selection,\r\n                            schema: {\r\n                                // tslint:disable-next-line:no-shadowed-variable\r\n                                nodes: { codeBlock },\r\n                            },\r\n                        },\r\n                    } = view\r\n                    const pluginState = getPluginState(view.state)\r\n                    const parentDOM = findParentDomRefOfType(codeBlock, view.domAtPos.bind(view))(\r\n                        selection,\r\n                    )\r\n                    const parent = findParentNodeOfType(codeBlock)(selection)\r\n                    let language\r\n\r\n                    if (parent) {\r\n                        language = parent.node.attrs.language\r\n                    }\r\n\r\n                    if (parentDOM !== pluginState.element || pluginState.language !== language) {\r\n                        const newState: CodeBlockState = {\r\n                            element: parentDOM as HTMLElement,\r\n                            language,\r\n                            toolbarVisible: !!parent,\r\n                        }\r\n                        setPluginState(newState)(view.state, view.dispatch)\r\n                        return true\r\n                    }\r\n\r\n                    /** Plugin dispatch needed to reposition the toolbar */\r\n                    dispatch(pluginKey, {\r\n                        ...pluginState,\r\n                        language\r\n                    })\r\n                },\r\n            }\r\n        },\r\n        props: {\r\n            nodeViews: {\r\n                codeBlock: codeBlockFactory,\r\n            },\r\n            handleDOMEvents: {\r\n                blur(view: EditorView, event) {\r\n                    const pluginState = getPluginState(view.state)\r\n                    if (pluginState.toolbarVisible) {\r\n                        setPluginState({\r\n                            toolbarVisible: false,\r\n                            element: null,\r\n                            language: null\r\n                        })(view.state, view.dispatch)\r\n                        return true\r\n                    }\r\n                    return false\r\n                },\r\n            },\r\n        },\r\n    })\r\n\r\nexport interface CodeBlockOptions {\r\n    enableKeybindingsForIDE?: boolean\r\n}\r\n\r\nexport function codeBlockPlugin(options: CodeBlockOptions = {}): EditorPlugin {\r\n    return {\r\n        nodes() {\r\n            return [{ name: \"codeBlock\", node: codeBlock }]\r\n        },\r\n\r\n        marks() {\r\n            return [{\r\n                name: 'code',\r\n                mark: code\r\n            }]\r\n        },\r\n\r\n        pmPlugins() {\r\n            return [\r\n                { name: \"codeBlock\", plugin: createPlugin },\r\n                {\r\n                    name: \"codeBlockKeyHandlers\",\r\n                    plugin: () => arrowHandlers\r\n                }\r\n                // {\r\n                //     name: \"codeBlockIDEKeyBindings\",\r\n                //     plugin: () => (options.enableKeybindingsForIDE ? ideUX : undefined),\r\n                // },\r\n                // {\r\n                //     name: \"codeBlockKeyMap\",\r\n                //     plugin: ({ schema }) => keymap(schema),\r\n                // },\r\n            ]\r\n        },\r\n    }\r\n}\r\n"]}