bioinsilico 3 anni fa
parent
commit
1cadf38165

+ 21 - 0
examples/local.html

@@ -0,0 +1,21 @@
+<html lang="en" class=""><head>
+    <meta charset="UTF-8">
+    <title>CodePen Demo</title>
+</head>
+<body>
+
+
+<script src="../build/dist/rcsb-saguaro-3d.js" type="text/javascript"></script>
+
+
+
+<div id="pfv" style="position: absolute" ></div>
+
+<script>
+    const fv = new RcsbFv3D.assembly({config:{entryId:" 7B9F", title: "3D Protein Feature View:  7BAA", subtitle: "Cys-42-tethered stabilizer 12 of 14-3-3(sigma)/ERa PPI"}});
+    fv.render();
+</script>
+
+
+
+</body></html>

File diff suppressed because it is too large
+ 7376 - 274
package-lock.json


+ 5 - 5
package.json

@@ -51,7 +51,9 @@
     "@babel/core": "^7.10.4",
     "@babel/plugin-proposal-class-properties": "^7.10.4",
     "@babel/preset-env": "^7.10.4",
+    "@types/react": "^17.0.11",
     "@types/react-dom": "^17.0.0",
+    "@types/react-select": "^3.0.11",
     "babel-loader": "^8.1.0",
     "concurrently": "^5.3.0",
     "css-loader": "^3.6.0",
@@ -74,12 +76,10 @@
     "webpack-cli": "^3.3.12"
   },
   "dependencies": {
-    "@rcsb/rcsb-molstar": "^1.6.2",
+    "@rcsb/rcsb-molstar": "file:../rcsb-molstar",
     "@rcsb/rcsb-saguaro": "^1.8.0",
-    "@rcsb/rcsb-saguaro-app": "^1.4.8",
-    "@types/react": "^17.0.0",
-    "@types/react-select": "^3.0.11",
-    "molstar": "^2.0.5",
+    "@rcsb/rcsb-saguaro-app": "^2.0.0-beta.3",
+    "molstar": "^2.0.7",
     "react-select": "^3.0.8"
   },
   "bugs": {

+ 6 - 7
src/RcsbFv3D/RcsbFv3DAbstract.tsx

@@ -1,8 +1,8 @@
 import * as React from "react";
 import * as ReactDom from "react-dom";
 import {RcsbFv3DComponent} from './RcsbFv3DComponent';
-import {StructureViewInterface} from "../RcsbFvStructure/RcsbFvStructure";
-import {SequenceViewInterface} from "../RcsbFvSequence/RcsbFvSequence";
+import {RcsbFvStructureInterface} from "../RcsbFvStructure/RcsbFvStructure";
+import {RcsbFvSequenceInterface} from "../RcsbFvSequence/RcsbFvSequence";
 import {EventType, RcsbFvContextManager} from "../RcsbFvContextManager/RcsbFvContextManager";
 import {PluginContext} from "molstar/lib/mol-plugin/context";
 import {CSSProperties} from "react";
@@ -17,12 +17,11 @@ export interface RcsbFv3DAbstractInterface {
     }
 }
 
-
 export abstract class RcsbFv3DAbstract {
 
     protected elementId: string;
-    protected structureConfig: StructureViewInterface;
-    protected sequenceConfig: SequenceViewInterface;
+    protected structureConfig: RcsbFvStructureInterface;
+    protected sequenceConfig: RcsbFvSequenceInterface;
     protected ctxManager: RcsbFvContextManager = new RcsbFvContextManager();
     private fullScreenFlag: boolean = false;
     protected cssConfig:{
@@ -36,7 +35,7 @@ export abstract class RcsbFv3DAbstract {
             this.init(config);
     }
 
-    protected init(config: any){}
+    protected abstract init(config: any): void;
 
     public render(): void{
         if(this.elementId == null )
@@ -75,7 +74,7 @@ export abstract class RcsbFv3DAbstract {
         }
     }
 
-    public updateConfig(config: {structurePanelConfig?: StructureViewInterface; sequencePanelConfig?: SequenceViewInterface;}){
+    public updateConfig(config: {structurePanelConfig?: RcsbFvStructureInterface; sequencePanelConfig?: RcsbFvSequenceInterface;}){
         this.ctxManager.next({eventType: EventType.UPDATE_CONFIG, eventData:config});
     }
 

+ 8 - 8
src/RcsbFv3D/RcsbFv3DComponent.tsx

@@ -5,8 +5,8 @@ import {MolstarPlugin} from '../RcsbFvStructure/StructurePlugins/MolstarPlugin';
 import {SaguaroPluginInterface} from '../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface';
 
 import '../styles/RcsbFvMolstarStyle.module.scss';
-import {RcsbFvSequence, SequenceViewInterface} from "../RcsbFvSequence/RcsbFvSequence";
-import {RcsbFvStructure, StructureViewInterface} from "../RcsbFvStructure/RcsbFvStructure";
+import {RcsbFvSequence, RcsbFvSequenceInterface} from "../RcsbFvSequence/RcsbFvSequence";
+import {RcsbFvStructure, RcsbFvStructureInterface} from "../RcsbFvStructure/RcsbFvStructure";
 import {
     EventType,
     RcsbFvContextManager,
@@ -19,8 +19,8 @@ import {RcsbFvSelection} from "../RcsbFvSelection/RcsbFvSelection";
 import {CSSProperties} from "react";
 
 export interface RcsbFv3DComponentInterface {
-    structurePanelConfig:StructureViewInterface;
-    sequencePanelConfig: SequenceViewInterface;
+    structurePanelConfig:RcsbFvStructureInterface;
+    sequencePanelConfig: RcsbFvSequenceInterface;
     id: string;
     ctxManager: RcsbFvContextManager;
     cssConfig?:{
@@ -32,14 +32,14 @@ export interface RcsbFv3DComponentInterface {
     fullScreen: boolean;
 }
 
-export class RcsbFv3DComponent extends React.Component <RcsbFv3DComponentInterface, {structurePanelConfig:StructureViewInterface, sequencePanelConfig:SequenceViewInterface}> {
+export class RcsbFv3DComponent extends React.Component <RcsbFv3DComponentInterface, {structurePanelConfig:RcsbFvStructureInterface, sequencePanelConfig:RcsbFvSequenceInterface}> {
 
     private readonly pfvScreenFraction = 0.55;
     private readonly plugin: SaguaroPluginInterface;
     private readonly selection: RcsbFvSelection = new RcsbFvSelection();
     private subscription: Subscription;
 
-    readonly state: {structurePanelConfig:StructureViewInterface, sequencePanelConfig:SequenceViewInterface} = {
+    readonly state: {structurePanelConfig:RcsbFvStructureInterface, sequencePanelConfig:RcsbFvSequenceInterface} = {
         structurePanelConfig: this.props.structurePanelConfig,
         sequencePanelConfig: this.props.sequencePanelConfig
     }
@@ -116,8 +116,8 @@ export class RcsbFv3DComponent extends React.Component <RcsbFv3DComponentInterfa
     }
 
     private updateConfig(config:UpdateConfigInterface){
-        const structureConfig: StructureViewInterface | undefined = config.structurePanelConfig;
-        const sequenceConfig: SequenceViewInterface | undefined = config.sequencePanelConfig;
+        const structureConfig: RcsbFvStructureInterface | undefined = config.structurePanelConfig;
+        const sequenceConfig: RcsbFvSequenceInterface | undefined = config.sequencePanelConfig;
         if(structureConfig != null && sequenceConfig != null){
             this.setState({structurePanelConfig:structureConfig, sequencePanelConfig:sequenceConfig});
         }else if(structureConfig != null){

+ 2 - 2
src/RcsbFv3D/RcsbFv3DCustom.tsx

@@ -1,10 +1,10 @@
 
-import {StructureViewInterface} from "../RcsbFvStructure/RcsbFvStructure";
+import {RcsbFvStructureInterface} from "../RcsbFvStructure/RcsbFvStructure";
 import {CustomViewInterface} from "../RcsbFvSequence/SequenceViews/CustomView";
 import {RcsbFv3DAbstract, RcsbFv3DAbstractInterface} from "./RcsbFv3DAbstract";
 
 export interface RcsbFv3DCustomInterface extends RcsbFv3DAbstractInterface {
-    structurePanelConfig: StructureViewInterface;
+    structurePanelConfig: RcsbFvStructureInterface;
     sequencePanelConfig: {
         config: CustomViewInterface;
         title?: string;

+ 4 - 4
src/RcsbFvContextManager/RcsbFvContextManager.ts

@@ -1,6 +1,6 @@
 import {Subject, Subscription} from 'rxjs';
-import {StructureViewInterface} from "../RcsbFvStructure/RcsbFvStructure";
-import {SequenceViewInterface} from "../RcsbFvSequence/RcsbFvSequence";
+import {RcsbFvStructureInterface} from "../RcsbFvStructure/RcsbFvStructure";
+import {RcsbFvSequenceInterface} from "../RcsbFvSequence/RcsbFvSequence";
 import {PluginContext} from "molstar/lib/mol-plugin/context";
 
 /**Main Event Data Object Interface*/
@@ -16,8 +16,8 @@ export enum EventType {
 }
 
 export interface UpdateConfigInterface {
-    structurePanelConfig?:StructureViewInterface;
-    sequencePanelConfig?:SequenceViewInterface;
+    structurePanelConfig?:RcsbFvStructureInterface;
+    sequencePanelConfig?:RcsbFvSequenceInterface;
 }
 
 /**rxjs Event Handler Object. It allows objects to subscribe methods and then, get(send) events to(from) other objects*/

+ 2 - 2
src/RcsbFvSequence/RcsbFvSequence.tsx

@@ -6,7 +6,7 @@ import {PluginContext} from "molstar/lib/mol-plugin/context";
 import {RcsbFv, RcsbFvTrackDataElementInterface} from "@rcsb/rcsb-saguaro";
 import {RcsbFvSelection} from "../RcsbFvSelection/RcsbFvSelection";
 
-export interface SequenceViewInterface{
+export interface RcsbFvSequenceInterface{
     type: "custom" | "assembly";
     config: AssemblyViewInterface | CustomViewInterface;
     title?: string;
@@ -18,7 +18,7 @@ interface CallbackConfig {
     sequenceCallback?: (rcsbFv: RcsbFv)=>void;
 }
 
-export class RcsbFvSequence extends React.Component <SequenceViewInterface & CallbackConfig & {unmount:(flag:boolean)=>void, plugin: SaguaroPluginInterface, selection:RcsbFvSelection, componentId:string}, SequenceViewInterface > {
+export class RcsbFvSequence extends React.Component <RcsbFvSequenceInterface & CallbackConfig & {unmount:(flag:boolean)=>void, plugin: SaguaroPluginInterface, selection:RcsbFvSelection, componentId:string}, RcsbFvSequenceInterface > {
 
     render() {
         if(this.props.type == "custom"){

+ 15 - 21
src/RcsbFvSequence/SequenceViews/AbstractView.tsx

@@ -6,6 +6,7 @@ import {
     SaguaroPluginModelMapType
 } from "../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
 import {RcsbFvSelection, ResidueSelectionInterface} from "../../RcsbFvSelection/RcsbFvSelection";
+import {SequenceViewInterface} from "./SequenceViewInterface";
 
 export interface AbstractViewInterface {
     componentId: string;
@@ -16,16 +17,16 @@ export interface AbstractViewInterface {
     unmount:(flag:boolean)=>void;
 }
 
-export abstract class AbstractView<P,S> extends React.Component <P & AbstractViewInterface, S> {
+export abstract class AbstractView<P,S> extends React.Component <P & AbstractViewInterface, S> implements SequenceViewInterface {
 
-    protected componentDivId: string;
-    protected pfvDivId: string;
-    protected updateDimTimeout: number = 0;
+    protected readonly componentDivId: string;
+    protected readonly rcsbFvDivId: string;
+    private updateDimTimeout: number = 0;
 
     constructor(props:P & AbstractViewInterface) {
         super(props);
         this.componentDivId = props.componentId+"_"+RcsbFvDOMConstants.PFV_DIV;
-        this.pfvDivId = props.componentId+"_"+RcsbFvDOMConstants.PFV_APP_ID;
+        this.rcsbFvDivId = props.componentId+"_"+RcsbFvDOMConstants.PFV_APP_ID;
     }
 
     render():JSX.Element {
@@ -36,7 +37,7 @@ export abstract class AbstractView<P,S> extends React.Component <P & AbstractVie
                         {this.createSubtitle()}
                         {this.additionalContent()}
                     </div>
-                    <div id ={this.pfvDivId} />
+                    <div id ={this.rcsbFvDivId} />
                 </div>
         );
     }
@@ -54,11 +55,11 @@ export abstract class AbstractView<P,S> extends React.Component <P & AbstractVie
         window.removeEventListener('resize', this.resizeCallback);
     }
 
-    protected resizeCallback: ()=>void =  () => {
+    private resizeCallback: ()=>void =  () => {
         window.clearTimeout(this.updateDimTimeout);
         this.updateDimTimeout = window.setTimeout(()=> {
             this.updateDimensions();
-        },100);
+        },300);
     };
 
     private createTitle(): JSX.Element | null{
@@ -73,18 +74,11 @@ export abstract class AbstractView<P,S> extends React.Component <P & AbstractVie
         return null;
     }
 
-    protected structureSelectionCallback(): void{}
-
-    protected structureHoverCallback(): void{}
-
-    protected representationChangeCallback(): void{}
-
-    protected modelChangeCallback(modelMap:SaguaroPluginModelMapType): void{}
-
-    protected updateDimensions(): void{}
-
-    protected additionalContent(): JSX.Element | null {
-        return <></>;
-    }
+    abstract structureSelectionCallback(): void;
+    abstract structureHoverCallback(): void;
+    abstract representationChangeCallback(): void;
+    abstract modelChangeCallback(modelMap:SaguaroPluginModelMapType): void;
+    abstract updateDimensions(): void;
+    abstract additionalContent(): JSX.Element | null;
 
 }

+ 150 - 207
src/RcsbFvSequence/SequenceViews/AssemblyView/AssemblyView.tsx

@@ -3,15 +3,16 @@ import * as React from "react";
 import {
     buildInstanceSequenceFv,
     buildMultipleInstanceSequenceFv,
-    getRcsbFv,
-    setBoardConfig,
     unmount
 } from "@rcsb/rcsb-saguaro-app";
 import {AbstractView, AbstractViewInterface} from "../AbstractView";
 import {InstanceSequenceOnchangeInterface} from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/RcsbFvBuilder/RcsbFvInstanceBuilder";
-import {RcsbFvTrackDataElementInterface} from "@rcsb/rcsb-saguaro";
+import {RcsbFvBoardConfigInterface, RcsbFvTrackDataElementInterface} from "@rcsb/rcsb-saguaro";
 import {ChainSelectionInterface} from "../../../RcsbFvSelection/RcsbFvSelection";
-import {SaguaroPluginModelMapType} from "../../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
+import {
+    SaguaroPluginInterface,
+    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";
 
@@ -19,12 +20,7 @@ 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";
+import {RcsbFvModulePublicInterface} from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/RcsbFvModule/RcsbFvModuleInterface";
 
 export interface AssemblyViewInterface {
     entryId: string;
@@ -40,15 +36,11 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
     private innerSelectionFlag: boolean = false;
     private currentSelectedComponentId: string;
     private currentModelMap:SaguaroPluginModelMapType;
+    private boardConfig: Partial<RcsbFvBoardConfigInterface>;
+    private rcsbFvModule: RcsbFvModulePublicInterface | null;
     //private readonly componentSet = new Map<string, {current: Set<string>, previous: Set<string>}>();
 
-    constructor(props: AssemblyViewInterface & AbstractViewInterface) {
-        super({
-            ...props
-        });
-    }
-
-    protected additionalContent(): JSX.Element {
+    additionalContent(): JSX.Element {
         return (
             <div style={{marginTop:10}}>
                 <div>
@@ -75,104 +67,55 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         if(width == null)
             return;
         const trackWidth: number = width - 190 - 55;
-        setBoardConfig({
+        this.boardConfig = {
             trackWidth: trackWidth,
+            highlightHoverPosition:true,
+            highlightHoverElement:true,
             elementClickCallBack:(e:RcsbFvTrackDataElementInterface)=>{
-                this.props.plugin.clearFocus();
-                if(this.currentSelectedComponentId != null)
-                    this.props.plugin.removeComponent(this.currentSelectedComponentId);
-                if(e == null)
-                    return;
-                const x = e.begin;
-                const y = e.end ?? e.begin;
-                if(e.isEmpty){
-                    this.props.plugin.cameraFocus(this.currentModelId, this.currentLabelAsymId, [x,y]);
-                    this.currentSelectedComponentId = this.currentLabelAsymId +":"+ ((x === y) ? x.toString() : x.toString()+","+y.toString());
-                    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.cameraFocus(this.currentModelId, this.currentLabelAsymId, x, y);
-                    if((y-x)<this.createComponentThreshold){
-                        this.currentSelectedComponentId = this.currentLabelAsymId +":"+ (x === y ? x.toString() : x.toString()+"-"+y.toString());
-                        setTimeout(()=>{
-                            this.props.plugin.createComponent(
-                                this.currentSelectedComponentId,
-                                this.currentModelId,
-                                processGaps(this.currentModelId, this.currentLabelAsymId, e),
-                                'ball-and-stick'
-                            ).then(()=>{
-                                if(x === y)
-                                    setTimeout(()=>{
-                                        this.props.plugin.setFocus(this.currentModelId, this.currentLabelAsymId, x, y);
-                                    },200);
-                            });
-                        },100);
-                    }
-                }
+                this.elementClickCallback(e);
             },
             selectionChangeCallBack:(selection: Array<SelectionInterface>)=>{
-                if(this.innerSelectionFlag)
-                    return;
-                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.resetPluginView();
-                    return;
-                }
-                this.select(selection);
+                this.selectionChangeCallback(selection);
             },
-            highlightHoverPosition:true,
-            highlightHoverElement:true,
             highlightHoverCallback:(selection: RcsbFvTrackDataElementInterface[])=>{
-                this.props.plugin.clearSelection('hover');
-                if(selection != null && selection.length > 0) {
-                    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.select(
-                            selectionList,
-                            'hover',
-                            'add'
-                        );
-                    }else {
-                        this.props.plugin.select(processMultipleGaps(this.currentModelId, this.currentLabelAsymId, selection), 'hover', 'set');
-                    }
-                }
+                this.highlightHoverCallback(selection);
             },
-        });
+        };
     }
 
     componentWillUnmount() {
         super.componentWillUnmount();
-        unmount(this.pfvDivId);
+        unmount(this.rcsbFvDivId);
     }
 
-    protected structureSelectionCallback(): void{
-        this.pluginSelectCallback('select');
+    async structureSelectionCallback(): Promise<void> {
+        await this.pluginSelectCallback('select');
     }
 
-    protected structureHoverCallback(): void{
-        this.pluginSelectCallback('hover');
+    async structureHoverCallback(): Promise<void> {
+        await this.pluginSelectCallback('hover');
     }
 
-    protected representationChangeCallback(): void{
+    representationChangeCallback(): void{
         //TODO
     }
 
-    private pluginSelectCallback(mode:'select'|'hover'): void{
-        if(getRcsbFv(this.pfvDivId) == null)
+    async updateDimensions(): Promise<void> {
+        const width: number = window.document.getElementById(this.componentDivId)?.getBoundingClientRect().width ?? 0;
+        const trackWidth: number = width - 190 - 55;
+        this.boardConfig.trackWidth = trackWidth;
+        await this.rcsbFvModule?.getFv().updateBoardConfig({boardConfigData:{trackWidth:trackWidth}})
+        await this.structureSelectionCallback();
+        return void 0;
+    }
+
+    private resetPluginView(): void {
+        this.props.plugin.clearFocus();
+        this.props.plugin.resetCamera();
+    }
+
+    private async pluginSelectCallback(mode:'select'|'hover'): Promise<void> {
+        if(this.rcsbFvModule == null)
             return;
         this.innerSelectionFlag = true;
         if(mode === 'select' && this.currentSelectedComponentId != null){
@@ -180,28 +123,28 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         }
         const allSel: Array<ChainSelectionInterface> | undefined = this.props.selection.getSelection(mode);
         if(allSel == null || allSel.length ===0) {
-            getRcsbFv(this.pfvDivId).clearSelection(mode);
+            this.rcsbFvModule?.getFv().clearSelection(mode);
             if(mode === 'select')
                 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, authId);
+            await this.modelChangeCallback(this.currentModelMap, authId);
         }else{
             const sel: ChainSelectionInterface | undefined = this.props.selection.getSelectionWithCondition(this.currentModelId, this.currentLabelAsymId, mode);
             if (sel == null) {
-                getRcsbFv(this.pfvDivId).clearSelection(mode);
+                this.rcsbFvModule?.getFv().clearSelection(mode);
                 if(mode === 'select')
                     this.resetPluginView();
             } else {
-                getRcsbFv(this.pfvDivId).setSelection({elements: sel.regions, mode: mode});
+                this.rcsbFvModule?.getFv().setSelection({elements: sel.regions, mode: mode});
             }
         }
         this.innerSelectionFlag = false;
     }
 
-    protected async modelChangeCallback(modelMap:SaguaroPluginModelMapType, defaultAuthId?: string): Promise<void> {
+    async modelChangeCallback(modelMap:SaguaroPluginModelMapType, defaultAuthId?: string): Promise<void> {
         this.currentModelMap = modelMap;
         this.props.plugin.clearFocus();
         const onChangeCallback: Map<string, (x: InstanceSequenceOnchangeInterface)=>void> = new Map<string, (x: InstanceSequenceOnchangeInterface) => {}>();
@@ -218,31 +161,64 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
             });
             filterInstances.set(v.entryId,new Set<string>(v.chains.map(d=>d.label)));
         });
-        unmount(this.pfvDivId);
+        this.unmountRcsbFv();
         const entryId: string = Array.from(modelMap.values()).map(d=>d.entryId)[0];
-        if(entryId != null)
-            buildInstanceSequenceFv(
-                this.pfvDivId,
+        if(entryId != null) {
+            this.rcsbFvModule = await buildInstanceSequenceFv(
+                this.rcsbFvDivId,
                 RcsbFvDOMConstants.SELECT_INSTANCE_PFV_ID,
-                entryId, {
+                entryId,
+                {
                     defaultValue: defaultAuthId,
                     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>)
+                    selectButtonOptionProps: (props: OptionProps<OptionPropsInterface>) => (components.Option &&
+                        <div style={{display: 'flex'}}>
+                            <ChainDisplay plugin={this.props.plugin} label={props.data.label}/>
+                            <components.Option {...props}/>
+                        </div>)
+                },
+                {
+                    boardConfig: this.boardConfig
                 }
             );
+        }
         if(!defaultAuthId)
-            await this.createComponents(modelMap);
+            await createComponents(this.props.plugin, modelMap);
     }
 
-    protected updateDimensions(): void{
-        const width: number = window.document.getElementById(this.componentDivId)?.getBoundingClientRect().width ?? 0;
-        const trackWidth: number = width - 190 - 55;
-        getRcsbFv(this.pfvDivId).updateBoardConfig({boardConfigData:{trackWidth:trackWidth}}).then(()=>{
-            this.structureSelectionCallback();
-        });
+    private unmountRcsbFv(): void {
+        this.rcsbFvModule = null;
+        unmount(this.rcsbFvDivId);
+    }
+
+    private highlightHoverCallback(selection: RcsbFvTrackDataElementInterface[]): void {
+        this.props.plugin.clearSelection('hover');
+        if(selection != null && selection.length > 0) {
+            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.select(
+                    selectionList,
+                    'hover',
+                    'add'
+                );
+            }else {
+                this.props.plugin.select(processMultipleGaps(this.currentModelId, this.currentLabelAsymId, selection), 'hover', 'set');
+            }
+        }
+    }
+
+    private selectionChangeCallback(selection: Array<SelectionInterface>): void {
+        if(this.innerSelectionFlag)
+            return;
+        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.resetPluginView();
+        }else{
+            this.select(selection);
+        }
     }
 
     private select(selection: Array<SelectionInterface>): void{
@@ -263,104 +239,50 @@ export class AssemblyView extends AbstractView<AssemblyViewInterface & AbstractV
         });
     }
 
-    private resetPluginView(): void {
+    private elementClickCallback(e:RcsbFvTrackDataElementInterface): void {
         this.props.plugin.clearFocus();
-        this.props.plugin.resetCamera();
-    }
-
-    private async createComponents(modelMap:SaguaroPluginModelMapType): Promise<void> {
-        await this.props.plugin.displayComponent("Water", false);
-        await this.props.plugin.colorComponent("Polymer", 'chain-id');
-        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=>{
-                if(ch.type === "polymer") {
-                    chains.push({modelId: modelId, auth: ch.auth, label: ch.label});
-                }
-            });
-        });
-        this.props.plugin.removeComponent();
-        this.props.plugin.clearFocus();
-        for(const ch of chains) {
-            const label: string = ch.auth === ch.label ? ch.label : `${ch.label} [auth ${ch.auth}]`;
-            await this.props.plugin.createComponent(label, ch.modelId, ch.label, 'cartoon');
-            await this.props.plugin.colorComponent(label, 'chain-id');
-        }
-        /*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'
-            }]);
-
-        });*/
-        await this.props.plugin.removeComponent("Polymer");
-    }
+        if(this.currentSelectedComponentId != null)
+            this.props.plugin.removeComponent(this.currentSelectedComponentId);
+        if(e == null)
+            return;
+        const x = e.begin;
+        const y = e.end ?? e.begin;
+        if(e.isEmpty){
+            this.props.plugin.cameraFocus(this.currentModelId, this.currentLabelAsymId, [x,y]);
+            this.currentSelectedComponentId = this.currentLabelAsymId +":"+ ((x === y) ? x.toString() : x.toString()+","+y.toString());
+            setTimeout(async ()=>{
+                await this.props.plugin.createComponent(
+                    this.currentSelectedComponentId,
+                    this.currentModelId,
+                    [{asymId: this.currentLabelAsymId, position: x}, {asymId: this.currentLabelAsymId, position: y}],
+                    'ball-and-stick'
+                )
+                if(x === y)
+                    setTimeout(()=>{
+                        this.props.plugin.setFocus(this.currentModelId, this.currentLabelAsymId, x, y);
+                    },200);
+            },100);
 
-    /*private removeComponents(labelAsymId?:string){
-        if(labelAsymId != null){
-            this.componentSet.get(labelAsymId)?.current.forEach(componentId=>{
-                this.props.plugin.removeComponent(componentId);
-            });
         }else{
-            Array.from(this.componentSet.keys()).forEach(labelAsymId=>{
-                this.componentSet.get(labelAsymId)?.current.forEach(componentId=>{
-                    this.props.plugin.removeComponent(componentId);
-                });
-            });
-        }
-    }
-
-    private removeObsoleteComponents(): void{
-        this.componentSet.get(this.currentLabelAsymId)?.previous.forEach(componentId=>{
-            if(!this.componentSet.get(this.currentLabelAsymId)?.current.has(componentId)) {
-                this.props.plugin.removeComponent(componentId);
+            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());
+                setTimeout(async ()=>{
+                    await this.props.plugin.createComponent(
+                        this.currentSelectedComponentId,
+                        this.currentModelId,
+                        processGaps(this.currentModelId, this.currentLabelAsymId, e),
+                        'ball-and-stick'
+                    )
+                    if(x === y)
+                        setTimeout(()=>{
+                            this.props.plugin.setFocus(this.currentModelId, this.currentLabelAsymId, x, y);
+                        },200);
+                },100);
             }
-        });
+        }
     }
 
-    private resetComponentKeys(): void {
-        if(!this.componentSet.has(this.currentLabelAsymId))
-            this.componentSet.set(this.currentLabelAsymId, {current: new Set<string>(), previous: new Set<string>()});
-        this.componentSet.get(this.currentLabelAsymId)?.previous.clear();
-        this.componentSet.get(this.currentLabelAsymId)?.current.forEach(e=>{
-            this.componentSet.get(this.currentLabelAsymId)?.previous.add(e);
-        });
-        this.componentSet.get(this.currentLabelAsymId)?.current.clear();
-    }*/
-
 }
 
 function processGaps(modelId: string, asymId: string, e: RcsbFvTrackDataElementInterface): Array<{modelId: string; asymId: string; begin: number; end: number;}>{
@@ -391,3 +313,24 @@ function processMultipleGaps(modelId: string, asymId: string, list: Array<RcsbFv
     });
     return regions;
 }
+
+async function createComponents(plugin: SaguaroPluginInterface, modelMap:SaguaroPluginModelMapType): Promise<void> {
+    await plugin.displayComponent("Water", false);
+    await plugin.colorComponent("Polymer", 'chain-id');
+    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=>{
+            if(ch.type === "polymer") {
+                chains.push({modelId: modelId, auth: ch.auth, label: ch.label});
+            }
+        });
+    });
+    plugin.removeComponent();
+    plugin.clearFocus();
+    for(const ch of chains) {
+        const label: string = ch.auth === ch.label ? ch.label : `${ch.label} [auth ${ch.auth}]`;
+        await plugin.createComponent(label, ch.modelId, ch.label, 'cartoon');
+        await plugin.colorComponent(label, 'chain-id');
+    }
+    await plugin.removeComponent("Polymer");
+}

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

@@ -148,7 +148,7 @@ export class CustomView extends AbstractView<CustomViewInterface & AbstractViewI
         });
     }
 
-    protected structureSelectionCallback(): void {
+    structureSelectionCallback(): void {
         this.blockMap.get(this.blockViewSelector.getActiveBlock())?.forEach(boardId=>{
             const pfv: RcsbFv | undefined = this.rcsbFvMap.get(boardId);
             if(pfv == null)
@@ -157,28 +157,29 @@ export class CustomView extends AbstractView<CustomViewInterface & AbstractViewI
         });
     }
 
-    protected structureHoverCallback(): void{
+    structureHoverCallback(): void{
         //TODO;
     }
 
-    protected representationChangeCallback(): void{
+    representationChangeCallback(): void{
         //TODO
     }
 
-    protected additionalContent(): JSX.Element {
+    additionalContent(): JSX.Element {
         if(this.state.additionalContent == null)
             return <></>;
         return this.state.additionalContent(this.blockViewSelector);
     }
 
-    protected modelChangeCallback(modelMap:SaguaroPluginModelMapType): void {
+    modelChangeCallback(modelMap:SaguaroPluginModelMapType): void {
         if(this.firstModelLoad){
             this.firstModelLoad = false;
             return;
         }
         if(typeof this.props.modelChangeCallback === "function") {
-            const newConfig: CustomViewStateInterface | void = this.props.modelChangeCallback(modelMap);
-            if(newConfig != null){
+            let newConfig: CustomViewStateInterface | void = this.props.modelChangeCallback(modelMap);
+            if(newConfig != null && typeof newConfig !== "function"){
+                newConfig = newConfig as CustomViewStateInterface;
                 if(newConfig.blockConfig != null && newConfig.additionalContent != null){
                     this.mapBlocks(newConfig.blockConfig);
                     this.setState({blockConfig: newConfig.blockConfig, additionalContent: newConfig.additionalContent})
@@ -192,7 +193,7 @@ export class CustomView extends AbstractView<CustomViewInterface & AbstractViewI
         }
     }
 
-    protected updateDimensions(): void {
+    updateDimensions(): void {
         const div: HTMLElement | undefined | null = document.getElementById(this.componentDivId)?.parentElement;
         const width: number = window.document.getElementById(this.componentDivId)?.getBoundingClientRect().width ?? 0;
         if(div == null || (div.style.width && !div.style.width.includes("%")) )

+ 10 - 0
src/RcsbFvSequence/SequenceViews/SequenceViewInterface.ts

@@ -0,0 +1,10 @@
+import {SaguaroPluginModelMapType} from "../../RcsbFvStructure/StructurePlugins/SaguaroPluginInterface";
+
+export interface SequenceViewInterface {
+    structureSelectionCallback(): void;
+    structureHoverCallback(): void;
+    representationChangeCallback(): void;
+    modelChangeCallback(modelMap:SaguaroPluginModelMapType): void;
+    updateDimensions(): void;
+    additionalContent(): JSX.Element | null;
+}

+ 2 - 2
src/RcsbFvStructure/RcsbFvStructure.tsx

@@ -5,12 +5,12 @@ import {ViewerProps} from "@rcsb/rcsb-molstar/build/src/viewer";
 import {LoadMolstarInterface} from "./StructurePlugins/MolstarPlugin";
 import {RcsbFvSelection} from "../RcsbFvSelection/RcsbFvSelection";
 
-export interface StructureViewInterface {
+export interface RcsbFvStructureInterface {
     loadConfig: LoadMolstarInterface;
     pluginConfig?: Partial<ViewerProps>;
 }
 
-export class RcsbFvStructure extends React.Component <StructureViewInterface & {plugin: SaguaroPluginInterface, componentId: string, selection: RcsbFvSelection}, StructureViewInterface > {
+export class RcsbFvStructure extends React.Component <RcsbFvStructureInterface & {plugin: SaguaroPluginInterface, componentId: string, selection: RcsbFvSelection}, RcsbFvStructureInterface > {
 
     render():JSX.Element {
         return (

+ 63 - 52
src/RcsbFvStructure/StructurePlugins/MolstarPlugin.ts

@@ -22,8 +22,8 @@ import {
 } 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} from "molstar/lib/mol-state";
-import {StructureRef} from "molstar/lib/mol-plugin-state/manager/structure/hierarchy-state";
+import {State, StateObject} from "molstar/lib/mol-state";
+import {StructureComponentRef, StructureRef} from "molstar/lib/mol-plugin-state/manager/structure/hierarchy-state";
 import {RcsbFvSelection, ResidueSelectionInterface} from "../../RcsbFvSelection/RcsbFvSelection";
 import {AbstractPlugin} from "./AbstractPlugin";
 import {Subscription} from "rxjs";
@@ -33,6 +33,7 @@ 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";
+import {StateObjectCell} from "molstar/lib/mol-state/object";
 
 export enum LoadMethod {
     loadPdbId = "loadPdbId",
@@ -60,7 +61,7 @@ interface LoadParams {
 }
 
 export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterface {
-    private plugin: Viewer;
+    private viewer: Viewer;
     private innerSelectionFlag: boolean = false;
     private loadingFlag: boolean = false;
     private selectCallbackSubs: Subscription;
@@ -75,12 +76,11 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public init(target: string | HTMLElement, props?: Partial<ViewerProps>) {
-        this.plugin = new Viewer(target, {layoutShowControls:false, layoutShowSequence: true, ...props});
-        this.plugin.getPlugin().representation.structure.registry
+        this.viewer = new Viewer(target, {layoutShowControls:false, layoutShowSequence: true, ...props});
     }
 
     public clear(): void{
-        this.plugin.clear();
+        this.viewer.clear();
     }
 
     async load(loadConfig: LoadMolstarInterface): Promise<void>{
@@ -88,24 +88,24 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         if(MolstarPlugin.checkLoadData(loadConfig)) {
             if (loadConfig.method == LoadMethod.loadPdbId) {
                 const config: LoadParams = loadConfig.params as LoadParams;
-                await this.plugin.loadPdbId(config.pdbId!, config.props, config.matrix);
+                await this.viewer.loadPdbId(config.pdbId!, config.props, config.matrix);
             } else if (loadConfig.method == LoadMethod.loadPdbIds) {
                 const config: Array<LoadParams> = loadConfig.params as Array<LoadParams>;
-                await this.plugin.loadPdbIds(config.map((d) => {
+                await this.viewer.loadPdbIds(config.map((d) => {
                     return {pdbId: d.pdbId!, props: d.props, matrix: d.matrix}
                 }));
             } else if (loadConfig.method == LoadMethod.loadStructureFromUrl) {
                 const config: LoadParams = loadConfig.params as LoadParams;
-                await this.plugin.loadStructureFromUrl(config.url!, config.format!, config.isBinary!);
+                await this.viewer.loadStructureFromUrl(config.url!, config.format!, config.isBinary!);
             } else if (loadConfig.method == LoadMethod.loadSnapshotFromUrl) {
                 const config: LoadParams = loadConfig.params as LoadParams;
-                await this.plugin.loadSnapshotFromUrl(config.url!, config.type!);
+                await this.viewer.loadSnapshotFromUrl(config.url!, config.type!);
             } else if (loadConfig.method == LoadMethod.loadStructureFromData) {
                 const config: LoadParams = loadConfig.params as LoadParams;
-                await this.plugin.loadStructureFromData(config.data!, config.format!, config.isBinary!);
+                await this.viewer.loadStructureFromData(config.data!, config.format!, config.isBinary!);
             }
         }
-        this.plugin.getPlugin().selectionMode = true;
+        this.viewer.plugin.selectionMode = true;
         this.loadingFlag = false;
         this.mapModels(loadConfig.params);
         this.modelChangeCallback(this.getChains());
@@ -156,40 +156,41 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         if(mode == null || mode === 'select') {
             this.innerSelectionFlag = true;
         }
-        this.plugin.select(this.getModelId(modelId), asymId, begin, end, mode, operation);
+        this.viewer.select({modelId:this.getModelId(modelId), label_asym_id: asymId, label_seq_range:{beg: begin, end:end}}, mode,operation);
         this.innerSelectionFlag = false;
     }
     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.viewer.select(selection.map(r=>({modelId: this.getModelId(r.modelId), label_seq_id:r.position, label_asym_id: r.asymId})), mode, operation);
         this.innerSelectionFlag = false;
     }
     private selectMultipleRanges(selection: Array<{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(selection.map(r=>{return{modelId: this.getModelId(r.modelId), begin:r.begin, end: r.end, asymId: r.asymId}}), mode, operation);
+        this.viewer.select(selection.map(r=>({modelId: this.getModelId(r.modelId), label_asym_id: r.asymId, label_seq_range:{beg:r.begin, end: r.end}})), mode, operation);
         this.innerSelectionFlag = false;
     }
+
     public clearSelection(mode:'select'|'hover', option?:{modelId:string; labelAsymId:string;}): void {
         if(mode === 'select') {
-            this.plugin.clearFocus();
+            this.viewer.clearFocus();
             this.innerSelectionFlag = true;
         }
         if(option != null)
-            this.plugin.clearSelection(mode, {modelId: this.getModelId(option.modelId), labelAsymId: option.labelAsymId});
+            this.viewer.clearSelection(mode, {modelId: this.getModelId(option.modelId), label_asym_id: option.labelAsymId});
         else
-            this.plugin.clearSelection(mode);
+            this.viewer.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);
+        this.viewer.setFocus({modelId: this.getModelId(modelId), label_asym_id: asymId, label_seq_range:{beg:begin, end: end}});
     }
     public clearFocus(): void {
-        this.plugin.clearFocus();
+        this.viewer.clearFocus();
     }
 
     public cameraFocus(modelId: string, asymId: string, positions:Array<number>): void;
@@ -202,7 +203,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         }
     }
     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));
+        const data: Structure | undefined = getStructureWithModelId(this.viewer.plugin.managers.structure.hierarchy.current.structures, this.getModelId(modelId));
         if (data == null) return;
         const sel: StructureSelection = Script.getStructureSelection(Q => Q.struct.generator.atomGroups({
             'chain-test': Q.core.rel.eq([asymId, MolScriptBuilder.ammp('label_asym_id')]),
@@ -210,9 +211,9 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         }), data);
         const loci: Loci = StructureSelection.toLociWithSourceUnits(sel);
         if(!StructureElement.Loci.isEmpty(loci))
-            this.plugin.getPlugin().managers.camera.focusLoci(loci);
+            this.viewer.plugin.managers.camera.focusLoci(loci);
         else
-            this.plugin.getPlugin().managers.camera.reset();
+            this.viewer.plugin.managers.camera.reset();
     }
     private focusRange(modelId: string, asymId: string, begin: number, end: number): void{
         const seqIds: Array<number> = new Array<number>();
@@ -230,14 +231,24 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         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]);
-        else if(args.length === 6)
-            await this.plugin.createComponent(args[0], this.getModelId(args[1]), args[2], args[3], args[4], args[5]);
+        if(args.length === 4){
+            if( args[2] instanceof Array && args[2].length > 0 ) {
+                if(typeof args[2][0].position === "number"){
+                    await this.viewer.createComponent(args[0], args[2].map(r=>({modelId: this.getModelId(args[1]), label_asym_id: r.asymId, label_seq_id: r.position})), args[3]);
+                }else{
+                    await this.viewer.createComponent(args[0], args[2].map(r=>({modelId: this.getModelId(args[1]), label_asym_id: r.asymId, label_seq_range:{beg:r.begin, end: r.end}})), args[3]);
+                }
+            }else{
+                await this.viewer.createComponent(args[0], {modelId: this.getModelId(args[1]), label_asym_id:args[2]}, args[3]);
+            }
+        }
+        else if(args.length === 6){
+            await this.viewer.createComponent(args[0], {modelId: this.getModelId(args[1]), label_asym_id: args[2], label_seq_range:{beg:args[3], end:args[4]}}, args[5]);
+        }
     }
 
     public isComponent(componentLabel: string): boolean{
-        for(const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups){
+        for(const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups){
             for(const comp of c){
                 if(comp.cell.obj?.label === componentLabel) {
                     return true;
@@ -248,10 +259,10 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public async colorComponent(componentLabel: string, color: ColorTheme.BuiltIn): Promise<void>{
-        for(const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups){
+        for(const c of this.viewer.plugin.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 });
+                    await this.viewer.plugin.managers.structure.component.updateRepresentationsTheme([comp], { color: color });
                     return;
                 }
             }
@@ -260,7 +271,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
 
     public getComponentSet(): Set<string>{
         const out: Set<string> = new Set<string>();
-        this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups.forEach(c=>{
+        this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups.forEach((c)=>{
             for(const comp of c){
                 if(comp.cell.obj?.label != null && out.has(comp.cell.obj?.label)) {
                     break;
@@ -275,12 +286,12 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     public removeComponent(componentLabel?: string): void{
         if(componentLabel == null){
             this.componentSet.forEach(id=>{
-                this.plugin.removeComponent(id);
+                this.viewer.removeComponent(id);
             })
             this.componentSet.clear();
             this.componentVisibility.clear();
         }else{
-            this.plugin.removeComponent(componentLabel);
+            this.viewer.removeComponent(componentLabel);
             this.componentSet.delete(componentLabel);
             this.componentVisibility.delete(componentLabel);
         }
@@ -298,11 +309,11 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         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 c of this.viewer.plugin.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]);
+                            this.viewer.plugin.managers.structure.component.toggleVisibility([comp]);
                         return void 0;
                     }
                 }
@@ -310,7 +321,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
         }
     }
     private getComponentDisplay(componentLabel: string): boolean{
-        for (const c of this.plugin.getPlugin().managers.structure.hierarchy.currentComponentGroups) {
+        for (const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups) {
             for (const comp of c) {
                 if (comp.cell.obj?.label === componentLabel) {
                     return this.componentVisibility.get(componentLabel) ?? false;
@@ -321,7 +332,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public setRepresentationChangeCallback(g:()=>void){
-        this.plugin.getPlugin().state.events.cell.stateUpdated.subscribe(o=>{
+        this.viewer.plugin.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);
@@ -331,9 +342,9 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public setHoverCallback(g:()=>void){
-        this.plugin.getPlugin().behaviors.interaction.hover.subscribe((r: InteractivityManager.HoverEvent)=>{
+        this.viewer.plugin.behaviors.interaction.hover.subscribe((r)=>{
             const sequenceData: Array<ResidueSelectionInterface> = new Array<ResidueSelectionInterface>();
-            const loci: Loci = r.current.loci;
+            const loci:Loci = r.current.loci;
             if(StructureElement.Loci.is(loci)){
                 const loc = StructureElement.Location.create(loci.structure);
                 for (const e of loci.elements) {
@@ -357,12 +368,12 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public setSelectCallback(g:(flag?:boolean)=>void){
-        this.selectCallbackSubs = this.plugin.getPlugin().managers.structure.selection.events.changed.subscribe(()=>{
+        this.selectCallbackSubs = this.viewer.plugin.managers.structure.selection.events.changed.subscribe(()=>{
             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;
+            if(this.viewer.plugin.managers.structure.selection.additionsHistory.length > 0) {
+                const currentLoci: Loci = this.viewer.plugin.managers.structure.selection.additionsHistory[0].loci;
                 const loc: StructureElement.Location = StructureElement.Location.create(currentLoci.structure);
                 StructureElement.Location.set(
                     loc,
@@ -387,7 +398,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
                         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);
+                        this.viewer.plugin.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]);
@@ -415,10 +426,10 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
                 this.selection.setLastSelection('select', null);
             }
             const sequenceData: Array<ResidueSelectionInterface> = new Array<ResidueSelectionInterface>();
-            for(const structure of this.plugin.getPlugin().managers.structure.hierarchy.current.structures){
+            for(const structure of this.viewer.plugin.managers.structure.hierarchy.current.structures){
                 const data: Structure | undefined = structure.cell.obj?.data;
                 if(data == null) return;
-                const loci: Loci = this.plugin.getPlugin().managers.structure.selection.getLoci(data);
+                const loci: Loci = this.viewer.plugin.managers.structure.selection.getLoci(data);
                 if(StructureElement.Loci.is(loci)){
                     const loc = StructureElement.Location.create(loci.structure);
                     for (const e of loci.elements) {
@@ -443,12 +454,12 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public pluginCall(f: (plugin: PluginContext) => void){
-        this.plugin.pluginCall(f);
+        this.viewer.pluginCall(f);
     }
 
     public setModelChangeCallback(f:(modelMap:SaguaroPluginModelMapType)=>void){
         this.modelChangeCallback = f;
-        this.modelChangeCallbackSubs = this.plugin.getPlugin().state.events.object.updated.subscribe((o)=>{
+        this.modelChangeCallbackSubs = this.viewer.plugin.state.events.object.updated.subscribe((o:{obj: StateObject, action: "in-place" | "recreate"})=>{
             if(this.loadingFlag)
                 return;
             if(o.obj.type.name === "Behavior" && o.action === "in-place") {
@@ -460,10 +471,10 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     private getChains(): SaguaroPluginModelMapType{
-        const structureRefList = getStructureOptions(this.plugin.getPlugin());
+        const structureRefList = getStructureOptions(this.viewer.plugin);
         const out: SaguaroPluginModelMapType = new Map<string, {entryId: string; chains: Array<{label:string;auth:string;entityId:string;title:string;type:ChainType;}>}>();
         structureRefList.forEach((structureRef,i)=>{
-            const structure = getStructure(structureRef[0], this.plugin.getPlugin().state.data);
+            const structure = getStructure(structureRef[0], this.viewer.plugin.state.data);
             let modelEntityId = getModelEntityOptions(structure)[0][0];
             const chains: [{modelId:string;entryId:string},{auth:string;label:string;entityId:string;title:string;type:ChainType;}[]] = getChainValues(structure, modelEntityId);
             out.set(this.getModelId(chains[0].modelId),{entryId:chains[0].entryId, chains: chains[1]});
@@ -473,9 +484,9 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
 
     private mapModels(loadParams: LoadParams | Array<LoadParams>): void{
         const loadParamList: Array<LoadParams> = loadParams instanceof Array ? loadParams : [loadParams];
-        const structureRefList = getStructureOptions(this.plugin.getPlugin());
+        const structureRefList = getStructureOptions(this.viewer.plugin);
         structureRefList.forEach((structureRef,i)=>{
-            const structure = getStructure(structureRef[0], this.plugin.getPlugin().state.data);
+            const structure = getStructure(structureRef[0], this.viewer.plugin.state.data);
             let modelEntityId = getModelEntityOptions(structure)[0][0];
             const chains: [{modelId:string, entryId:string},{auth:string,label:string;entityId:string;title:string;type:ChainType;}[]] = getChainValues(structure, modelEntityId);
             this.modelMap.set(chains[0].modelId,loadParamList[i].id);
@@ -494,7 +505,7 @@ export class MolstarPlugin extends AbstractPlugin implements SaguaroPluginInterf
     }
 
     public resetCamera(): void {
-        this.plugin.getPlugin().managers.camera.reset();
+        this.viewer.plugin.managers.camera.reset();
     }
 
 }

+ 3 - 3
src/examples/custom-panel/example.tsx

@@ -1,6 +1,6 @@
 import {RcsbFv3DCustomBuilder} from "../../RcsbFv3D/RcsbFv3DCustom";
-import {StructureViewInterface} from "../../RcsbFvStructure/RcsbFvStructure";
-import {SequenceViewInterface} from "../../RcsbFvSequence/RcsbFvSequence";
+import {RcsbFvStructureInterface} from "../../RcsbFvStructure/RcsbFvStructure";
+import {RcsbFvSequenceInterface} from "../../RcsbFvSequence/RcsbFvSequence";
 
 import './example.html';
 import {LoadMethod} from "../../RcsbFvStructure/StructurePlugins/MolstarPlugin";
@@ -171,7 +171,7 @@ const sequenceConfig = {
     config: customConfig
 };
 
-const structureConfig:StructureViewInterface = {
+const structureConfig:RcsbFvStructureInterface = {
     loadConfig: {
         method: LoadMethod.loadPdbIds,
         params: [{

+ 2 - 2
webpack.config.js

@@ -44,7 +44,7 @@ const appConfig = {
         'RcsbFv3DCustom':'./build/src/RcsbFv3D/RcsbFv3DCustom.js',
         'rcsb-saguaro-3d':'./build/src/RcsbSaguaro3D.js'
     },
-    mode: "production",
+    mode: "development",
     output: {
         filename: '[name].js',
         library: 'RcsbFv3D',
@@ -52,7 +52,7 @@ const appConfig = {
         umdNamedDefine: true,
         path: path.resolve(__dirname, 'build/dist')
     },
-    devtool: false//'source-map'
+    devtool: 'source-map'
 }
 
 module.exports = [appConfig];

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