parent
f4abf04994
commit
561fbbed3e
3 changed files with 144 additions and 2 deletions
@ -0,0 +1,116 @@ |
|||||||
|
import 'package:quill_delta/quill_delta.dart'; |
||||||
|
import 'package:tuple/tuple.dart'; |
||||||
|
|
||||||
|
import 'document.dart'; |
||||||
|
|
||||||
|
class History { |
||||||
|
final HistoryStack stack = HistoryStack.empty(); |
||||||
|
|
||||||
|
/// used for disable redo or undo function |
||||||
|
bool ignoreChange; |
||||||
|
|
||||||
|
int lastRecorded; |
||||||
|
|
||||||
|
/// Collaborative editing's conditions should be true |
||||||
|
final bool userOnly; |
||||||
|
|
||||||
|
///max operation count for undo |
||||||
|
final int maxStack; |
||||||
|
|
||||||
|
///record delay |
||||||
|
final int interval; |
||||||
|
|
||||||
|
History( |
||||||
|
{this.ignoreChange = false, |
||||||
|
this.interval = 400, |
||||||
|
this.maxStack = 100, |
||||||
|
this.userOnly = false, |
||||||
|
this.lastRecorded = 0}); |
||||||
|
|
||||||
|
void handleDocChange(Tuple3<Delta, Delta, ChangeSource> change) { |
||||||
|
if (ignoreChange) return; |
||||||
|
if (!userOnly || change.item3 == ChangeSource.LOCAL) { |
||||||
|
record(change.item2, change.item1); |
||||||
|
} else { |
||||||
|
transform(change.item2); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void clear() { |
||||||
|
stack.clear(); |
||||||
|
} |
||||||
|
|
||||||
|
void record(Delta change, Delta before) { |
||||||
|
if (change.isEmpty) return; |
||||||
|
stack.redo.clear(); |
||||||
|
Delta undoDelta = change.invert(before); |
||||||
|
final timeStamp = DateTime.now().millisecondsSinceEpoch; |
||||||
|
|
||||||
|
if (lastRecorded + interval > timeStamp && stack.undo.isNotEmpty) { |
||||||
|
final lastDelta = stack.undo.removeLast(); |
||||||
|
undoDelta = undoDelta.compose(lastDelta); |
||||||
|
} else { |
||||||
|
lastRecorded = timeStamp; |
||||||
|
} |
||||||
|
|
||||||
|
if (undoDelta.isEmpty) return; |
||||||
|
stack.undo.add(undoDelta); |
||||||
|
|
||||||
|
if (stack.undo.length > maxStack) { |
||||||
|
stack.undo.removeAt(0); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/// |
||||||
|
///It will override pre local undo delta,replaced by remote change |
||||||
|
/// |
||||||
|
void transform(Delta delta) { |
||||||
|
transformStack(this.stack.undo, delta); |
||||||
|
transformStack(this.stack.redo, delta); |
||||||
|
} |
||||||
|
|
||||||
|
void transformStack(List<Delta> stack, Delta delta) { |
||||||
|
for (int i = stack.length - 1; i >= 0; i -= 1) { |
||||||
|
final oldDelta = stack[i]; |
||||||
|
stack[i] = delta.transform(oldDelta, true); |
||||||
|
delta = oldDelta.transform(delta, false); |
||||||
|
if (stack[i].length == 0) { |
||||||
|
stack.removeAt(i); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void _change(Document doc, List<Delta> source, List<Delta> dest) { |
||||||
|
if (source.length == 0) return; |
||||||
|
Delta delta = source.removeLast(); |
||||||
|
Delta base = doc.toDelta(); |
||||||
|
Delta inverseDelta = delta.invert(base); |
||||||
|
dest.add(inverseDelta); |
||||||
|
this.lastRecorded = 0; |
||||||
|
this.ignoreChange = true; |
||||||
|
doc.compose(delta, ChangeSource.LOCAL, history: true); |
||||||
|
this.ignoreChange = false; |
||||||
|
} |
||||||
|
|
||||||
|
void undo(Document doc) { |
||||||
|
_change(doc, stack.undo, stack.redo); |
||||||
|
} |
||||||
|
|
||||||
|
void redo(Document doc) { |
||||||
|
_change(doc, stack.redo, stack.undo); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class HistoryStack { |
||||||
|
final List<Delta> undo; |
||||||
|
final List<Delta> redo; |
||||||
|
|
||||||
|
HistoryStack.empty() |
||||||
|
: undo = [], |
||||||
|
redo = []; |
||||||
|
|
||||||
|
void clear() { |
||||||
|
undo.clear(); |
||||||
|
redo.clear(); |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue