MolstarActionManager.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  1. import {
  2. SaguaroChain,
  3. SaguaroPosition,
  4. SaguaroRange,
  5. ViewerActionManagerInterface,
  6. ViewerModelMapManagerInterface
  7. } from "../../StructureViewerInterface";
  8. import {Viewer} from "@rcsb/rcsb-molstar/build/src/viewer";
  9. import {DataContainer} from "../../../Utils/DataContainer";
  10. import {Structure, StructureElement, StructureSelection} from "molstar/lib/mol-model/structure";
  11. import {Expression} from "molstar/lib/commonjs/mol-script/language/expression";
  12. import {MolScriptBuilder as MS} from "molstar/lib/mol-script/language/builder";
  13. import {Script} from "molstar/lib/mol-script/script";
  14. import {SetUtils} from "molstar/lib/mol-util/set";
  15. import {Loci} from "molstar/lib/mol-model/loci";
  16. import {StructureRef} from "molstar/lib/mol-plugin-state/manager/structure/hierarchy-state";
  17. import {ColorTheme} from "molstar/lib/mol-theme/color";
  18. import {StructureRepresentationRegistry} from "molstar/lib/mol-repr/structure/registry";
  19. import {PresetProps} from "@rcsb/rcsb-molstar/build/src/viewer/helpers/preset";
  20. import {Mat4} from "molstar/lib/mol-math/linear-algebra";
  21. import {BuiltInTrajectoryFormat} from "molstar/lib/mol-plugin-state/formats/trajectory";
  22. import {PluginState} from "molstar/lib/mol-plugin/state";
  23. import {TrajectoryHierarchyPresetProvider} from "molstar/lib/mol-plugin-state/builder/structure/hierarchy-preset";
  24. import {StateObject, StateObjectSelector} from "molstar/lib/mol-state";
  25. import {PluginStateObject} from "molstar/lib/mol-plugin-state/objects";
  26. import {StateTransformer} from "molstar/lib/mol-state/transformer";
  27. import {
  28. StructureRepresentationPresetProvider
  29. } from "molstar/lib/mol-plugin-state/builder/structure/representation-preset";
  30. // import { TmDetDescriptorCache } from "@rcsb/rcsb-molstar/build/src/tmdet-extension/prop";
  31. // import { DebugUtil } from "../../../TmFv3DApp/tmdet-extension/debug-utils";
  32. export enum LoadMethod {
  33. loadPdbId = "loadPdbId",
  34. loadStructureFromUrl = "loadStructureFromUrl",
  35. loadSnapshotFromUrl = "loadSnapshotFromUrl",
  36. loadStructureFromData = "loadStructureFromData"
  37. }
  38. export interface LoadMolstarInterface<P,L> {
  39. loadMethod: LoadMethod;
  40. loadParams: LoadParams<P,L>;
  41. }
  42. export type LoadMolstarReturnType = {
  43. model?: StateObjectSelector<PluginStateObject.Molecule.Model, StateTransformer<StateObject<any, StateObject.Type<any>>, StateObject<any, StateObject.Type<any>>, any>>,
  44. modelProperties?: StateObjectSelector<PluginStateObject.Molecule.Model, StateTransformer<StateObject<any, StateObject.Type<any>>, StateObject<any, StateObject.Type<any>>, any>>,
  45. structure?: StateObjectSelector<PluginStateObject.Molecule.Structure, StateTransformer<StateObject<any, StateObject.Type<any>>, StateObject<any, StateObject.Type<any>>, any>>,
  46. structureProperties?: StateObjectSelector<PluginStateObject.Molecule.Structure, StateTransformer<StateObject<any, StateObject.Type<any>>, StateObject<any, StateObject.Type<any>>, any>>,
  47. representation?: StructureRepresentationPresetProvider.Result
  48. };
  49. interface LoadParams<P,L> {
  50. entryId?: string;
  51. props?: PresetProps;
  52. matrix?: Mat4;
  53. url?: string,
  54. format?: BuiltInTrajectoryFormat,
  55. isBinary?: boolean,
  56. type?: PluginState.SnapshotType,
  57. data?: string | number[]
  58. id?:string;
  59. reprProvider?: TrajectoryHierarchyPresetProvider<P,L>;
  60. params?:P;
  61. }
  62. export class MolstarActionManager<P,L> implements ViewerActionManagerInterface<LoadMolstarInterface<P,L>,L>{
  63. private readonly viewer: Viewer;
  64. private readonly innerSelectionFlag: DataContainer<boolean>;
  65. private readonly innerReprChangeFlag: DataContainer<boolean>;
  66. private readonly modelMapManager: ViewerModelMapManagerInterface<LoadMolstarInterface<P,L>,L>;
  67. private readonly loadingFlag: DataContainer<boolean>;
  68. constructor(config:{viewer: Viewer;modelMapManager: ViewerModelMapManagerInterface<LoadMolstarInterface<P,L>,L>;innerSelectionFlag: DataContainer<boolean>; innerReprChangeFlag: DataContainer<boolean>; loadingFlag: DataContainer<boolean>;}) {
  69. this.viewer = config.viewer;
  70. this.modelMapManager = config.modelMapManager;
  71. this.innerSelectionFlag = config.innerSelectionFlag;
  72. this.innerReprChangeFlag = config.innerReprChangeFlag;
  73. this.loadingFlag = config.loadingFlag;
  74. }
  75. async load(loadConfig: LoadMolstarInterface<P,L>): Promise<L|undefined>;
  76. async load(loadConfig: LoadMolstarInterface<P,L>[]): Promise<(L|undefined)[]>;
  77. async load(loadConfig: LoadMolstarInterface<P,L>|LoadMolstarInterface<P,L>[]): Promise<L|undefined|(L|undefined)[]>{
  78. this.loadingFlag.set(true);
  79. const out: (L|undefined)[] = [];
  80. for (const lC of (Array.isArray(loadConfig) ? loadConfig : [loadConfig])) {
  81. if(checkLoadData(lC)) {
  82. if (lC.loadMethod == LoadMethod.loadPdbId) {
  83. const config: LoadParams<P,L> = lC.loadParams as LoadParams<P,L>;
  84. out.push(await this.viewer.loadPdbId(config.entryId!, {props: config.props, matrix: config.matrix, reprProvider: config.reprProvider, params: config.params}) as L|undefined);
  85. } else if (lC.loadMethod == LoadMethod.loadStructureFromUrl) {
  86. const config: LoadParams<P,L> = lC.loadParams as LoadParams<P,L>;
  87. out.push(await this.viewer.loadStructureFromUrl(config.url!, config.format!, config.isBinary!,{props: config.props, matrix: config.matrix, reprProvider: config.reprProvider, params: config.params}) as L|undefined);
  88. // DebugUtil.log('LOAD STRUCT. URL RESULT:', out[0]);
  89. // const pdbtmDescriptor = TmDetDescriptorCache.get('1afo'); // TODO
  90. // let matrix = undefined;
  91. // if (pdbtmDescriptor) {
  92. // matrix = DebugUtil.descriptorMxToMat4(pdbtmDescriptor!.additional_entry_annotations.membrane.transformation_matrix as any);
  93. // }
  94. // this.viewer.plugin.managers.structure.hierarchy.applyPreset()
  95. // const selector = await this.viewer.plugin.builders.structure.hierarchy.applyPreset(
  96. // trajectory, 'default', { representationPreset: TMDET_STRUCTURE_PRESET_ID as any });
  97. // if (matrix && selector?.structureProperties) {
  98. // const params = {
  99. // transform: {
  100. // name: 'matrix' as const,
  101. // params: { data: matrix, transpose: false }
  102. // }
  103. // };
  104. // setTimeout(async () => {
  105. // console.log(this.viewer.plugin);
  106. // // const b = this.plugin.state.data.build().to(selector.structureProperties)
  107. // // .insert(StateTransforms.Model.TransformStructureConformation, params);
  108. // // await this.plugin.runTask(this.plugin.state.data.updateTree(b));
  109. // },
  110. // 2000
  111. // );
  112. // }
  113. } else if (lC.loadMethod == LoadMethod.loadSnapshotFromUrl) {
  114. const config: LoadParams<P,L> = lC.loadParams as LoadParams<P,L>;
  115. await this.viewer.loadSnapshotFromUrl(config.url!, config.type!);
  116. } else if (lC.loadMethod == LoadMethod.loadStructureFromData) {
  117. const config: LoadParams<P,L> = lC.loadParams as LoadParams<P,L>;
  118. out.push(await this.viewer.loadStructureFromData(config.data!, config.format!, config.isBinary!, {props: config.props, matrix: config.matrix, reprProvider: config.reprProvider, params: config.params}) as L|undefined);
  119. }
  120. const trajectory = out[out.length-1];
  121. if(trajectory)
  122. this.modelMapManager.add(lC,trajectory);
  123. }
  124. }
  125. this.loadingFlag.set(false);
  126. return out.length == 1 ? out[0] : out;
  127. }
  128. async removeStructure(loadConfig: LoadMolstarInterface<P,L>|Array<LoadMolstarInterface<P,L>>): Promise<void>{
  129. loadConfig = Array.isArray(loadConfig) ? loadConfig : [loadConfig];
  130. loadConfig.forEach(lC=>{
  131. (Array.isArray(lC.loadParams) ? lC.loadParams : [lC.loadParams]).forEach(loadParams=>{
  132. if(typeof loadParams.id === "string") {
  133. const pdbStr: StructureRef | undefined = this.viewer.plugin.managers.structure.hierarchy.current.structures.find(s => s.properties?.cell?.obj?.data?.units[0]?.model?.id == this.modelMapManager.getModelId(loadParams.id!));
  134. if (pdbStr) {
  135. this.viewer.plugin.managers.structure.hierarchy.remove([pdbStr]);
  136. }
  137. }
  138. });
  139. })
  140. }
  141. public select(modelId:string, labelAsymId: string, begin: number, end: number, mode: 'select'|'hover', operation:'add'|'set', operatorName?:string): void;
  142. public select(selection: Array<SaguaroPosition>, mode: 'select'|'hover', operation:'add'|'set'): void;
  143. public select(selection: Array<SaguaroRange>, mode: 'select'|'hover', operation:'add'|'set'): void;
  144. public select(...args: any[]): void{
  145. if(args[5] != undefined){
  146. this.selectRange(args[0],args[1],args[2],args[3],args[4],args[5],args[6]);
  147. }else if(Array.isArray(args[0]) && args[0].length > 0 && typeof args[0][0].position === 'number'){
  148. this.selectSet(args[0],args[1],args[2]);
  149. }else if(Array.isArray(args[0]) && args[0].length > 0 && typeof args[0][0].begin === 'number'){
  150. this.selectMultipleRanges(args[0],args[1],args[2]);
  151. }
  152. }
  153. private selectRange(modelId:string, labelAsymId: string, begin: number, end: number, mode: 'select'|'hover', operation:'add'|'set', operatorName?:string): void {
  154. if(mode == null || mode === 'select') {
  155. this.innerSelectionFlag.set(true);
  156. }
  157. this.viewer.select({modelId:this.modelMapManager.getModelId(modelId), labelAsymId: labelAsymId, labelSeqRange:{beg: begin, end:end}, operatorName: operatorName}, mode,operation);
  158. this.innerSelectionFlag.set(false);
  159. }
  160. private selectSet(selection: Array<SaguaroPosition>, mode: 'select'|'hover', operation:'add'|'set'): void {
  161. if(mode == null || mode === 'select') {
  162. this.innerSelectionFlag.set(true);
  163. }
  164. this.viewer.select(selection.map(r=>({modelId: this.modelMapManager.getModelId(r.modelId), labelSeqId:r.position, labelAsymId: r.labelAsymId, operatorName: r.operatorName})), mode, operation);
  165. this.innerSelectionFlag.set(false);
  166. }
  167. private selectMultipleRanges(selection: Array<SaguaroRange>, mode: 'select'|'hover', operation:'add'|'set'): void {
  168. if(mode == null || mode === 'select') {
  169. this.innerSelectionFlag.set(true);
  170. }
  171. this.viewer.select(selection.map(r=>({modelId: this.modelMapManager.getModelId(r.modelId), labelAsymId: r.labelAsymId, labelSeqRange:{beg:r.begin, end: r.end}, operatorName: r.operatorName})), mode, operation);
  172. this.innerSelectionFlag.set(false);
  173. }
  174. public async clear(): Promise<void>{
  175. await this.viewer.clear();
  176. }
  177. public clearSelection(mode:'select'|'hover', option?:SaguaroChain): void {
  178. if(mode === 'select') {
  179. this.viewer.clearFocus();
  180. this.innerSelectionFlag.set(true);
  181. }
  182. if(option != null)
  183. this.viewer.clearSelection(mode, {...option, modelId: this.modelMapManager.getModelId(option.modelId)});
  184. else
  185. this.viewer.clearSelection(mode);
  186. this.innerSelectionFlag.set(false);
  187. }
  188. public setFocus(modelId: string, labelAsymId: string, begin: number, end: number, operatorName?:string): void{
  189. this.viewer.setFocus({modelId: this.modelMapManager.getModelId(modelId), labelAsymId: labelAsymId, labelSeqRange:{beg:begin, end: end}, operatorName: operatorName});
  190. }
  191. public clearFocus(): void {
  192. this.viewer.clearFocus();
  193. }
  194. public cameraFocus(modelId: string, labelAsymId: string, positions:Array<number>, operatorName?:string): void;
  195. public cameraFocus(modelId: string, labelAsymId: string, begin: number, end: number, operatorName?:string): void;
  196. public cameraFocus(...args: any[]): void{
  197. if(Array.isArray(args[2])){
  198. this.focusPositions(args[0],args[1],args[2],args[3]);
  199. }else{
  200. this.focusRange(args[0],args[1],args[2],args[3],args[4]);
  201. }
  202. }
  203. private focusRange(modelId: string, labelAsymId: string, begin: number, end: number, operatorName?:string): void{
  204. const seqIds: Array<number> = new Array<number>();
  205. for(let n = begin; n <= end; n++){
  206. seqIds.push(n);
  207. }
  208. this.focusPositions(modelId, labelAsymId, seqIds, operatorName);
  209. }
  210. private focusPositions(modelId: string, labelAsymId: string, positions:Array<number>, operatorName?:string): void{
  211. const structure: Structure | undefined = getStructureWithModelId(this.viewer.plugin.managers.structure.hierarchy.current.structures, this.modelMapManager.getModelId(modelId));
  212. if (structure == null) return;
  213. const chainTests: Expression[] = [MS.core.rel.eq([MS.ammp('label_asym_id'), labelAsymId])];
  214. if(operatorName)
  215. chainTests.push(MS.core.rel.eq([operatorName, MS.acp('operatorName')]));
  216. const sel: StructureSelection = Script.getStructureSelection(Q => Q.struct.generator.atomGroups({
  217. 'chain-test': Q.core.logic.and(chainTests),
  218. 'residue-test': Q.core.set.has([MS.set(...SetUtils.toArray(new Set(positions))), MS.ammp('label_seq_id')])
  219. }), structure);
  220. const loci: Loci = StructureSelection.toLociWithSourceUnits(sel);
  221. if(!StructureElement.Loci.isEmpty(loci)) {
  222. this.viewer.plugin.managers.camera.focusLoci(loci);
  223. }else{
  224. this.resetCamera();
  225. }
  226. }
  227. public async createComponent(componentLabel: string, modelId:string, labelAsymId: string, begin: number, end : number, representationType: StructureRepresentationRegistry.BuiltIn, operatorName?:string): Promise<void>;
  228. public async createComponent(componentLabel: string, modelId:string, labelAsymId: string, representationType: StructureRepresentationRegistry.BuiltIn, operatorName?:string): Promise<void>;
  229. public async createComponent(componentLabel: string, residues: Array<SaguaroPosition>, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
  230. public async createComponent(componentLabel: string, residues: Array<SaguaroRange>, representationType: StructureRepresentationRegistry.BuiltIn): Promise<void>;
  231. public async createComponent(...args: any[]): Promise<void> {
  232. this.innerReprChangeFlag.set(true);
  233. await this.removeComponent(args[0]);
  234. if(Array.isArray(args[1])){
  235. if( args[1].length > 0 ) {
  236. if(typeof args[1][0].position === "number"){
  237. await this.viewer.createComponent(args[0], args[1].map(r=>({modelId: this.modelMapManager.getModelId(r.modelId), labelAsymId: r.labelAsymId, labelSeqId: r.position, operatorName: r.operatorName})), args[2]);
  238. }else{
  239. await this.viewer.createComponent(args[0], args[1].map(r=>({modelId: this.modelMapManager.getModelId(r.modelId), labelAsymId: r.labelAsymId, labelSeqRange:{beg:r.begin, end: r.end}, operatorName: r.operatorName})), args[2]);
  240. }
  241. }
  242. }else if(args[5] != undefined){
  243. await this.viewer.createComponent(args[0], {modelId: this.modelMapManager.getModelId(args[1]), labelAsymId: args[2], labelSeqRange:{beg:args[3], end:args[4]}, operatorName: args[6]}, args[5]);
  244. }else{
  245. await this.viewer.createComponent(args[0], {modelId: this.modelMapManager.getModelId(args[1]), labelAsymId:args[2], operatorName: args[4]}, args[3]);
  246. }
  247. this.innerReprChangeFlag.set(false);
  248. }
  249. public isComponent(componentLabel: string): boolean{
  250. for(const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups){
  251. for(const comp of c){
  252. if(comp.cell.obj?.label === componentLabel) {
  253. return true;
  254. }
  255. }
  256. }
  257. return false;
  258. }
  259. public async colorComponent(componentLabel: string, color: ColorTheme.BuiltIn): Promise<void>{
  260. for(const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups){
  261. for(const comp of c){
  262. if(comp.cell.obj?.label === componentLabel) {
  263. await this.viewer.plugin.managers.structure.component.updateRepresentationsTheme([comp], { color: color });
  264. return;
  265. }
  266. }
  267. }
  268. }
  269. public getComponentSet(): Set<string>{
  270. const out: Set<string> = new Set<string>();
  271. this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups.forEach((c)=>{
  272. for(const comp of c){
  273. if(comp.cell.obj?.label != null && out.has(comp.cell.obj?.label)) {
  274. break;
  275. }else if(comp.cell.obj?.label != null){
  276. out.add(comp.cell.obj?.label);
  277. }
  278. }
  279. });
  280. return out;
  281. }
  282. public async removeComponent(componentLabel?: string): Promise<void>{
  283. if(componentLabel == null){
  284. this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups.forEach(c=>this.viewer.plugin.managers.structure.hierarchy.remove(c))
  285. }else{
  286. await this.viewer.removeComponent(componentLabel);
  287. }
  288. }
  289. public displayComponent(componentLabel: string): boolean;
  290. public displayComponent(componentLabel: string, visibilityFlag: boolean): void;
  291. public displayComponent(componentLabel: string, visibilityFlag?: boolean): void|boolean {
  292. if(typeof visibilityFlag === 'boolean')
  293. return this.changeComponentDisplay(componentLabel, visibilityFlag);
  294. else
  295. return this.getComponentDisplay(componentLabel);
  296. }
  297. private changeComponentDisplay(componentLabel: string, visibilityFlag: boolean): void{
  298. this.innerReprChangeFlag.set(true);
  299. for (const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups) {
  300. for (const comp of c) {
  301. if(comp.cell.obj?.label === componentLabel) {
  302. if(!comp.cell.state.isHidden != visibilityFlag) {
  303. this.viewer.plugin.managers.structure.component.toggleVisibility(c);
  304. return void 0;
  305. }
  306. }
  307. }
  308. }
  309. this.innerReprChangeFlag.set(false);
  310. }
  311. private getComponentDisplay(componentLabel: string): boolean | undefined{
  312. for (const c of this.viewer.plugin.managers.structure.hierarchy.currentComponentGroups) {
  313. for (const comp of c) {
  314. if(comp.cell.obj?.label === componentLabel) {
  315. return !comp.cell.state.isHidden;
  316. }
  317. }
  318. }
  319. return false;
  320. }
  321. public resetCamera(): void {
  322. this.viewer.plugin.managers.camera.reset();
  323. }
  324. public async exportLoadedStructures(): Promise<void> {
  325. await this.viewer.exportLoadedStructures();
  326. }
  327. }
  328. function getStructureWithModelId(structures: StructureRef[], modelId: string): Structure|undefined{
  329. for(const structure of structures){
  330. if(!structure.cell?.obj?.data?.units)
  331. continue;
  332. const unit = structure.cell.obj.data.units[0];
  333. const id:string = unit.model.id;
  334. if(id === modelId)
  335. return structure.cell.obj.data
  336. }
  337. }
  338. function checkLoadData<P,S>(loadConfig: LoadMolstarInterface<P,S>): boolean{
  339. const method: LoadMethod = loadConfig.loadMethod;
  340. const params: LoadParams<P,S> | Array<LoadParams<P,S>> = loadConfig.loadParams;
  341. if( method == LoadMethod.loadPdbId ){
  342. if(params instanceof Array || params.entryId == null)
  343. throw loadConfig.loadMethod+": missing pdbId";
  344. }else if( method == LoadMethod.loadStructureFromUrl ){
  345. if(params instanceof Array || params.url == null || params.isBinary == null || params.format == null)
  346. throw loadConfig.loadMethod+": arguments needed url, format, isBinary"
  347. }else if( method == LoadMethod.loadSnapshotFromUrl ){
  348. if(params instanceof Array || params.url == null || params.type == null)
  349. throw loadConfig.loadMethod+": arguments needed url, type"
  350. }else if( method == LoadMethod.loadStructureFromData ){
  351. if(params instanceof Array || params.data == null || params.format == null || params.isBinary == null)
  352. throw loadConfig.loadMethod+": arguments needed data, format, isBinary"
  353. }
  354. return true;
  355. }