Преглед на файлове

Change chain on plugin selection && hide instances

bioinsilico преди 4 години
родител
ревизия
ce361377e1

Файловите разлики са ограничени, защото са твърде много
+ 400 - 6663
package-lock.json


+ 11 - 9
package.json

@@ -1,6 +1,6 @@
 {
   "name": "@rcsb/rcsb-saguaro-3d",
-  "version": "0.0.16",
+  "version": "0.0.17",
   "description": "RCSB Molstar/Saguaro Web App",
   "main": "build/dist/RcsbFv3DBuilder.js",
   "files": [
@@ -9,7 +9,7 @@
     "build/dist/*ts"
   ],
   "scripts": {
-    "tsc": "tsc",
+    "tsc": "tsc --incremental",
     "tscExamples": "tsc --project ./tsconfig.examples.json",
     "build": "webpack --config ./webpack.config.js",
     "buildApp": "npm run cleanAll && npm run tsc && npm run cpStyles && npm run copyConfig && npm run build && npm run tscExamples && npm run copyHtml && npm run buildExamples && npm run clean",
@@ -51,7 +51,7 @@
     "@babel/core": "^7.10.4",
     "@babel/plugin-proposal-class-properties": "^7.10.4",
     "@babel/preset-env": "^7.10.4",
-    "@types/react-dom": "^16.9.8",
+    "@types/react-dom": "^17.0.0",
     "babel-loader": "^8.1.0",
     "concurrently": "^5.3.0",
     "css-loader": "^3.6.0",
@@ -60,8 +60,8 @@
     "mini-css-extract-plugin": "^0.9.0",
     "ncp": "^2.0.0",
     "node-sass": "^4.14.1",
-    "react": "^16.13.1",
-    "react-dom": "^16.13.1",
+    "react": "^17.0.1",
+    "react-dom": "^17.0.1",
     "react-icons": "^3.11.0",
     "rxjs": "^6.6.3",
     "sass-loader": "^7.3.1",
@@ -74,11 +74,13 @@
     "webpack-cli": "^3.3.12"
   },
   "dependencies": {
-    "@rcsb/rcsb-molstar": "file:../rcsb-molstar",
+    "@rcsb/rcsb-molstar": "^1.3.0-beta.saguaro.1",
     "@rcsb/rcsb-saguaro": "^1.0.13",
-    "@rcsb/rcsb-saguaro-app": "^1.0.22",
-    "@types/react": "^16.9.49",
-    "molstar": "^1.2.3"
+    "@rcsb/rcsb-saguaro-app": "^1.0.23",
+    "@types/react": "^17.0.0",
+    "@types/react-select": "^3.0.11",
+    "molstar": "^1.3.1",
+    "react-select": "^3.0.8"
   },
   "bugs": {
     "url": "https://github.com/rcsb/rcsb-saguaro-3d/issues"

+ 5 - 1
src/RcsbFv3D/RcsbFv3DAssembly.tsx

@@ -22,7 +22,11 @@ export class RcsbFv3DAssembly extends RcsbFv3DAbstract{
                 method: LoadMethod.loadPdbId,
                 params: {
                     pdbId:assemblyData.config.entryId,
-                    id:assemblyData.config.entryId
+                    id:assemblyData.config.entryId,
+                    props:{
+                        kind:'empty',
+                        assemblyId:'1'
+                    }
                 }
             }
         };

+ 9 - 0
src/RcsbFvSelection/RcsbFvSelection.ts

@@ -19,6 +19,7 @@ export interface ChainSelectionInterface {
 
 export class RcsbFvSelection {
 
+    private lastSelection: ChainSelectionInterface | null = null;
     private selection: Array<ChainSelectionInterface> = new Array<ChainSelectionInterface>();
     private hover: Array<ChainSelectionInterface> = new Array<ChainSelectionInterface>();
 
@@ -83,6 +84,14 @@ export class RcsbFvSelection {
             return this.hover;
     }
 
+    public getLastSelection(mode:'select'|'hover'): ChainSelectionInterface | null{
+       return this.lastSelection;
+    }
+
+    public setLastSelection(mode:'select'|'hover', selection: ChainSelectionInterface | null): void {
+        this.lastSelection = selection;
+    }
+
     public getSelectionWithCondition(modelId: string, labelAsymId: string, mode:'select'|'hover'): ChainSelectionInterface | undefined{
         const sel: Array<ChainSelectionInterface> = mode === 'select' ?
             this.selection.filter(d=>(d.modelId===modelId && d.labelAsymId === labelAsymId)) :

+ 1 - 1
src/RcsbFvSequence/RcsbFvSequence.tsx

@@ -1,5 +1,5 @@
 import * as React from "react";
-import {AssemblyView, AssemblyViewInterface} from "./SequenceViews/AssemblyView";
+import {AssemblyView, AssemblyViewInterface} from "./SequenceViews/AssemblyView/AssemblyView";
 import {CustomView, CustomViewInterface} from "./SequenceViews/CustomView";
 import {SaguaroPluginInterface} from "../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
 import {PluginContext} from "molstar/lib/mol-plugin/context";

+ 3 - 0
src/RcsbFvSequence/SequenceViews/AbstractView.tsx

@@ -45,6 +45,7 @@ export abstract class AbstractView<P,S> extends React.Component <P & AbstractVie
         this.props.plugin.setSelectCallback(this.structureSelectionCallback.bind(this));
         this.props.plugin.setModelChangeCallback(this.modelChangeCallback.bind(this));
         this.props.plugin.setHoverCallback(this.structureHoverCallback.bind(this));
+        this.props.plugin.setRepresentationChangeCallback(this.representationChangeCallback.bind(this));
         window.addEventListener('resize', this.resizeCallback);
     }
 
@@ -76,6 +77,8 @@ export abstract class AbstractView<P,S> extends React.Component <P & AbstractVie
 
     protected structureHoverCallback(): void{}
 
+    protected representationChangeCallback(): void{}
+
     protected modelChangeCallback(modelMap:SaguaroPluginModelMapType): void{}
 
     protected updateDimensions(): void{}

+ 124 - 29
src/RcsbFvSequence/SequenceViews/AssemblyView.tsx → src/RcsbFvSequence/SequenceViews/AssemblyView/AssemblyView.tsx

@@ -1,4 +1,4 @@
-import {RcsbFvDOMConstants} from "../../RcsbFvConstants/RcsbFvConstants";
+import {RcsbFvDOMConstants} from "../../../RcsbFvConstants/RcsbFvConstants";
 import * as React from "react";
 import {
     buildInstanceSequenceFv,
@@ -7,12 +7,24 @@ import {
     setBoardConfig,
     unmount
 } from "@rcsb/rcsb-saguaro-app";
-import {AbstractView, AbstractViewInterface} from "./AbstractView";
+import {AbstractView, AbstractViewInterface} from "../AbstractView";
 import {InstanceSequenceOnchangeInterface} from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/RcsbFvBuilder/RcsbFvInstanceBuilder";
 import {RcsbFvTrackDataElementInterface} from "@rcsb/rcsb-saguaro";
-import {ChainSelectionInterface} from "../../RcsbFvSelection/RcsbFvSelection";
-import {SaguaroPluginModelMapType} from "../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
+import {ChainSelectionInterface} from "../../../RcsbFvSelection/RcsbFvSelection";
+import {SaguaroPluginModelMapType} from "../../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
 import {SelectionInterface} from "@rcsb/rcsb-saguaro/build/RcsbBoard/RcsbSelection";
+import {OptionPropsInterface} from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/WebTools/SelectButton";
+
+import {OptionProps} from "react-select/src/components/Option";
+import {components} from 'react-select';
+import {ChainDisplay} from "./ChainDisplay";
+
+import {
+    StructureSelectionQueries as Q,
+    StructureSelectionQuery
+} from 'molstar/lib/mol-plugin-state/helpers/structure-selection-query';
+import {StructureRepresentationRegistry} from "molstar/lib/mol-repr/structure/registry";
+import Expression from "molstar/lib/mol-script/language/expression";
 
 export interface AssemblyViewInterface {
     entryId: string;
@@ -28,6 +40,7 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
     private createComponentThreshold: number = 9;
     private innerSelectionFlag: boolean = false;
     private currentSelectedComponentId: string;
+    private currentModelMap:SaguaroPluginModelMapType;
     //private readonly componentSet = new Map<string, {current: Set<string>, previous: Set<string>}>();
 
     constructor(props: AssemblyViewInterface & AbstractViewInterface) {
@@ -54,6 +67,7 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         setBoardConfig({
             trackWidth: trackWidth,
             elementClickCallBack:(e:RcsbFvTrackDataElementInterface)=>{
+                this.props.plugin.clearFocus();
                 if(this.currentSelectedComponentId != null)
                     this.props.plugin.removeComponent(this.currentSelectedComponentId);
                 if(e == null)
@@ -61,19 +75,34 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
                 const x = e.begin;
                 const y = e.end ?? e.begin;
                 if(e.isEmpty){
-                    this.props.plugin.focusPositions(this.currentModelId, this.currentLabelAsymId, [x,y]);
+                    this.props.plugin.cameraFocus(this.currentModelId, this.currentLabelAsymId, [x,y]);
                     this.currentSelectedComponentId = this.currentLabelAsymId +":"+ ((x === y) ? x.toString() : x.toString()+","+y.toString());
-                    this.props.plugin.createComponent(
-                        this.currentSelectedComponentId,
-                        this.currentModelId,
-                        [{asymId: this.currentLabelAsymId, position: x}, {asymId: this.currentLabelAsymId, position: y}],
-                        'ball-and-stick'
-                    );
+                    setTimeout(()=>{
+                        this.props.plugin.createComponent(
+                            this.currentSelectedComponentId,
+                            this.currentModelId,
+                            [{asymId: this.currentLabelAsymId, position: x}, {asymId: this.currentLabelAsymId, position: y}],
+                            'ball-and-stick'
+                        ).then(()=>{
+                            if(x === y)
+                                setTimeout(()=>{
+                                    this.props.plugin.setFocus(this.currentModelId, this.currentLabelAsymId, x, y);
+                                },200);
+                        });
+                    },100);
+
                 }else{
-                    this.props.plugin.focusRange(this.currentModelId, this.currentLabelAsymId, x, y);
+                    this.props.plugin.cameraFocus(this.currentModelId, this.currentLabelAsymId, x, y);
                     if((y-x)<this.createComponentThreshold){
                         this.currentSelectedComponentId = this.currentLabelAsymId +":"+ (x === y ? x.toString() : x.toString()+"-"+y.toString());
-                        this.props.plugin.createComponent(this.currentSelectedComponentId, this.currentModelId, this.currentLabelAsymId, x, y, 'ball-and-stick');
+                        setTimeout(()=>{
+                            this.props.plugin.createComponent(this.currentSelectedComponentId, this.currentModelId, this.currentLabelAsymId, x, y, 'ball-and-stick').then(()=>{
+                                if(x === y)
+                                    setTimeout(()=>{
+                                        this.props.plugin.setFocus(this.currentModelId, this.currentLabelAsymId, x, y);
+                                    },200);
+                            });
+                        },100);
                     }
                 }
             },
@@ -83,7 +112,7 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
                 this.props.plugin.clearSelection('select', {modelId: this.currentModelId, labelAsymId: this.currentLabelAsymId});
                 this.props.selection.clearSelection('select', this.currentLabelAsymId);
                 if(selection == null || selection.length === 0) {
-                    this.props.plugin.resetCamera();
+                    this.resetPluginView();
                     return;
                 }
                 this.select(selection);
@@ -96,13 +125,13 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
                     if(selection[0].isEmpty){
                         const selectionList = [{modelId: this.currentModelId, asymId: this.currentLabelAsymId, position: selection[0].begin}];
                         if(selection[0].end != null) selectionList.push({modelId: this.currentModelId, asymId: this.currentLabelAsymId, position: selection[0].end})
-                        this.props.plugin.selectSet(
+                        this.props.plugin.select(
                             selectionList,
                             'hover',
                             'add'
                         );
                     }else {
-                        this.props.plugin.selectRange(this.currentModelId, this.currentLabelAsymId, selection[0].begin, selection[0].end ?? selection[0].begin, 'hover', 'set');
+                        this.props.plugin.select(this.currentModelId, this.currentLabelAsymId, selection[0].begin, selection[0].end ?? selection[0].begin, 'hover', 'set');
                     }
                 }
             },
@@ -122,6 +151,10 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         this.pluginSelectCallback('hover');
     }
 
+    protected representationChangeCallback(): void{
+        //TODO
+    }
+
     private pluginSelectCallback(mode:'select'|'hover'): void{
         if(getRcsbFv(this.pfvDivId) == null)
             return;
@@ -133,13 +166,18 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         if(allSel == null || allSel.length ===0) {
             getRcsbFv(this.pfvDivId).clearSelection(mode);
             if(mode === 'select')
-                this.props.plugin.resetCamera();
+                this.resetPluginView();
+        }else if(mode === 'select' && this.props.selection.getLastSelection('select')?.labelAsymId != null && this.props.selection.getLastSelection('select')?.labelAsymId != this.currentLabelAsymId){
+            const authId: string | undefined = this.currentModelMap
+                .get(this.currentModelId)?.chains
+                .filter(ch=>(ch.label===this.props.selection.getLastSelection('select')?.labelAsymId))[0]?.auth;
+            this.modelChangeCallback(this.currentModelMap, this.currentEntryId+"."+authId);
         }else{
             const sel: ChainSelectionInterface | undefined = this.props.selection.getSelectionWithCondition(this.currentModelId, this.currentLabelAsymId, mode);
             if (sel == null) {
                 getRcsbFv(this.pfvDivId).clearSelection(mode);
                 if(mode === 'select')
-                    this.props.plugin.resetCamera();
+                    this.resetPluginView();
             } else {
                 getRcsbFv(this.pfvDivId).setSelection({elements: sel.regions, mode: mode});
             }
@@ -147,7 +185,9 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         this.innerSelectionFlag = false;
     }
 
-    protected modelChangeCallback(modelMap:SaguaroPluginModelMapType): void {
+    protected async modelChangeCallback(modelMap:SaguaroPluginModelMapType, defaultAsymId?: string): Promise<void> {
+        this.currentModelMap = modelMap;
+        this.props.plugin.clearFocus();
         const onChangeCallback: Map<string, (x: InstanceSequenceOnchangeInterface)=>void> = new Map<string, (x: InstanceSequenceOnchangeInterface) => {}>();
         const filterInstances: Map<string, Set<string>> = new Map<string, Set<string>>();
         modelMap.forEach((v,k)=>{
@@ -155,8 +195,8 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
                 this.currentEntryId = v.entryId;
                 this.currentLabelAsymId = x.asymId;
                 this.currentModelId = k;
-                this.props.plugin.resetCamera();
                 setTimeout(()=>{
+                    this.props.selection.setLastSelection('select', null);
                     this.structureSelectionCallback();
                 },1000);
             });
@@ -168,15 +208,20 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
             buildInstanceSequenceFv(
                 this.pfvDivId,
                 RcsbFvDOMConstants.SELECT_INSTANCE_PFV_ID,
-                entryId,
-                undefined,
-                onChangeCallback.get(entryId),
-                filterInstances.get(entryId)
+                entryId, {
+                    defaultValue: defaultAsymId,
+                    onChangeCallback: onChangeCallback.get(entryId),
+                    filterInstances: filterInstances.get(entryId),
+                    selectButtonOptionProps:(props:OptionProps<OptionPropsInterface>)=>(components.Option && <div style={{display:'flex'}}>
+                        <ChainDisplay plugin={this.props.plugin} label={props.data.label}/><components.Option {...props}/>
+                    </div>)
+                }
             ).then(()=>{
                 const length: number = getRcsbFv(this.pfvDivId).getBoardConfig().length ?? 0;
                 this.createComponentThreshold = (((Math.floor(length/100))+1)*this.createComponentThresholdBatch)-1;
             });
-        this.createChainComponents(modelMap);
+        if(!defaultAsymId)
+            await this.createComponents(modelMap);
     }
 
     protected updateDimensions(): void{
@@ -191,19 +236,24 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
             const x = e.begin;
             const y = e.end ?? e.begin;
             if(e.isEmpty){
-                this.props.plugin.selectSet(
+                this.props.plugin.select(
                     [{modelId: this.currentModelId, asymId: this.currentLabelAsymId, position: x},{modelId: this.currentModelId, asymId: this.currentLabelAsymId, position: y}], 'select',
                     'add'
                 );
                 this.props.selection.addSelectionFromRegion(this.currentModelId, this.currentLabelAsymId, {begin:x, end:y, isEmpty: true, source: 'sequence'}, 'select');
             }else{
-                this.props.plugin.selectRange(this.currentModelId, this.currentLabelAsymId,x,y, 'select', 'add');
+                this.props.plugin.select(this.currentModelId, this.currentLabelAsymId,x,y, 'select', 'add');
                 this.props.selection.addSelectionFromRegion(this.currentModelId, this.currentLabelAsymId, {begin:x, end:y, source: 'sequence'}, 'select');
             }
         });
     }
 
-    private async createChainComponents(modelMap:SaguaroPluginModelMapType): Promise<void> {
+    private resetPluginView(): void {
+        this.props.plugin.clearFocus();
+        this.props.plugin.resetCamera();
+    }
+
+    private async createComponents(modelMap:SaguaroPluginModelMapType): Promise<void> {
         const chains: Array<{modelId: string; auth: string; label: string;}> = new Array<{modelId: string; auth: string; label: string;}>();
         modelMap.forEach((entry, modelId)=>{
             entry.chains.forEach(ch=>{
@@ -213,9 +263,54 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
             });
         });
         this.props.plugin.removeComponent();
+        this.props.plugin.clearFocus();
         for(const ch of chains) {
-            await this.props.plugin.createComponent("Chain " + ch.auth, ch.modelId, ch.label, 'cartoon');
+            const label: string = "chain " + ch.auth;
+            await this.props.plugin.createComponent(label, ch.modelId, ch.label, 'cartoon');
+            await this.props.plugin.colorComponent(label, 'entity-source');
         }
+
+        this.props.plugin.pluginCall((plugin)=>{
+            const createComponent = (label: string, tag: string, expression: Expression, representationType: StructureRepresentationRegistry.BuiltIn) => {
+                return plugin.managers.structure.component.add({
+                    selection: StructureSelectionQuery(tag, expression),
+                    options: { checkExisting: false, label: label },
+                    representation: representationType,
+
+                });
+            }
+            const recursive = (componentList: {label: string; tag: string; expression: Expression; representationType: StructureRepresentationRegistry.BuiltIn;}[])=>{
+                if(componentList.length>0){
+                    const component = componentList.shift()!;
+                    createComponent(component.label, component.tag, component.expression, component.representationType).then(()=>{
+                        recursive(componentList);
+                    });
+                }
+            };
+            recursive([{
+                label: 'Ligands',
+                tag: 'ligand',
+                expression: Q.ligand.expression,
+                representationType: 'ball-and-stick'
+            },{
+                label: 'Carbohydrates',
+                tag: 'carbohydrate',
+                expression: Q.branched.expression,
+                representationType: 'carbohydrate'
+            },{
+                label: 'Ions',
+                tag: 'ion',
+                expression: Q.ion.expression,
+                representationType: 'ball-and-stick'
+            },{
+                label: 'Lipids',
+                tag: 'lipid',
+                expression: Q.lipid.expression,
+                representationType: 'ball-and-stick'
+            }]);
+
+        });
+
     }
 
     /*private removeComponents(labelAsymId?:string){

+ 35 - 0
src/RcsbFvSequence/SequenceViews/AssemblyView/ChainDisplay.tsx

@@ -0,0 +1,35 @@
+import * as React from "react";
+import {SaguaroPluginInterface} from "../../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
+
+interface ChainDisplayInterface {
+    plugin: SaguaroPluginInterface;
+    label: string;
+}
+
+interface ChainDisplayState {
+    display: 'visible' | 'hidden';
+}
+
+export class ChainDisplay extends React.Component<ChainDisplayInterface, ChainDisplayState>{
+
+    readonly state: ChainDisplayState = {
+        display: this.props.plugin.displayComponent(this.props.label) ? 'visible' : 'hidden'
+    };
+
+    private changeDisplay(): void{
+        if(this.state.display === 'visible') {
+            this.props.plugin.displayComponent(this.props.label, false);
+            this.setState({display: 'hidden'});
+        }else{
+            this.props.plugin.displayComponent(this.props.label, true);
+            this.setState({display: 'visible'});
+        }
+    }
+
+    render(): JSX.Element{
+        return(
+                <input style={{marginLeft:5, marginRight:5}} type={'checkbox'} checked={this.state.display === 'visible'} onChange={this.changeDisplay.bind(this)}/>
+        );
+    }
+
+}

+ 8 - 0
src/RcsbFvSequence/SequenceViews/CustomView.tsx

@@ -157,6 +157,14 @@ export class CustomView extends AbstractView<CustomViewInterface & AbstractViewI
         });
     }
 
+    protected structureHoverCallback(): void{
+        //TODO;
+    }
+
+    protected representationChangeCallback(): void{
+        //TODO
+    }
+
     protected additionalContent(): JSX.Element {
         if(this.state.additionalContent == null)
             return <></>;

+ 163 - 20
src/RcsbFvStructure/StructurePlugins/MolstarPlugin.ts

@@ -12,14 +12,17 @@ import {Mat4} from "molstar/lib/mol-math/linear-algebra";
 import {BuiltInTrajectoryFormat} from "molstar/lib/mol-plugin-state/formats/trajectory";
 import {PluginState} from "molstar/lib/mol-plugin/state";
 import {
+    ResidueIndex,
     Structure,
     StructureElement,
     StructureProperties as SP,
-    StructureSelection
+    StructureSelection,
+    Queries as Q,
+    StructureQuery
 } from "molstar/lib/mol-model/structure";
 import {OrderedSet} from "molstar/lib/mol-data/int";
 import { PluginStateObject as PSO } from 'molstar/lib/mol-plugin-state/objects';
-import {State, StateSelection} from "molstar/lib/mol-state";
+import {State} from "molstar/lib/mol-state";
 import {StructureRef} from "molstar/lib/mol-plugin-state/manager/structure/hierarchy-state";
 import {RcsbFvSelection, ResidueSelectionInterface} from "../../RcsbFvSelection/RcsbFvSelection";
 import {AbstractPlugin} from "./AbstractPlugin";
@@ -29,6 +32,7 @@ import {Script} from "molstar/lib/mol-script/script";
 import {MolScriptBuilder} from "molstar/lib/mol-script/language/builder";
 import {SetUtils} from "molstar/lib/mol-util/set";
 import {StructureRepresentationRegistry} from "molstar/lib/mol-repr/structure/registry";
+import {ColorTheme} from "molstar/lib/mol-theme/color";
 
 export enum LoadMethod {
     loadPdbId = "loadPdbId",
@@ -55,7 +59,7 @@ interface LoadParams {
     id?:string;
 }
 
-export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterface, SaguaroPluginPublicInterface {
+export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterface {
     private plugin: Viewer;
     private innerSelectionFlag: boolean = false;
     private loadingFlag: boolean = false;
@@ -64,6 +68,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     private modelChangeCallbackSubs: Subscription;
     private modelMap: Map<string,string|undefined> = new Map<string, string>();
     private readonly componentSet: Set<string> = new Set<string>();
+    private readonly componentVisibility: Map<string,boolean> = new Map<string, boolean>();
 
     constructor(props: RcsbFvSelection) {
         super(props);
@@ -100,6 +105,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
                 await this.plugin.loadStructureFromData(config.data!, config.format!, config.isBinary!);
             }
         }
+        this.plugin.getPlugin().selectionMode = true;
         this.loadingFlag = false;
         this.mapModels(loadConfig.params);
         this.modelChangeCallback(this.getChains());
@@ -134,22 +140,59 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     public setBackground(color: number) {
     }
 
-    public selectRange(modelId:string, asymId: string, begin: number, end: number, mode: 'select'|'hover', operation:'add'|'set'): void {
+    public select(modelId:string, asymId: string, begin: number, end: number, mode: 'select'|'hover', operation:'add'|'set'): void;
+    public select(selection: Array<{modelId:string; asymId: string; position: number;}>, mode: 'select'|'hover', operation:'add'|'set'): void;
+    public select(...args: any[]): void{
+        if(args.length === 6){
+            this.selectRange(args[0],args[1],args[2],args[3],args[4],args[5]);
+        }else if(args.length === 3){
+            this.selectSet(args[0],args[1],args[2]);
+        }
+    }
+    private selectRange(modelId:string, asymId: string, begin: number, end: number, mode: 'select'|'hover', operation:'add'|'set'): void {
         if(mode == null || mode === 'select') {
             this.innerSelectionFlag = true;
         }
+
         this.plugin.select(this.getModelId(modelId), asymId, begin, end, mode, operation);
         this.innerSelectionFlag = false;
     }
-    public selectSet(selection: Array<{modelId:string; asymId: string; position: number;}>, mode: 'select'|'hover', operation:'add'|'set'): void {
+    private selectSet(selection: Array<{modelId:string; asymId: string; position: number;}>, mode: 'select'|'hover', operation:'add'|'set'): void {
         if(mode == null || mode === 'select') {
             this.innerSelectionFlag = true;
         }
         this.plugin.select(selection.map(r=>{return{modelId: this.getModelId(r.modelId), position:r.position, asymId: r.asymId}}), mode, operation);
         this.innerSelectionFlag = false;
     }
+    public clearSelection(mode:'select'|'hover', option?:{modelId:string; labelAsymId:string;}): void {
+        if(mode === 'select') {
+            this.plugin.clearFocus();
+            this.innerSelectionFlag = true;
+        }
+        if(option != null)
+            this.plugin.clearSelection(mode, {modelId: this.getModelId(option.modelId), labelAsymId: option.labelAsymId});
+        else
+            this.plugin.clearSelection(mode);
+        this.innerSelectionFlag = false;
+    }
+
+    public setFocus(modelId: string, asymId: string, begin: number, end: number): void{
+        this.plugin.setFocus(this.getModelId(modelId), asymId, begin, end);
+    }
+    public clearFocus(): void {
+        this.plugin.clearFocus();
+    }
 
-    public focusPositions(modelId: string, asymId: string, positions:Array<number>): void{
+    public cameraFocus(modelId: string, asymId: string, positions:Array<number>): void;
+    public cameraFocus(modelId: string, asymId: string, begin: number, end: number): void;
+    public cameraFocus(...args: any[]): void{
+        if(args.length === 3){
+            this.focusPositions(args[0],args[1],args[2]);
+        }else if(args.length === 4){
+            this.focusRange(args[0],args[1],args[2],args[3]);
+        }
+    }
+    private focusPositions(modelId: string, asymId: string, positions:Array<number>): void{
         const data: Structure | undefined = getStructureWithModelId(this.plugin.getPlugin().managers.structure.hierarchy.current.structures, this.getModelId(modelId));
         if (data == null) return;
         const sel: StructureSelection = Script.getStructureSelection(Q => Q.struct.generator.atomGroups({
@@ -162,8 +205,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         else
             this.plugin.getPlugin().managers.camera.reset();
     }
-
-    public focusRange(modelId: string, asymId: string, begin: number, end: number): void{
+    private focusRange(modelId: string, asymId: string, begin: number, end: number): void{
         const seqIds: Array<number> = new Array<number>();
         for(let n = begin; n <= end; n++){
             seqIds.push(n);
@@ -175,6 +217,8 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     public async createComponent(componentLabel: string, modelId:string, asymId: string, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
     public async createComponent(componentLabel: string, modelId:string, residues: Array<{asymId: string; position: number;}>, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
     public async createComponent(...args: any[]): Promise<void> {
+        this.removeComponent(args[0]);
+        this.componentVisibility.set(args[0], true);
         this.componentSet.add(args[0]);
         if(args.length === 4)
             await this.plugin.createComponent(args[0], this.getModelId(args[1]), args[2], args[3]);
@@ -193,6 +237,17 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         return false;
     }
 
+    public async colorComponent(componentLabel: string, color: ColorTheme.BuiltIn): Promise<void>{
+        for(const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups){
+            for(const comp of c){
+                if(comp.cell.obj?.label === componentLabel) {
+                    await this.plugin.getPlugin().managers.structure.component.updateRepresentationsTheme([comp], { color: color });
+                    return;
+                }
+            }
+        }
+    }
+
     public getComponentSet(): Set<string>{
         const out: Set<string> = new Set<string>();
         this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups.forEach(c=>{
@@ -213,11 +268,57 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
                 this.plugin.removeComponent(id);
             })
             this.componentSet.clear();
+            this.componentVisibility.clear();
         }else{
             this.plugin.removeComponent(componentLabel);
             this.componentSet.delete(componentLabel);
+            this.componentVisibility.delete(componentLabel);
+        }
+    }
+
+    public displayComponent(componentLabel: string): boolean;
+    public displayComponent(componentLabel: string, visibilityFlag: boolean): void;
+    public displayComponent(componentLabel: string, visibilityFlag?: boolean): void|boolean {
+        if(typeof visibilityFlag === 'boolean')
+            return this.changeComponentDisplay(componentLabel, visibilityFlag);
+        else
+            return this.getComponentDisplay(componentLabel);
+    }
+    private changeComponentDisplay(componentLabel: string, visibilityFlag: boolean): void{
+        if(this.isComponent(componentLabel) && !this.componentVisibility.has(componentLabel))
+            this.componentVisibility.set(componentLabel, true);
+        if(this.componentVisibility.get(componentLabel) != visibilityFlag) {
+            for (const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups) {
+                for (const comp of c) {
+                    if (comp.cell.obj?.label === componentLabel) {
+                        if(this.getComponentDisplay(componentLabel) != visibilityFlag)
+                            this.plugin.getPlugin().managers.structure.component.toggleVisibility([comp]);
+                        return void 0;
+                    }
+                }
+            }
         }
     }
+    private getComponentDisplay(componentLabel: string): boolean{
+        for (const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups) {
+            for (const comp of c) {
+                if (comp.cell.obj?.label === componentLabel) {
+                    return this.componentVisibility.get(componentLabel) ?? false;
+                }
+            }
+        }
+        return false;
+    }
+
+    public setRepresentationChangeCallback(g:()=>void){
+        this.plugin.getPlugin().state.events.cell.stateUpdated.subscribe(o=>{
+            if(o.cell.obj?.type.name === "Structure" && this.componentSet.has(o.cell.obj?.label)){
+                if(o.cell.state.isHidden != null) {
+                    this.componentVisibility.set(o.cell.obj?.label, !o.cell.state.isHidden);
+                }
+            }
+        });
+    }
 
     public setHoverCallback(g:()=>void){
         this.plugin.getPlugin().behaviors.interaction.hover.subscribe((r: InteractivityManager.HoverEvent)=>{
@@ -250,6 +351,59 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
             if(this.innerSelectionFlag) {
                 return;
             }
+            if(this.plugin.getPlugin().managers.structure.selection.additionsHistory.length > 0) {
+                const currentLoci: Loci = this.plugin.getPlugin().managers.structure.selection.additionsHistory[0].loci;
+                const loc: StructureElement.Location = StructureElement.Location.create(currentLoci.structure);
+                StructureElement.Location.set(
+                    loc,
+                    currentLoci.structure,
+                    currentLoci.elements[0].unit,
+                    currentLoci.elements[0].unit.elements[OrderedSet.getAt(currentLoci.elements[0].indices,0)]
+                );
+                const currentModelId: string = this.getModelId(currentLoci.structure.model.id);
+                if(currentLoci.elements.length > 0)
+                    if(SP.entity.type(loc) === 'non-polymer') {
+                        const resAuthId: number = SP.residue.auth_seq_id(loc);
+                        const chainLabelId: string = SP.chain.label_asym_id(loc);
+                        const query: StructureQuery = Q.modifiers.includeSurroundings(
+                            Q.generators.residues({
+                                residueTest:l=>SP.residue.auth_seq_id(l.element) === resAuthId,
+                                chainTest:l=>SP.chain.label_asym_id(l.element) === chainLabelId
+                            }),
+                            {
+                                radius: 5,
+                                wholeResidues: true
+                            });
+                        this.innerSelectionFlag = true;
+                        const sel: StructureSelection = StructureQuery.run(query, currentLoci.structure);
+                        const surroundingsLoci: Loci = StructureSelection.toLociWithSourceUnits(sel);
+                        this.plugin.getPlugin().managers.structure.selection.fromLoci('add', surroundingsLoci);
+                        const surroundingsLoc = StructureElement.Location.create(surroundingsLoci.structure);
+                        for (const e of surroundingsLoci.elements) {
+                            StructureElement.Location.set(surroundingsLoc, surroundingsLoci.structure, e.unit, e.unit.elements[0]);
+                            if(SP.entity.type(surroundingsLoc) === 'polymer'){
+                                const currentAsymId: string = SP.chain.label_asym_id(surroundingsLoc);
+                                this.selection.setLastSelection('select', {
+                                    modelId: currentModelId,
+                                    labelAsymId: currentAsymId,
+                                    regions: []
+                                });
+                            }
+                        }
+                        this.innerSelectionFlag = false;
+                    }else if( SP.entity.type(loc) === 'polymer' ) {
+                        const currentAsymId: string = SP.chain.label_asym_id(loc);
+                        this.selection.setLastSelection('select', {
+                            modelId: currentModelId,
+                            labelAsymId: currentAsymId,
+                            regions: []
+                        });
+                    }else{
+                        this.selection.setLastSelection('select', null);
+                    }
+            }else{
+                this.selection.setLastSelection('select', null);
+            }
             const sequenceData: Array<ResidueSelectionInterface> = new Array<ResidueSelectionInterface>();
             for(const structure of this.plugin.getPlugin().managers.structure.hierarchy.current.structures){
                 const data: Structure | undefined = structure.cell.obj?.data;
@@ -258,8 +412,8 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
                 if(StructureElement.Loci.is(loci)){
                     const loc = StructureElement.Location.create(loci.structure);
                     for (const e of loci.elements) {
+                        StructureElement.Location.set(loc, loci.structure, e.unit, e.unit.elements[0]);
                         const seqIds = new Set<number>();
-                        loc.unit = e.unit;
                         for (let i = 0, il = OrderedSet.size(e.indices); i < il; ++i) {
                             loc.element = e.unit.elements[OrderedSet.getAt(e.indices, i)];
                             seqIds.add(SP.residue.label_seq_id(loc));
@@ -278,17 +432,6 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         });
     }
 
-    public clearSelection(mode:'select'|'hover', option?:{modelId:string; labelAsymId:string;}): void {
-        if(mode === 'select') {
-            this.innerSelectionFlag = true;
-        }
-        if(option != null)
-            this.plugin.clearSelection(mode, {modelId: this.getModelId(option.modelId), labelAsymId: option.labelAsymId});
-        else
-            this.plugin.clearSelection(mode);
-        this.innerSelectionFlag = false;
-    }
-
     public pluginCall(f: (plugin: PluginContext) => void){
         this.plugin.pluginCall(f);
     }

+ 11 - 4
src/RcsbFvStructure/StructurePlugins/SaguaroPluginInterface.ts

@@ -1,6 +1,7 @@
 import {LoadMolstarInterface} from "./MolstarPlugin";
 import {PluginContext} from "molstar/lib/mol-plugin/context";
 import {StructureRepresentationRegistry} from "molstar/lib/mol-repr/structure/registry";
+import {ColorTheme} from "molstar/lib/mol-theme/color";
 
 export type SaguaroPluginModelMapType = Map<string,{entryId: string; chains:Array<{label:string, auth:string; entityId: string; title: string; type:"polymer"|"water"|"branched"|"non-polymer"|"macrolide";}>;}>;
 
@@ -12,20 +13,26 @@ export interface SaguaroPluginInterface extends SaguaroPluginPublicInterface{
     setSelectCallback: (g:(flag?:boolean)=>void)=>void;
     setModelChangeCallback: (f:(modelMap:SaguaroPluginModelMapType)=>void)=>void;
     setHoverCallback:(g:(flag?:boolean)=>void)=>void;
+    setRepresentationChangeCallback:(g:(flag?:boolean)=>void)=>void;
     unsetCallbacks:()=>void;
 }
 
 export interface SaguaroPluginPublicInterface {
-    selectRange: (modelId:string, asymId: string, x: number, y: number, mode: 'select'|'hover', operation:'set'|'add') => void;
-    selectSet: (selection: Array<{modelId:string; asymId: string; position: number;}>, mode: 'select'|'hover', operation:'add'|'set') => void;
+    select(modelId:string, asymId: string, x: number, y: number, mode: 'select'|'hover', operation:'set'|'add'): void;
+    select(selection: Array<{modelId:string; asymId: string; position: number;}>, mode: 'select'|'hover', operation:'add'|'set'): void;
     clearSelection: (mode:'select'|'hover', option?:{modelId:string; labelAsymId:string;}) => void;
     createComponent(componentId: string, modelId:string, asymId: string, begin: number, end : number, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
     createComponent(componentId: string, modelId:string, asymId: string, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
     createComponent(componentId: string, modelId:string, residues: Array<{asymId: string; position: number;}>, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
+    colorComponent(componentId: string, color: ColorTheme.BuiltIn): Promise<void>;
     removeComponent: (componentId?: string) => void;
     isComponent: (componentId: string) => boolean;
+    displayComponent(componentLabel: string, visibilityFlag: boolean): void;
+    displayComponent(componentLabel: string): boolean;
     getComponentSet: () => Set<string>;
-    focusPositions: (modelId: string, asymId: string, positions:Array<number>)=> void;
-    focusRange: (modelId: string, asymId: string, begin: number, end: number) => void;
+    setFocus(modelId: string, asymId: string, begin: number, end: number): void;
+    clearFocus(): void;
+    cameraFocus(modelId: string, asymId: string, positions:Array<number>): void;
+    cameraFocus(modelId: string, asymId: string, begin: number, end: number): void;
     resetCamera: ()=>void;
 }

+ 1 - 1
webpack.config.js

@@ -52,7 +52,7 @@ const appConfig = {
         umdNamedDefine: true,
         path: path.resolve(__dirname, 'build/dist')
     },
-    devtool: 'source-map'
+    devtool: false//'source-map'
 }
 
 module.exports = [appConfig];

Някои файлове не бяха показани, защото твърде много файлове са промени