diff --git a/wax-prosemirror-services/src/CommentsService/plugins/AnnoationDecoration.js b/wax-prosemirror-services/src/CommentsService/plugins/AnnotationDecoration.js similarity index 100% rename from wax-prosemirror-services/src/CommentsService/plugins/AnnoationDecoration.js rename to wax-prosemirror-services/src/CommentsService/plugins/AnnotationDecoration.js diff --git a/wax-prosemirror-services/src/CommentsService/plugins/AnnotationState.js b/wax-prosemirror-services/src/CommentsService/plugins/AnnotationState.js index 86d422ce503b96c92756b66a7565755732f46f17..6829fb0fc8f7533ec1a2612b98980d5e67528515 100644 --- a/wax-prosemirror-services/src/CommentsService/plugins/AnnotationState.js +++ b/wax-prosemirror-services/src/CommentsService/plugins/AnnotationState.js @@ -1,6 +1,9 @@ +import { Decoration, DecorationSet } from 'prosemirror-view'; +import AnnotationDecoration from './AnnotationDecoration'; + export default class AnnotationState { constructor(options) { - this.decorations = view_1.DecorationSet.empty; + this.decorations = DecorationSet.empty; this.options = options; } @@ -30,7 +33,7 @@ export default class AnnotationState { termsAt(position, to) { return this.decorations.find(position, to || position).map(decoration => { - return new annotation_decoration_1.AnnotationDecoration(decoration); + return new AnnotationDecoration(decoration); }); } diff --git a/wax-prosemirror-services/src/CommentsService/plugins/rendering/annotation.js b/wax-prosemirror-services/src/CommentsService/plugins/rendering/annotation.js new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/wax-prosemirror-services/src/CommentsService/plugins/rendering/engine.js b/wax-prosemirror-services/src/CommentsService/plugins/rendering/engine.js new file mode 100644 index 0000000000000000000000000000000000000000..5d94ee16e9ca221455be812904518a4cd7015dfe --- /dev/null +++ b/wax-prosemirror-services/src/CommentsService/plugins/rendering/engine.js @@ -0,0 +1,128 @@ +export const isConflicting = (fromA, toA, fromB, toB) => { + // case 1: (non-conflicting) A is before B + if (fromA < toB && toA < fromB) return false; + // case 2: (non-conflicting) B is before A + if (fromB < toA && toB < fromA) return false; + // case 3: (conflicting) some kind of overlap + return true; +}; + +export const createAnnotationRendering = annotations => { + const renderedAnnotations = []; + const openAnnotationStack = []; + // const actionMap: Map<number, ActionKeyframe[]> = new Map(); + const actionMap = []; + const annotationFragmentation = []; + // annotations = sortAnnotationsByStart(annotations); + // STEP 1: Create a Map, containing the rendering actions for each index in the document. + // this could be opening or closing an annotation + annotations.forEach((term, index) => { + // create an opening action keyframe + const open = { + action: 'open', + annotationIndex: index, + textAnchor: term.from, + }; + // create a closing action keyframe + const close = { + action: 'close', + annotationIndex: index, + textAnchor: term.to, + }; + const openMapElement = actionMap[open.textAnchor]; + // create empty actions list if necessary + if (!openMapElement) actionMap[open.textAnchor] = []; + actionMap[open.textAnchor].push(open); + const closeMapElement = actionMap[close.textAnchor]; + if (!closeMapElement) actionMap[close.textAnchor] = []; + actionMap[close.textAnchor].push(close); + }); + actionMap // STEP 2: iterate the actionMap and generate the annotation UI elements + .forEach((actions, _) => { + actions.forEach(action => { + // check if there are still open annotations + if (openAnnotationStack.length !== 0) { + const actionStackPeek = + openAnnotationStack[openAnnotationStack.length - 1]; + if ( + actionStackPeek.action === 'open' && + actionStackPeek.annotationIndex === action.annotationIndex && + action.action === 'close' + ) { + // base case: the last opened annotation is closed by next action + openAnnotationStack.pop(); + const rendering = annotationFragmentation[action.annotationIndex] + ? 'fragment-right' + : 'normal'; + const from = annotationFragmentation[action.annotationIndex] + ? renderedAnnotations[renderedAnnotations.length - 1].to + : annotations[action.annotationIndex].from; + const normalTerm = Object.assign( + Object.assign({}, annotations[action.annotationIndex]), + { from, rendering }, + ); + renderedAnnotations.push(normalTerm); + } else if ( + actionStackPeek.action === 'open' && + action.action === 'close' + ) { + // annotation is closed while being overlapped by another annotation + // -> find "open" action and remove it, otherwise a new truncated segment would be created + const indexOfActionToRemove = openAnnotationStack.findIndex(a => { + return ( + a.textAnchor === annotations[action.annotationIndex].from && + a.annotationIndex === action.annotationIndex && + a.action === 'open' + ); + }); + if (indexOfActionToRemove > -1) { + openAnnotationStack.splice(indexOfActionToRemove, 1); + } else { + throw Error( + "Couldn't find opening keyframe for annotation " + + action.annotationIndex, + ); + } + } else if ( + actionStackPeek.action === 'open' && + action.action === 'open' + ) { + let fragment; + if (annotationFragmentation[actionStackPeek.annotationIndex]) { + // n-th truncation (n > 1): render a middle fragment + fragment = Object.assign( + Object.assign({}, annotations[actionStackPeek.annotationIndex]), + { + rendering: 'fragment-middle', + // start where the last rendered annotation ends + 1 + from: renderedAnnotations[renderedAnnotations.length - 1].to, + // stop where the next annotation begins - 1 + to: annotations[action.annotationIndex].from, + }, + ); + } else { + // first-time-truncation: a new annotation begins, truncating the old open annotation + fragment = Object.assign( + Object.assign({}, annotations[actionStackPeek.annotationIndex]), + { + rendering: 'fragment-left', + to: annotations[action.annotationIndex].from, + }, + ); + // mark the previous annotation as fragmented, by saving where the fragment ends + annotationFragmentation[actionStackPeek.annotationIndex] = true; + } + renderedAnnotations.push(fragment); + openAnnotationStack.push(action); + } + } else if (action.action === 'open') { + openAnnotationStack.push(action); + } + }); + }); + return renderedAnnotations; +}; + +export const sortAnnotationsByStart = annotations => { + return annotations.sort((a, b) => a.from - b.from); +};