|
@@ -3,10 +3,12 @@ import 'dart:developer';
|
3
|
3
|
|
4
|
4
|
import 'package:defer_pointer/defer_pointer.dart';
|
5
|
5
|
import 'package:flutter/material.dart';
|
|
6
|
+import 'package:flutter_canvas_editor/history.dart';
|
6
|
7
|
import 'package:flutter_canvas_editor/style/canvas_style.dart';
|
7
|
8
|
import 'package:flutter_canvas_editor/widgets/elements.dart';
|
8
|
9
|
import 'package:toastification/toastification.dart';
|
9
|
10
|
import 'package:uuid/uuid.dart';
|
|
11
|
+import 'package:collection/collection.dart';
|
10
|
12
|
|
11
|
13
|
import '../main.dart';
|
12
|
14
|
|
|
@@ -65,12 +67,91 @@ class Editor extends ChangeNotifier {
|
65
|
67
|
qrScale: 3
|
66
|
68
|
)
|
67
|
69
|
];
|
68
|
|
- // int _reservedIdUntil = 0;
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+ // ? History stack
|
|
75
|
+ List<CanvasHistory> undoStack = [];
|
|
76
|
+ List<List<ElementProperty>> redoStack = [];
|
|
77
|
+
|
|
78
|
+
|
|
79
|
+ void addUndoEntry(CanvasHistoryModifyType canvasHistoryType, List<ElementProperty> elementProperties) {
|
|
80
|
+ undoStack.add(CanvasHistory(canvasHistoryType, _cloneCurrentState(elementProperties)));
|
|
81
|
+
|
|
82
|
+ redoStack.clear();
|
|
83
|
+
|
|
84
|
+ notifyListeners();
|
|
85
|
+ }
|
|
86
|
+
|
|
87
|
+ void undo() {
|
|
88
|
+ addRedoEntry(elementProperties);
|
|
89
|
+
|
|
90
|
+ // apply to current state
|
|
91
|
+ if (undoStack.last.type == CanvasHistoryModifyType.textEdit && undoStack.last.getState != elementProperties) {
|
|
92
|
+ undoStack.removeLast();
|
|
93
|
+ }
|
|
94
|
+
|
|
95
|
+ elementProperties = undoStack.last.getState;
|
|
96
|
+
|
|
97
|
+ // unselect element
|
|
98
|
+ if (elementProperties.firstWhereOrNull((e) => e.id == selectedElmId) == null) {
|
|
99
|
+ unSelectElm();
|
|
100
|
+ }
|
|
101
|
+
|
|
102
|
+ undoStack.removeLast();
|
|
103
|
+
|
|
104
|
+ notifyListeners();
|
|
105
|
+ }
|
|
106
|
+
|
|
107
|
+
|
|
108
|
+ void addRedoEntry(List<ElementProperty> elementProperties) {
|
|
109
|
+ redoStack.add(_cloneCurrentState(elementProperties));
|
|
110
|
+
|
|
111
|
+ notifyListeners();
|
|
112
|
+ }
|
|
113
|
+
|
|
114
|
+ void redo() {
|
|
115
|
+ undoStack.add(CanvasHistory(CanvasHistoryModifyType.redo, _cloneCurrentState(elementProperties)));
|
|
116
|
+
|
|
117
|
+ // apply to current state
|
|
118
|
+ elementProperties = redoStack.last;
|
|
119
|
+
|
|
120
|
+ redoStack.removeLast();
|
|
121
|
+
|
|
122
|
+ notifyListeners();
|
|
123
|
+ }
|
|
124
|
+
|
|
125
|
+ List<ElementProperty> _cloneCurrentState(List<ElementProperty> elementProperties) {
|
|
126
|
+ List<ElementProperty> clonedElementProperties = elementProperties.map((elementProperty) {
|
|
127
|
+ return ElementProperty(
|
|
128
|
+ id: elementProperty.id,
|
|
129
|
+ valueController: TextEditingController(text: elementProperty.valueController.text),
|
|
130
|
+ type: elementProperty.type,
|
|
131
|
+ position: ElementPosition(top: elementProperty.position.top, left: elementProperty.position.left),
|
|
132
|
+ width: elementProperty.width,
|
|
133
|
+ quarterTurns: elementProperty.quarterTurns,
|
|
134
|
+ elementKey: elementProperty.elementKey,
|
|
135
|
+ qrScale: elementProperty.qrScale,
|
|
136
|
+ fontScale: elementProperty.fontScale,
|
|
137
|
+ variableType: elementProperty.variableType,
|
|
138
|
+ isLocked: elementProperty.isLocked
|
|
139
|
+ );
|
|
140
|
+ }).toList();
|
|
141
|
+
|
|
142
|
+ return clonedElementProperties;
|
|
143
|
+ }
|
|
144
|
+
|
|
145
|
+
|
|
146
|
+
|
69
|
147
|
|
70
|
148
|
// ? udpate canvas
|
71
|
149
|
void updateCanvasProperty(BuildContext context, double width, double height) {
|
72
|
150
|
canvasProperty.height = height;
|
73
|
151
|
canvasProperty.width = width;
|
|
152
|
+
|
|
153
|
+ undoStack.clear();
|
|
154
|
+ redoStack.clear();
|
74
|
155
|
|
75
|
156
|
_adjustOutOfBoundElement();
|
76
|
157
|
|
|
@@ -122,6 +203,10 @@ class Editor extends ChangeNotifier {
|
122
|
203
|
);
|
123
|
204
|
|
124
|
205
|
|
|
206
|
+ // Save History
|
|
207
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
208
|
+
|
|
209
|
+
|
125
|
210
|
elementProperties.add(element);
|
126
|
211
|
// _reservedIdUntil = elementProperties.length - 1;r
|
127
|
212
|
notifyListeners();
|
|
@@ -142,6 +227,9 @@ class Editor extends ChangeNotifier {
|
142
|
227
|
elementKey: GlobalKey()
|
143
|
228
|
);
|
144
|
229
|
|
|
230
|
+ // Save History
|
|
231
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
232
|
+
|
145
|
233
|
elementProperties.add(element);
|
146
|
234
|
notifyListeners();
|
147
|
235
|
|
|
@@ -164,6 +252,9 @@ class Editor extends ChangeNotifier {
|
164
|
252
|
elementKey: GlobalKey()
|
165
|
253
|
);
|
166
|
254
|
|
|
255
|
+ // Save History
|
|
256
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
257
|
+
|
167
|
258
|
elementProperties.add(element);
|
168
|
259
|
notifyListeners();
|
169
|
260
|
|
|
@@ -184,6 +275,9 @@ class Editor extends ChangeNotifier {
|
184
|
275
|
elementKey: GlobalKey()
|
185
|
276
|
);
|
186
|
277
|
|
|
278
|
+ // Save History
|
|
279
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
280
|
+
|
187
|
281
|
elementProperties.add(element);
|
188
|
282
|
notifyListeners();
|
189
|
283
|
|
|
@@ -204,6 +298,9 @@ class Editor extends ChangeNotifier {
|
204
|
298
|
elementKey: GlobalKey()
|
205
|
299
|
);
|
206
|
300
|
|
|
301
|
+ // Save History
|
|
302
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
303
|
+
|
207
|
304
|
elementProperties.add(element);
|
208
|
305
|
notifyListeners();
|
209
|
306
|
|
|
@@ -224,6 +321,9 @@ class Editor extends ChangeNotifier {
|
224
|
321
|
elementKey: GlobalKey()
|
225
|
322
|
);
|
226
|
323
|
|
|
324
|
+ // Save History
|
|
325
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
326
|
+
|
227
|
327
|
elementProperties.add(element);
|
228
|
328
|
notifyListeners();
|
229
|
329
|
|
|
@@ -244,6 +344,9 @@ class Editor extends ChangeNotifier {
|
244
|
344
|
elementKey: GlobalKey()
|
245
|
345
|
);
|
246
|
346
|
|
|
347
|
+ // Save History
|
|
348
|
+ addUndoEntry(CanvasHistoryModifyType.add, elementProperties);
|
|
349
|
+
|
247
|
350
|
elementProperties.add(element);
|
248
|
351
|
notifyListeners();
|
249
|
352
|
|
|
@@ -279,6 +382,9 @@ class Editor extends ChangeNotifier {
|
279
|
382
|
void toggleLockElement() {
|
280
|
383
|
if (selectedElm == null) return;
|
281
|
384
|
|
|
385
|
+ // ? Save History
|
|
386
|
+ addUndoEntry(CanvasHistoryModifyType.lock, elementProperties);
|
|
387
|
+
|
282
|
388
|
selectedElm!.isLocked = !selectedElm!.isLocked;
|
283
|
389
|
notifyListeners();
|
284
|
390
|
}
|
|
@@ -417,7 +523,9 @@ class Editor extends ChangeNotifier {
|
417
|
523
|
}
|
418
|
524
|
|
419
|
525
|
if (![ElementType.text, ElementType.textbox].contains(element.type)) return;
|
420
|
|
-
|
|
526
|
+
|
|
527
|
+ // ? Save history
|
|
528
|
+ addUndoEntry(CanvasHistoryModifyType.rotate, elementProperties);
|
421
|
529
|
|
422
|
530
|
if (element.type == ElementType.text) _rotateText(element);
|
423
|
531
|
|
|
@@ -463,6 +571,9 @@ class Editor extends ChangeNotifier {
|
463
|
571
|
return;
|
464
|
572
|
}
|
465
|
573
|
|
|
574
|
+ // ? Save History
|
|
575
|
+ addUndoEntry(CanvasHistoryModifyType.resize, elementProperties);
|
|
576
|
+
|
466
|
577
|
selectedElm!.fontScale = fontSize;
|
467
|
578
|
notifyListeners();
|
468
|
579
|
}
|
|
@@ -478,6 +589,9 @@ class Editor extends ChangeNotifier {
|
478
|
589
|
}
|
479
|
590
|
// check if value is allowed for resize
|
480
|
591
|
if (CanvasStyle.fontSizeMap.containsKey(incrementTo)) {
|
|
592
|
+ // ? Save History
|
|
593
|
+ addUndoEntry(CanvasHistoryModifyType.resize, elementProperties);
|
|
594
|
+
|
481
|
595
|
selectedElm!.fontScale = incrementTo;
|
482
|
596
|
print('kepenjet increase');
|
483
|
597
|
} else {
|
|
@@ -498,6 +612,9 @@ class Editor extends ChangeNotifier {
|
498
|
612
|
}
|
499
|
613
|
// check if value is allowed for resize
|
500
|
614
|
if (CanvasStyle.fontSizeMap.containsKey(decrementTo)) {
|
|
615
|
+ // ? Save History
|
|
616
|
+ addUndoEntry(CanvasHistoryModifyType.resize, elementProperties);
|
|
617
|
+
|
501
|
618
|
selectedElm!.fontScale = decrementTo;
|
502
|
619
|
} else {
|
503
|
620
|
print('cant decrement');
|
|
@@ -596,6 +713,9 @@ class Editor extends ChangeNotifier {
|
596
|
713
|
|
597
|
714
|
if (!shouldDelete) return;
|
598
|
715
|
|
|
716
|
+ // ? Save History
|
|
717
|
+ addUndoEntry(CanvasHistoryModifyType.remove, elementProperties);
|
|
718
|
+
|
599
|
719
|
elementProperties.removeWhere((e) => e.id == selectedElm!.id);
|
600
|
720
|
unSelectElm();
|
601
|
721
|
|