UNPKG

@zodiac-ui/editor

Version:

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

118 lines 17.3 kB
import { EventEmitter, Inject, Injectable, Optional } from "@angular/core"; import { EditorView } from "prosemirror-view"; import { EditorState, Selection } from "prosemirror-state"; import { Node } from "prosemirror-model"; import { findChangedNodesFromTransaction, validateNodes } from "./utils/nodes"; import { EventDispatcher } from "./interfaces/editor-config"; import { createPMPlugins } from "./utils/create-plugins"; import { createSchema } from "./utils/create-schema"; import { createDispatch } from "./utils/create-dispatch"; import { createConfig } from "./utils/create-config"; import { EDITOR_PLUGIN, STATE_HANDLER } from "./constants"; import { ReplaySubject } from "rxjs"; export const defaultState = { content: [], type: "doc", }; export class EditorService { constructor(plugins, handlers) { this.eventDispatcher = new EventDispatcher(); this.viewChange = new EventEmitter(); this.stateChange = new ReplaySubject(1); this.plugins = plugins; this.handlers = handlers; } runTool(tool) { tool.run(this.state, this.view.dispatch, this.view); } updateState(state) { if (this.view) { this.createEditorState(state); this.view.updateState(this.state); } } createEditorState(state) { this.config = createConfig(this.plugins, {}); const schema = createSchema(this.config); const dispatch = createDispatch(this.eventDispatcher); const doc = Node.fromJSON(schema, state ? state.doc : defaultState); const selection = state ? Selection.fromJSON(doc, state.selection) : undefined; // const errorReporter = createErrorReporter(errorReporterHandler); const plugins = createPMPlugins({ schema, dispatch, editorConfig: this.config, eventDispatcher: this.eventDispatcher, }); // let doc; // if (options.replaceDoc) { // doc = // this.contentTransformer && typeof defaultValue === 'string' // ? this.contentTransformer.parse(defaultValue) // : processRawValue(schema, defaultValue); // } // let selection: Selection | undefined; // if (doc) { // // ED-4759: Don't set selection at end for full-page editor - should be at start // selection = // options.props.editorProps.appearance === 'full-page' // ? Selection.atStart(doc) // : Selection.atEnd(doc); // } // // Workaround for ED-3507: When media node is the last element, scrollIntoView throws an error // const patchedSelection = selection // ? Selection.findFrom(selection.$head, -1, true) || undefined // : undefined; this.state = EditorState.create({ schema, plugins, doc, selection }); } createEditorView(node) { // Creates the editor-view from this.editorState. If an editor has been mounted // previously, this will contain the previous state of the editor. this.view = new EditorView({ mount: node }, { state: this.state, dispatchTransaction: (transaction) => { if (!this.view) { return; } const nodes = findChangedNodesFromTransaction(transaction); if (validateNodes(nodes)) { // go ahead and update the state now we know the transaction is good const editorState = this.view.state.apply(transaction); this.view.updateState(editorState); if (transaction.docChanged) { this.viewChange.emit(this); } this.state = editorState; this.stateChange.next(this); } // else { // const documents = { // new: getDocStructure(transaction.doc), // prev: getDocStructure(transaction.docs[0]), // }; // } }, }); this.viewChange.emit(this); this.stateChange.next(this); if (this.handlers) { this.stateChange.subscribe((editor) => { this.handlers.forEach(handler => handler(editor)); }); } } } EditorService.decorators = [ { type: Injectable } ]; /** @nocollapse */ EditorService.ctorParameters = () => [ { type: Array, decorators: [{ type: Inject, args: [EDITOR_PLUGIN,] }] }, { type: Array, decorators: [{ type: Optional }, { type: Inject, args: [STATE_HANDLER,] }] } ]; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor.service.js","sourceRoot":"ng://@zodiac-ui/editor/","sources":["lib/editor.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAA;AAE1E,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAA;AAC7C,OAAO,EAAE,WAAW,EAAe,SAAS,EAAE,MAAM,mBAAmB,CAAA;AACvE,OAAO,EAAE,IAAI,EAAE,MAAM,mBAAmB,CAAA;AACxC,OAAO,EAAE,+BAA+B,EAAE,aAAa,EAAE,MAAM,eAAe,CAAA;AAC9E,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAA;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAA;AACxD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAA;AACpD,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAA;AACxD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAA;AACpD,OAAO,EAAE,aAAa,EAAE,aAAa,EAAE,MAAM,aAAa,CAAA;AAG1D,OAAO,EAAE,aAAa,EAAE,MAAM,MAAM,CAAA;AAEpC,MAAM,CAAC,MAAM,YAAY,GAAG;IACxB,OAAO,EAAE,EAAE;IACX,IAAI,EAAE,KAAK;CACd,CAAA;AAGD,MAAM,OAAO,aAAa;IAWtB,YAAmC,OAAuB,EAAqC,QAAgB;QAC3G,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,EAAE,CAAC;QAC7C,IAAI,CAAC,UAAU,GAAG,IAAI,YAAY,EAAE,CAAA;QACpC,IAAI,CAAC,WAAW,GAAG,IAAI,aAAa,CAAC,CAAC,CAAC,CAAA;QACvC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAA;QACtB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAA;IAC5B,CAAC;IAED,OAAO,CAAC,IAAgB;QACpB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,CAAA;IACvD,CAAC;IAED,WAAW,CAAC,KAAK;QACb,IAAI,IAAI,CAAC,IAAI,EAAE;YACX,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAA;YAC7B,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;SACpC;IACL,CAAC;IAED,iBAAiB,CAAC,KAAK;QACnB,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAA;QAC5C,MAAM,MAAM,GAAG,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;QACxC,MAAM,QAAQ,GAAG,cAAc,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QACtD,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,CAAA;QACnE,MAAM,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;QAC9E,mEAAmE;QAEnE,MAAM,OAAO,GAAG,eAAe,CAAC;YAC5B,MAAM;YACN,QAAQ;YACR,YAAY,EAAE,IAAI,CAAC,MAAM;YACzB,eAAe,EAAE,IAAI,CAAC,eAAe;SACxC,CAAC,CAAA;QAEF,WAAW;QACX,4BAA4B;QAC5B,YAAY;QACZ,sEAAsE;QACtE,4DAA4D;QAC5D,uDAAuD;QACvD,IAAI;QACJ,wCAAwC;QACxC,aAAa;QACb,uFAAuF;QACvF,kBAAkB;QAClB,+DAA+D;QAC/D,uCAAuC;QACvC,sCAAsC;QACtC,IAAI;QACJ,iGAAiG;QACjG,qCAAqC;QACrC,mEAAmE;QACnE,mBAAmB;QAEnB,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC,MAAM,CAAC;YAC5B,MAAM;YACN,OAAO;YACP,GAAG;YACH,SAAS;SACZ,CAAC,CAAA;IACN,CAAC;IAED,gBAAgB,CAAC,IAAI;QACjB,+EAA+E;QAC/E,kEAAkE;QAClE,IAAI,CAAC,IAAI,GAAG,IAAI,UAAU,CACtB,EAAE,KAAK,EAAE,IAAI,EAAE,EACf;YACI,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,mBAAmB,EAAE,CAAC,WAAwB,EAAE,EAAE;gBAC9C,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE;oBACZ,OAAO;iBACV;gBAED,MAAM,KAAK,GAAW,+BAA+B,CAAC,WAAW,CAAC,CAAC;gBACnE,IAAI,aAAa,CAAC,KAAK,CAAC,EAAE;oBACtB,oEAAoE;oBACpE,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;oBACvD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;oBACnC,IAAI,WAAW,CAAC,UAAU,EAAE;wBACxB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;qBAC9B;oBACD,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC;oBAEzB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;iBAC9B;gBACD,SAAS;gBACT,0BAA0B;gBAC1B,iDAAiD;gBACjD,sDAAsD;gBACtD,SAAS;gBACT,IAAI;YACR,CAAC;SAGJ,CACJ,CAAC;QAEF,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC1B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAE3B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACf,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;gBAClC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAA;YACrD,CAAC,CAAC,CAAA;SACL;IACL,CAAC;;;YAtHJ,UAAU;;;;wCAYM,MAAM,SAAC,aAAa;wCAA4B,QAAQ,YAAI,MAAM,SAAC,aAAa","sourcesContent":["import { EventEmitter, Inject, Injectable, Optional } from \"@angular/core\"\r\nimport { Editor } from \"./interfaces\"\r\nimport { EditorView } from \"prosemirror-view\"\r\nimport { EditorState, Transaction, Selection } from \"prosemirror-state\"\r\nimport { Node } from \"prosemirror-model\"\r\nimport { findChangedNodesFromTransaction, validateNodes } from \"./utils/nodes\"\r\nimport { EventDispatcher } from \"./interfaces/editor-config\"\r\nimport { createPMPlugins } from \"./utils/create-plugins\"\r\nimport { createSchema } from \"./utils/create-schema\"\r\nimport { createDispatch } from \"./utils/create-dispatch\"\r\nimport { createConfig } from \"./utils/create-config\"\r\nimport { EDITOR_PLUGIN, STATE_HANDLER } from \"./constants\"\r\nimport { EditorPlugin } from \"./interfaces/editor-plugin\"\r\nimport { EditorTool } from \"./editor-toolbar/interfaces\"\r\nimport { ReplaySubject } from \"rxjs\"\r\n\r\nexport const defaultState = {\r\n    content: [],\r\n    type: \"doc\",\r\n}\r\n\r\n@Injectable()\r\nexport class EditorService implements Editor {\r\n    public state: EditorState\r\n    public view: EditorView\r\n    private config\r\n    private readonly eventDispatcher\r\n    private readonly plugins\r\n    private readonly handlers?: any[]\r\n\r\n    public viewChange: EventEmitter<any>\r\n    public stateChange: ReplaySubject<any>\r\n\r\n    constructor(@Inject(EDITOR_PLUGIN) plugins: EditorPlugin[], @Optional() @Inject(STATE_HANDLER) handlers?: any[]) {\r\n        this.eventDispatcher = new EventDispatcher();\r\n        this.viewChange = new EventEmitter()\r\n        this.stateChange = new ReplaySubject(1)\r\n        this.plugins = plugins\r\n        this.handlers = handlers\r\n    }\r\n\r\n    runTool(tool: EditorTool) {\r\n        tool.run(this.state, this.view.dispatch, this.view)\r\n    }\r\n\r\n    updateState(state) {\r\n        if (this.view) {\r\n            this.createEditorState(state)\r\n            this.view.updateState(this.state)\r\n        }\r\n    }\r\n\r\n    createEditorState(state) {\r\n        this.config = createConfig(this.plugins, {})\r\n        const schema = createSchema(this.config)\r\n        const dispatch = createDispatch(this.eventDispatcher);\r\n        const doc = Node.fromJSON(schema, state ? state.doc : defaultState)\r\n        const selection = state ? Selection.fromJSON(doc, state.selection) : undefined\r\n        // const errorReporter = createErrorReporter(errorReporterHandler);\r\n\r\n        const plugins = createPMPlugins({\r\n            schema,\r\n            dispatch,\r\n            editorConfig: this.config,\r\n            eventDispatcher: this.eventDispatcher,\r\n        })\r\n\r\n        // let doc;\r\n        // if (options.replaceDoc) {\r\n        //     doc =\r\n        //         this.contentTransformer && typeof defaultValue === 'string'\r\n        //             ? this.contentTransformer.parse(defaultValue)\r\n        //             : processRawValue(schema, defaultValue);\r\n        // }\r\n        // let selection: Selection | undefined;\r\n        // if (doc) {\r\n        //     // ED-4759: Don't set selection at end for full-page editor - should be at start\r\n        //     selection =\r\n        //         options.props.editorProps.appearance === 'full-page'\r\n        //             ? Selection.atStart(doc)\r\n        //             : Selection.atEnd(doc);\r\n        // }\r\n        // // Workaround for ED-3507: When media node is the last element, scrollIntoView throws an error\r\n        // const patchedSelection = selection\r\n        //     ? Selection.findFrom(selection.$head, -1, true) || undefined\r\n        //     : undefined;\r\n\r\n        this.state = EditorState.create({\r\n            schema,\r\n            plugins,\r\n            doc,\r\n            selection\r\n        })\r\n    }\r\n\r\n    createEditorView(node) {\r\n        // Creates the editor-view from this.editorState. If an editor has been mounted\r\n        // previously, this will contain the previous state of the editor.\r\n        this.view = new EditorView(\r\n            { mount: node },\r\n            {\r\n                state: this.state,\r\n                dispatchTransaction: (transaction: Transaction) => {\r\n                    if (!this.view) {\r\n                        return;\r\n                    }\r\n\r\n                    const nodes: Node[] = findChangedNodesFromTransaction(transaction);\r\n                    if (validateNodes(nodes)) {\r\n                        // go ahead and update the state now we know the transaction is good\r\n                        const editorState = this.view.state.apply(transaction);\r\n                        this.view.updateState(editorState);\r\n                        if (transaction.docChanged) {\r\n                            this.viewChange.emit(this);\r\n                        }\r\n                        this.state = editorState;\r\n\r\n                        this.stateChange.next(this)\r\n                    }\r\n                    // else {\r\n                    //     const documents = {\r\n                    //         new: getDocStructure(transaction.doc),\r\n                    //         prev: getDocStructure(transaction.docs[0]),\r\n                    //     };\r\n                    // }\r\n                },\r\n                // Disables the contentEditable attribute of the editor if the editor is disabled\r\n                // editable: state => !this.props.editorProps.disabled,\r\n            },\r\n        );\r\n\r\n        this.viewChange.emit(this)\r\n        this.stateChange.next(this)\r\n\r\n        if (this.handlers) {\r\n            this.stateChange.subscribe((editor) => {\r\n                this.handlers.forEach(handler => handler(editor))\r\n            })\r\n        }\r\n    }\r\n}\r\n"]}