From 7d4c3ea57ba3a04f7812983cdafa7f578bd047b3 Mon Sep 17 00:00:00 2001
From: David Sehnal <david.sehnal@gmail.com>
Date: Wed, 3 Apr 2019 13:43:58 +0200
Subject: [PATCH] wip mol-state

---
 src/mol-plugin/behavior/dynamic/labels.ts     |  2 +-
 .../behavior/static/representation.ts         |  2 +-
 src/mol-plugin/behavior/static/state.ts       |  2 +-
 src/mol-plugin/ui/state/tree.tsx              | 26 +++++++++----------
 src/mol-state/object.ts                       |  2 +-
 src/mol-state/state.ts                        | 11 +++-----
 src/mol-state/transform.ts                    | 13 ++++++++++
 src/mol-state/tree/transient.ts               | 12 ++++++---
 8 files changed, 43 insertions(+), 27 deletions(-)

diff --git a/src/mol-plugin/behavior/dynamic/labels.ts b/src/mol-plugin/behavior/dynamic/labels.ts
index b3597932c..313ce27bf 100644
--- a/src/mol-plugin/behavior/dynamic/labels.ts
+++ b/src/mol-plugin/behavior/dynamic/labels.ts
@@ -118,7 +118,7 @@ export const SceneLabels = PluginBehavior.create<SceneLabelsProps>({
             for (const s of structures) {
                 const rootStructure = getRootStructure(s, state)
                 if (!rootStructure || !SO.Molecule.Structure.is(rootStructure.obj)) continue
-                if (!s.state.isHidden) {
+                if (!s.transform.state.isHidden) {
                     rootStructures.add(rootStructure.obj)
                 }
             }
diff --git a/src/mol-plugin/behavior/static/representation.ts b/src/mol-plugin/behavior/static/representation.ts
index df0310e77..d17291691 100644
--- a/src/mol-plugin/behavior/static/representation.ts
+++ b/src/mol-plugin/behavior/static/representation.ts
@@ -92,5 +92,5 @@ export function UpdateRepresentationVisibility(ctx: PluginContext) {
 }
 
 function updateVisibility(cell: StateObjectCell, r: Representation<any>) {
-    r.setState({ visible: !cell.state.isHidden });
+    r.setState({ visible: !cell.transform.state.isHidden });
 }
\ No newline at end of file
diff --git a/src/mol-plugin/behavior/static/state.ts b/src/mol-plugin/behavior/static/state.ts
index c4111b371..0237d3263 100644
--- a/src/mol-plugin/behavior/static/state.ts
+++ b/src/mol-plugin/behavior/static/state.ts
@@ -87,7 +87,7 @@ export function ToggleExpanded(ctx: PluginContext) {
 }
 
 export function ToggleVisibility(ctx: PluginContext) {
-    PluginCommands.State.ToggleVisibility.subscribe(ctx, ({ state, ref }) => setVisibility(state, ref, !state.cells.get(ref)!.state.isHidden));
+    PluginCommands.State.ToggleVisibility.subscribe(ctx, ({ state, ref }) => setVisibility(state, ref, !state.cells.get(ref)!.transform.state.isHidden));
 }
 
 function setVisibility(state: State, root: StateTransform.Ref, value: boolean) {
diff --git a/src/mol-plugin/ui/state/tree.tsx b/src/mol-plugin/ui/state/tree.tsx
index 5e0815d2d..d7680c969 100644
--- a/src/mol-plugin/ui/state/tree.tsx
+++ b/src/mol-plugin/ui/state/tree.tsx
@@ -54,8 +54,8 @@ class StateTreeNode extends PluginUIComponent<{ cell: StateObjectCell, depth: nu
         this.subscribe(this.plugin.events.state.cell.stateUpdated, e => {
             if (this.props.cell === e.cell && this.is(e) && e.state.cells.has(this.ref)) {
                 this.forceUpdate();
-                // if (!!this.props.cell.state.isCollapsed !== this.state.isCollapsed) {
-                //     this.setState({ isCollapsed: !!e.cell.state.isCollapsed });
+                // if (!!this.props.cell.transform.state.isCollapsed !== this.state.isCollapsed) {
+                //     this.setState({ isCollapsed: !!e.cell.transform.state.isCollapsed });
                 // }
             }
         });
@@ -74,22 +74,22 @@ class StateTreeNode extends PluginUIComponent<{ cell: StateObjectCell, depth: nu
     }
 
     state = {
-        isCollapsed: !!this.props.cell.state.isCollapsed
+        isCollapsed: !!this.props.cell.transform.state.isCollapsed
     }
 
     static getDerivedStateFromProps(props: _Props<StateTreeNode>, state: _State<StateTreeNode>): _State<StateTreeNode> | null {
-        if (!!props.cell.state.isCollapsed === state.isCollapsed) return null;
-        return { isCollapsed: !!props.cell.state.isCollapsed };
+        if (!!props.cell.transform.state.isCollapsed === state.isCollapsed) return null;
+        return { isCollapsed: !!props.cell.transform.state.isCollapsed };
     }
 
     render() {
         const cell = this.props.cell;
-        if (!cell || cell.obj === StateObject.Null) {
+        if (!cell || cell.obj === StateObject.Null || !cell.parent.tree.transforms.has(cell.transform.ref)) {
             return null;
         }
 
-        const cellState = cell.state;
-        const showLabel = cell.status !== 'ok' || !cell.state.isGhost;
+        const cellState = cell.transform.state;
+        const showLabel = cell.status !== 'ok' || !cell.transform.state.isGhost;
         const children = cell.parent.tree.children.get(this.ref);
         const newDepth = showLabel ? this.props.depth + 1 : this.props.depth;
 
@@ -140,7 +140,7 @@ class StateTreeNodeLabel extends PluginUIComponent<
             if (e.state.transforms.has(this.ref)) {
                 this.setState({
                     isCurrent: this.props.cell.parent.current === this.ref,
-                    isCollapsed: !!this.props.cell.state.isCollapsed
+                    isCollapsed: !!this.props.cell.transform.state.isCollapsed
                 });
             }
         });
@@ -148,12 +148,12 @@ class StateTreeNodeLabel extends PluginUIComponent<
 
     state = {
         isCurrent: this.props.cell.parent.current === this.ref,
-        isCollapsed: !!this.props.cell.state.isCollapsed
+        isCollapsed: !!this.props.cell.transform.state.isCollapsed
     }
 
     static getDerivedStateFromProps(props: _Props<StateTreeNodeLabel>, state: _State<StateTreeNodeLabel>): _State<StateTreeNodeLabel> | null {
         const isCurrent = props.cell.parent.current === props.cell.transform.ref;
-        const isCollapsed = !!props.cell.state.isCollapsed;
+        const isCollapsed = !!props.cell.transform.state.isCollapsed;
 
         if (state.isCollapsed === isCollapsed && state.isCurrent === isCurrent) return null;
         return { isCurrent, isCollapsed };
@@ -227,7 +227,7 @@ class StateTreeNodeLabel extends PluginUIComponent<
         }
 
         const children = cell.parent.tree.children.get(this.ref);
-        const cellState = cell.state;
+        const cellState = cell.transform.state;
 
         const visibility = <button onClick={this.toggleVisible} className={`msp-btn msp-btn-link msp-tree-visibility${cellState.isHidden ? ' msp-tree-visibility-hidden' : ''}`}>
             <span className='msp-icon msp-icon-visual-visibility' />
@@ -244,7 +244,7 @@ class StateTreeNodeLabel extends PluginUIComponent<
             {children.size > 0 &&  <button onClick={this.toggleExpanded} className='msp-btn msp-btn-link msp-tree-toggle-exp-button'>
                 <span className={`msp-icon msp-icon-${cellState.isCollapsed ? 'expand' : 'collapse'}`} />
             </button>}
-            {!cell.state.isLocked && <button onClick={this.remove} className='msp-btn msp-btn-link msp-tree-remove-button'>
+            {!cell.transform.state.isLocked && <button onClick={this.remove} className='msp-btn msp-btn-link msp-tree-remove-button'>
                 <span className='msp-icon msp-icon-remove' />
             </button>}{visibility}
         </div>;
diff --git a/src/mol-state/object.ts b/src/mol-state/object.ts
index dd4c5c1d3..a144ca8f9 100644
--- a/src/mol-state/object.ts
+++ b/src/mol-state/object.ts
@@ -64,7 +64,7 @@ interface StateObjectCell<T extends StateObject = StateObject, F extends StateTr
     sourceRef: StateTransform.Ref | undefined,
 
     status: StateObjectCell.Status,
-    state: StateTransform.State,
+    // state: StateTransform.State,
 
     params: {
         definition: ParamDefinition.Params,
diff --git a/src/mol-state/state.ts b/src/mol-state/state.ts
index 5d0d00850..213ddb465 100644
--- a/src/mol-state/state.ts
+++ b/src/mol-state/state.ts
@@ -63,7 +63,6 @@ class State {
     private spine = new StateTreeSpine.Impl(this.cells);
 
     getSnapshot(): State.Snapshot {
-        this.cells.forEach(c => this._tree.updateState(c.transform.ref, c.state));
         return { tree: StateTree.toJSON(this._tree) };
     }
 
@@ -80,10 +79,10 @@ class State {
         const cell = this.cells.get(ref);
         if (!cell) return;
 
-        const update = typeof stateOrProvider === 'function' ? stateOrProvider(cell.state) : stateOrProvider;
+        const update = typeof stateOrProvider === 'function' ? stateOrProvider(cell.transform.state) : stateOrProvider;
 
-        if (StateTransform.assignState(cell.state, update)) {
-            // this._tree.updateCellState(ref, update)) {
+        if (this._tree.updateState(cell.transform.ref, update)) {
+            cell.transform = this._tree.transforms.get(cell.transform.ref);
             this.events.cell.stateUpdated.next({ state: this, ref, cell });
         }
     }
@@ -211,7 +210,6 @@ class State {
             sourceRef: void 0,
             obj: rootObject,
             status: 'ok',
-            state: { ...root.state },
             errorText: void 0,
             params: {
                 definition: {},
@@ -409,7 +407,7 @@ function findDeletes(ctx: UpdateContext): Ref[] {
 
 function syncNewStatesVisitor(n: StateTransform, tree: StateTree, ctx: UpdateContext) {
     const cell = ctx.cells.get(n.ref);
-    if (!cell || !StateTransform.assignState(cell.state, n.state)) return;
+    if (!cell || !StateTransform.syncState(cell.transform.state, n.state)) return;
     ctx.parent.events.cell.stateUpdated.next({ state: ctx.parent, ref: n.ref, cell });
 }
 
@@ -447,7 +445,6 @@ function initCellsVisitor(transform: StateTransform, _: any, { ctx, added }: Ini
         transform,
         sourceRef: void 0,
         status: 'pending',
-        state: { ...transform.state },
         errorText: void 0,
         params: void 0,
         cache: void 0
diff --git a/src/mol-state/transform.ts b/src/mol-state/transform.ts
index 0e11a042e..fe4b251bf 100644
--- a/src/mol-state/transform.ts
+++ b/src/mol-state/transform.ts
@@ -53,6 +53,19 @@ namespace Transform {
     export function assignState(a: State, b?: Partial<State>): boolean {
         if (!b) return false;
 
+        let changed = false;
+        for (const k of Object.keys(b)) {
+            const s = (b as any)[k], t = (a as any)[k];
+            if (!!s === !!t) continue;
+            changed = true;
+            (a as any)[k] = s;
+        }
+        return changed;
+    }
+
+    export function syncState(a: State, b?: Partial<State>): boolean {
+        if (!b) return false;
+
         let changed = false;
         for (const k of Object.keys(b)) {
             const s = (b as any)[k], t = (a as any)[k];
diff --git a/src/mol-state/tree/transient.ts b/src/mol-state/tree/transient.ts
index 068a5264d..e23b81aba 100644
--- a/src/mol-state/tree/transient.ts
+++ b/src/mol-state/tree/transient.ts
@@ -19,6 +19,7 @@ class TransientTree implements StateTree {
     private changedChildren = false;
 
     private _childMutations: Map<StateTransform.Ref, OrderedSet<StateTransform.Ref>> | undefined = void 0;
+    private _stateUpdates: Set<StateTransform.Ref> | undefined = void 0;
 
     private get childMutations() {
         if (this._childMutations) return this._childMutations;
@@ -145,9 +146,14 @@ class TransientTree implements StateTree {
         const old = this.transforms.get(ref);
         if (!StateTransform.isStateChange(old.state, state)) return false;
 
-        this.changeNodes();
-        // TODO: cache these changes?
-        this.transforms.set(ref, StateTransform.withState(old, state));
+        if (this._stateUpdates && this._stateUpdates.has(old.ref)) {
+            StateTransform.assignState(old.state, state);
+        } else {
+            if (!this._stateUpdates) this._stateUpdates = new Set();
+            this._stateUpdates.add(old.ref);
+            this.changeNodes();
+            this.transforms.set(ref, StateTransform.withState(old, state));
+        }
 
         return true;
     }
-- 
GitLab