|
@@ -8,16 +8,16 @@ import {
|
|
|
Structure,
|
|
|
StructureElement,
|
|
|
StructureProperties as SP,
|
|
|
- StructureQuery,
|
|
|
StructureSelection
|
|
|
} from "molstar/lib/mol-model/structure";
|
|
|
import {OrderedSet} from "molstar/lib/mol-data/int";
|
|
|
-import {Queries as Q} from "molstar/lib/mol-model/structure/query";
|
|
|
import {PluginContext} from "molstar/lib/mol-plugin/context";
|
|
|
import {Viewer} from "@rcsb/rcsb-molstar/build/src/viewer";
|
|
|
import {Subscription} from "rxjs";
|
|
|
import {DataContainer, DataContainerReader} from "../../../Utils/DataContainer";
|
|
|
import {RcsbFvStateInterface} from "../../../RcsbFvState/RcsbFvStateInterface";
|
|
|
+import {MolScriptBuilder as MS} from "molstar/lib/mol-script/language/builder";
|
|
|
+import {Script} from "molstar/lib/mol-script/script";
|
|
|
|
|
|
type ModelMapType = Omit<ViewerModelMapManagerInterface<unknown,unknown>,'add'|'delete'>;
|
|
|
export class MolstarCallbackManager implements ViewerCallbackManagerInterface{
|
|
@@ -29,7 +29,9 @@ export class MolstarCallbackManager implements ViewerCallbackManagerInterface{
|
|
|
private readonly innerSelectionFlag: DataContainer<boolean>;
|
|
|
private readonly innerReprChangeFlag: DataContainer<boolean>;
|
|
|
|
|
|
- private selectSubs: Subscription;
|
|
|
+ private addSubs: Subscription;
|
|
|
+ private removeSubs: Subscription;
|
|
|
+ private clearSubs: Subscription;
|
|
|
private hoverSubs: Subscription;
|
|
|
private modelChangeSubs: Subscription;
|
|
|
private reprChangeSubs: Subscription;
|
|
@@ -82,97 +84,75 @@ export class MolstarCallbackManager implements ViewerCallbackManagerInterface{
|
|
|
}
|
|
|
|
|
|
public subscribeSelection(): Subscription {
|
|
|
- this.selectSubs = this.viewer.plugin.managers.structure.selection.events.changed.subscribe(()=>{
|
|
|
+ this.addSubs = this.viewer.plugin.managers.structure.selection.events.loci.add.subscribe((currentLoci)=>{
|
|
|
if(this.innerSelectionFlag.get())
|
|
|
return;
|
|
|
- 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,
|
|
|
- currentLoci.structure,
|
|
|
- currentLoci.elements[0].unit,
|
|
|
- currentLoci.elements[0].unit.elements[OrderedSet.getAt(currentLoci.elements[0].indices,0)]
|
|
|
- );
|
|
|
- const currentModelId: string = this.modelMapManager.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.set(true);
|
|
|
- const sel: StructureSelection = StructureQuery.run(query, currentLoci.structure);
|
|
|
- const surroundingsLoci: Loci = StructureSelection.toLociWithSourceUnits(sel);
|
|
|
- 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]);
|
|
|
- if(SP.entity.type(surroundingsLoc) === 'polymer'){
|
|
|
- this.stateManager.selectionState.setLastSelection({
|
|
|
- modelId: currentModelId,
|
|
|
- labelAsymId: SP.chain.label_asym_id(surroundingsLoc),
|
|
|
- source:"structure",
|
|
|
- regions: []
|
|
|
- });
|
|
|
- }
|
|
|
- }
|
|
|
- this.innerSelectionFlag.set(false);
|
|
|
- }else if( SP.entity.type(loc) === 'polymer' ) {
|
|
|
+ const loc: StructureElement.Location = createLocation(currentLoci);
|
|
|
+ const currentModelId: string = this.modelMapManager.getModelId(currentLoci.structure.model.id);
|
|
|
+ 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 surrCore = MS.struct.generator.atomGroups({
|
|
|
+ 'residue-test': MS.core.rel.eq([MS.ammp('auth_seq_id'), resAuthId]),
|
|
|
+ 'chain-test': MS.core.rel.eq([MS.ammp('label_asym_id'), chainLabelId]),
|
|
|
+ 'entity-test': MS.core.rel.eq([MS.ammp('entityType'), 'non-polymer']),
|
|
|
+ });
|
|
|
+ const surrExp = MS.struct.modifier.includeSurroundings({
|
|
|
+ 0: surrCore,
|
|
|
+ radius: 5,
|
|
|
+ "as-whole-residues": true
|
|
|
+ })
|
|
|
+ const polymerExp = MS.struct.generator.atomGroups({
|
|
|
+ 'residue-test': MS.core.rel.eq([MS.ammp('entityType'),'non-polymer'])
|
|
|
+ });
|
|
|
+ const sel: StructureSelection = Script.getStructureSelection(Q=>Q.struct.modifier.exceptBy({
|
|
|
+ 0: surrExp,
|
|
|
+ by: polymerExp
|
|
|
+ }), currentLoci.structure);
|
|
|
+ this.innerSelectionFlag.set(true);
|
|
|
+ const surroundingsLoci: StructureElement.Loci = StructureSelection.toLociWithSourceUnits(sel);
|
|
|
+ this.viewer.plugin.managers.structure.selection.fromLoci('add', StructureSelection.toLociWithSourceUnits(sel));
|
|
|
+ const surroundingsLoc = StructureElement.Location.create(surroundingsLoci.structure);
|
|
|
+ let currentSelLength = 0;
|
|
|
+ for (const e of surroundingsLoci.elements) {
|
|
|
+ StructureElement.Location.set(surroundingsLoc, surroundingsLoci.structure, e.unit, e.unit.elements[0]);
|
|
|
+ if(SP.entity.type(surroundingsLoc) === 'polymer' && ((typeof e.indices !== "object" && currentSelLength == 0) || (e.indices as unknown as []).length >= currentSelLength)){
|
|
|
+ currentSelLength = typeof e.indices !== "object" ? 0 : (e.indices as unknown as []).length;
|
|
|
this.stateManager.selectionState.setLastSelection({
|
|
|
modelId: currentModelId,
|
|
|
- labelAsymId: SP.chain.label_asym_id(loc),
|
|
|
- operatorName: SP.unit.operator_name(loc),
|
|
|
+ labelAsymId: SP.chain.label_asym_id(surroundingsLoc),
|
|
|
source:"structure",
|
|
|
regions: []
|
|
|
});
|
|
|
- }else{
|
|
|
- this.stateManager.selectionState.setLastSelection(null);
|
|
|
}
|
|
|
- else
|
|
|
- this.stateManager.selectionState.setLastSelection(null);
|
|
|
+ }
|
|
|
+ this.innerSelectionFlag.set(false);
|
|
|
+ }else if( SP.entity.type(loc) === 'polymer' ) {
|
|
|
+ const labelAsymId= SP.chain.label_asym_id(loc);
|
|
|
+ const operatorName = SP.unit.operator_name(loc);
|
|
|
+ this.stateManager.selectionState.setLastSelection({
|
|
|
+ modelId: currentModelId,
|
|
|
+ labelAsymId: labelAsymId,
|
|
|
+ operatorName: operatorName,
|
|
|
+ source: "structure",
|
|
|
+ regions: []
|
|
|
+ });
|
|
|
}else{
|
|
|
this.stateManager.selectionState.setLastSelection(null);
|
|
|
}
|
|
|
- const sequenceData: Array<SaguaroSet> = new Array<SaguaroSet>();
|
|
|
- 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.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) {
|
|
|
- StructureElement.Location.set(loc, loci.structure, e.unit, e.unit.elements[0]);
|
|
|
- if(SP.entity.type(loc) === 'polymer'){
|
|
|
- const seqIds = new Set<number>();
|
|
|
- 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));
|
|
|
- }
|
|
|
- if(seqIds.size > 0)
|
|
|
- sequenceData.push({
|
|
|
- modelId: this.modelMapManager.getModelId(data.model.id),
|
|
|
- labelAsymId: SP.chain.label_asym_id(loc),
|
|
|
- operatorName: SP.unit.operator_name(loc),
|
|
|
- seqIds
|
|
|
- });
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- this.stateManager.selectionState.setSelectionFromResidueSelection(sequenceData, 'select', 'structure');
|
|
|
- if(sequenceData.length == 0)
|
|
|
- this.stateManager.selectionState.setLastSelection(null);
|
|
|
- this.stateManager.next({type:"selection-change", view:"3d-view"});
|
|
|
+ this.updateStateManager()
|
|
|
});
|
|
|
- return this.selectSubs;
|
|
|
+ this.removeSubs = this.viewer.plugin.managers.structure.selection.events.loci.remove.subscribe(()=> {
|
|
|
+ if (this.innerSelectionFlag.get())
|
|
|
+ return;
|
|
|
+ this.updateStateManager();
|
|
|
+ });
|
|
|
+ this.viewer.plugin.managers.structure.selection.events.loci.clear.subscribe(()=> {
|
|
|
+ if (this.innerSelectionFlag.get())
|
|
|
+ return;
|
|
|
+ this.updateStateManager();
|
|
|
+ });
|
|
|
+ return this.addSubs;
|
|
|
}
|
|
|
|
|
|
public pluginCall(f: (plugin: PluginContext) => void){
|
|
@@ -194,10 +174,55 @@ export class MolstarCallbackManager implements ViewerCallbackManagerInterface{
|
|
|
}
|
|
|
|
|
|
public unsubscribe(): void {
|
|
|
- this.selectSubs?.unsubscribe();
|
|
|
+ this.addSubs?.unsubscribe();
|
|
|
+ this.removeSubs?.unsubscribe();
|
|
|
+ this.clearSubs?.unsubscribe();
|
|
|
this.modelChangeSubs?.unsubscribe();
|
|
|
this.hoverSubs?.unsubscribe();
|
|
|
}
|
|
|
|
|
|
+ private updateStateManager(): void{
|
|
|
+ const sequenceData: Array<SaguaroSet> = new Array<SaguaroSet>();
|
|
|
+ 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.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) {
|
|
|
+ StructureElement.Location.set(loc, loci.structure, e.unit, e.unit.elements[0]);
|
|
|
+ if(SP.entity.type(loc) === 'polymer'){
|
|
|
+ const seqIds = new Set<number>();
|
|
|
+ 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));
|
|
|
+ }
|
|
|
+ if(seqIds.size > 0)
|
|
|
+ sequenceData.push({
|
|
|
+ modelId: this.modelMapManager.getModelId(data.model.id),
|
|
|
+ labelAsymId: SP.chain.label_asym_id(loc),
|
|
|
+ operatorName: SP.unit.operator_name(loc),
|
|
|
+ seqIds
|
|
|
+ });
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ this.stateManager.selectionState.setSelectionFromResidueSelection(sequenceData, 'select', 'structure');
|
|
|
+ if(sequenceData.length == 0)
|
|
|
+ this.stateManager.selectionState.setLastSelection(null);
|
|
|
+ this.stateManager.next({type:"selection-change", view:"3d-view"});
|
|
|
+ }
|
|
|
|
|
|
}
|
|
|
+
|
|
|
+function createLocation(loci: StructureElement.Loci){
|
|
|
+ const loc: StructureElement.Location = StructureElement.Location.create(loci.structure);
|
|
|
+ StructureElement.Location.set(
|
|
|
+ loc,
|
|
|
+ loci.structure,
|
|
|
+ loci.elements[0].unit,
|
|
|
+ loci.elements[0].unit.elements[OrderedSet.getAt(loci.elements[0].indices,0)]
|
|
|
+ );
|
|
|
+ return loc;
|
|
|
+}
|