Fix Undo throw Offset cannot be negative

pull/13/head
singerdmx 4 years ago
parent 590fac0408
commit 0706656937
  1. 12
      lib/models/documents/document.dart
  2. 19
      lib/models/documents/history.dart
  3. 14
      lib/widgets/controller.dart
  4. 39
      lib/widgets/toolbar.dart

@ -155,14 +155,18 @@ class Document {
_history.handleDocChange(change); _history.handleDocChange(change);
} }
void undo() { bool undo() {
_history.undo(this); return _history.undo(this);
} }
void redo() { bool redo() {
_history.redo(this); return _history.redo(this);
} }
get hasUndo => _history.hasUndo;
get hasRedo => _history.hasRedo;
static Delta _transform(Delta delta) { static Delta _transform(Delta delta) {
Delta res = Delta(); Delta res = Delta();
for (Operation op in delta.toList()) { for (Operation op in delta.toList()) {

@ -6,6 +6,10 @@ import 'document.dart';
class History { class History {
final HistoryStack stack = HistoryStack.empty(); final HistoryStack stack = HistoryStack.empty();
get hasUndo => stack.undo.isNotEmpty;
get hasRedo => stack.redo.isNotEmpty;
/// used for disable redo or undo function /// used for disable redo or undo function
bool ignoreChange; bool ignoreChange;
@ -80,8 +84,10 @@ class History {
} }
} }
void _change(Document doc, List<Delta> source, List<Delta> dest) { bool _change(Document doc, List<Delta> source, List<Delta> dest) {
if (source.length == 0) return; if (source.length == 0) {
return false;
}
Delta delta = source.removeLast(); Delta delta = source.removeLast();
Delta base = doc.toDelta(); Delta base = doc.toDelta();
Delta inverseDelta = delta.invert(base); Delta inverseDelta = delta.invert(base);
@ -90,14 +96,15 @@ class History {
this.ignoreChange = true; this.ignoreChange = true;
doc.compose(delta, ChangeSource.LOCAL); doc.compose(delta, ChangeSource.LOCAL);
this.ignoreChange = false; this.ignoreChange = false;
return true;
} }
void undo(Document doc) { bool undo(Document doc) {
_change(doc, stack.undo, stack.redo); return _change(doc, stack.undo, stack.redo);
} }
void redo(Document doc) { bool redo(Document doc) {
_change(doc, stack.redo, stack.undo); return _change(doc, stack.redo, stack.undo);
} }
} }

@ -43,15 +43,21 @@ class QuillController extends ChangeNotifier {
} }
void undo() { void undo() {
document.undo(); if (document.undo()) {
notifyListeners(); notifyListeners();
}
} }
void redo() { void redo() {
document.redo(); if (document.redo()) {
notifyListeners(); notifyListeners();
}
} }
get hasUndo => document.hasUndo;
get hasRedo => document.hasRedo;
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);

@ -537,27 +537,56 @@ class HistoryButton extends StatefulWidget {
} }
class _HistoryButtonState extends State<HistoryButton> { class _HistoryButtonState extends State<HistoryButton> {
Color _iconColor;
ThemeData theme;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final theme = Theme.of(context); theme = Theme.of(context);
final iconColor = theme.iconTheme.color; _setIconColor();
final fillColor = theme.canvasColor; final fillColor = theme.canvasColor;
widget.controller.changes.listen((event) async {
_setIconColor();
});
return QuillIconButton( return QuillIconButton(
highlightElevation: 0, highlightElevation: 0,
hoverElevation: 0, hoverElevation: 0,
size: 32, size: 32,
icon: Icon(widget.icon, size: 18, color: iconColor), icon: Icon(widget.icon, size: 18, color: _iconColor),
fillColor: fillColor, fillColor: fillColor,
onPressed: _changeHistory, onPressed: _changeHistory,
); );
} }
void _setIconColor() {
if (widget.undo) {
setState(() {
_iconColor = widget.controller.hasUndo
? theme.iconTheme.color
: theme.disabledColor;
});
} else {
setState(() {
_iconColor = widget.controller.hasRedo
? theme.iconTheme.color
: theme.disabledColor;
});
}
}
void _changeHistory() { void _changeHistory() {
if (widget.undo) { if (widget.undo) {
widget.controller.undo(); if (widget.controller.hasUndo) {
widget.controller.undo();
}
} else { } else {
widget.controller.redo(); if (widget.controller.hasRedo) {
widget.controller.redo();
}
} }
_setIconColor();
} }
} }

Loading…
Cancel
Save