Quellcode durchsuchen

MVS extension - additional work (#991)

* MVS extension: deterministic transform refs, updated metadata structure

* Perf-test for `sortIfNeeded`

* MVS extension: README

* MVS extension: show loading errors in the Mol* console

* MVS extension: auto-fix rotation matrix imprecisions

* MVS extension: data format provider

* MVS extension: Updated README

* MVS extension: rename deletePrevious -> replaceExisting, default to false in "Load MVS Data" to allow loading multiple files

* Perf-test for sortIfNeeded uses Benchmark.js
midlik vor 1 Jahr
Ursprung
Commit
c592a3b93d

BIN
docs/extensions/mvs/1cbs.png


+ 129 - 0
docs/extensions/mvs/README.md

@@ -0,0 +1,129 @@
+# Mol* MolViewSpec extension
+
+**MolViewSpec (MVS)** is a tool for standardized description of reproducible molecular visualizations shareable across software applications.
+
+MolViewSpec provides a generic description of typical visual scenes that may occur as part of molecular visualizations. A tree format allows the composition of complex scene descriptors by combining reoccurring nodes that serve as building blocks.
+
+
+## More sources:
+
+- MolViewSpec home page: https://molstar.org/mol-view-spec/
+- Python library `molviewspec` for building MolViewSpec views: https://pypi.org/project/molviewspec/
+- Python library `molviewspec` in action: https://colab.research.google.com/drive/1O2TldXlS01s-YgkD9gy87vWsfCBTYuz9
+
+
+## MolViewSpec data structure
+
+MVS is based on a tree format, i.e. a molecular view is described as a tree where individual node types represent common data operations needed to create the view (e.g. download, parse, color). Each node can have parameters that provide additional details for the operation. 
+
+A simple example of a MVS tree showing PDB structure 1cbs:
+
+![Example MolViewSpec - 1cbs with labelled protein and ligand](./1cbs.png "Example MolViewSpec")
+
+```txt
+- root {}
+  - download {url: "https://www.ebi.ac.uk/pdbe/entry-files/1cbs.bcif"}
+    - parse {format: "bcif"}
+      - structure {type: "model"}
+        - component {selector: "polymer"}
+          - representation {type: "cartoon"}
+            - color {color: "green"}
+            - color {selector: {label_asym_id: "A", beg_label_seq_id: 1, end_label_seq_id: 50}, color: "#6688ff"}
+          - label {text: "Protein"}
+        - component {selector: "ligand"}
+          - representation {type: "ball_and_stick"}
+            - color {color: "#cc3399"}
+          - label {text: "Retinoic Acid"}
+  - canvas {background_color: "#ffffee"}
+  - camera {target: [17,21,27], position: [41,34,69], up: [-0.129,0.966,-0.224]}
+```
+
+(This is just a human-friendly representation of the tree, not the actual data format!)
+
+A complete list of supported node types and their parameters is described by the [MVS tree schema](./mvs-tree-schema.md).
+
+### Encoding
+
+A MolViewSpec tree can be encoded and stored in `.mvsj` format, which is basically a JSON representation of the tree with additional metadata:
+
+```json
+{
+  "metadata": {
+    "title": "Example MolViewSpec - 1cbs with labelled protein and ligand",
+    "version": "1",
+    "timestamp": "2023-11-24T10:38:17.483Z"
+  },
+  "root": {
+    "kind": "root",
+    "children": [
+      {
+        "kind": "download",
+        "params": {"url": "https://www.ebi.ac.uk/pdbe/entry-files/1cbs.bcif"},
+        "children": [
+          {
+            "kind": "parse",
+            "params": {"format": "bcif"},
+            "children": [
+    ...
+```
+Complete file: [1cbs.mvsj](../../../examples/mvs/1cbs.mvsj)
+
+
+## MolViewSpec extension functionality
+
+Mol* MolViewSpec extension provides functionality for building, validating, and visualizing MVS views.
+
+### Graphical user interface
+
+- **Drag&drop support:** The easiest way to load a MVS view into Mol* Viewer is to drag a `.mvsj` file and drop it in a browser window with Mol* Viewer.
+
+- **Load via menu:** Another way to load a MVS view is to use "Download File" or "Open Files" action, available in the "Home" tab in the left panel. For these actions, the "Format" parameter must be set to "MVSJ" (in the "Miscellaneous" category) or "Auto".
+
+- **URL parameters:** Mol* Viewer supports `mvs-url`, `mvs-data`, and `mvs-format` URL parameters to specify a MVS view to be loaded when the viewer is initialized.
+  - `mvs-url` specifies the address from which the MVS view should be retrieved.
+  - `mvs-data` specifies the MVS view data directly. Keep in mind that some characters must be escaped to be used in the URL. Also beware that URLs longer than 2000 character may not work in all browsers.
+  - `mvs-format` specifies the format of the MVS view data (from `mvs-url` or `mvs-data`). The only allowed (and default) value is `mvsj`, as this is currently the only supported format.
+  
+  Examples of URL parameter usage:
+
+  - https://molstar.org/viewer?mvs-format=mvsj&mvs-url=https://raw.githubusercontent.com/molstar/molstar/master/examples/mvs/1cbs.mvsj
+
+  - https://molstar.org/viewer?mvs-format=mvsj&mvs-data=%7B%22metadata%22%3A%7B%22title%22%3A%22Example%20MolViewSpec%20-%201cbs%20with%20labelled%20protein%20and%20ligand%22%2C%22version%22%3A%221%22%2C%22timestamp%22%3A%222023-11-24T10%3A38%3A17.483%22%7D%2C%22root%22%3A%7B%22kind%22%3A%22root%22%2C%22children%22%3A%5B%7B%22kind%22%3A%22download%22%2C%22params%22%3A%7B%22url%22%3A%22https%3A//www.ebi.ac.uk/pdbe/entry-files/1cbs.bcif%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22parse%22%2C%22params%22%3A%7B%22format%22%3A%22bcif%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22structure%22%2C%22params%22%3A%7B%22type%22%3A%22model%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22component%22%2C%22params%22%3A%7B%22selector%22%3A%22polymer%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22representation%22%2C%22params%22%3A%7B%22type%22%3A%22cartoon%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22color%22%2C%22params%22%3A%7B%22color%22%3A%22green%22%7D%7D%2C%7B%22kind%22%3A%22color%22%2C%22params%22%3A%7B%22selector%22%3A%7B%22label_asym_id%22%3A%22A%22%2C%22beg_label_seq_id%22%3A1%2C%22end_label_seq_id%22%3A50%7D%2C%22color%22%3A%22%236688ff%22%7D%7D%5D%7D%2C%7B%22kind%22%3A%22label%22%2C%22params%22%3A%7B%22text%22%3A%22Protein%22%7D%7D%5D%7D%2C%7B%22kind%22%3A%22component%22%2C%22params%22%3A%7B%22selector%22%3A%22ligand%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22representation%22%2C%22params%22%3A%7B%22type%22%3A%22ball_and_stick%22%7D%2C%22children%22%3A%5B%7B%22kind%22%3A%22color%22%2C%22params%22%3A%7B%22color%22%3A%22%23cc3399%22%7D%7D%5D%7D%2C%7B%22kind%22%3A%22label%22%2C%22params%22%3A%7B%22text%22%3A%22Retinoic%20Acid%22%7D%7D%5D%7D%5D%7D%5D%7D%5D%7D%2C%7B%22kind%22%3A%22canvas%22%2C%22params%22%3A%7B%22background_color%22%3A%22%23ffffee%22%7D%7D%2C%7B%22kind%22%3A%22camera%22%2C%22params%22%3A%7B%22target%22%3A%5B17%2C21%2C27%5D%2C%22position%22%3A%5B41%2C34%2C69%5D%2C%22up%22%3A%5B-0.129%2C0.966%2C-0.224%5D%7D%7D%5D%7D%7D
+
+
+### Programming interface
+
+Most functions for manipulation of MVS data (including parsing, encoding, validating, and building) are provided by the `MVSData` object (defined in [src/extensions/mvs/mvs-data.ts](../../../src/extensions/mvs/mvs-data.ts)). In TypeScript, `MVSData` is also the type for a MVS view.
+
+The `loadMVS` function (defined in [src/extensions/mvs/load.ts](../../../src/extensions/mvs/load.ts)) can be used to load MVS view data into Mol* Viewer.
+
+Example usage:
+
+```ts
+// Fetch a MVS, validate, and load
+const response = await fetch('https://raw.githubusercontent.com/molstar/molstar/master/examples/mvs/1cbs.mvsj');
+const rawData = await response.text();
+const mvsData: MVSData = MVSData.fromMVSJ(rawData);
+if (!MVSData.isValid(mvsData)) throw new Error(`Oh no: ${MVSData.validationIssues(mvsData)}`);
+await loadMVS(this.plugin, mvsData, { replaceExisting: true });
+console.log('Loaded this:', MVSData.toPrettyString(mvsData));
+console.log('Loaded this:', MVSData.toMVSJ(mvsData));
+
+// Build a MVS and load
+const builder = MVSData.createBuilder();
+const structure = builder
+    .download({ url: 'https://www.ebi.ac.uk/pdbe/entry-files/download/1og2_updated.cif' })
+    .parse({ format: 'mmcif' })
+    .modelStructure();
+structure
+    .component({ selector: 'polymer' })
+    .representation({ type: 'cartoon' });
+structure
+    .component({ selector: 'ligand' })
+    .representation({ type: 'ball_and_stick' })
+    .color({ color: '#aa55ff' as any });
+const mvsData2: MVSData = builder.getState();
+await loadMVS(this.plugin, mvsData2, { replaceExisting: false });
+```
+
+When using the pre-built Mol* plugin bundle, `MVSData` and `loadMVS` are exposed as `molstar.PluginExtensions.mvs.MVSData` and `molstar.PluginExtensions.mvs.loadMVS`. Furthermore, the `molstar.Viewer` class has `loadMvsFromUrl` and `loadMvsData` methods, providing the same functionality as `mvs-url` and `mvs-data` URL parameters.

+ 3 - 4
docs/extensions/mvs/MVS-tree-documentation.md → docs/extensions/mvs/mvs-tree-schema.md

@@ -1,7 +1,6 @@
-(This documentation was auto-generated by the `treeSchemaToMarkdown` function)
+# MolViewSpec tree schema
 
-
-Tree schema:
+(This documentation was auto-generated by calling `treeSchemaToMarkdown(MVSTreeSchema, MVSDefaults)`)
 
   - **`root`**
 
@@ -45,7 +44,7 @@ Tree schema:
 
     Params:
 
-      - **`kind: `**`"model" | "assembly" | "symmetry" | "symmetry_mates"`
+      - **`type: `**`"model" | "assembly" | "symmetry" | "symmetry_mates"`
 
         Type of structure to be created (`"model"` for original model coordinates, `"assembly"` for assembly structure, `"symmetry"` for a set of crystal unit cells based on Miller indices, `"symmetry_mates"` for a set of asymmetric units within a radius from the original model).
 

+ 12 - 8
examples/mvs/1cbs-focus.mvsj

@@ -1,14 +1,12 @@
 {
- "version": 6,
+ "metadata": {
+  "title": "Example MolViewSpec - 1cbs with labelled and zoomed ligand",
+  "version": "1",
+  "timestamp": "2023-11-24T10:45:49.873Z"
+ },
  "root": {
   "kind": "root",
   "children": [
-   {
-    "kind": "canvas",
-    "params": {
-     "background_color": "#ffffee"
-    }
-   },
    {
     "kind": "download",
     "params": {
@@ -75,7 +73,7 @@
             "kind": "focus",
             "params": {
              "direction": [0.5, 0, -1],
-             "up": [0.5, 1, 0]
+             "up": [0.365, 0.913, 0.183]
             }
            },
            {
@@ -105,6 +103,12 @@
       ]
      }
     ]
+   },
+   {
+    "kind": "canvas",
+    "params": {
+     "background_color": "#ffffee"
+    }
    }
   ]
  }

+ 20 - 14
examples/mvs/1cbs.mvsj

@@ -1,21 +1,12 @@
 {
- "version": 6,
+ "metadata": {
+  "title": "Example MolViewSpec - 1cbs with labelled protein and ligand",
+  "version": "1",
+  "timestamp": "2023-11-24T10:38:17.483Z"
+ },
  "root": {
   "kind": "root",
   "children": [
-   {
-    "kind": "canvas",
-    "params": {
-     "background_color": "#ffffee"
-    }
-   },
-   {
-    "kind": "camera",
-    "params": {
-     "target": [17, 21, 27],
-     "position": [54, 41, 91]
-    }
-   },
    {
     "kind": "download",
     "params": {
@@ -57,6 +48,7 @@
               "params": {
                "selector": {
                 "label_asym_id": "A",
+                "beg_label_seq_id": 1,
                 "end_label_seq_id": 50
                },
                "color": "#6688ff"
@@ -105,6 +97,20 @@
       ]
      }
     ]
+   },
+   {
+    "kind": "canvas",
+    "params": {
+     "background_color": "#ffffee"
+    }
+   },
+   {
+    "kind": "camera",
+    "params": {
+     "target": [17, 21, 27],
+     "position": [41, 34, 69],
+     "up": [-0.129,0.966,-0.224]
+    }
    }
   ]
  }

+ 5 - 1
examples/mvs/1h9t_domain_colors.mvsj

@@ -1,5 +1,9 @@
 {
- "version": 6,
+ "metadata": {
+  "title": "Example MolViewSpec - 1h9t colored by external annotation",
+  "version": "1",
+  "timestamp": "2023-11-24T10:47:33.182Z"
+ },
  "root": {
   "kind": "root",
   "children": [

+ 6 - 6
examples/mvs/1h9t_domain_labels.mvsj

@@ -1,5 +1,9 @@
 {
- "version": 6,
+ "metadata": {
+  "title": "Example MolViewSpec - 1h9t colored and labelled by external annotation",
+  "version": "1",
+  "timestamp": "2023-11-24T10:48:28.677Z"
+ },
  "root": {
   "kind": "root",
   "children": [
@@ -557,11 +561,7 @@
            {
             "kind": "focus",
             "params": {
-             "direction": [
-              -0.3,
-              -0.1,
-              -1
-             ]
+             "direction": [-0.3, -0.1, -1]
             }
            }
           ]

+ 1 - 0
src/apps/viewer/app.ts

@@ -556,4 +556,5 @@ export const ViewerAutoPreset = StructureRepresentationPresetProvider({
 
 export const PluginExtensions = {
     wwPDBStructConn: wwPDBStructConnExtensionFunctions,
+    mvs: { MVSData, loadMVS },
 };

+ 1 - 1
src/examples/mvs/mvs-render.ts

@@ -78,7 +78,7 @@ async function main(args: Args): Promise<void> {
         const mvsData = MVSData.fromMVSJ(data);
         removeLabelNodes(mvsData);
 
-        await loadMVS(plugin, mvsData, { sanityChecks: true, deletePrevious: true });
+        await loadMVS(plugin, mvsData, { sanityChecks: true, replaceExisting: true });
         fs.mkdirSync(path.dirname(output), { recursive: true });
         if (args.molj) {
             await plugin.saveStateSnapshot(withExtension(output, '.molj'));

+ 1 - 0
src/extensions/mvs/README.md

@@ -0,0 +1 @@
+Find the MVS extension documentation [here](../../../docs/extensions/mvs/README.md).

+ 46 - 0
src/extensions/mvs/_spec/mvs-data.spec.ts

@@ -0,0 +1,46 @@
+/**
+ * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
+ *
+ * @author Adam Midlik <midlik@gmail.com>
+ */
+
+import fs from 'fs';
+import { MVSData } from '../mvs-data';
+
+
+describe('MVSData', () => {
+    it('MVSData functions work', async () => {
+        const data = fs.readFileSync('examples/mvs/1cbs.mvsj', { encoding: 'utf8' });
+        const mvsData = MVSData.fromMVSJ(data);
+        expect(mvsData).toBeTruthy();
+
+        expect(MVSData.validationIssues(mvsData)).toEqual(undefined);
+
+        expect(MVSData.isValid(mvsData)).toEqual(true);
+
+        const reencoded = MVSData.toMVSJ(mvsData);
+        expect(reencoded.replace(/\s/g, '')).toEqual(data.replace(/\s/g, ''));
+
+        const prettyString = MVSData.toPrettyString(mvsData);
+        expect(typeof prettyString).toEqual('string');
+        expect(prettyString.length).toBeGreaterThan(0);
+    });
+
+    it('MVSData builder works', async () => {
+        const builder = MVSData.createBuilder();
+        expect(builder).toBeTruthy();
+
+        const mvsData = builder.getState();
+        expect(MVSData.validationIssues(mvsData)).toEqual(undefined);
+
+        builder
+            .download({ url: 'http://example.com' })
+            .parse({ format: 'mmcif' })
+            .assemblyStructure({ assembly_id: '1' })
+            .component({ selector: 'polymer' })
+            .representation()
+            .color({ color: 'green', selector: { label_asym_id: 'A' } });
+        const mvsData2 = builder.getState();
+        expect(MVSData.validationIssues(mvsData2)).toEqual(undefined);
+    });
+});

+ 24 - 1
src/extensions/mvs/behavior.ts

@@ -6,11 +6,13 @@
 
 import { CustomModelProperty } from '../../mol-model-props/common/custom-model-property';
 import { CustomStructureProperty } from '../../mol-model-props/common/custom-structure-property';
+import { DataFormatProvider } from '../../mol-plugin-state/formats/provider';
 import { PluginDragAndDropHandler } from '../../mol-plugin-state/manager/drag-and-drop';
 import { LociLabelProvider } from '../../mol-plugin-state/manager/loci-label';
 import { PluginBehavior } from '../../mol-plugin/behavior/behavior';
 import { PluginContext } from '../../mol-plugin/context';
 import { StructureRepresentationProvider } from '../../mol-repr/structure/representation';
+import { StateAction } from '../../mol-state';
 import { ColorTheme } from '../../mol-theme/color';
 import { ParamDefinition as PD } from '../../mol-util/param-definition';
 import { MVSAnnotationColorThemeProvider } from './components/annotation-color-theme';
@@ -19,6 +21,7 @@ import { MVSAnnotationsProvider } from './components/annotation-prop';
 import { MVSAnnotationTooltipsLabelProvider, MVSAnnotationTooltipsProvider } from './components/annotation-tooltips-prop';
 import { CustomLabelRepresentationProvider } from './components/custom-label/representation';
 import { CustomTooltipsLabelProvider, CustomTooltipsProvider } from './components/custom-tooltips-prop';
+import { LoadMvsData, MVSJFormatProvider } from './components/formats';
 import { makeMultilayerColorThemeProvider } from './components/multilayer-color-theme';
 import { loadMVS } from './load';
 import { MVSData } from './mvs-data';
@@ -32,6 +35,8 @@ interface Registrables {
     colorThemes?: ColorTheme.Provider[],
     lociLabels?: LociLabelProvider[],
     dragAndDropHandlers?: DragAndDropHandler[],
+    dataFormats?: { name: string, provider: DataFormatProvider }[],
+    actions?: StateAction[],
 }
 
 
@@ -67,6 +72,12 @@ export const MolViewSpec = PluginBehavior.create<{ autoAttach: boolean }>({
             dragAndDropHandlers: [
                 MVSDragAndDropHandler,
             ],
+            dataFormats: [
+                { name: 'MVSJ', provider: MVSJFormatProvider },
+            ],
+            actions: [
+                LoadMvsData,
+            ]
         };
 
         register(): void {
@@ -88,6 +99,12 @@ export const MolViewSpec = PluginBehavior.create<{ autoAttach: boolean }>({
             for (const handler of this.registrables.dragAndDropHandlers ?? []) {
                 this.ctx.managers.dragAndDrop.addHandler(handler.name, handler.handle);
             }
+            for (const format of this.registrables.dataFormats ?? []) {
+                this.ctx.dataFormats.add(format.name, format.provider);
+            }
+            for (const action of this.registrables.actions ?? []) {
+                this.ctx.state.data.actions.add(action);
+            }
         }
         update(p: { autoAttach: boolean }) {
             const updated = this.params.autoAttach !== p.autoAttach;
@@ -119,6 +136,12 @@ export const MolViewSpec = PluginBehavior.create<{ autoAttach: boolean }>({
             for (const handler of this.registrables.dragAndDropHandlers ?? []) {
                 this.ctx.managers.dragAndDrop.removeHandler(handler.name);
             }
+            for (const format of this.registrables.dataFormats ?? []) {
+                this.ctx.dataFormats.remove(format.name);
+            }
+            for (const action of this.registrables.actions ?? []) {
+                this.ctx.state.data.actions.remove(action);
+            }
         }
     },
     params: () => ({
@@ -144,7 +167,7 @@ const MVSDragAndDropHandler: DragAndDropHandler = {
             if (file.name.toLowerCase().endsWith('.mvsj')) {
                 const data = await file.text();
                 const mvsData = MVSData.fromMVSJ(data);
-                await loadMVS(plugin, mvsData, { sanityChecks: true, deletePrevious: !applied });
+                await loadMVS(plugin, mvsData, { sanityChecks: true, replaceExisting: !applied });
                 applied = true;
             }
         }

+ 9 - 3
src/extensions/mvs/camera.ts

@@ -31,12 +31,15 @@ const DefaultFocusOptions = {
 const DefaultCanvasBackgroundColor = ColorNames.white;
 
 
+const _tmpVec = Vec3();
+
 /** Set the camera based on a camera node params. */
 export async function setCamera(plugin: PluginContext, params: ParamsOfKind<MolstarTree, 'camera'>) {
     const target = Vec3.create(...params.target);
     let position = Vec3.create(...params.position);
     if (plugin.canvas3d) position = fovAdjustedPosition(target, position, plugin.canvas3d.camera.state.mode, plugin.canvas3d.camera.state.fov);
     const up = Vec3.create(...params.up);
+    Vec3.orthogonalize(up, Vec3.sub(_tmpVec, target, position), up);
     const snapshot: Partial<Camera.Snapshot> = { target, position, up, radius: 10_000, 'radiusMax': 10_000 };
     await PluginCommands.Camera.SetSnapshot(plugin, { snapshot });
 }
@@ -57,11 +60,14 @@ export async function setFocus(plugin: PluginContext, structureNodeSelector: Sta
     const boundingSphere = structure ? Loci.getBoundingSphere(Structure.Loci(structure)) : getPluginBoundingSphere(plugin);
     if (boundingSphere && plugin.canvas3d) {
         const extraRadius = structure ? DefaultFocusOptions.extraRadiusForFocus : DefaultFocusOptions.extraRadiusForZoomAll;
+        const direction = Vec3.create(...params.direction);
+        const up = Vec3.create(...params.up);
+        Vec3.orthogonalize(up, direction, up);
         const snapshot = snapshotFromSphereAndDirections(plugin.canvas3d.camera, {
             center: boundingSphere.center,
             radius: boundingSphere.radius + extraRadius,
-            up: Vec3.create(...params.up),
-            direction: Vec3.create(...params.direction),
+            up,
+            direction,
         });
         await PluginCommands.Camera.SetSnapshot(plugin, { snapshot });
     }
@@ -75,7 +81,7 @@ function snapshotFromSphereAndDirections(camera: Camera, options: { center: Vec3
     const { center, direction, up } = options;
     const radius = Math.max(options.radius, DefaultFocusOptions.minRadius);
     const distance = camera.getTargetDistance(radius);
-    const deltaDirection = Vec3.setMagnitude(Vec3(), direction, distance);
+    const deltaDirection = Vec3.setMagnitude(_tmpVec, direction, distance);
     const position = Vec3.sub(Vec3(), center, deltaDirection);
     return { target: center, position, up, radius };
 }

+ 67 - 0
src/extensions/mvs/components/formats.ts

@@ -0,0 +1,67 @@
+/**
+ * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
+ *
+ * @author Adam Midlik <midlik@gmail.com>
+ */
+
+import { DataFormatProvider } from '../../../mol-plugin-state/formats/provider';
+import { PluginStateObject as SO } from '../../../mol-plugin-state/objects';
+import { PluginContext } from '../../../mol-plugin/context';
+import { StateAction, StateObjectRef } from '../../../mol-state';
+import { Task } from '../../../mol-task';
+import { ParamDefinition as PD } from '../../../mol-util/param-definition';
+import { loadMVS } from '../load';
+import { MVSData } from '../mvs-data';
+import { MVSTransform } from './annotation-structure-component';
+
+
+/** Plugin state object storing `MVSData` */
+export class Mvs extends SO.Create<MVSData>({ name: 'MVS Data', typeClass: 'Data' }) { }
+
+/** Transformer for parsing data in MVSJ format */
+export const ParseMVSJ = MVSTransform({
+    name: 'mvs-parse-mvsj',
+    display: { name: 'MVS Annotation Component', description: 'A molecular structure component defined by MVS annotation data.' },
+    from: SO.Data.String,
+    to: Mvs,
+})({
+    apply({ a }) {
+        const mvsData = MVSData.fromMVSJ(a.data);
+        return new Mvs(mvsData);
+    },
+});
+
+
+/** Params for the `LoadMvsData` action */
+const LoadMvsDataParams = {
+    replaceExisting: PD.Boolean(false, { description: 'If true, the loaded MVS view will replace the current state; if false, the MVS view will be added to the current state.' }),
+};
+
+/** State action which loads a MVS view into Mol* */
+export const LoadMvsData = StateAction.build({
+    display: { name: 'Load MVS Data' },
+    from: Mvs,
+    params: LoadMvsDataParams,
+})(({ a, params }, plugin: PluginContext) => Task.create('Load MVS Data', async () => {
+    const mvsData = a.data;
+    await loadMVS(plugin, mvsData, { replaceExisting: params.replaceExisting });
+}));
+
+
+/** Data format provider for MVSJ format.
+ * If Visuals:On, it will load the parsed MVS view;
+ * otherwise it will just create a plugin state object with parsed data. */
+export const MVSJFormatProvider: DataFormatProvider<{}, StateObjectRef<Mvs>, any> = DataFormatProvider({
+    label: 'MVSJ',
+    description: 'MVSJ',
+    category: 'Miscellaneous',
+    stringExtensions: ['mvsj'],
+    parse: async (plugin, data) => {
+        return plugin.state.data.build().to(data).apply(ParseMVSJ).commit();
+    },
+    visuals: async (plugin, data) => {
+        const ref = StateObjectRef.resolveRef(data);
+        const params = PD.getDefaultValues(LoadMvsDataParams);
+        return await plugin.state.data.applyAction(LoadMvsData, params, ref).run();
+    },
+});

+ 7 - 2
src/extensions/mvs/helpers/utils.ts

@@ -80,11 +80,16 @@ export function filterDefined<T>(elements: (T | undefined | null)[]): T[] {
     return elements.filter(x => x !== undefined && x !== null) as T[];
 }
 
-/** Create an 8-hex-character hash for a given input string, e.g. 'spanish inquisition' -> 'bd65e59a' */
-export function stringHash(input: string): string {
+/** Create an 8-hex-character hash for a given input string, e.g. 'spanish inquisition' -> '7f9ac4be' */
+function stringHash32(input: string): string {
     const uint32hash = hashString(input) >>> 0; // >>>0 converts to uint32, LOL
     return uint32hash.toString(16).padStart(8, '0');
 }
+/** Create an 16-hex-character hash for a given input string, e.g. 'spanish inquisition' -> '7f9ac4be544330be'*/
+export function stringHash(input: string): string {
+    const reversed = input.split('').reverse().join('');
+    return stringHash32(input) + stringHash32(reversed);
+}
 
 /** Return type of elements in a set */
 export type ElementOfSet<S> = S extends Set<infer T> ? T : never

+ 110 - 16
src/extensions/mvs/load-helpers.ts

@@ -9,7 +9,7 @@ import { StructureComponentParams } from '../../mol-plugin-state/helpers/structu
 import { StructureFromModel, TransformStructureConformation } from '../../mol-plugin-state/transforms/model';
 import { StructureRepresentation3D } from '../../mol-plugin-state/transforms/representation';
 import { PluginContext } from '../../mol-plugin/context';
-import { StateBuilder, StateObjectSelector, StateTransformer } from '../../mol-state';
+import { StateBuilder, StateObject, StateObjectSelector, StateTransform, StateTransformer } from '../../mol-state';
 import { arrayDistinct } from '../../mol-util/array';
 import { canonicalJsonString } from '../../mol-util/json';
 import { MVSAnnotationColorThemeProps, MVSAnnotationColorThemeProvider } from './components/annotation-color-theme';
@@ -30,30 +30,29 @@ import { DefaultColor } from './tree/mvs/mvs-defaults';
 
 
 /** Function responsible for loading a tree node `node` into Mol*.
- * Should apply changes within `update` but not commit them.
+ * Should apply changes within `updateParent.update` but not commit them.
  * Should modify `context` accordingly, if it is needed for loading other nodes later.
- * `msParent` is the result of loading the node's parent into Mol* state hierarchy (or the hierarchy root in case of root node). */
-export type LoadingAction<TNode extends Tree, TContext> = (update: StateBuilder.Root, msParent: StateObjectSelector, node: TNode, context: TContext) => StateObjectSelector | undefined
+ * `updateParent.selector` is the result of loading the node's parent into Mol* state hierarchy (or the hierarchy root in case of root node). */
+export type LoadingAction<TNode extends Tree, TContext> = (updateParent: UpdateTarget, node: TNode, context: TContext) => UpdateTarget | undefined
 
 /** Loading actions for loading a tree into Mol*, per node kind. */
 export type LoadingActions<TTree extends Tree, TContext> = { [kind in Kind<SubTree<TTree>>]?: LoadingAction<SubTreeOfKind<TTree, kind>, TContext> }
 
 /** Load a tree into Mol*, by applying loading actions in DFS order and then commiting at once.
- * If `deletePrevious`, remove all objects in the current Mol* state; otherwise add to the current state. */
-export async function loadTree<TTree extends Tree, TContext>(plugin: PluginContext, tree: TTree, loadingActions: LoadingActions<TTree, TContext>, context: TContext, options?: { deletePrevious?: boolean }) {
-    const mapping = new Map<SubTree<TTree>, StateObjectSelector | undefined>();
-    const update = plugin.build();
-    const msRoot = update.toRoot().selector;
-    if (options?.deletePrevious) {
-        update.currentTree.children.get(msRoot.ref).forEach(child => update.delete(child));
+ * If `options.replaceExisting`, remove all objects in the current Mol* state; otherwise add to the current state. */
+export async function loadTree<TTree extends Tree, TContext>(plugin: PluginContext, tree: TTree, loadingActions: LoadingActions<TTree, TContext>, context: TContext, options?: { replaceExisting?: boolean }) {
+    const mapping = new Map<SubTree<TTree>, UpdateTarget | undefined>();
+    const updateRoot: UpdateTarget = UpdateTarget.create(plugin, options?.replaceExisting ?? false);
+    if (options?.replaceExisting) {
+        UpdateTarget.deleteChildren(updateRoot);
     }
     dfs<TTree>(tree, (node, parent) => {
         const kind: Kind<typeof node> = node.kind;
         const action = loadingActions[kind] as LoadingAction<typeof node, TContext> | undefined;
         if (action) {
-            const msParent = parent ? mapping.get(parent) : msRoot;
-            if (msParent) {
-                const msNode = action(update, msParent, node, context);
+            const updateParent = parent ? mapping.get(parent) : updateRoot;
+            if (updateParent) {
+                const msNode = action(updateParent, node, context);
                 mapping.set(node, msNode);
             } else {
                 console.warn(`No target found for this "${node.kind}" node`);
@@ -61,7 +60,84 @@ export async function loadTree<TTree extends Tree, TContext>(plugin: PluginConte
             }
         }
     });
-    await update.commit();
+    await UpdateTarget.commit(updateRoot);
+}
+
+
+/** A wrapper for updating Mol* state, while using deterministic transform refs.
+ * ```
+ * updateTarget = UpdateTarget.create(plugin); // like update = plugin.build();
+ * UpdateTarget.apply(updateTarget, transformer, params); // like update.to(selector).apply(transformer, params);
+ * await UpdateTarget.commit(updateTarget); // like await update.commit();
+ * ```
+ */
+export interface UpdateTarget {
+    readonly update: StateBuilder.Root,
+    readonly selector: StateObjectSelector,
+    readonly refManager: RefManager,
+}
+export const UpdateTarget = {
+    /** Create a new update, with `selector` pointing to the root. */
+    create(plugin: PluginContext, replaceExisting: boolean): UpdateTarget {
+        const update = plugin.build();
+        const msTarget = update.toRoot().selector;
+        const refManager = new RefManager(plugin, replaceExisting);
+        return { update, selector: msTarget, refManager };
+    },
+    /** Add a child node to `target.selector`, return a new `UpdateTarget` pointing to the new child. */
+    apply<A extends StateObject, B extends StateObject, P extends {}>(target: UpdateTarget, transformer: StateTransformer<A, B, P>, params?: Partial<P>, options?: Partial<StateTransform.Options>): UpdateTarget {
+        let refSuffix: string = transformer.id;
+        if (transformer.id === StructureRepresentation3D.id) {
+            const reprType = (params as any)?.type?.name ?? '';
+            refSuffix += `:${reprType}`;
+        }
+        const ref = target.refManager.getChildRef(target.selector, refSuffix);
+        const msResult = target.update.to(target.selector).apply(transformer, params, { ...options, ref }).selector;
+        return { ...target, selector: msResult };
+    },
+    /** Delete all children of `target.selector`. */
+    deleteChildren(target: UpdateTarget): UpdateTarget {
+        const children = target.update.currentTree.children.get(target.selector.ref);
+        children.forEach(child => target.update.delete(child));
+        return target;
+    },
+    /** Commit all changes done in the current update. */
+    commit(target: UpdateTarget): Promise<void> {
+        return target.update.commit();
+    },
+};
+
+/** Manages transform refs in a deterministic way. Uses refs like !mvs:3ce3664304d32c5d:0 */
+class RefManager {
+    /** For each hash (e.g. 3ce3664304d32c5d), store the number of already used refs with that hash. */
+    private _counter: Record<string, number> = {};
+    constructor(plugin: PluginContext, replaceExisting: boolean) {
+        if (!replaceExisting) {
+            plugin.state.data.cells.forEach(cell => {
+                const ref = cell.transform.ref;
+                if (ref.startsWith('!mvs:')) {
+                    const [_, hash, idNumber] = ref.split(':');
+                    const nextIdNumber = parseInt(idNumber) + 1;
+                    if (nextIdNumber > (this._counter[hash] ?? 0)) {
+                        this._counter[hash] = nextIdNumber;
+                    }
+                }
+            });
+        }
+    }
+    /** Return ref for a new node with given `hash`; update the counter accordingly. */
+    private nextRef(hash: string): string {
+        this._counter[hash] ??= 0;
+        const idNumber = this._counter[hash]++;
+        return `!mvs:${hash}:${idNumber}`;
+    }
+    /** Return ref for a new node based on parent and desired suffix. */
+    getChildRef(parent: StateObjectSelector, suffix: string): string {
+        const hashBase = parent.ref.replace(/^!mvs:/, '') + ':' + suffix;
+        const hash = stringHash(hashBase);
+        const result = this.nextRef(hash);
+        return result;
+    }
 }
 
 
@@ -78,7 +154,9 @@ export function transformFromRotationTranslation(rotation: number[] | null | und
     if (translation && translation.length !== 3) throw new Error(`'translation' param for 'transform' node must be array of 3 elements, found ${translation}`);
     const T = Mat4.identity();
     if (rotation) {
-        Mat4.fromMat3(T, Mat3.fromArray(Mat3(), rotation, 0));
+        const rotMatrix = Mat3.fromArray(Mat3(), rotation, 0);
+        ensureRotationMatrix(rotMatrix, rotMatrix);
+        Mat4.fromMat3(T, rotMatrix);
     }
     if (translation) {
         Mat4.setTranslation(T, Vec3.fromArray(Vec3(), translation, 0));
@@ -87,6 +165,22 @@ export function transformFromRotationTranslation(rotation: number[] | null | und
     return T;
 }
 
+/** Adjust values in a close-to-rotation matrix `a` to ensure it is a proper rotation matrix
+ * (i.e. its columns and rows are orthonormal and determinant equal to 1, within available precission). */
+function ensureRotationMatrix(out: Mat3, a: Mat3) {
+    const x = Vec3.fromArray(_tmpVecX, a, 0);
+    const y = Vec3.fromArray(_tmpVecY, a, 3);
+    const z = Vec3.fromArray(_tmpVecZ, a, 6);
+    Vec3.normalize(x, x);
+    Vec3.orthogonalize(y, x, y);
+    Vec3.normalize(z, Vec3.cross(z, x, y));
+    Mat3.fromColumns(out, x, y, z);
+    return out;
+}
+const _tmpVecX = Vec3();
+const _tmpVecY = Vec3();
+const _tmpVecZ = Vec3();
+
 /** Create an array of props for `TransformStructureConformation` transformers from all 'transform' nodes applied to a 'structure' node. */
 export function transformProps(node: SubTreeOfKind<MolstarTree, 'structure'>): StateTransformer.Params<TransformStructureConformation>[] {
     const result = [] as StateTransformer.Params<TransformStructureConformation>[];

+ 76 - 70
src/extensions/mvs/load.ts

@@ -8,16 +8,16 @@ import { Download, ParseCif } from '../../mol-plugin-state/transforms/data';
 import { CustomModelProperties, CustomStructureProperties, ModelFromTrajectory, StructureComponent, StructureFromModel, TrajectoryFromMmCif, TrajectoryFromPDB, TransformStructureConformation } from '../../mol-plugin-state/transforms/model';
 import { StructureRepresentation3D } from '../../mol-plugin-state/transforms/representation';
 import { PluginContext } from '../../mol-plugin/context';
-import { StateBuilder, StateObjectSelector } from '../../mol-state';
+import { StateObjectSelector } from '../../mol-state';
 import { canonicalJsonString } from '../../mol-util/json';
+import { MolViewSpec } from './behavior';
+import { setCamera, setCanvas, setFocus } from './camera';
 import { MVSAnnotationsProvider } from './components/annotation-prop';
 import { MVSAnnotationStructureComponent } from './components/annotation-structure-component';
 import { MVSAnnotationTooltipsProvider } from './components/annotation-tooltips-prop';
 import { CustomLabelProps, CustomLabelRepresentationProvider } from './components/custom-label/representation';
 import { CustomTooltipsProvider } from './components/custom-tooltips-prop';
-import { MolViewSpec } from './behavior';
-import { setCamera, setCanvas, setFocus } from './camera';
-import { AnnotationFromSourceKind, AnnotationFromUriKind, LoadingActions, collectAnnotationReferences, collectAnnotationTooltips, collectInlineTooltips, colorThemeForNode, componentFromXProps, componentPropsFromSelector, isPhantomComponent, labelFromXProps, loadTree, makeNearestReprMap, representationProps, structureProps, transformProps } from './load-helpers';
+import { AnnotationFromSourceKind, AnnotationFromUriKind, LoadingActions, UpdateTarget, collectAnnotationReferences, collectAnnotationTooltips, collectInlineTooltips, colorThemeForNode, componentFromXProps, componentPropsFromSelector, isPhantomComponent, labelFromXProps, loadTree, makeNearestReprMap, representationProps, structureProps, transformProps } from './load-helpers';
 import { MVSData } from './mvs-data';
 import { ParamsOfKind, SubTreeOfKind, validateTree } from './tree/generic/tree-schema';
 import { convertMvsToMolstar, mvsSanityCheck } from './tree/molstar/conversion';
@@ -26,22 +26,27 @@ import { MVSTreeSchema } from './tree/mvs/mvs-tree';
 
 
 /** Load a MolViewSpec (MVS) tree into the Mol* plugin.
- * If `options.deletePrevious`, remove all objects in the current Mol* state; otherwise add to the current state.
+ * If `options.replaceExisting`, remove all objects in the current Mol* state; otherwise add to the current state.
  * If `options.sanityChecks`, run some sanity checks and print potential issues to the console. */
-export async function loadMVS(plugin: PluginContext, data: MVSData, options: { deletePrevious?: boolean, sanityChecks?: boolean } = {}) {
-    // console.log(`MVS tree (v${data.version}):\n${treeToString(data.root)}`);
-    validateTree(MVSTreeSchema, data.root, 'MVS');
-    if (options.sanityChecks) mvsSanityCheck(data.root);
-    const molstarTree = convertMvsToMolstar(data.root);
-    // console.log(`Converted MolStar tree:\n${treeToString(molstarTree)}`);
-    validateTree(MolstarTreeSchema, molstarTree, 'Converted Molstar');
-    await loadMolstarTree(plugin, molstarTree, options);
+export async function loadMVS(plugin: PluginContext, data: MVSData, options: { replaceExisting?: boolean, sanityChecks?: boolean } = {}) {
+    try {
+        // console.log(`MVS tree (v${data.version}):\n${treeToString(data.root)}`);
+        validateTree(MVSTreeSchema, data.root, 'MVS');
+        if (options.sanityChecks) mvsSanityCheck(data.root);
+        const molstarTree = convertMvsToMolstar(data.root);
+        // console.log(`Converted MolStar tree:\n${treeToString(molstarTree)}`);
+        validateTree(MolstarTreeSchema, molstarTree, 'Converted Molstar');
+        await loadMolstarTree(plugin, molstarTree, options);
+    } catch (err) {
+        plugin.log.error(`${err}`);
+        throw err;
+    }
 }
 
 
 /** Load a `MolstarTree` into the Mol* plugin.
- * If `deletePrevious`, remove all objects in the current Mol* state; otherwise add to the current state. */
-async function loadMolstarTree(plugin: PluginContext, tree: MolstarTree, options?: { deletePrevious?: boolean }) {
+ * If `replaceExisting`, remove all objects in the current Mol* state; otherwise add to the current state. */
+async function loadMolstarTree(plugin: PluginContext, tree: MolstarTree, options?: { replaceExisting?: boolean }) {
     const mvsExtensionLoaded = plugin.state.hasBehavior(MolViewSpec);
     if (!mvsExtensionLoaded) throw new Error('MolViewSpec extension is not loaded.');
 
@@ -72,66 +77,67 @@ export interface MolstarLoadingContext {
 
 /** Loading actions for loading a `MolstarTree`, per node kind. */
 const MolstarLoadingActions: LoadingActions<MolstarTree, MolstarLoadingContext> = {
-    root(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'root'>, context: MolstarLoadingContext): StateObjectSelector {
+    root(updateParent: UpdateTarget, node: MolstarNode<'root'>, context: MolstarLoadingContext): UpdateTarget {
         context.nearestReprMap = makeNearestReprMap(node);
-        return msParent;
+        return updateParent;
     },
-    download(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'download'>): StateObjectSelector {
-        return update.to(msParent).apply(Download, {
+    download(updateParent: UpdateTarget, node: MolstarNode<'download'>): UpdateTarget {
+        return UpdateTarget.apply(updateParent, Download, {
             url: node.params.url,
             isBinary: node.params.is_binary,
-        }).selector;
+        });
     },
-    parse(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'parse'>): StateObjectSelector | undefined {
+    parse(updateParent: UpdateTarget, node: MolstarNode<'parse'>): UpdateTarget | undefined {
         const format = node.params.format;
         if (format === 'cif') {
-            return update.to(msParent).apply(ParseCif, {}).selector;
+            return UpdateTarget.apply(updateParent, ParseCif, {});
         } else if (format === 'pdb') {
-            return msParent;
+            return updateParent;
         } else {
             console.error(`Unknown format in "parse" node: "${format}"`);
             return undefined;
         }
     },
-    trajectory(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'trajectory'>): StateObjectSelector | undefined {
+    trajectory(updateParent: UpdateTarget, node: MolstarNode<'trajectory'>): UpdateTarget | undefined {
         const format = node.params.format;
         if (format === 'cif') {
-            return update.to(msParent).apply(TrajectoryFromMmCif, {
+            return UpdateTarget.apply(updateParent, TrajectoryFromMmCif, {
                 blockHeader: node.params.block_header ?? '', // Must set to '' because just undefined would get overwritten by createDefaults
                 blockIndex: node.params.block_index ?? undefined,
-            }).selector;
+            });
         } else if (format === 'pdb') {
-            return update.to(msParent).apply(TrajectoryFromPDB, {}).selector;
+            return UpdateTarget.apply(updateParent, TrajectoryFromPDB, {});
         } else {
             console.error(`Unknown format in "trajectory" node: "${format}"`);
             return undefined;
         }
     },
-    model(update: StateBuilder.Root, msParent: StateObjectSelector, node: SubTreeOfKind<MolstarTree, 'model'>, context: MolstarLoadingContext): StateObjectSelector {
+    model(updateParent: UpdateTarget, node: SubTreeOfKind<MolstarTree, 'model'>, context: MolstarLoadingContext): UpdateTarget {
         const annotations = collectAnnotationReferences(node, context);
-        return update.to(msParent)
-            .apply(ModelFromTrajectory, {
-                modelIndex: node.params.model_index,
-            })
-            .apply(CustomModelProperties, {
-                properties: {
-                    [MVSAnnotationsProvider.descriptor.name]: { annotations }
-                },
-                autoAttach: [
-                    MVSAnnotationsProvider.descriptor.name
-                ],
-            }).selector;
+        const model = UpdateTarget.apply(updateParent, ModelFromTrajectory, {
+            modelIndex: node.params.model_index,
+        });
+        UpdateTarget.apply(model, CustomModelProperties, {
+            properties: {
+                [MVSAnnotationsProvider.descriptor.name]: { annotations }
+            },
+            autoAttach: [
+                MVSAnnotationsProvider.descriptor.name
+            ],
+        });
+        return model;
     },
-    structure(update: StateBuilder.Root, msParent: StateObjectSelector, node: SubTreeOfKind<MolstarTree, 'structure'>, context: MolstarLoadingContext): StateObjectSelector {
+    structure(updateParent: UpdateTarget, node: SubTreeOfKind<MolstarTree, 'structure'>, context: MolstarLoadingContext): UpdateTarget {
         const props = structureProps(node);
-        let result: StateObjectSelector = update.to(msParent).apply(StructureFromModel, props).selector;
+        const struct = UpdateTarget.apply(updateParent, StructureFromModel, props);
+        let transformed = struct;
         for (const t of transformProps(node)) {
-            result = update.to(result).apply(TransformStructureConformation, t).selector;
+            transformed = UpdateTarget.apply(transformed, TransformStructureConformation, t); // applying to the result of previous transform, to get the correct transform order
         }
         const annotationTooltips = collectAnnotationTooltips(node, context);
         const inlineTooltips = collectInlineTooltips(node, context);
         if (annotationTooltips.length + inlineTooltips.length > 0) {
-            update.to(result).apply(CustomStructureProperties, {
+            UpdateTarget.apply(struct, CustomStructureProperties, {
                 properties: {
                     [MVSAnnotationTooltipsProvider.descriptor.name]: { tooltips: annotationTooltips },
                     [CustomTooltipsProvider.descriptor.name]: { tooltips: inlineTooltips },
@@ -142,73 +148,73 @@ const MolstarLoadingActions: LoadingActions<MolstarTree, MolstarLoadingContext>
                 ],
             });
         }
-        return result;
+        return struct;
     },
     tooltip: undefined, // No action needed, already loaded in `structure`
     tooltip_from_uri: undefined, // No action needed, already loaded in `structure`
     tooltip_from_source: undefined, // No action needed, already loaded in `structure`
-    component(update: StateBuilder.Root, msParent: StateObjectSelector, node: SubTreeOfKind<MolstarTree, 'component'>): StateObjectSelector | undefined {
+    component(updateParent: UpdateTarget, node: SubTreeOfKind<MolstarTree, 'component'>): UpdateTarget | undefined {
         if (isPhantomComponent(node)) {
-            return msParent;
+            return updateParent;
         }
         const selector = node.params.selector;
-        return update.to(msParent).apply(StructureComponent, {
+        return UpdateTarget.apply(updateParent, StructureComponent, {
             type: componentPropsFromSelector(selector),
             label: canonicalJsonString(selector),
             nullIfEmpty: false,
-        }).selector;
+        });
     },
-    component_from_uri(update: StateBuilder.Root, msParent: StateObjectSelector, node: SubTreeOfKind<MolstarTree, 'component_from_uri'>, context: MolstarLoadingContext): StateObjectSelector | undefined {
+    component_from_uri(updateParent: UpdateTarget, node: SubTreeOfKind<MolstarTree, 'component_from_uri'>, context: MolstarLoadingContext): UpdateTarget | undefined {
         if (isPhantomComponent(node)) return undefined;
         const props = componentFromXProps(node, context);
-        return update.to(msParent).apply(MVSAnnotationStructureComponent, props).selector;
+        return UpdateTarget.apply(updateParent, MVSAnnotationStructureComponent, props);
     },
-    component_from_source(update: StateBuilder.Root, msParent: StateObjectSelector, node: SubTreeOfKind<MolstarTree, 'component_from_source'>, context: MolstarLoadingContext): StateObjectSelector | undefined {
+    component_from_source(updateParent: UpdateTarget, node: SubTreeOfKind<MolstarTree, 'component_from_source'>, context: MolstarLoadingContext): UpdateTarget | undefined {
         if (isPhantomComponent(node)) return undefined;
         const props = componentFromXProps(node, context);
-        return update.to(msParent).apply(MVSAnnotationStructureComponent, props).selector;
+        return UpdateTarget.apply(updateParent, MVSAnnotationStructureComponent, props);
     },
-    representation(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'representation'>, context: MolstarLoadingContext): StateObjectSelector {
-        return update.to(msParent).apply(StructureRepresentation3D, {
+    representation(updateParent: UpdateTarget, node: MolstarNode<'representation'>, context: MolstarLoadingContext): UpdateTarget {
+        return UpdateTarget.apply(updateParent, StructureRepresentation3D, {
             ...representationProps(node.params),
             colorTheme: colorThemeForNode(node, context),
-        }).selector;
+        });
     },
     color: undefined, // No action needed, already loaded in `structure`
     color_from_uri: undefined, // No action needed, already loaded in `structure`
     color_from_source: undefined, // No action needed, already loaded in `structure`
-    label(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'label'>, context: MolstarLoadingContext): StateObjectSelector {
+    label(updateParent: UpdateTarget, node: MolstarNode<'label'>, context: MolstarLoadingContext): UpdateTarget {
         const item: CustomLabelProps['items'][number] = {
             text: node.params.text,
             position: { name: 'selection', params: {} },
         };
         const nearestReprNode = context.nearestReprMap?.get(node);
-        return update.to(msParent).apply(StructureRepresentation3D, {
+        return UpdateTarget.apply(updateParent, StructureRepresentation3D, {
             type: {
                 name: CustomLabelRepresentationProvider.name,
                 params: { items: [item] } satisfies Partial<CustomLabelProps>
             },
             colorTheme: colorThemeForNode(nearestReprNode, context),
-        }).selector;
+        });
     },
-    label_from_uri(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'label_from_uri'>, context: MolstarLoadingContext): StateObjectSelector {
+    label_from_uri(updateParent: UpdateTarget, node: MolstarNode<'label_from_uri'>, context: MolstarLoadingContext): UpdateTarget {
         const props = labelFromXProps(node, context);
-        return update.to(msParent).apply(StructureRepresentation3D, props).selector;
+        return UpdateTarget.apply(updateParent, StructureRepresentation3D, props);
     },
-    label_from_source(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'label_from_source'>, context: MolstarLoadingContext): StateObjectSelector {
+    label_from_source(updateParent: UpdateTarget, node: MolstarNode<'label_from_source'>, context: MolstarLoadingContext): UpdateTarget {
         const props = labelFromXProps(node, context);
-        return update.to(msParent).apply(StructureRepresentation3D, props).selector;
+        return UpdateTarget.apply(updateParent, StructureRepresentation3D, props);
     },
-    focus(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'focus'>, context: MolstarLoadingContext): StateObjectSelector {
-        context.focus = { kind: 'focus', focusTarget: msParent, params: node.params };
-        return msParent;
+    focus(updateParent: UpdateTarget, node: MolstarNode<'focus'>, context: MolstarLoadingContext): UpdateTarget {
+        context.focus = { kind: 'focus', focusTarget: updateParent.selector, params: node.params };
+        return updateParent;
     },
-    camera(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'camera'>, context: MolstarLoadingContext): StateObjectSelector {
+    camera(updateParent: UpdateTarget, node: MolstarNode<'camera'>, context: MolstarLoadingContext): UpdateTarget {
         context.focus = { kind: 'camera', params: node.params };
-        return msParent;
+        return updateParent;
     },
-    canvas(update: StateBuilder.Root, msParent: StateObjectSelector, node: MolstarNode<'canvas'>, context: MolstarLoadingContext): StateObjectSelector {
+    canvas(updateParent: UpdateTarget, node: MolstarNode<'canvas'>, context: MolstarLoadingContext): UpdateTarget {
         context.canvas = node.params;
-        return msParent;
+        return updateParent;
     },
 };

+ 39 - 6
src/extensions/mvs/mvs-data.ts

@@ -5,6 +5,7 @@
  */
 
 import { treeValidationIssues } from './tree/generic/tree-schema';
+import { treeToString } from './tree/generic/tree-utils';
 import { Root, createMVSBuilder } from './tree/mvs/mvs-builder';
 import { MVSTree, MVSTreeSchema } from './tree/mvs/mvs-tree';
 
@@ -13,8 +14,21 @@ import { MVSTree, MVSTreeSchema } from './tree/mvs/mvs-tree';
 export interface MVSData {
     /** MolViewSpec tree */
     root: MVSTree,
-    /** Integer defining the major version of MolViewSpec format (e.g. 1 for version '1.0.8') */
-    version: number,
+    /** Associated metadata */
+    metadata: MVSMetadata,
+}
+
+interface MVSMetadata {
+    /** Version of the spec used to write this tree */
+    version: string,
+    /** Name of this view */
+    title?: string,
+    /** Detailed description of this view */
+    description?: string,
+    /** Format of the description */
+    description_format?: 'markdown' | 'plaintext',
+    /** Timestamp when this view was exported */
+    timestamp: string,
 }
 
 export const MVSData = {
@@ -24,8 +38,11 @@ export const MVSData = {
     /** Parse MVSJ (MolViewSpec-JSON) format to `MVSData`. Does not include any validation. */
     fromMVSJ(mvsjString: string): MVSData {
         const result: MVSData = JSON.parse(mvsjString);
-        if (result?.version > MVSData.SupportedVersion) {
-            console.warn(`Loaded MVS is of higher version (${result?.version}) than currently supported version (${MVSData.SupportedVersion}). Some features may not work as expected.`);
+        const major = majorVersion(result?.metadata?.version);
+        if (major === undefined) {
+            console.error('Loaded MVS does not contain valid version info.');
+        } else if (major > (majorVersion(MVSData.SupportedVersion) ?? 0)) {
+            console.warn(`Loaded MVS is of higher version (${result.metadata.version}) than currently supported version (${MVSData.SupportedVersion}). Some features may not work as expected.`);
         }
         return result;
     },
@@ -44,11 +61,18 @@ export const MVSData = {
     /** Validate `MVSData`. Return `undefined` if OK; list of issues if not OK.
      * If `options.noExtra` is true, presence of any extra node parameters is treated as an issue. */
     validationIssues(mvsData: MVSData, options: { noExtra?: boolean } = {}): string[] | undefined {
-        if (typeof mvsData.version !== 'number') return [`"version" in MVS must be a number, not ${mvsData.version}`];
+        const version = mvsData?.metadata?.version;
+        if (typeof version !== 'string') return [`"version" in MVS must be a string, not ${typeof version}: ${version}`];
         if (mvsData.root === undefined) return [`"root" missing in MVS`];
         return treeValidationIssues(MVSTreeSchema, mvsData.root, options);
     },
 
+    /** Return a human-friendly textual representation of `mvsData`. */
+    toPrettyString(mvsData: MVSData): string {
+        const title = mvsData.metadata.title !== undefined ? ` "${mvsData.metadata.title}"` : '';
+        return `MolViewSpec tree${title} (version ${mvsData.metadata.version}, created ${mvsData.metadata.timestamp}):\n${treeToString(mvsData.root)}`;
+    },
+
     /** Create a new MolViewSpec builder containing only a root node. Example of MVS builder usage:
      *
      * ```
@@ -56,10 +80,19 @@ export const MVSData = {
      * builder.canvas({ background_color: 'white' });
      * const struct = builder.download({ url: 'https://www.ebi.ac.uk/pdbe/entry-files/download/1og2_updated.cif' }).parse({ format: 'mmcif' }).modelStructure();
      * struct.component().representation().color({ color: HexColor('#3050F8') });
-     * console.log(JSON.stringify(builder.getState()));
+     * console.log(MVSData.toPrettyString(builder.getState()));
      * ```
      */
     createBuilder(): Root {
         return createMVSBuilder();
     },
 };
+
+
+/** Get the major version from a semantic version string, e.g. '1.0.8' -> 1 */
+function majorVersion(semanticVersion: string | number): number | undefined {
+    if (typeof semanticVersion === 'string') return parseInt(semanticVersion.split('.')[0]);
+    if (typeof semanticVersion === 'number') return Math.floor(semanticVersion);
+    console.error(`Version should be a string, not ${typeof semanticVersion}: ${semanticVersion}`);
+    return undefined;
+}

+ 19 - 0
src/extensions/mvs/tree/mvs/_spec/mvs-builder.spec.ts

@@ -0,0 +1,19 @@
+/**
+ * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
+ *
+ * @author Adam Midlik <midlik@gmail.com>
+ */
+
+import { treeValidationIssues } from '../../generic/tree-schema';
+import { builderDemo } from '../mvs-builder';
+import { MVSTreeSchema } from '../mvs-tree';
+
+
+describe('mvs-builder', () => {
+    it('mvs-builder demo works', async () => {
+        const mvsData = builderDemo();
+        expect(typeof mvsData.metadata.version).toEqual('string');
+        expect(typeof mvsData.metadata.timestamp).toEqual('string');
+        expect(treeValidationIssues(MVSTreeSchema, mvsData.root)).toEqual(undefined);
+    });
+});

+ 15 - 3
src/extensions/mvs/tree/mvs/mvs-builder.ts

@@ -4,7 +4,7 @@
  * @author Adam Midlik <midlik@gmail.com>
  */
 
-import { pickObjectKeys } from '../../../../mol-util/object';
+import { deepClone, pickObjectKeys } from '../../../../mol-util/object';
 import { HexColor } from '../../helpers/utils';
 import { MVSData } from '../../mvs-data';
 import { ParamsOfKind, SubTreeOfKind } from '../generic/tree-schema';
@@ -55,8 +55,15 @@ export class Root extends _Base<'root'> {
         (this._root as Root) = this;
     }
     /** Return the current state of the builder as object in MVS format. */
-    getState(): MVSData {
-        return { version: MVSData.SupportedVersion, root: this._node };
+    getState(metadata?: Partial<Pick<MVSData['metadata'], 'title' | 'description' | 'description_format'>>): MVSData {
+        return {
+            root: deepClone(this._node),
+            metadata: {
+                ...metadata,
+                version: `${MVSData.SupportedVersion}`,
+                timestamp: utcNowISO(),
+            },
+        };
     }
     // omitting `saveState`, filesystem operations are responsibility of the caller code (platform-dependent)
 
@@ -248,3 +255,8 @@ export function builderDemo() {
 
     return builder.getState();
 }
+
+/** Return the current universal time, in ISO format, e.g. '2023-11-24T10:45:49.873Z' */
+function utcNowISO(): string {
+    return new Date().toISOString();
+}

+ 13 - 0
src/mol-math/linear-algebra/3d/mat3.ts

@@ -90,6 +90,19 @@ namespace Mat3 {
         return a;
     }
 
+    export function fromColumns(out: Mat3, left: Vec3, middle: Vec3, right: Vec3) {
+        out[0] = left[0];
+        out[1] = left[1];
+        out[2] = left[2];
+        out[3] = middle[0];
+        out[4] = middle[1];
+        out[5] = middle[2];
+        out[6] = right[0];
+        out[7] = right[1];
+        out[8] = right[2];
+        return out;
+    }
+
     /**
      * Copies the upper-left 3x3 values into the given mat3.
      */

+ 3 - 1
src/mol-util/array.ts

@@ -177,14 +177,16 @@ export function range(start: number, end?: number): number[] {
 /** Copy all elements from `src` to the end of `dst`.
  * Equivalent to `dst.push(...src)`, but avoids storing element on call stack. Faster that `extend` from Underscore.js.
  * `extend(a, a)` will double the array.
+ * Returns the modified `dst` array.
  */
-export function arrayExtend<T>(dst: T[], src: ArrayLike<T>): void {
+export function arrayExtend<T>(dst: T[], src: ArrayLike<T>): T[] {
     const offset = dst.length;
     const nCopy = src.length;
     dst.length += nCopy;
     for (let i = 0; i < nCopy; i++) {
         dst[offset + i] = src[i];
     }
+    return dst;
 }
 
 /** Check whether `array` is sorted, sort if not. */

+ 69 - 0
src/perf-tests/array.ts

@@ -0,0 +1,69 @@
+/**
+ * Copyright (c) 2018-2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
+ *
+ * @author Adam Midlik <midlik@gmail.com>
+ */
+
+import * as B from 'benchmark';
+import { arrayExtend, range, sortIfNeeded } from '../mol-util/array';
+
+
+function randomFloats(n: number) {
+    const SCALE = 1000;
+    const data = new Array(n);
+    for (let i = 0; i < n; i++) {
+        data[i] = SCALE * Math.random();
+    }
+    return data;
+}
+
+function le(x: number, y: number) { return x - y; }
+
+interface Copies<T> {
+    init: () => T,
+    copies: T[],
+    offset: number,
+}
+const Copies = {
+    create<T>(init: () => T, nCopies: number): Copies<T> {
+        return { init, offset: 0, copies: range(nCopies).map(init) };
+    },
+    get<T>(copies: Copies<T>): T {
+        return (copies.offset < copies.copies.length) ? copies.copies[copies.offset++] : copies.init();
+    },
+};
+
+export function runBenchmarks(arrayLength: number) {
+    const _data = randomFloats(arrayLength);
+    const _sortedData = arrayExtend([], _data).sort(le);
+    const _worstData = arrayExtend([], _sortedData);
+    [_worstData[arrayLength - 1], _worstData[arrayLength - 2]] = [_worstData[arrayLength - 2], _worstData[arrayLength - 1]];
+
+    const nCopies = 100;
+    let randomData: Copies<number[]>, sortedData: Copies<number[]>, worstData: Copies<number[]>;
+
+    function prepareData() {
+        randomData = Copies.create(() => arrayExtend([], _data), nCopies);
+        sortedData = Copies.create(() => arrayExtend([], _sortedData), nCopies);
+        worstData = Copies.create(() => arrayExtend([], _worstData), nCopies);
+    }
+    prepareData();
+
+    const suite = new B.Suite();
+    suite
+        .add(`native sort (${arrayLength}, pre-sorted)`, () => Copies.get(sortedData).sort(le))
+        .add(`sortIfNeeded (${arrayLength}, pre-sorted)`, () => sortIfNeeded(Copies.get(sortedData), le))
+        .add(`native sort (${arrayLength}, not sorted)`, () => Copies.get(randomData).sort(le))
+        .add(`sortIfNeeded (${arrayLength}, not sorted)`, () => sortIfNeeded(Copies.get(randomData), le))
+        .add(`native sort (${arrayLength}, worst case)`, () => Copies.get(worstData).sort(le))
+        .add(`sortIfNeeded (${arrayLength}, worst case)`, () => sortIfNeeded(Copies.get(worstData), le))
+        .on('cycle', (e: any) => {
+            console.log(String(e.target));
+            prepareData();
+        })
+        .run();
+    console.log('---------------------');
+    console.log('`sortIfNeeded` should be faster than native `sort` on pre-sorted data, same speed on non-sorted data and worst case data (almost sorted array when only the two last elements are swapped)');
+}
+
+runBenchmarks(10 ** 6);