From d8b2ef410a1e0dee6c0f5d665ff216b2620a4c7e Mon Sep 17 00:00:00 2001 From: chris <kokosias@yahoo.gr> Date: Mon, 17 Jan 2022 03:15:55 +0200 Subject: [PATCH] add editors --- .../src/EssayService/EssayAnswerNodeView.js | 23 +- .../src/EssayService/EssayQuestionNodeView.js | 23 +- .../components/EssayAnswerComponent.js | 194 ++++++++++++++++- .../components/EssayQuestionComponent.js | 196 +++++++++++++++++- 4 files changed, 422 insertions(+), 14 deletions(-) diff --git a/wax-prosemirror-services/src/EssayService/EssayAnswerNodeView.js b/wax-prosemirror-services/src/EssayService/EssayAnswerNodeView.js index 01c9fd33d..36cda93ba 100644 --- a/wax-prosemirror-services/src/EssayService/EssayAnswerNodeView.js +++ b/wax-prosemirror-services/src/EssayService/EssayAnswerNodeView.js @@ -23,14 +23,29 @@ export default class EssayAnswerNodeView extends AbstractNodeView { } update(node) { + this.node = node; + if (this.context.view[node.attrs.id]) { + const { state } = this.context.view[node.attrs.id]; + const start = node.content.findDiffStart(state.doc.content); + if (start != null) { + let { a: endA, b: endB } = node.content.findDiffEnd(state.doc.content); + const overlap = start - Math.min(endA, endB); + if (overlap > 0) { + endA += overlap; + endB += overlap; + } + this.context.view[node.attrs.id].dispatch( + state.tr + .replace(start, endB, node.slice(start, endA)) + .setMeta('fromOutside', true), + ); + } + } + return true; } stopEvent(event) { - console.log(event.target.type); - if (event.target.type === 'textarea') { - return true; - } const innerView = this.context.view[this.node.attrs.id]; return innerView && innerView.dom.contains(event.target); } diff --git a/wax-prosemirror-services/src/EssayService/EssayQuestionNodeView.js b/wax-prosemirror-services/src/EssayService/EssayQuestionNodeView.js index 867d0dcd1..c69fefc4c 100644 --- a/wax-prosemirror-services/src/EssayService/EssayQuestionNodeView.js +++ b/wax-prosemirror-services/src/EssayService/EssayQuestionNodeView.js @@ -23,14 +23,29 @@ export default class EssayQuestionNodeView extends AbstractNodeView { } update(node) { + this.node = node; + if (this.context.view[node.attrs.id]) { + const { state } = this.context.view[node.attrs.id]; + const start = node.content.findDiffStart(state.doc.content); + if (start != null) { + let { a: endA, b: endB } = node.content.findDiffEnd(state.doc.content); + const overlap = start - Math.min(endA, endB); + if (overlap > 0) { + endA += overlap; + endB += overlap; + } + this.context.view[node.attrs.id].dispatch( + state.tr + .replace(start, endB, node.slice(start, endA)) + .setMeta('fromOutside', true), + ); + } + } + return true; } stopEvent(event) { - console.log(event.target.type); - if (event.target.type === 'textarea') { - return true; - } const innerView = this.context.view[this.node.attrs.id]; return innerView && innerView.dom.contains(event.target); } diff --git a/wax-prosemirror-services/src/EssayService/components/EssayAnswerComponent.js b/wax-prosemirror-services/src/EssayService/components/EssayAnswerComponent.js index 42e07f530..9792280a5 100644 --- a/wax-prosemirror-services/src/EssayService/components/EssayAnswerComponent.js +++ b/wax-prosemirror-services/src/EssayService/components/EssayAnswerComponent.js @@ -1,7 +1,195 @@ +/* eslint-disable react/destructuring-assignment */ /* eslint-disable react/prop-types */ -import React from 'react'; + +import React, { useContext, useRef, useEffect } from 'react'; import styled from 'styled-components'; +import { EditorView } from 'prosemirror-view'; +import { EditorState, TextSelection, NodeSelection } from 'prosemirror-state'; +import { StepMap } from 'prosemirror-transform'; +import { keymap } from 'prosemirror-keymap'; +import { baseKeymap, chainCommands } from 'prosemirror-commands'; +import { undo, redo } from 'prosemirror-history'; +import { WaxContext } from 'wax-prosemirror-core'; +import { + splitListItem, + liftListItem, + sinkListItem, +} from 'prosemirror-schema-list'; +import Placeholder from '../plugins/placeholder'; + +const EditorWrapper = styled.div` + border: none; + display: flex; + flex: 2 1 auto; + justify-content: left; + + .ProseMirror { + white-space: break-spaces; + width: 100%; + word-wrap: break-word; + + &:focus { + outline: none; + } + + p.empty-node:first-child::before { + content: attr(data-content); + } + + .empty-node::before { + color: rgb(170, 170, 170); + float: left; + font-style: italic; + height: 0px; + pointer-events: none; + } + } +`; +const EssayAnswerComponent = ({ node, view, getPos }) => { + const editorRef = useRef(); + + const context = useContext(WaxContext); + let essayAnswerView; + const questionId = node.attrs.id; + const isEditable = context.view.main.props.editable(editable => { + return editable; + }); + + let finalPlugins = []; + + const createKeyBindings = () => { + const keys = getKeys(); + Object.keys(baseKeymap).forEach(key => { + if (keys[key]) { + keys[key] = chainCommands(keys[key], baseKeymap[key]); + } else { + keys[key] = baseKeymap[key]; + } + }); + return keys; + }; + + const pressEnter = (state, dispatch) => { + if (state.selection.node && state.selection.node.type.name === 'image') { + const { $from, to } = state.selection; + + const same = $from.sharedDepth(to); + + const pos = $from.before(same); + dispatch(state.tr.setSelection(NodeSelection.create(state.doc, pos))); + return true; + } + // LISTS + if (splitListItem(state.schema.nodes.list_item)(state)) { + splitListItem(state.schema.nodes.list_item)(state, dispatch); + return true; + } -export default ({ node, view, getPos }) => { - return <span>Answer</span>; + return false; + }; + + const getKeys = () => { + return { + 'Mod-z': () => undo(view.state, view.dispatch), + 'Mod-y': () => redo(view.state, view.dispatch), + 'Mod-[': liftListItem(view.state.schema.nodes.list_item), + 'Mod-]': sinkListItem(view.state.schema.nodes.list_item), + Enter: pressEnter, + }; + }; + + const plugins = [keymap(createKeyBindings()), ...context.app.getPlugins()]; + + // eslint-disable-next-line no-shadow + const createPlaceholder = placeholder => { + return Placeholder({ + content: placeholder, + }); + }; + + finalPlugins = finalPlugins.concat([ + createPlaceholder('Type your essay answer'), + ...plugins, + ]); + + useEffect(() => { + essayAnswerView = new EditorView( + { + mount: editorRef.current, + }, + { + editable: () => isEditable, + state: EditorState.create({ + doc: node, + plugins: finalPlugins, + }), + // This is the magic part + dispatchTransaction, + disallowedTools: ['MultipleChoice'], + handleDOMEvents: { + mousedown: () => { + context.updateView({}, questionId); + context.view.main.dispatch( + context.view.main.state.tr + .setMeta('outsideView', questionId) + .setSelection( + new TextSelection( + context.view.main.state.tr.doc.resolve( + getPos() + + 2 + + context.view[questionId].state.selection.to, + ), + ), + ), + ); + context.updateView({}, questionId); + + // Kludge to prevent issues due to the fact that the whole + // footnote is node-selected (and thus DOM-selected) when + // the parent editor is focused. + if (essayAnswerView.hasFocus()) essayAnswerView.focus(); + }, + }, + + attributes: { + spellcheck: 'false', + }, + }, + ); + + // Set Each note into Wax's Context + context.updateView( + { + [questionId]: essayAnswerView, + }, + questionId, + ); + if (essayAnswerView.hasFocus()) essayAnswerView.focus(); + }, []); + + const dispatchTransaction = tr => { + const outerTr = context.view.main.state.tr; + context.view.main.dispatch(outerTr.setMeta('outsideView', questionId)); + const { state, transactions } = essayAnswerView.state.applyTransaction(tr); + context.updateView({}, questionId); + essayAnswerView.updateState(state); + if (!tr.getMeta('fromOutside')) { + const offsetMap = StepMap.offset(getPos() + 1); + for (let i = 0; i < transactions.length; i++) { + const { steps } = transactions[i]; + for (let j = 0; j < steps.length; j++) + outerTr.step(steps[j].map(offsetMap)); + } + if (outerTr.docChanged) + context.view.main.dispatch(outerTr.setMeta('outsideView', questionId)); + } + }; + + return ( + <EditorWrapper> + <div ref={editorRef} /> + </EditorWrapper> + ); }; + +export default EssayAnswerComponent; diff --git a/wax-prosemirror-services/src/EssayService/components/EssayQuestionComponent.js b/wax-prosemirror-services/src/EssayService/components/EssayQuestionComponent.js index 8dcbcc26b..ea1b7b326 100644 --- a/wax-prosemirror-services/src/EssayService/components/EssayQuestionComponent.js +++ b/wax-prosemirror-services/src/EssayService/components/EssayQuestionComponent.js @@ -1,7 +1,197 @@ +/* eslint-disable react/destructuring-assignment */ /* eslint-disable react/prop-types */ -import React from 'react'; + +import React, { useContext, useRef, useEffect } from 'react'; import styled from 'styled-components'; +import { EditorView } from 'prosemirror-view'; +import { EditorState, TextSelection, NodeSelection } from 'prosemirror-state'; +import { StepMap } from 'prosemirror-transform'; +import { keymap } from 'prosemirror-keymap'; +import { baseKeymap, chainCommands } from 'prosemirror-commands'; +import { undo, redo } from 'prosemirror-history'; +import { WaxContext } from 'wax-prosemirror-core'; +import { + splitListItem, + liftListItem, + sinkListItem, +} from 'prosemirror-schema-list'; +import Placeholder from '../plugins/placeholder'; + +const EditorWrapper = styled.div` + border: none; + display: flex; + flex: 2 1 auto; + justify-content: left; + + .ProseMirror { + white-space: break-spaces; + width: 100%; + word-wrap: break-word; + + &:focus { + outline: none; + } + + p.empty-node:first-child::before { + content: attr(data-content); + } + + .empty-node::before { + color: rgb(170, 170, 170); + float: left; + font-style: italic; + height: 0px; + pointer-events: none; + } + } +`; +const EssayQuestionComponent = ({ node, view, getPos }) => { + const editorRef = useRef(); + + const context = useContext(WaxContext); + let essayQuestionView; + const questionId = node.attrs.id; + const isEditable = context.view.main.props.editable(editable => { + return editable; + }); + + let finalPlugins = []; + + const createKeyBindings = () => { + const keys = getKeys(); + Object.keys(baseKeymap).forEach(key => { + if (keys[key]) { + keys[key] = chainCommands(keys[key], baseKeymap[key]); + } else { + keys[key] = baseKeymap[key]; + } + }); + return keys; + }; + + const pressEnter = (state, dispatch) => { + if (state.selection.node && state.selection.node.type.name === 'image') { + const { $from, to } = state.selection; + + const same = $from.sharedDepth(to); + + const pos = $from.before(same); + dispatch(state.tr.setSelection(NodeSelection.create(state.doc, pos))); + return true; + } + // LISTS + if (splitListItem(state.schema.nodes.list_item)(state)) { + splitListItem(state.schema.nodes.list_item)(state, dispatch); + return true; + } -export default ({ node, view, getPos }) => { - return <span>Question</span>; + return false; + }; + + const getKeys = () => { + return { + 'Mod-z': () => undo(view.state, view.dispatch), + 'Mod-y': () => redo(view.state, view.dispatch), + 'Mod-[': liftListItem(view.state.schema.nodes.list_item), + 'Mod-]': sinkListItem(view.state.schema.nodes.list_item), + Enter: pressEnter, + }; + }; + + const plugins = [keymap(createKeyBindings()), ...context.app.getPlugins()]; + + // eslint-disable-next-line no-shadow + const createPlaceholder = placeholder => { + return Placeholder({ + content: placeholder, + }); + }; + + finalPlugins = finalPlugins.concat([ + createPlaceholder('Type your essay'), + ...plugins, + ]); + + useEffect(() => { + essayQuestionView = new EditorView( + { + mount: editorRef.current, + }, + { + editable: () => isEditable, + state: EditorState.create({ + doc: node, + plugins: finalPlugins, + }), + // This is the magic part + dispatchTransaction, + disallowedTools: ['MultipleChoice'], + handleDOMEvents: { + mousedown: () => { + context.updateView({}, questionId); + context.view.main.dispatch( + context.view.main.state.tr + .setMeta('outsideView', questionId) + .setSelection( + new TextSelection( + context.view.main.state.tr.doc.resolve( + getPos() + + 2 + + context.view[questionId].state.selection.to, + ), + ), + ), + ); + context.updateView({}, questionId); + + // Kludge to prevent issues due to the fact that the whole + // footnote is node-selected (and thus DOM-selected) when + // the parent editor is focused. + if (essayQuestionView.hasFocus()) essayQuestionView.focus(); + }, + }, + + attributes: { + spellcheck: 'false', + }, + }, + ); + + // Set Each note into Wax's Context + context.updateView( + { + [questionId]: essayQuestionView, + }, + questionId, + ); + if (essayQuestionView.hasFocus()) essayQuestionView.focus(); + }, []); + + const dispatchTransaction = tr => { + const outerTr = context.view.main.state.tr; + context.view.main.dispatch(outerTr.setMeta('outsideView', questionId)); + const { state, transactions } = essayQuestionView.state.applyTransaction( + tr, + ); + context.updateView({}, questionId); + essayQuestionView.updateState(state); + if (!tr.getMeta('fromOutside')) { + const offsetMap = StepMap.offset(getPos() + 1); + for (let i = 0; i < transactions.length; i++) { + const { steps } = transactions[i]; + for (let j = 0; j < steps.length; j++) + outerTr.step(steps[j].map(offsetMap)); + } + if (outerTr.docChanged) + context.view.main.dispatch(outerTr.setMeta('outsideView', questionId)); + } + }; + + return ( + <EditorWrapper> + <div ref={editorRef} /> + </EditorWrapper> + ); }; + +export default EssayQuestionComponent; -- GitLab