UNPKG

ngx-editor

Version:

Rich Text Editor for angular using ProseMirror

109 lines 13.5 kB
import { EditorState } from 'prosemirror-state'; import { EditorView } from 'prosemirror-view'; import { Subject } from 'rxjs'; import { isNil } from 'ngx-editor/utils'; import EditorCommands from './EditorCommands'; import defautlSchema from './schema'; import { parseContent } from './parsers'; import getDefaultPlugins from './defaultPlugins'; const defaultFeatures = { linkOnPaste: true, resizeImage: true, }; const DEFAULT_OPTIONS = { content: null, history: true, keyboardShortcuts: true, inputRules: true, schema: defautlSchema, plugins: [], nodeViews: {}, attributes: {}, features: defaultFeatures, handleScrollToSelection: null, }; class Editor { constructor(options = DEFAULT_OPTIONS) { this.valueChangesSubject = new Subject(); this.updateSubject = new Subject(); this.options = { ...DEFAULT_OPTIONS, ...options }; this.createEditor(); } get valueChanges() { return this.valueChangesSubject.asObservable(); } get update() { return this.updateSubject.asObservable(); } get schema() { return this.options.schema || defautlSchema; } get commands() { return new EditorCommands(this.view); } get features() { return { ...defaultFeatures, ...this.options.features }; } handleTransactions(tr) { const state = this.view.state.apply(tr); this.view.updateState(state); this.updateSubject.next(this.view); if (!tr.docChanged && !tr.getMeta('FORCE_EMIT')) { return; } const json = state.doc.toJSON(); this.valueChangesSubject.next(json); } createEditor() { const { options, schema } = this; const { content = null, nodeViews } = options; const { history = true, keyboardShortcuts = true, inputRules = true } = options; const doc = parseContent(content, schema); const plugins = options.plugins ?? []; const attributes = options.attributes ?? {}; const defaultPlugins = getDefaultPlugins(schema, { history, keyboardShortcuts, inputRules, }); this.view = new EditorView(null, { state: EditorState.create({ doc, schema, plugins: [...defaultPlugins, ...plugins], }), nodeViews, dispatchTransaction: this.handleTransactions.bind(this), attributes, handleScrollToSelection: options.handleScrollToSelection, }); } setContent(content) { if (isNil(content)) { return; } const { state } = this.view; const { tr, doc } = state; const newDoc = parseContent(content, this.schema); tr.replaceWith(0, state.doc.content.size, newDoc); // don't emit if both content is same if (doc.eq(tr.doc)) { return; } if (!tr.docChanged) { return; } this.view.dispatch(tr); } registerPlugin(plugin) { const { state } = this.view; const plugins = [...state.plugins, plugin]; const newState = state.reconfigure({ plugins }); this.view.updateState(newState); } destroy() { this.view.destroy(); } } export default Editor; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Editor.js","sourceRoot":"","sources":["../../../../projects/ngx-editor/src/lib/Editor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Schema } from 'prosemirror-model';\nimport { EditorState, Plugin, Transaction } from 'prosemirror-state';\nimport { EditorProps, EditorView } from 'prosemirror-view';\nimport { Observable, Subject } from 'rxjs';\n\nimport { isNil } from 'ngx-editor/utils';\n\nimport EditorCommands from './EditorCommands';\nimport defautlSchema from './schema';\nimport { parseContent } from './parsers';\nimport getDefaultPlugins from './defaultPlugins';\n\ntype JSONDoc = Record<string, any>;\ntype Content = string | null | JSONDoc;\n\ninterface Options {\n  content?: Content;\n  history?: boolean;\n  keyboardShortcuts?: boolean;\n  inputRules?: boolean;\n  schema?: Schema;\n  plugins?: Plugin[];\n  nodeViews?: EditorProps['nodeViews'];\n  attributes?: EditorProps['attributes'];\n  features?: EditorFeatures;\n  handleScrollToSelection?: EditorProps['handleScrollToSelection'];\n}\n\ninterface EditorFeatures {\n  linkOnPaste?: boolean;\n  resizeImage?: boolean;\n}\n\nconst defaultFeatures = {\n  linkOnPaste: true,\n  resizeImage: true,\n};\n\nconst DEFAULT_OPTIONS: Options = {\n  content: null,\n  history: true,\n  keyboardShortcuts: true,\n  inputRules: true,\n  schema: defautlSchema,\n  plugins: [],\n  nodeViews: {},\n  attributes: {},\n  features: defaultFeatures,\n  handleScrollToSelection: null,\n};\n\nclass Editor {\n  private options: Options;\n  view: EditorView;\n\n  constructor(options: Options = DEFAULT_OPTIONS) {\n    this.options = { ...DEFAULT_OPTIONS, ...options };\n    this.createEditor();\n  }\n\n  private valueChangesSubject = new Subject<JSONDoc>();\n  private updateSubject = new Subject<EditorView>();\n\n  get valueChanges(): Observable<JSONDoc> {\n    return this.valueChangesSubject.asObservable();\n  }\n\n  get update(): Observable<EditorView> {\n    return this.updateSubject.asObservable();\n  }\n\n  get schema(): Schema {\n    return this.options.schema || defautlSchema;\n  }\n\n  get commands(): EditorCommands {\n    return new EditorCommands(this.view);\n  }\n\n  get features(): EditorFeatures {\n    return { ...defaultFeatures, ...this.options.features };\n  }\n\n  private handleTransactions(tr: Transaction): void {\n    const state = this.view.state.apply(tr);\n    this.view.updateState(state);\n\n    this.updateSubject.next(this.view);\n\n    if (!tr.docChanged && !tr.getMeta('FORCE_EMIT')) {\n      return;\n    }\n\n    const json = state.doc.toJSON();\n    this.valueChangesSubject.next(json);\n  }\n\n  private createEditor(): void {\n    const { options, schema } = this;\n    const { content = null, nodeViews } = options;\n    const { history = true, keyboardShortcuts = true, inputRules = true } = options;\n\n    const doc = parseContent(content, schema);\n\n    const plugins: Plugin[] = options.plugins ?? [];\n    const attributes: EditorProps['attributes'] = options.attributes ?? {};\n\n    const defaultPlugins = getDefaultPlugins(schema, {\n      history,\n      keyboardShortcuts,\n      inputRules,\n    });\n\n    this.view = new EditorView(null, {\n      state: EditorState.create({\n        doc,\n        schema,\n        plugins: [...defaultPlugins, ...plugins],\n      }),\n      nodeViews,\n      dispatchTransaction: this.handleTransactions.bind(this),\n      attributes,\n      handleScrollToSelection: options.handleScrollToSelection,\n    });\n  }\n\n  setContent(content: Content): void {\n    if (isNil(content)) {\n      return;\n    }\n\n    const { state } = this.view;\n    const { tr, doc } = state;\n\n    const newDoc = parseContent(content, this.schema);\n\n    tr.replaceWith(0, state.doc.content.size, newDoc);\n\n    // don't emit if both content is same\n    if (doc.eq(tr.doc)) {\n      return;\n    }\n\n    if (!tr.docChanged) {\n      return;\n    }\n\n    this.view.dispatch(tr);\n  }\n\n  registerPlugin(plugin: Plugin): void {\n    const { state } = this.view;\n    const plugins = [...state.plugins, plugin];\n\n    const newState = state.reconfigure({ plugins });\n    this.view.updateState(newState);\n  }\n\n  destroy(): void {\n    this.view.destroy();\n  }\n}\n\nexport default Editor;\n"]}