/** * This code is mostly from the old Etherpad. Please help us to comment this code. * This helps other people to understand this code better and helps them to improve it. * TL;DR COMMENTS ON THIS FILE ARE HIGHLY APPRECIATED */ /** * Copyright 2009 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS-IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ const Changeset = require('./Changeset'); const _ = require('./underscore'); var undoModule = (function () { const stack = (function () { const stackElements = []; // two types of stackElements: // 1) { elementType: UNDOABLE_EVENT, eventType: "anything", [backset: ,] // [selStart: , selEnd: , selFocusAtStart: ] } // 2) { elementType: EXTERNAL_CHANGE, changeset: } // invariant: no two consecutive EXTERNAL_CHANGEs let numUndoableEvents = 0; const UNDOABLE_EVENT = 'undoableEvent'; const EXTERNAL_CHANGE = 'externalChange'; function clearStack() { stackElements.length = 0; stackElements.push( { elementType: UNDOABLE_EVENT, eventType: 'bottom', }); numUndoableEvents = 1; } clearStack(); function pushEvent(event) { const e = _.extend( {}, event); e.elementType = UNDOABLE_EVENT; stackElements.push(e); numUndoableEvents++; // dmesg("pushEvent backset: "+event.backset); } function pushExternalChange(cs) { const idx = stackElements.length - 1; if (stackElements[idx].elementType == EXTERNAL_CHANGE) { stackElements[idx].changeset = Changeset.compose(stackElements[idx].changeset, cs, getAPool()); } else { stackElements.push( { elementType: EXTERNAL_CHANGE, changeset: cs, }); } } function _exposeEvent(nthFromTop) { // precond: 0 <= nthFromTop < numUndoableEvents const targetIndex = stackElements.length - 1 - nthFromTop; let idx = stackElements.length - 1; while (idx > targetIndex || stackElements[idx].elementType == EXTERNAL_CHANGE) { if (stackElements[idx].elementType == EXTERNAL_CHANGE) { const ex = stackElements[idx]; const un = stackElements[idx - 1]; if (un.backset) { const excs = ex.changeset; const unbs = un.backset; un.backset = Changeset.follow(excs, un.backset, false, getAPool()); ex.changeset = Changeset.follow(unbs, ex.changeset, true, getAPool()); if ((typeof un.selStart) === 'number') { const newSel = Changeset.characterRangeFollow(excs, un.selStart, un.selEnd); un.selStart = newSel[0]; un.selEnd = newSel[1]; if (un.selStart == un.selEnd) { un.selFocusAtStart = false; } } } stackElements[idx - 1] = ex; stackElements[idx] = un; if (idx >= 2 && stackElements[idx - 2].elementType == EXTERNAL_CHANGE) { ex.changeset = Changeset.compose(stackElements[idx - 2].changeset, ex.changeset, getAPool()); stackElements.splice(idx - 2, 1); idx--; } } else { idx--; } } } function getNthFromTop(n) { // precond: 0 <= n < numEvents() _exposeEvent(n); return stackElements[stackElements.length - 1 - n]; } function numEvents() { return numUndoableEvents; } function popEvent() { // precond: numEvents() > 0 _exposeEvent(0); numUndoableEvents--; return stackElements.pop(); } return { numEvents, popEvent, pushEvent, pushExternalChange, clearStack, getNthFromTop, }; })(); // invariant: stack always has at least one undoable event let undoPtr = 0; // zero-index from top of stack, 0 == top function clearHistory() { stack.clearStack(); undoPtr = 0; } function _charOccurrences(str, c) { let i = 0; let count = 0; while (i >= 0 && i < str.length) { i = str.indexOf(c, i); if (i >= 0) { count++; i++; } } return count; } function _opcodeOccurrences(cs, opcode) { return _charOccurrences(Changeset.unpack(cs).ops, opcode); } function _mergeChangesets(cs1, cs2) { if (!cs1) return cs2; if (!cs2) return cs1; // Rough heuristic for whether changesets should be considered one action: // each does exactly one insertion, no dels, and the composition does also; or // each does exactly one deletion, no ins, and the composition does also. // A little weird in that it won't merge "make bold" with "insert char" // but will merge "make bold and insert char" with "insert char", // though that isn't expected to come up. const plusCount1 = _opcodeOccurrences(cs1, '+'); const plusCount2 = _opcodeOccurrences(cs2, '+'); const minusCount1 = _opcodeOccurrences(cs1, '-'); const minusCount2 = _opcodeOccurrences(cs2, '-'); if (plusCount1 == 1 && plusCount2 == 1 && minusCount1 == 0 && minusCount2 == 0) { var merge = Changeset.compose(cs1, cs2, getAPool()); var plusCount3 = _opcodeOccurrences(merge, '+'); var minusCount3 = _opcodeOccurrences(merge, '-'); if (plusCount3 == 1 && minusCount3 == 0) { return merge; } } else if (plusCount1 == 0 && plusCount2 == 0 && minusCount1 == 1 && minusCount2 == 1) { var merge = Changeset.compose(cs1, cs2, getAPool()); var plusCount3 = _opcodeOccurrences(merge, '+'); var minusCount3 = _opcodeOccurrences(merge, '-'); if (plusCount3 == 0 && minusCount3 == 1) { return merge; } } return null; } function reportEvent(event) { const topEvent = stack.getNthFromTop(0); function applySelectionToTop() { if ((typeof event.selStart) === 'number') { topEvent.selStart = event.selStart; topEvent.selEnd = event.selEnd; topEvent.selFocusAtStart = event.selFocusAtStart; } } if ((!event.backset) || Changeset.isIdentity(event.backset)) { applySelectionToTop(); } else { let merged = false; if (topEvent.eventType == event.eventType) { const merge = _mergeChangesets(event.backset, topEvent.backset); if (merge) { topEvent.backset = merge; // dmesg("reportEvent merge: "+merge); applySelectionToTop(); merged = true; } } if (!merged) { /* * Push the event on the undo stack only if it exists, and if it's * not a "clearauthorship". This disallows undoing the removal of the * authorship colors, but is a necessary stopgap measure against * https://github.com/ether/etherpad-lite/issues/2802 */ if (event && (event.eventType !== 'clearauthorship')) { stack.pushEvent(event); } } undoPtr = 0; } } function reportExternalChange(changeset) { if (changeset && !Changeset.isIdentity(changeset)) { stack.pushExternalChange(changeset); } } function _getSelectionInfo(event) { if ((typeof event.selStart) !== 'number') { return null; } else { return { selStart: event.selStart, selEnd: event.selEnd, selFocusAtStart: event.selFocusAtStart, }; } } // For "undo" and "redo", the change event must be returned // by eventFunc and NOT reported through the normal mechanism. // "eventFunc" should take a changeset and an optional selection info object, // or can be called with no arguments to mean that no undo is possible. // "eventFunc" will be called exactly once. function performUndo(eventFunc) { if (undoPtr < stack.numEvents() - 1) { const backsetEvent = stack.getNthFromTop(undoPtr); const selectionEvent = stack.getNthFromTop(undoPtr + 1); const undoEvent = eventFunc(backsetEvent.backset, _getSelectionInfo(selectionEvent)); stack.pushEvent(undoEvent); undoPtr += 2; } else { eventFunc(); } } function performRedo(eventFunc) { if (undoPtr >= 2) { const backsetEvent = stack.getNthFromTop(0); const selectionEvent = stack.getNthFromTop(1); eventFunc(backsetEvent.backset, _getSelectionInfo(selectionEvent)); stack.popEvent(); undoPtr -= 2; } else { eventFunc(); } } function getAPool() { return undoModule.apool; } return { clearHistory, reportEvent, reportExternalChange, performUndo, performRedo, enabled: true, apool: null, }; // apool is filled in by caller })(); exports.undoModule = undoModule;