UNPKG

ngx-editor

Version:

Rich Text Editor for angular using ProseMirror

117 lines 18.3 kB
import { keymap } from 'prosemirror-keymap'; import { toggleMark, baseKeymap, chainCommands, exitCode } from 'prosemirror-commands'; import { splitListItem, liftListItem, sinkListItem } from 'prosemirror-schema-list'; import { history, undo, redo } from 'prosemirror-history'; import { inputRules, wrappingInputRule, textblockTypeInputRule, smartQuotes, emDash, ellipsis, } from 'prosemirror-inputrules'; import { markInputRule } from 'ngx-editor/helpers'; const isMacOs = typeof navigator !== 'undefined' ? (/Mac/).test(navigator.platform) : false; // Input rules ref: https://github.com/ProseMirror/prosemirror-example-setup/ // : (NodeType) → InputRule // Given a blockquote node type, returns an input rule that turns `"> "` // at the start of a textblock into a blockquote. const blockQuoteRule = (nodeType) => { return wrappingInputRule(/^\s*>\s$/, nodeType); }; // : (NodeType) → InputRule // Given a list node type, returns an input rule that turns a number // followed by a dot at the start of a textblock into an ordered list. const orderedListRule = (nodeType) => { return wrappingInputRule(/^(?:\d+)\.\s$/, nodeType, (match) => ({ order: Number(match[1]) }), (match, node) => node.childCount + node.attrs['order'] === Number(match[1])); }; // : (NodeType) → InputRule // Given a list node type, returns an input rule that turns a bullet // (dash, plush, or asterisk) at the start of a textblock into a // bullet list. const bulletListRule = (nodeType) => { return wrappingInputRule(/^\s*(?:[-+*])\s$/, nodeType); }; // : (NodeType) → InputRule // Given a code block node type, returns an input rule that turns a // textblock starting with three backticks into a code block. const codeBlockRule = (nodeType) => { return textblockTypeInputRule(/^```$/, nodeType); }; // : (NodeType, number) → InputRule // Given a node type and a maximum level, creates an input rule that // turns up to that number of `#` characters followed by a space at // the start of a textblock into a heading whose level corresponds to // the number of `#` signs. const headingRule = (nodeType, maxLevel) => { return textblockTypeInputRule(new RegExp(`^(#{1,${maxLevel}})\\s$`), nodeType, (match) => ({ level: match[1].length })); }; // : (MarkType) → InputRule // Wraps matching text with bold mark const boldRule = (markType) => { return markInputRule(/(?:^|\s)(?:(?:\*\*|__)(?:(?:[^*_]+))(?:\*\*|__))$/, markType); }; // : (MarkType) → InputRule // Wraps matching text with em mark const emRule = (markType) => { return markInputRule(/(?:^|\s)(?:(?:\*|_)(?:(?:[^*_]+))(?:\*|_))$/, markType); }; // : (Schema) → Plugin // A set of input rules for creating the basic block quotes, lists, // code blocks, and heading. const buildInputRules = (schema) => { const rules = smartQuotes.concat(ellipsis, emDash); rules.push(boldRule(schema.marks['strong'])); rules.push(emRule(schema.marks['em'])); rules.push(blockQuoteRule(schema.nodes['blockquote'])); rules.push(orderedListRule(schema.nodes['ordered_list'])); rules.push(bulletListRule(schema.nodes['bullet_list'])); rules.push(codeBlockRule(schema.nodes['code_block'])); rules.push(headingRule(schema.nodes['heading'], 6)); return inputRules({ rules }); }; const getKeyboardShortcuts = (schema, options) => { const historyKeyMap = {}; historyKeyMap['Mod-z'] = undo; if (isMacOs) { historyKeyMap['Shift-Mod-z'] = redo; } else { historyKeyMap['Mod-y'] = redo; } const plugins = [ keymap({ 'Mod-b': toggleMark(schema.marks['strong']), 'Mod-i': toggleMark(schema.marks['em']), 'Mod-u': toggleMark(schema.marks['u']), 'Mod-`': toggleMark(schema.marks['code']), }), keymap({ 'Enter': splitListItem(schema.nodes['list_item']), 'Shift-Enter': chainCommands(exitCode, (state, dispatch) => { const { tr } = state; const br = schema.nodes['hard_break']; dispatch(tr.replaceSelectionWith(br.create()).scrollIntoView()); return true; }), 'Mod-[': liftListItem(schema.nodes['list_item']), 'Mod-]': sinkListItem(schema.nodes['list_item']), 'Tab': sinkListItem(schema.nodes['list_item']), }), keymap(baseKeymap), ]; if (options.history) { plugins.push(keymap(historyKeyMap)); } return plugins; }; const getDefaultPlugins = (schema, options) => { const plugins = []; if (options.keyboardShortcuts) { plugins.push(...getKeyboardShortcuts(schema, { history: options.history })); } if (options.history) { plugins.push(history()); } if (options.inputRules) { plugins.push(buildInputRules(schema)); } return plugins; }; export default getDefaultPlugins; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"defaultPlugins.js","sourceRoot":"","sources":["../../../../projects/ngx-editor/src/lib/defaultPlugins.ts"],"names":[],"mappings":"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","sourcesContent":["import { MarkType, NodeType, Schema } from 'prosemirror-model';\nimport { Plugin } from 'prosemirror-state';\nimport { keymap } from 'prosemirror-keymap';\nimport { toggleMark, baseKeymap, chainCommands, exitCode } from 'prosemirror-commands';\nimport { splitListItem, liftListItem, sinkListItem } from 'prosemirror-schema-list';\nimport { history, undo, redo } from 'prosemirror-history';\nimport {\n  inputRules, wrappingInputRule, textblockTypeInputRule,\n  smartQuotes, emDash, ellipsis, InputRule,\n} from 'prosemirror-inputrules';\n\nimport { markInputRule } from 'ngx-editor/helpers';\n\ninterface Options {\n  history: boolean;\n  keyboardShortcuts: boolean;\n  inputRules: boolean;\n}\n\ninterface ShortcutOptions {\n  history: boolean;\n}\n\nconst isMacOs = typeof navigator !== 'undefined'\n  ? (/Mac/).test(navigator.platform)\n  : false;\n\n// Input rules ref: https://github.com/ProseMirror/prosemirror-example-setup/\n\n// : (NodeType) → InputRule\n// Given a blockquote node type, returns an input rule that turns `\"> \"`\n// at the start of a textblock into a blockquote.\nconst blockQuoteRule = (nodeType: NodeType): InputRule => {\n  return wrappingInputRule(/^\\s*>\\s$/, nodeType);\n};\n\n// : (NodeType) → InputRule\n// Given a list node type, returns an input rule that turns a number\n// followed by a dot at the start of a textblock into an ordered list.\nconst orderedListRule = (nodeType: NodeType): InputRule => {\n  return wrappingInputRule(\n    /^(?:\\d+)\\.\\s$/,\n    nodeType,\n    (match) => ({ order: Number(match[1]) }),\n    (match, node) => node.childCount + node.attrs['order'] === Number(match[1]),\n  );\n};\n\n// : (NodeType) → InputRule\n// Given a list node type, returns an input rule that turns a bullet\n// (dash, plush, or asterisk) at the start of a textblock into a\n// bullet list.\nconst bulletListRule = (nodeType: NodeType): InputRule => {\n  return wrappingInputRule(/^\\s*(?:[-+*])\\s$/, nodeType);\n};\n\n// : (NodeType) → InputRule\n// Given a code block node type, returns an input rule that turns a\n// textblock starting with three backticks into a code block.\nconst codeBlockRule = (nodeType: NodeType): InputRule => {\n  return textblockTypeInputRule(/^```$/, nodeType);\n};\n\n// : (NodeType, number) → InputRule\n// Given a node type and a maximum level, creates an input rule that\n// turns up to that number of `#` characters followed by a space at\n// the start of a textblock into a heading whose level corresponds to\n// the number of `#` signs.\nconst headingRule = (nodeType: NodeType, maxLevel: number): InputRule => {\n  return textblockTypeInputRule(\n    new RegExp(`^(#{1,${maxLevel}})\\\\s$`),\n    nodeType,\n    (match) => ({ level: match[1].length }),\n  );\n};\n\n// : (MarkType) → InputRule\n// Wraps matching text with bold mark\nconst boldRule = (markType: MarkType): InputRule => {\n  return markInputRule(/(?:^|\\s)(?:(?:\\*\\*|__)(?:(?:[^*_]+))(?:\\*\\*|__))$/, markType);\n};\n\n// : (MarkType) → InputRule\n// Wraps matching text with em mark\nconst emRule = (markType: MarkType): InputRule => {\n  return markInputRule(/(?:^|\\s)(?:(?:\\*|_)(?:(?:[^*_]+))(?:\\*|_))$/, markType);\n};\n\n// : (Schema) → Plugin\n// A set of input rules for creating the basic block quotes, lists,\n// code blocks, and heading.\nconst buildInputRules = (schema: Schema): Plugin => {\n  const rules = smartQuotes.concat(ellipsis, emDash);\n\n  rules.push(boldRule(schema.marks['strong']));\n  rules.push(emRule(schema.marks['em']));\n  rules.push(blockQuoteRule(schema.nodes['blockquote']));\n  rules.push(orderedListRule(schema.nodes['ordered_list']));\n  rules.push(bulletListRule(schema.nodes['bullet_list']));\n  rules.push(codeBlockRule(schema.nodes['code_block']));\n  rules.push(headingRule(schema.nodes['heading'], 6));\n\n  return inputRules({ rules });\n};\n\nconst getKeyboardShortcuts = (schema: Schema, options: ShortcutOptions) => {\n  const historyKeyMap: Record<string, any> = {};\n\n  historyKeyMap['Mod-z'] = undo;\n  if (isMacOs) {\n    historyKeyMap['Shift-Mod-z'] = redo;\n  } else {\n    historyKeyMap['Mod-y'] = redo;\n  }\n\n  const plugins = [\n    keymap({\n      'Mod-b': toggleMark(schema.marks['strong']),\n      'Mod-i': toggleMark(schema.marks['em']),\n      'Mod-u': toggleMark(schema.marks['u']),\n      'Mod-`': toggleMark(schema.marks['code']),\n    }),\n    keymap({\n      'Enter': splitListItem(schema.nodes['list_item']),\n      'Shift-Enter': chainCommands(exitCode, (state, dispatch) => {\n        const { tr } = state;\n        const br = schema.nodes['hard_break'];\n        dispatch(tr.replaceSelectionWith(br.create()).scrollIntoView());\n        return true;\n      }),\n      'Mod-[': liftListItem(schema.nodes['list_item']),\n      'Mod-]': sinkListItem(schema.nodes['list_item']),\n      'Tab': sinkListItem(schema.nodes['list_item']),\n    }),\n    keymap(baseKeymap),\n  ];\n\n  if (options.history) {\n    plugins.push(keymap(historyKeyMap));\n  }\n\n  return plugins;\n};\n\nconst getDefaultPlugins = (schema: Schema, options: Options): Plugin[] => {\n  const plugins: Plugin[] = [];\n\n  if (options.keyboardShortcuts) {\n    plugins.push(...getKeyboardShortcuts(schema, { history: options.history }));\n  }\n\n  if (options.history) {\n    plugins.push(history());\n  }\n\n  if (options.inputRules) {\n    plugins.push(buildInputRules(schema));\n  }\n\n  return plugins;\n};\n\nexport default getDefaultPlugins;\n"]}