CustomView.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. import {asyncScheduler} from "rxjs";
  2. import {AbstractView, AbstractViewInterface} from "../AbstractView";
  3. import {
  4. RcsbFvBoardConfigInterface,
  5. RcsbFvRowConfigInterface,
  6. RcsbFv,
  7. RcsbFvTrackDataElementInterface
  8. } from "@rcsb/rcsb-saguaro";
  9. import * as React from "react";
  10. import {
  11. StructureViewerPublicInterface, ViewerActionManagerInterface, ViewerCallbackManagerInterface
  12. } from "../../../RcsbFvStructure/StructureViewerInterface";
  13. import uniqid from "uniqid";
  14. import {RcsbFvStateInterface} from "../../../RcsbFvState/RcsbFvStateInterface";
  15. export type CustomViewStateInterface<R,L> = Omit<Omit<CustomViewInterface<R,L>, "modelChangeCallback">, "structureViewer">;
  16. export interface CustomViewInterface<R,L> {
  17. customViewContainerId?: string;
  18. blockConfig: FeatureBlockInterface<R,L> | Array<FeatureBlockInterface<R,L>>;
  19. blockSelectorElement?: (blockSelector: BlockSelectorManager) => JSX.Element;
  20. modelChangeCallback?: () => CustomViewStateInterface<R,L>;
  21. blockChangeCallback?: (plugin: StructureViewerPublicInterface<R,L>, pfvList: Array<RcsbFv>, stateManager: RcsbFvStateInterface) => void;
  22. }
  23. export interface FeatureBlockInterface<R,L> {
  24. blockId:string;
  25. blockTitle?: string;
  26. blockShortName?: string;
  27. featureViewConfig: Array<FeatureViewInterface<R,L>> | FeatureViewInterface<R,L>;
  28. }
  29. export interface FeatureViewInterface<R,L> {
  30. boardId?:string;
  31. boardConfig: RcsbFvBoardConfigInterface;
  32. rowConfig: Array<RcsbFvRowConfigInterface>;
  33. sequenceSelectionChangeCallback: (plugin: StructureViewerPublicInterface<R,L>, stateManager: RcsbFvStateInterface, sequenceRegion: Array<RcsbFvTrackDataElementInterface>) => void;
  34. sequenceElementClickCallback: (plugin: StructureViewerPublicInterface<R,L>, stateManager: RcsbFvStateInterface, d: RcsbFvTrackDataElementInterface) => void;
  35. sequenceHoverCallback: (plugin: StructureViewerPublicInterface<R,L>, stateManager: RcsbFvStateInterface, hoverRegion: Array<RcsbFvTrackDataElementInterface>) => void;
  36. structureSelectionCallback: (plugin: StructureViewerPublicInterface<R,L>, pfv: RcsbFv, stateManager: RcsbFvStateInterface) => void;
  37. structureHoverCallback: (plugin: StructureViewerPublicInterface<R,L>, pfv: RcsbFv, stateManager: RcsbFvStateInterface) => void;
  38. }
  39. export class BlockSelectorManager {
  40. private blockId: string;
  41. private previousBlockId: string;
  42. private readonly blockChangeCallback: ()=>void = ()=>{};
  43. constructor(f:()=>void){
  44. this.blockChangeCallback = f;
  45. }
  46. setActiveBlock(blockId:string): void{
  47. this.previousBlockId = this.blockId;
  48. this.blockId = blockId;
  49. this.blockChangeCallback();
  50. }
  51. getActiveBlock(): string{
  52. return this.blockId;
  53. }
  54. getPreviousBlock(): string{
  55. return this.previousBlockId;
  56. }
  57. }
  58. export class CustomView<R,L> extends AbstractView<CustomViewInterface<R,L> & {structureViewer: ViewerActionManagerInterface<R,L>;}, CustomViewStateInterface<R,L>> {
  59. private blockViewSelector: BlockSelectorManager = new BlockSelectorManager( this.blockChange.bind(this) );
  60. private boardMap: Map<string, FeatureViewInterface<R,L>> = new Map<string, FeatureViewInterface<R,L>>();
  61. private blockMap: Map<string, Array<string>> = new Map<string, Array<string>>();
  62. private rcsbFvMap: Map<string, RcsbFv> = new Map<string, RcsbFv>();
  63. private firstModelLoad: boolean = true;
  64. private innerSelectionFlag: boolean = false;
  65. private updateContext:"state-change"|null = null;
  66. readonly state: CustomViewStateInterface<R,L> = {
  67. customViewContainerId: this.props.customViewContainerId,
  68. blockConfig: this.props.blockConfig,
  69. blockSelectorElement: this.props.blockSelectorElement,
  70. blockChangeCallback: this.props.blockChangeCallback
  71. };
  72. constructor(props: CustomViewInterface<R,L> & {structureViewer: ViewerActionManagerInterface<R,L>;} & AbstractViewInterface) {
  73. super(props);
  74. this.mapBlocks(props.blockConfig);
  75. }
  76. render():JSX.Element {
  77. return (
  78. this.props.customViewContainerId ? <div id={this.rcsbFvDivId}></div> : super.render()
  79. );
  80. }
  81. componentDidMount(): void {
  82. super.componentDidMount();
  83. this.blockViewSelector.setActiveBlock( (this.state.blockConfig instanceof Array ? this.state.blockConfig : [this.state.blockConfig])[0].blockId! );
  84. }
  85. componentWillUnmount() {
  86. super.componentWillUnmount();
  87. this.rcsbFvMap.forEach((pfv,id)=>{
  88. pfv.unmount();
  89. });
  90. }
  91. componentDidUpdate(prevProps: Readonly<CustomViewInterface<R,L> & AbstractViewInterface>, prevState: Readonly<CustomViewStateInterface<R,L>>, snapshot?: any) {
  92. if(this.updateContext != "state-change") {
  93. this.updateContext = "state-change";
  94. this.mapBlocks(this.props.blockConfig);
  95. this.setState( {
  96. blockConfig: this.props.blockConfig,
  97. blockSelectorElement: this.props.blockSelectorElement,
  98. blockChangeCallback: this.props.blockChangeCallback
  99. });
  100. }
  101. }
  102. private mapBlocks(config: FeatureBlockInterface<R,L> | Array<FeatureBlockInterface<R,L>>){
  103. this.rcsbFvMap.forEach((pfv, id) => {
  104. pfv.unmount();
  105. });
  106. this.blockMap.clear();
  107. this.boardMap.clear();
  108. ( config instanceof Array ? config : [config]).forEach(block=>{
  109. if(block.blockId == null)block.blockId = uniqid("block_");
  110. if(!this.blockMap.has(block.blockId))this.blockMap.set(block.blockId, new Array<string>());
  111. (block.featureViewConfig instanceof Array ? block.featureViewConfig : [block.featureViewConfig]).forEach(board=>{
  112. if(board.boardId == null)board.boardId = uniqid("board_");
  113. this.blockMap.get(block.blockId!)?.push(board.boardId);
  114. this.boardMap.set(board.boardId, board);
  115. });
  116. });
  117. }
  118. private blockChange(): void{
  119. this.unmountBlockFv();
  120. this.buildBlockFv();
  121. asyncScheduler.schedule(()=>{
  122. if(typeof this.state.blockChangeCallback === "function")
  123. this.state.blockChangeCallback(this.props.structureViewer, Array.from(this.blockMap.get(this.blockViewSelector.getActiveBlock())!.values()).map(boardId=>(this.rcsbFvMap.get(boardId)!)), this.props.stateManager);
  124. else
  125. this.structureSelectionCallback();
  126. },1000);
  127. }
  128. private unmountBlockFv(){
  129. this.blockMap.get(this.blockViewSelector.getPreviousBlock())?.forEach(boardId=>{
  130. if(this.rcsbFvMap.get(boardId) == null)
  131. return;
  132. this.rcsbFvMap.get(boardId)!.unmount();
  133. document.getElementById("boardDiv_"+boardId)?.remove()
  134. });
  135. this.rcsbFvMap.clear();
  136. }
  137. private buildBlockFv(){
  138. this.blockMap.get(this.blockViewSelector.getActiveBlock())?.forEach(boardId=>{
  139. if(this.boardMap.get(boardId) == null)
  140. return;
  141. const div: HTMLDivElement = document.createElement<"div">("div");
  142. div.setAttribute("id", "boardDiv_"+boardId);
  143. document.getElementById(this.componentDivId)?.append(div);
  144. const width: number = window.document.getElementById(this.componentDivId)?.getBoundingClientRect().width ?? 0;
  145. const trackWidth: number = width - (this.boardMap.get(boardId)!.boardConfig?.rowTitleWidth ?? 190) - 55;
  146. const rcsbFv: RcsbFv = new RcsbFv({
  147. elementId: "boardDiv_"+boardId,
  148. boardConfigData:{
  149. highlightHoverPosition:true,
  150. highlightHoverElement:true,
  151. ...this.boardMap.get(boardId)!.boardConfig,
  152. trackWidth:this.boardMap.get(boardId)!.boardConfig?.trackWidth ? this.boardMap.get(boardId)!.boardConfig?.trackWidth!-4 : trackWidth,
  153. selectionChangeCallBack:(selection: RcsbFvTrackDataElementInterface[])=>{
  154. if(this.innerSelectionFlag)
  155. return;
  156. this.boardMap.get(boardId)!.sequenceSelectionChangeCallback(this.props.structureViewer, this.props.stateManager, selection);
  157. },
  158. highlightHoverCallback:(elements:Array<RcsbFvTrackDataElementInterface>)=>{
  159. this.boardMap.get(boardId)!.sequenceHoverCallback(this.props.structureViewer, this.props.stateManager, elements);
  160. },
  161. elementClickCallBack: (element: RcsbFvTrackDataElementInterface)=>{
  162. this.boardMap.get(boardId)!.sequenceElementClickCallback(this.props.structureViewer, this.props.stateManager, element);
  163. }
  164. },
  165. rowConfigData: this.boardMap.get(boardId)!.rowConfig
  166. });
  167. this.rcsbFvMap.set(boardId, rcsbFv);
  168. });
  169. /*this.props.structureViewer.setSelectCallback(()=>{
  170. this.structureSelectionCallback();
  171. });*/
  172. }
  173. structureSelectionCallback(): void {
  174. this.innerSelectionFlag = true;
  175. this.blockMap.get(this.blockViewSelector.getActiveBlock())?.forEach(boardId=>{
  176. const pfv: RcsbFv | undefined = this.rcsbFvMap.get(boardId);
  177. if(pfv == null)
  178. return;
  179. this.boardMap.get(boardId)?.structureSelectionCallback(this.props.structureViewer, pfv, this.props.stateManager);
  180. });
  181. this.innerSelectionFlag = false;
  182. }
  183. structureHoverCallback(): void{
  184. this.blockMap.get(this.blockViewSelector.getActiveBlock())?.forEach(boardId=>{
  185. const pfv: RcsbFv | undefined = this.rcsbFvMap.get(boardId);
  186. if(pfv == null)
  187. return;
  188. this.boardMap.get(boardId)?.structureHoverCallback(this.props.structureViewer, pfv, this.props.stateManager);
  189. });
  190. }
  191. representationChangeCallback(): void{
  192. //TODO
  193. }
  194. additionalContent(): JSX.Element {
  195. if(this.state.blockSelectorElement == null)
  196. return <></>;
  197. return this.state.blockSelectorElement(this.blockViewSelector);
  198. }
  199. modelChangeCallback(): void {
  200. if(this.firstModelLoad){
  201. this.firstModelLoad = false;
  202. return;
  203. }
  204. if(typeof this.props.modelChangeCallback === "function") {
  205. let newConfig: CustomViewStateInterface<R,L> = this.props.modelChangeCallback();
  206. if(newConfig != null ){
  207. this.updateContext = "state-change";
  208. if(newConfig.blockConfig != null && newConfig.blockSelectorElement != null){
  209. this.mapBlocks(newConfig.blockConfig);
  210. this.setState({blockConfig: newConfig.blockConfig, blockSelectorElement: newConfig.blockSelectorElement})
  211. }else if(newConfig.blockConfig == null && newConfig.blockSelectorElement != null){
  212. this.setState({blockSelectorElement: newConfig.blockSelectorElement})
  213. }else if(newConfig.blockConfig != null && newConfig.blockSelectorElement == null){
  214. this.mapBlocks(newConfig.blockConfig);
  215. this.setState({blockConfig: newConfig.blockConfig})
  216. }
  217. }
  218. }
  219. }
  220. setState<K extends keyof CustomViewStateInterface<R,L>>(
  221. state: ((
  222. prevState: Readonly<CustomViewStateInterface<R,L>>,
  223. props: Readonly<CustomViewInterface<R,L> & AbstractViewInterface>
  224. ) => (Pick<CustomViewStateInterface<R,L>, K> | CustomViewStateInterface<R,L> | null)) | Pick<CustomViewStateInterface<R,L>, K> | CustomViewStateInterface<R,L> | null, callback?: () => void
  225. ) {
  226. super.setState(state, ()=>{
  227. this.blockViewSelector.setActiveBlock( (this.state.blockConfig instanceof Array ? this.state.blockConfig : [this.state.blockConfig])[0].blockId! )
  228. if(typeof callback === "function") callback();
  229. this.updateContext = null
  230. });
  231. }
  232. updateDimensions(): void {
  233. const div: HTMLElement | undefined | null = document.getElementById(this.componentDivId)?.parentElement;
  234. const width: number = window.document.getElementById(this.componentDivId)?.getBoundingClientRect().width ?? 0;
  235. if(div == null || (div.style.width && !div.style.width.includes("%")) )
  236. return;
  237. this.rcsbFvMap.forEach((rcsbFv, boardId)=>{
  238. const trackWidth: number = width - (this.boardMap.get(boardId)!.boardConfig?.rowTitleWidth ?? 190) - 55;
  239. rcsbFv.updateBoardConfig({boardConfigData:{trackWidth:this.boardMap.get(boardId)!.boardConfig?.trackWidth ? this.boardMap.get(boardId)!.boardConfig?.trackWidth!-4 : trackWidth}});
  240. });
  241. }
  242. }