@zodiac-ui/editor
Version:
A rich text editor for Angular based on `@atlaskit/editor-core`.
118 lines • 17.3 kB
JavaScript
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":"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","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"]}