Add history

pull/13/head
singerdmx 4 years ago
parent f4abf04994
commit 561fbbed3e
  1. 18
      lib/models/documents/document.dart
  2. 116
      lib/models/documents/history.dart
  3. 12
      lib/widgets/controller.dart

@ -9,6 +9,7 @@ import 'package:tuple/tuple.dart';
import '../rules/rule.dart'; import '../rules/rule.dart';
import 'attribute.dart'; import 'attribute.dart';
import 'history.dart';
import 'nodes/embed.dart'; import 'nodes/embed.dart';
import 'nodes/node.dart'; import 'nodes/node.dart';
@ -30,6 +31,8 @@ class Document {
final StreamController<Tuple3<Delta, Delta, ChangeSource>> _observer = final StreamController<Tuple3<Delta, Delta, ChangeSource>> _observer =
StreamController.broadcast(); StreamController.broadcast();
final History _history = History();
Stream<Tuple3> get changes => _observer.stream; Stream<Tuple3> get changes => _observer.stream;
Document() : _delta = Delta() Document() : _delta = Delta()
@ -115,7 +118,7 @@ class Document {
return block.queryChild(res.offset, true); return block.queryChild(res.offset, true);
} }
compose(Delta delta, ChangeSource changeSource) { compose(Delta delta, ChangeSource changeSource, {bool history}) {
assert(!_observer.isClosed); assert(!_observer.isClosed);
delta.trim(); delta.trim();
assert(delta.isNotEmpty); assert(delta.isNotEmpty);
@ -148,7 +151,17 @@ class Document {
if (_delta != _root.toDelta()) { if (_delta != _root.toDelta()) {
throw ('Compose failed'); throw ('Compose failed');
} }
_observer.add(Tuple3(originalDelta, delta, changeSource)); final change = Tuple3(originalDelta, delta, changeSource);
_observer.add(change);
_history.handleDocChange(change);
}
void undo() {
_history.undo(this);
}
void redo() {
_history.redo(this);
} }
static Delta _transform(Delta delta) { static Delta _transform(Delta delta) {
@ -172,6 +185,7 @@ class Document {
close() { close() {
_observer.close(); _observer.close();
_history.clear();
} }
String toPlainText() => _root.children.map((e) => e.toPlainText()).join(''); String toPlainText() => _root.children.map((e) => e.toPlainText()).join('');

@ -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();
}
}

@ -34,6 +34,18 @@ class QuillController extends ChangeNotifier {
.mergeAll(toggledStyle); .mergeAll(toggledStyle);
} }
void undo() {
document.undo();
updateSelection(
TextSelection.collapsed(offset: document.length), ChangeSource.REMOTE);
}
void redo() {
document.redo();
updateSelection(
TextSelection.collapsed(offset: document.length), ChangeSource.REMOTE);
}
replaceText(int index, int len, Object data, TextSelection textSelection) { replaceText(int index, int len, Object data, TextSelection textSelection) {
assert(data is String || data is Embeddable); assert(data is String || data is Embeddable);

Loading…
Cancel
Save