AssemblyPfvFactory.tsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. import {
  2. RcsbFvModulePublicInterface
  3. } from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/RcsbFvModule/RcsbFvModuleInterface";
  4. import {
  5. ChainInfo, OperatorInfo,
  6. SaguaroPluginInterface,
  7. SaguaroPluginModelMapType
  8. } from "../../../../RcsbFvStructure/SaguaroPluginInterface";
  9. import {
  10. InstanceSequenceConfig
  11. } from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/RcsbFvBuilder/RcsbFvInstanceBuilder";
  12. import {asyncScheduler} from "rxjs";
  13. import {buildInstanceSequenceFv, FeatureType, RcsbFvUI, RcsbRequestContextManager} from "@rcsb/rcsb-saguaro-app";
  14. import {RcsbFvDOMConstants} from "../../../../RcsbFvConstants/RcsbFvConstants";
  15. import {SelectOptionProps} from "@rcsb/rcsb-saguaro-app/build/dist/RcsbFvWeb/WebTools/SelectButton";
  16. import {ChainDisplay} from "./ChainDisplay";
  17. import * as React from "react";
  18. import {AnnotationFeatures, Source, Type} from "@rcsb/rcsb-api-tools/build/RcsbGraphQL/Types/Borrego/GqlTypes";
  19. import {
  20. PolymerEntityInstanceInterface
  21. } from "@rcsb/rcsb-saguaro-app/build/dist/RcsbCollectTools/DataCollectors/PolymerEntityInstancesCollector";
  22. import {
  23. InterfaceInstanceTranslate
  24. } from "@rcsb/rcsb-saguaro-app/build/dist/RcsbUtils/Translators/InterfaceInstanceTranslate";
  25. import {DataContainer} from "../../../../Utils/DataContainer";
  26. import {BuildPfvInterface, PfvAbstractFactory, PfvFactoryConfigInterface} from "../PfvFactoryInterface";
  27. interface AssemblyPfvFactoryInterface extends PfvFactoryConfigInterface{
  28. useOperatorsFlag:boolean | undefined;
  29. instanceSequenceConfig: InstanceSequenceConfig | undefined;
  30. }
  31. export class AssemblyPfvFactory extends PfvAbstractFactory<{instanceSequenceConfig: InstanceSequenceConfig|undefined}> {
  32. private readonly instanceSequenceConfig: InstanceSequenceConfig|undefined;
  33. private readonly useOperatorsFlag:boolean | undefined;
  34. private readonly OPERATOR_DROPDOWN_TITLE: string = "Symmetry Partner";
  35. constructor(config: AssemblyPfvFactoryInterface) {
  36. super(config);
  37. this.instanceSequenceConfig = config.instanceSequenceConfig;
  38. this.useOperatorsFlag = config.useOperatorsFlag;
  39. }
  40. async getPfv(config: BuildPfvInterface): Promise<RcsbFvModulePublicInterface | undefined> {
  41. this.assemblyModelSate.setMap(config.modelMap);
  42. this.plugin.clearFocus();
  43. const onChangeCallback: Map<string, (x: PolymerEntityInstanceInterface)=>void> = new Map<string, (x: PolymerEntityInstanceInterface) => {}>();
  44. const assemblyInstances: Map<string, Set<string>> = new Map<string, Set<string>>();
  45. this.assemblyModelSate.forEach((v,k)=>{
  46. assemblyInstances.set(v.entryId,new Set<string>(v.chains.map(d=>d.label)));
  47. onChangeCallback.set(v.entryId,(x)=>{
  48. this.assemblyModelSate.set({entryId: v.entryId, labelAsymId: x.asymId, modelId: k});
  49. asyncScheduler.schedule(()=>{
  50. this.selectorManager.setLastSelection('select', null);
  51. this.pfvChangeCallback();
  52. },1000);
  53. });
  54. });
  55. const operatorNameContainer: DataContainer<string> = new DataContainer<string>(config.defaultOperatorName);
  56. let module: RcsbFvModulePublicInterface | undefined = undefined;
  57. if(this.assemblyModelSate.get("entryId") != null) {
  58. module = await buildInstanceSequenceFv(
  59. this.rcsbFvDivId,
  60. RcsbFvDOMConstants.SELECT_BUTTON_PFV_ID,
  61. this.assemblyModelSate.getString("entryId"),
  62. {
  63. ...this.instanceSequenceConfig,
  64. defaultValue: config.defaultAuthId,
  65. onChangeCallback: onChangeCallback.get(this.assemblyModelSate.getString("entryId")),
  66. beforeChangeCallback: (x: PolymerEntityInstanceInterface)=>{
  67. this.assemblyModelSate.set({entryId:x.entryId, labelAsymId: x.asymId});
  68. const entryMap:[string, {entryId: string, assemblyId: string, chains: ChainInfo[]}] | undefined = Array.from(this.assemblyModelSate.entries()).find((e)=>(e[1].entryId === x.entryId));
  69. if(!entryMap){
  70. throw `Error: no modelId was found for ${x.entryId}`;
  71. }
  72. const operator: OperatorInfo|undefined = getOperator(this.assemblyModelSate.getMap().get(entryMap[0])!, config.defaultAuthId, operatorNameContainer.get());
  73. this.addOperatorButton(operator?.name);
  74. this.assemblyModelSate.setOperator(x.asymId,operator?.name);
  75. operatorNameContainer.set(undefined);
  76. if(typeof this.additionalConfig?.operatorChangeCallback === "function" && this.assemblyModelSate.getOperator()){
  77. this.additionalConfig.operatorChangeCallback(this.assemblyModelSate.getOperator()!);
  78. }
  79. if((this.assemblyModelSate.getChainInfo()?.operators?.length ?? 0) > 1)
  80. return {
  81. operatorIds: operator?.ids
  82. }
  83. },
  84. filterInstances: assemblyInstances.get(this.assemblyModelSate.getString("entryId")),
  85. selectButtonOptionProps: (props: SelectOptionProps) => (
  86. <div style={{display: 'flex'}}>
  87. <ChainDisplay plugin={this.plugin} label={props.data.label}/>
  88. {props.children}
  89. </div>)
  90. },
  91. {
  92. ...this.additionalConfig,
  93. boardConfig: this.boardConfigContainer.get(),
  94. externalTrackBuilder:{
  95. filterFeatures: this.filterFeatures.bind(this)
  96. }
  97. }
  98. );
  99. }
  100. if(!config.defaultAuthId)
  101. await createComponents(this.plugin, this.assemblyModelSate.getMap());
  102. return module;
  103. }
  104. private addOperatorButton(operatorName?: string): void{
  105. const currentChainInfo: ChainInfo|undefined = this.assemblyModelSate.getChainInfo();
  106. if(this.useOperatorsFlag && currentChainInfo && currentChainInfo.operators.length >1 ){
  107. this.assemblyModelSate.setOperator(undefined,operatorName);
  108. RcsbFvUI.addSelectButton(
  109. this.rcsbFvDivId,
  110. RcsbFvDOMConstants.SELECT_BUTTON_PFV_ID,
  111. currentChainInfo.operators.map(op=>({
  112. label:`${op.ids.join("-")} (${op.name})`,
  113. optId:op.name,
  114. onChange: async ()=>{
  115. this.assemblyModelSate.set({operator:op});
  116. await this.getPfv({
  117. modelMap: this.assemblyModelSate.getMap(),
  118. defaultAuthId: this.assemblyModelSate.getChainInfo()?.auth,
  119. defaultOperatorName: op.name
  120. })
  121. }
  122. })),
  123. {
  124. defaultValue: this.assemblyModelSate.getOperator()?.name,
  125. dropdownTitle:this.OPERATOR_DROPDOWN_TITLE
  126. }
  127. );
  128. }
  129. }
  130. private filterFeatures(data: {annotations: Array<AnnotationFeatures>; rcsbContext:Partial<PolymerEntityInstanceInterface>}): Promise<Array<AnnotationFeatures>> {
  131. return new Promise<Array<AnnotationFeatures>>(async resolve => {
  132. let annotations: Array<AnnotationFeatures> = [];
  133. (await Promise.all(data.annotations.map(async ann=>{
  134. if(ann.source == Source.PdbInterface && ann.target_id && data.rcsbContext?.asymId) {
  135. const interfaceToInstance: InterfaceInstanceTranslate = await RcsbRequestContextManager.getInterfaceToInstance(ann.target_id);
  136. if(typeof ann.target_identifiers?.interface_partner_index === "number" && ann.target_identifiers.assembly_id === this.assemblyModelSate.getString("assemblyId")) {
  137. const operatorIds:string[][] = interfaceToInstance.getOperatorIds(ann.target_id)[ann.target_identifiers.interface_partner_index];
  138. if(ann.features && this.assemblyModelSate.getOperator() && operatorIds.map(o=>o.join("|")).includes( this.assemblyModelSate.getOperator()!.ids.join("|") )){
  139. ann.features = ann.features.filter(f=>(f && f.type == FeatureType.BurialFraction));
  140. if(ann.features.length > 0)
  141. return ann;
  142. }
  143. }
  144. }else if(ann.source == Source.PdbInstance && ann.features){
  145. ann.features = ann.features?.filter(f=>(f?.type!==Type.Asa));
  146. return ann;
  147. }else if(ann.source != Source.PdbInterface){
  148. return ann;
  149. }
  150. }))).forEach((value,index,array)=>{
  151. if(value)
  152. annotations = annotations.concat(value);
  153. });
  154. resolve(annotations);
  155. });
  156. }
  157. }
  158. async function createComponents(plugin: SaguaroPluginInterface, modelMap:SaguaroPluginModelMapType): Promise<void> {
  159. plugin.displayComponent("Water", false);
  160. await plugin.colorComponent("Polymer", 'chain-id');
  161. const chains: Array<{modelId: string; auth: string; label: string;}> = new Array<{modelId: string; auth: string; label: string;}>();
  162. modelMap.forEach((entry, modelId)=>{
  163. entry.chains.forEach(ch=>{
  164. if(ch.type === "polymer") {
  165. chains.push({modelId: modelId, auth: ch.auth, label: ch.label});
  166. }
  167. });
  168. });
  169. plugin.removeComponent();
  170. plugin.clearFocus();
  171. for(const ch of chains) {
  172. const label: string = ch.auth === ch.label ? ch.label : `${ch.label} [auth ${ch.auth}]`;
  173. await plugin.createComponent(label, ch.modelId, ch.label, 'cartoon');
  174. await plugin.colorComponent(label, 'chain-id');
  175. }
  176. await plugin.removeComponent("Polymer");
  177. }
  178. function getOperator(entryInfo: {entryId: string; assemblyId: string, chains:Array<ChainInfo>;}, defaultAuthId?: string, defaultOperatorName?:string): OperatorInfo | undefined{
  179. const chainInfo: ChainInfo | undefined = defaultAuthId ? entryInfo.chains.find(ch=>ch.auth === defaultAuthId) : entryInfo.chains[0];
  180. if(chainInfo){
  181. const operatorInfo: OperatorInfo | undefined = defaultOperatorName ? chainInfo.operators.find(op=>op.name === defaultOperatorName) : chainInfo.operators[0];
  182. if(operatorInfo)
  183. return operatorInfo;
  184. }
  185. return undefined;
  186. }