|
@@ -0,0 +1,108 @@
|
|
|
+import { BehaviorSubject } from "rxjs";
|
|
|
+import {reactive, computed} from "vue"
|
|
|
+
|
|
|
+export class ValueSnap {
|
|
|
+ Id:string;
|
|
|
+ Value: any;
|
|
|
+ OldValue: any;
|
|
|
+
|
|
|
+ Rx: BehaviorSubject<any>;
|
|
|
+ constructor(id:string, value:any, oldValue:any, rx: BehaviorSubject<any>) {
|
|
|
+ this.Id = id;
|
|
|
+ this.Value = value;
|
|
|
+ this.OldValue = oldValue;
|
|
|
+ this.Rx = rx
|
|
|
+ }
|
|
|
+ redo() {
|
|
|
+ this.Rx.next({value: this.Value, _hstry:false});
|
|
|
+ }
|
|
|
+ undo() {
|
|
|
+ this.Rx.next({value: this.OldValue, _hstry:false});
|
|
|
+ }
|
|
|
+
|
|
|
+ clone() {
|
|
|
+ return new ValueSnap(this.Id, this.Value,this.OldValue, this.Rx);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+export class HistoryController {
|
|
|
+
|
|
|
+ _valueIndex = 0;
|
|
|
+ createValueSnap(value:any, oldValue:any, rx:BehaviorSubject<any>) {
|
|
|
+ let i = this._valueIndex + 1;
|
|
|
+ this._valueIndex +=1;
|
|
|
+ return new ValueSnap(i+"", value, oldValue, rx);
|
|
|
+ }
|
|
|
+
|
|
|
+ state = reactive({
|
|
|
+ currLen: 0, //操作栈的长度
|
|
|
+ maxLen: 100, //操作栈总长度
|
|
|
+ opIndex: -1, //操作栈的指针
|
|
|
+ });
|
|
|
+
|
|
|
+ refCanUndo = computed(() => {
|
|
|
+ return this.state.opIndex >= 0;
|
|
|
+ });
|
|
|
+ refCanRedo = computed(() => {
|
|
|
+ return this.state.opIndex < this.state.currLen - 1;
|
|
|
+ });
|
|
|
+
|
|
|
+ queues: Map<string, ValueSnap>[] = [];
|
|
|
+ cacheSnapValues = new Map<string , ValueSnap>();
|
|
|
+
|
|
|
+ // 添加缓存记录
|
|
|
+ record(snap: ValueSnap) {
|
|
|
+ const first = this.cacheSnapValues.get(snap.Id)
|
|
|
+ if (first) {
|
|
|
+ snap.OldValue = first.OldValue;
|
|
|
+ }
|
|
|
+ this.cacheSnapValues.set(snap.Id, snap);
|
|
|
+ }
|
|
|
+
|
|
|
+ // 保存缓存记录到历史栈中
|
|
|
+ submit() {
|
|
|
+ if (this.cacheSnapValues.size < 1) return;
|
|
|
+
|
|
|
+ const state = this.state;
|
|
|
+ const queue = this.queues;
|
|
|
+
|
|
|
+ // 将缓存操作记录保存到当前指针的下一栈中
|
|
|
+ queue[++state.opIndex] = this.cacheSnapValues;
|
|
|
+ this.cacheSnapValues = new Map<string, ValueSnap>();
|
|
|
+
|
|
|
+ // 设置栈的长度为指针的长度,舍弃后面的记录
|
|
|
+ queue.length = state.opIndex + 1;
|
|
|
+ // 若栈长度超过上限, 舍弃之前的记录
|
|
|
+ if (queue.length > state.maxLen) {
|
|
|
+ queue.splice(0, queue.length - state.maxLen);
|
|
|
+ state.opIndex = state.maxLen - 1;
|
|
|
+ }
|
|
|
+ // 更新当前长度状态
|
|
|
+ state.currLen = queue.length;
|
|
|
+ }
|
|
|
+
|
|
|
+ undo() {
|
|
|
+ if (!this.refCanUndo.value) return;
|
|
|
+
|
|
|
+ this.cacheSnapValues = new Map<string, ValueSnap>();
|
|
|
+ const snaps = this.queues[this.state.opIndex--]
|
|
|
+
|
|
|
+ snaps.forEach((vn)=>vn.undo())
|
|
|
+ }
|
|
|
+
|
|
|
+ redo() {
|
|
|
+ if (!this.refCanRedo.value ) return;
|
|
|
+
|
|
|
+ this.cacheSnapValues = new Map<string, ValueSnap>();
|
|
|
+ const snaps = this.queues[++this.state.opIndex];
|
|
|
+ snaps.forEach(vn=>vn.redo());
|
|
|
+ }
|
|
|
+
|
|
|
+ //清除操作
|
|
|
+ clear() {
|
|
|
+ this.queues = [];
|
|
|
+ this.state.currLen = 0;
|
|
|
+ this.state.opIndex = -1;
|
|
|
+ this.cacheSnapValues = new Map<string, ValueSnap>();
|
|
|
+ }
|
|
|
+}
|