Browse Source

dedicated ligand viewer

Sebastian Bittrich 1 year ago
parent
commit
9eceadb62a

File diff suppressed because it is too large
+ 388 - 191
package-lock.json


+ 9 - 9
package.json

@@ -37,35 +37,35 @@
     "author": "RCSB PDB and Mol* Contributors",
     "license": "MIT",
     "devDependencies": {
-        "@typescript-eslint/eslint-plugin": "^6.4.0",
-        "@typescript-eslint/parser": "^6.4.0",
+        "@typescript-eslint/eslint-plugin": "^6.5.0",
+        "@typescript-eslint/parser": "^6.5.0",
         "buffer": "^6.0.3",
-        "concurrently": "^8.2.0",
+        "concurrently": "^8.2.1",
         "cpx2": "^5.0.0",
         "crypto-browserify": "^3.12.0",
         "css-loader": "^6.8.1",
-        "eslint": "^8.47.0",
+        "eslint": "^8.48.0",
         "extra-watch-webpack-plugin": "^1.0.3",
         "file-loader": "^6.2.0",
         "fs-extra": "^11.1.1",
         "mini-css-extract-plugin": "^2.7.6",
         "path-browserify": "^1.0.1",
         "raw-loader": "^4.0.2",
-        "sass": "^1.65.1",
+        "sass": "^1.66.1",
         "sass-loader": "^13.3.2",
         "stream-browserify": "^3.0.0",
         "style-loader": "^3.3.3",
-        "typescript": "^5.1.6",
+        "typescript": "^5.2.2",
         "webpack": "^5.88.2",
         "webpack-cli": "^5.1.4"
     },
     "dependencies": {
-        "@types/react": "^18.2.20",
+        "@types/react": "^18.2.21",
         "@types/react-dom": "^18.2.7",
-        "molstar": "^3.38.3",
+        "molstar": "file://../molstar/molstar-3.38.3.tgz",
         "react": "^18.2.0",
         "react-dom": "^18.2.0",
         "rxjs": "^7.8.1",
-        "tslib": "^2.6.1"
+        "tslib": "^2.6.2"
     }
 }

+ 1 - 0
src/viewer/assets.ts

@@ -1,3 +1,4 @@
 import './index.html';
+import './ligand.html';
 import './favicon.ico';
 require('./skin/rcsb.scss');

+ 1 - 3
src/viewer/helpers/model.ts

@@ -23,9 +23,7 @@ export class ModelLoader {
             ? (await this.plugin.builders.data.readFile({ file: Asset.File(fileOrUrl), isBinary, label: props?.dataLabel })).data
             : await this.plugin.builders.data.download({ url: fileOrUrl, isBinary, label: props?.dataLabel });
 
-        const hierarchy = await this.handleTrajectory<P, S>(data, format, props, matrix, reprProvider, params) as any;
-
-        return hierarchy;
+        return await this.handleTrajectory<P, S>(data, format, props, matrix, reprProvider, params) as any;
     }
 
     async parse<P = any, S = {}>(parse: ParseParams, props?: PresetProps & { dataLabel?: string }, matrix?: Mat4, reprProvider?: TrajectoryHierarchyPresetProvider<P, S>, params?: P) {

+ 218 - 2
src/viewer/index.ts

@@ -10,7 +10,7 @@
 import { BehaviorSubject } from 'rxjs';
 import { Plugin } from 'molstar/lib/mol-plugin-ui/plugin';
 import { PluginCommands } from 'molstar/lib/mol-plugin/commands';
-import { ViewerState, CollapsedState, ModelUrlProvider } from './types';
+import { ViewerState, CollapsedState, ModelUrlProvider, LigandUrlProvider, LigandViewerState } from './types';
 import { PluginSpec } from 'molstar/lib/mol-plugin/spec';
 
 import { ColorNames } from 'molstar/lib/mol-util/color/names';
@@ -50,6 +50,9 @@ import { PartialCanvas3DProps } from 'molstar/lib/mol-canvas3d/canvas3d';
 import { RSCCScore } from './helpers/rscc/behavior';
 import { createRoot } from 'react-dom/client';
 import { AssemblySymmetry } from 'molstar/lib/extensions/rcsb/assembly-symmetry/prop';
+import { wwPDBChemicalComponentDictionary } from 'molstar/lib/extensions/wwpdb/ccd/behavior';
+import { ChemicalCompontentTrajectoryHierarchyPreset } from 'molstar/lib/extensions/wwpdb/ccd/representation';
+import { StateTransforms } from 'molstar/lib/mol-plugin-state/transforms';
 
 /** package version, filled in at bundle build time */
 declare const __RCSB_MOLSTAR_VERSION__: string;
@@ -118,10 +121,52 @@ const DefaultViewerProps = {
     backgroundColor: ColorNames.white,
     manualReset: false, // switch to 'true' for 'motif' preset
     pickingAlphaThreshold: 0.5, // lower to 0.2 to accommodate 'motif' preset
-    showWelcomeToast: true
+    showWelcomeToast: true,
 };
 export type ViewerProps = typeof DefaultViewerProps & { canvas3d: PartialCanvas3DProps }
 
+const LigandExtensions = {
+    'wwpdb-chemical-component-dictionary': PluginSpec.Behavior(wwPDBChemicalComponentDictionary),
+};
+
+const DefaultLigandViewerProps = {
+    showImportControls: false,
+    showSessionControls: false,
+    showStructureSourceControls: true,
+    showMeasurementsControls: true,
+    showQuickStylesControls: false,
+    showStructureComponentControls: true,
+
+    ligandUrlProviders: [
+        (id: string) => ({
+            url: id.length <= 5 ? `https://files.rcsb.org/ligands/view/${id.toUpperCase()}.cif` : `https://files.rcsb.org/birds/view/${id.toUpperCase()}.cif`,
+            format: 'mmcif',
+            isBinary: false
+        })
+    ] as LigandUrlProvider[],
+
+    extensions: ObjectKeys(LigandExtensions),
+    layoutIsExpanded: false,
+    layoutShowControls: true,
+    layoutControlsDisplay: 'reactive' as PluginLayoutControlsDisplay,
+    layoutShowSequence: false,
+    layoutShowLog: false,
+
+    viewportShowExpand: true,
+    viewportShowSelectionMode: true,
+    volumeStreamingServer: 'https://maps.rcsb.org/',
+
+    backgroundColor: ColorNames.white,
+    manualReset: false,
+    pickingAlphaThreshold: 0.5,
+    showWelcomeToast: true,
+
+    ignoreHydrogens: true,
+    showLabels: false,
+    shownCoordinateType: 'ideal' as const
+};
+export type LigandViewerProps = typeof DefaultLigandViewerProps & { canvas3d: PartialCanvas3DProps }
+
 export class Viewer {
     private readonly _plugin: PluginUIContext;
     private readonly modelUrlProviders: ModelUrlProvider[];
@@ -349,4 +394,175 @@ export class Viewer {
     }
 }
 
+export class LigandViewer {
+    private readonly _plugin: PluginUIContext;
+    private readonly ligandUrlProviders: LigandUrlProvider[];
 
+    constructor(elementOrId: string | HTMLElement, props: Partial<LigandViewerProps> = {}) {
+        const element = typeof elementOrId === 'string' ? document.getElementById(elementOrId)! : elementOrId;
+        if (!element) throw new Error(`Could not get element with id '${elementOrId}'`);
+
+        const o = { ...DefaultLigandViewerProps, ...props };
+
+        const defaultSpec = DefaultPluginUISpec();
+        const spec: PluginUISpec = {
+            ...defaultSpec,
+            actions: defaultSpec.actions,
+            behaviors: [
+                ...defaultSpec.behaviors,
+                ...o.extensions.map(e => LigandExtensions[e]),
+            ],
+            animations: [...defaultSpec.animations?.filter(a => a.name !== AnimateStateSnapshots.name) || []],
+            layout: {
+                initial: {
+                    isExpanded: o.layoutIsExpanded,
+                    showControls: o.layoutShowControls,
+                    controlsDisplay: o.layoutControlsDisplay,
+                },
+            },
+            canvas3d: {
+                ...defaultSpec.canvas3d,
+                ...o.canvas3d,
+                renderer: {
+                    ...defaultSpec.canvas3d?.renderer,
+                    ...o.canvas3d?.renderer,
+                    backgroundColor: o.backgroundColor,
+                    pickingAlphaThreshold: o.pickingAlphaThreshold
+                },
+                camera: {
+                    // desirable for alignment view so that the display doesn't "jump around" as more structures get loaded
+                    manualReset: o.manualReset
+                }
+            },
+            components: {
+                ...defaultSpec.components,
+                controls: {
+                    ...defaultSpec.components?.controls,
+                    top: o.layoutShowSequence ? undefined : 'none',
+                    bottom: o.layoutShowLog ? undefined : 'none',
+                    left: 'none',
+                    right: ControlsWrapper,
+                },
+                remoteState: 'none',
+            },
+            config: [
+                [PluginConfig.Viewport.ShowExpand, o.viewportShowExpand],
+                [PluginConfig.Viewport.ShowSelectionMode, o.viewportShowSelectionMode],
+                [PluginConfig.Viewport.ShowAnimation, false],
+                [PluginConfig.VolumeStreaming.DefaultServer, o.volumeStreamingServer],
+                [PluginConfig.Download.DefaultPdbProvider, 'rcsb'],
+                [PluginConfig.Download.DefaultEmdbProvider, 'rcsb'],
+                [PluginConfig.Structure.DefaultRepresentationPreset, PresetStructureRepresentations.auto.id],
+                // wboit & webgl1 checks are needed to work properly on recent Safari versions
+                [PluginConfig.General.EnableWboit, PluginFeatureDetection.preferWebGl1],
+                [PluginConfig.General.PreferWebGl1, PluginFeatureDetection.preferWebGl1]
+            ]
+        };
+
+        this._plugin = new PluginUIContext(spec);
+        this.ligandUrlProviders = o.ligandUrlProviders;
+
+        (this._plugin.customState as LigandViewerState) = {
+            showMeasurementsControls: o.showMeasurementsControls,
+            showStructureComponentControls: o.showStructureComponentControls,
+            modelLoader: new ModelLoader(this._plugin),
+            collapsed: new BehaviorSubject<CollapsedState>({
+                selection: true,
+                measurements: true,
+                strucmotifSubmit: true,
+                superposition: true,
+                quickStyles: true,
+                component: false,
+                volume: true,
+                assemblySymmetry: true,
+                validationReport: true,
+                custom: true,
+            }),
+            ignoreHydrogens: o.ignoreHydrogens,
+            showLabels: o.showLabels,
+            shownCoordinateType: o.shownCoordinateType
+        };
+
+        this._plugin.init()
+            .then(async () => {
+                const root = createRoot(element);
+                root.render(React.createElement(Plugin, { plugin: this._plugin }));
+
+                if (o.showWelcomeToast) {
+                    await PluginCommands.Toast.Show(this._plugin, {
+                        title: 'Welcome',
+                        message: `RCSB PDB Mol* Ligand Viewer ${RCSB_MOLSTAR_VERSION} [${BUILD_DATE.toLocaleString()}]`,
+                        key: 'toast-welcome',
+                        timeoutMs: 5000
+                    });
+                }
+
+                // allow picking of individual atoms
+                this._plugin.managers.interactivity.setProps({ granularity: 'element' });
+            });
+    }
+
+    private get customState() {
+        return this._plugin.customState as LigandViewerState;
+    }
+
+    clear() {
+        const state = this._plugin.state.data;
+        return PluginCommands.State.RemoveObject(this._plugin, { state, ref: state.tree.root.ref });
+    }
+
+    async loadLigandId(id: string) {
+        for (const provider of this.ligandUrlProviders) {
+            try {
+                const p = provider(id);
+                await this.customState.modelLoader.load<any, any>({ fileOrUrl: p.url, format: p.format, isBinary: p.isBinary }, undefined, undefined, ChemicalCompontentTrajectoryHierarchyPreset, { shownCoordinateType: this.customState.shownCoordinateType });
+                await this.syncHydrogenState();
+
+                for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
+                    for (const c of s.components) {
+                        const isHidden = c.cell.state.isHidden === true || !this.customState.showLabels;
+                        await this._plugin.builders.structure.representation.addRepresentation(c.cell, { type: 'label', typeParams: { level: 'element', ignoreHydrogens: this.customState.ignoreHydrogens } }, { initialState: { isHidden } });
+                    }
+                }
+            } catch (e) {
+                console.warn(`loading '${id}' failed with '${e}', trying next ligand-loader-provider`);
+            }
+        }
+    }
+
+    async toggleHydrogen() {
+        this.customState.ignoreHydrogens = !this.customState.ignoreHydrogens;
+        await this.syncHydrogenState();
+    }
+
+    async syncHydrogenState() {
+        const update = this._plugin.build();
+        for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
+            for (const c of s.components) {
+                for (const r of c.representations) {
+                    update.to(r.cell).update(StateTransforms.Representation.StructureRepresentation3D, old => {
+                        old.type.params.ignoreHydrogens = this.customState.ignoreHydrogens;
+                    });
+                }
+            }
+        }
+        await update.commit();
+    }
+
+    async toggleLabels() {
+        this.customState.showLabels = !this.customState.showLabels;
+        await this.syncLabelState();
+    }
+
+    async syncLabelState() {
+        for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
+            for (const c of s.components) {
+                if (c.cell.state.isHidden) continue;
+                for (const r of c.representations) {
+                    if (r.cell.obj?.label !== 'Label') continue;
+                    this._plugin.managers.structure.hierarchy.toggleVisibility([r], this.customState.showLabels ? 'show' : 'hide');
+                }
+            }
+        }
+    }
+}

+ 173 - 0
src/viewer/ligand.html

@@ -0,0 +1,173 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="utf-8" />
+    <meta name="viewport" content="width=device-width, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0">
+    <link rel="icon" href="./favicon.ico" type="image/x-icon">
+    <title>RCSB PDB Mol* Ligand Test Page</title>
+    <style>
+        * {
+            margin: 0;
+            padding: 0;
+            box-sizing: border-box;
+        }
+        #viewer {
+            position: absolute;
+            left: 5%;
+            top: 100px;
+            min-width: 90%;
+            height: 85%;
+        }
+
+        .msp-layout-expanded {
+            z-index: 10;
+        }
+
+        #menu {
+            position: absolute;
+            left: 5%;
+            top: 20px;
+        }
+
+        #menu > select {
+            width: 200px;
+        }
+    </style>
+    <link rel="stylesheet" type="text/css" href="rcsb-molstar.css" />
+    <script type="text/javascript" src="./rcsb-molstar.js"></script>
+</head>
+<body>
+<div id="viewer"></div>
+<script>
+    // create an instance of the plugin
+    const viewer = new rcsbMolstar.LigandViewer('viewer', {
+        layoutShowLog: true,
+        layoutShowControls: true,
+    })
+</script>
+<div id="menu">
+    <h2> RCSB PDB Mol* Viewer - Ligand Test Page</h2>
+    <label for="examples">Examples</label>
+    <select id="examples" onchange="loadExample(parseInt(this.value))">
+        <option value=''></option>
+    </select>
+
+    &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
+
+    Clear
+    <button style="padding: 3px;" onclick="viewer.clear()">all</button>
+
+    &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
+
+    <button style="padding: 3px;" onclick="toggleHydrogen()">Hydrogen</button>
+
+    &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
+
+    <button style="padding: 3px;" onclick="toggleLabels()">Labels</button>
+</div>
+<script>
+
+    function loadExample(index) {
+        const e = examples[index];
+        return viewer.loadLigandId(e.id);
+    }
+
+    const examples = [
+        {
+            id: 'UNK',
+            info: 'Unknown'
+        },
+        {
+            id: 'UNL',
+            info: 'Unknown ligand'
+        },
+        {
+            id: 'UNX',
+            info: 'Unknown atom or ion'
+        },
+        {
+            id: '2NR',
+            info: 'dual rep: N-{(2S)-2-[(N-acetyl-L-threonyl-L-isoleucyl)amino]hexyl}-L-norleucyl-L-glutaminyl-N~5~-[amino(iminio)methyl]-L-ornithinamide'
+        },
+        {
+            id: 'PRDCC_000001',
+            info: 'PRDCC: Actinomycin D'
+        },
+        {
+            id: 'FE',
+            info: '+3 oxidation state'
+        },
+        {
+            id: 'FE2',
+            info: '+2 oxidation state'
+        },
+        {
+            id: 'RUC',
+            info: 'Transition metal'
+        },
+        {
+            id: 'SF4',
+            info: 'Fe-S cluster'
+        },
+        {
+            id: 'TBR',
+            info: 'HEXATANTALUM DODECABROMIDE'
+        },
+        {
+            id: 'OER',
+            info: 'SR-MN4-O5 CLUSTER'
+        },
+        {
+            id: 'FEA',
+            info: 'Charges: MONOAZIDO-MU-OXO-DIIRON'
+        },
+        {
+            id: 'PR2',
+            info: 'Orientation differs: THIENO[3,2-B]PYRIDINE-2-SULFONIC ACID [2-OXO-1-(1H-PYRROLO[2,3-C]PYRIDIN-2-YLMETHYL)-PYRROLIDIN-3-YL]-AMIDE'
+        },
+        {
+            id: 'O3R',
+            info: 'Some atoms missing: 6-{[(3,4-dichlorophenyl)methyl](methyl)amino}pyridine-3-sulfonamide'
+        },
+        {
+            id: 'O2U',
+            info: 'Many atoms missing: [(3S)-1-hydroxy-2,5-dioxopyrrolidin-3-yl]acetic acid'
+        },
+        {
+            id: 'HC0',
+            info: 'No ideal coordinates: 2 IRON/2 SULFUR/6 CARBONYL/1 WATER INORGANIC CLUSTER',
+        },
+        {
+            id: 'Q6O',
+            info: 'No model coordinates: N-(4-chloro-2,5-dimethoxyphenyl)acetamide'
+        },
+        {
+            id: 'H0C',
+            info: 'Huge: [(2~{R},3~{R},4~{R},5~{S},6~{S})-2-[[(1~{R},3~{S},5~{S},8~{R},9~{S},10~{R},11~{R},13~{R},14~{S},17~{R})-10-(hydroxymethyl)-13-methyl-1,5,11,14-tetrakis(oxidanyl)-17-(5-oxidanylidene-2~{H}-furan-3-yl)-2,3,4,6,7,8,9,11,12,15,16,17-dodecahydro-1~{H}-cyclopenta[a]phenanthren-3-yl]oxy]-6-methyl-3,5-bis(oxidanyl)oxan-4-yl] anthracene-9-carboxylate'
+        },
+        {
+            id: 'HEM',
+            info: 'PROTOPORPHYRIN IX CONTAINING FE'
+        },
+    ];
+
+    const examplesSelect = document.getElementById('examples');
+    for (let i = 0, il = examples.length; i < il; ++i) {
+        const e = examples[i]
+        const option = document.createElement('option')
+        Object.assign(option, { text: '[' + e.id + '] ' + e.info, value: i })
+        examplesSelect.appendChild(option)
+    }
+
+    //
+
+    function toggleHydrogen() {
+        viewer.toggleHydrogen();
+    }
+
+    function toggleLabels() {
+        viewer.toggleLabels();
+    }
+</script>
+</body>
+</html>

+ 18 - 0
src/viewer/types.ts

@@ -15,6 +15,12 @@ export type ModelUrlProvider = (pdbId: string) => {
     isBinary: boolean
 }
 
+export type LigandUrlProvider = (id: string) => {
+    url: string,
+    format: BuiltInTrajectoryFormat,
+    isBinary: boolean
+}
+
 interface SharedParams {
     /** A supported file format extension string */
     format: BuiltInTrajectoryFormat,
@@ -65,6 +71,18 @@ export interface ViewerState {
     detachedFromSierra: boolean
 }
 
+export interface LigandViewerState {
+    showMeasurementsControls: boolean
+    showStructureComponentControls: boolean
+    ignoreHydrogens: boolean
+    showLabels: boolean
+    shownCoordinateType: 'ideal' | 'model' | 'both'
+
+    modelLoader: ModelLoader
+
+    collapsed: BehaviorSubject<CollapsedState>
+}
+
 export function ViewerState(plugin: PluginContext) {
     return plugin.customState as ViewerState;
 }

+ 1 - 1
src/viewer/ui/strucmotif/helpers.ts

@@ -90,7 +90,7 @@ export function extractResidues(ctx: StrucmotifCtx, loci: StructureSelectionHist
         ctx.residueIds.push(residueId);
 
         // retrieve CA/C4', used to compute residue distance
-        const coords = [x(location), y(location), z(location)] as Vec3;
+        const coords = [x(location), y(location), z(location)] as unknown as Vec3;
         ctx.coordinates.push({ coords, residueId });
 
         // handle potential exchanges - can be empty if deselected by users

Some files were not shown because too many files changed in this diff