|
@@ -1,27 +1,71 @@
|
|
|
+import { Filter } from "queenjs/typing";
|
|
|
import { HistoryController } from "./history";
|
|
|
-import {BehaviorSubject, Subject, concatAll} from "rxjs";
|
|
|
+import {BehaviorSubject} from "rxjs";
|
|
|
import { ref } from "vue";
|
|
|
|
|
|
-type RxValueType<T> = {
|
|
|
+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 type RxValueType<T> = {
|
|
|
value: T,
|
|
|
_hstry?: boolean
|
|
|
}
|
|
|
+
|
|
|
+
|
|
|
function createRxValue<T>(value: T, histry:boolean) {
|
|
|
return new BehaviorSubject< RxValueType<T> >({value:value, _hstry: histry})
|
|
|
}
|
|
|
|
|
|
-class RxValue {
|
|
|
+let _valueIndex = 0;
|
|
|
+export function createValueSnap(value:any, oldValue:any, rx:BehaviorSubject<any>) {
|
|
|
+ let i = _valueIndex + 1;
|
|
|
+ _valueIndex +=1;
|
|
|
+ return new ValueSnap(i+"", value, oldValue, rx);
|
|
|
+}
|
|
|
|
|
|
- static create<T extends {[key:string]: any}>(fields:T, histroy?: HistoryController ) {
|
|
|
+class RxValue {
|
|
|
+ static create<T extends {[key:string]: any}>(_fields:T, histroy?: HistoryController ) {
|
|
|
let obj = {} as any;
|
|
|
let refs = {} as any;
|
|
|
|
|
|
- const names = Object.keys(fields);
|
|
|
+ obj._historySnap = {} as any;
|
|
|
+ obj._historySub = {} as any;
|
|
|
+ obj._rxs = {} as any;
|
|
|
+ obj._fields = _fields;
|
|
|
+ obj._history = histroy;
|
|
|
+
|
|
|
+ const names = Object.keys(_fields);
|
|
|
|
|
|
names.forEach(name=>{
|
|
|
const currName = name;
|
|
|
- const initValue = fields[currName]
|
|
|
+ const initValue = _fields[currName]
|
|
|
const f = createRxValue(initValue, !!histroy);
|
|
|
+ obj._rxs[name] = f;
|
|
|
+
|
|
|
+ const snap = createValueSnap(initValue, initValue, f);
|
|
|
+ obj._historySnap[name] = snap;
|
|
|
+
|
|
|
Object.defineProperty(obj, currName, {
|
|
|
get: function(){
|
|
|
return f.getValue().value;
|
|
@@ -31,12 +75,13 @@ class RxValue {
|
|
|
}
|
|
|
})
|
|
|
|
|
|
+
|
|
|
obj["set"+currName] = function(value:T, nohistory = false){
|
|
|
f.next({value, _hstry: !nohistory});
|
|
|
}
|
|
|
-
|
|
|
- obj["on"+currName + "Changed"] = function(subscribe: (value:T)=>void){
|
|
|
- return f.subscribe((v)=>subscribe(v.value))
|
|
|
+
|
|
|
+ obj["on"+currName + "Changed"] = function(subscribe: (value:T, oldValue:T)=>void){
|
|
|
+ return f.subscribe((v)=>subscribe(v.value, snap.OldValue))
|
|
|
}
|
|
|
|
|
|
obj["ref"+currName] = function(){
|
|
@@ -47,31 +92,52 @@ class RxValue {
|
|
|
refs[currName].value = v.value;
|
|
|
})
|
|
|
}
|
|
|
- return refs[currName];
|
|
|
+ return refs[currName].value;
|
|
|
}
|
|
|
|
|
|
- //历史回退
|
|
|
- if (histroy) {
|
|
|
- const snap = histroy.createValueSnap(initValue, initValue, f);
|
|
|
- f.subscribe((v)=>{
|
|
|
- if (!v._hstry) return;
|
|
|
+ obj._historySub[name] = f.subscribe((v)=>{
|
|
|
+ if (!v._hstry) return;
|
|
|
+ snap.OldValue = v.value;
|
|
|
+ if (obj._history) {
|
|
|
const s = snap.clone();
|
|
|
s.Value = v.value;
|
|
|
- snap.OldValue = s.Value;
|
|
|
-
|
|
|
- histroy.record(s);
|
|
|
- })
|
|
|
- }
|
|
|
+ obj._history.record(s);
|
|
|
+ }
|
|
|
+ })
|
|
|
});
|
|
|
|
|
|
|
|
|
- return obj as typeof fields & {
|
|
|
- [K in keyof typeof fields as `set${Capitalize<string & K>}`]: (value: typeof fields[K], nohistory?:boolean) => void;
|
|
|
+ obj["setHistory"] = function(h: HistoryController){
|
|
|
+ obj._history = h;
|
|
|
+ }
|
|
|
+ obj["toJson"] = function() {
|
|
|
+ const out:any = {};
|
|
|
+ const names = Object.keys(_fields);
|
|
|
+ names.forEach(name=>{
|
|
|
+ out[name] = obj._rxs[name].getValue().value;
|
|
|
+ })
|
|
|
+ return out;
|
|
|
+ }
|
|
|
+ obj["fromJson"] = function(json:any) {
|
|
|
+ const out:any = {};
|
|
|
+ const names = Object.keys(_fields);
|
|
|
+ names.forEach(name=>{
|
|
|
+ obj._rxs[name].next({value: json[name], _hstry: false})
|
|
|
+ })
|
|
|
+ return out;
|
|
|
+ }
|
|
|
+
|
|
|
+ return obj as typeof _fields & {
|
|
|
+ [K in keyof typeof _fields as `set${Capitalize<string & K>}`]: (value: typeof _fields[K], nohistory?:boolean) => void;
|
|
|
+ } & {
|
|
|
+ [K in keyof typeof _fields as `on${Capitalize<string & K>}Changed`]: (subscribe: (value: typeof _fields[K], oldValue:typeof _fields[K])=>void) => void;
|
|
|
} & {
|
|
|
- [K in keyof typeof fields as `on${Capitalize<string & K>}Changed`]: (subscribe: (value: typeof fields[K])=>void) => void;
|
|
|
+ [K in keyof typeof _fields as `ref${Capitalize<string & K>}`]: () => typeof _fields[K];
|
|
|
} & {
|
|
|
- [K in keyof typeof fields as `ref${Capitalize<string & K>}`]: () => ReturnType< typeof ref<typeof fields[K]> >;
|
|
|
- };
|
|
|
+ setHistory: (history: HistoryController)=>void
|
|
|
+ toJson:()=>typeof _fields
|
|
|
+ fromJson:(json:typeof _fields)=>void
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|