TmFv1DComponent.tsx 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. import React from 'react';
  2. import { Root, createRoot } from 'react-dom/client';
  3. import { updateSiteColors } from '../tmdet-extension/tmdet-color-theme';
  4. import { RcsbFv, RcsbFvBoardConfigInterface } from '@rcsb/rcsb-saguaro';
  5. import { fetchDescriptor } from '../UniTmpHelper';
  6. import { setExternalRowTitleComponent, setRegionColors } from '../FeatureViewConfig';
  7. import { TmRowTitleComponent } from './TmRowTitleComponent';
  8. const defaultViewerWidth = 600;
  9. const defaultRowTitleWidth = 100;
  10. const ULTIMATE_GAP_CONSTANT = 4; // I do not know why it is 4.
  11. type TmFv1DParams = {
  12. elementId: string,
  13. url: string,
  14. side1: string,
  15. trackWidth?: number,
  16. rowTitleWidth?: number,
  17. autoResize?: boolean
  18. };
  19. type TmFv1DState = {
  20. featureViewer?: RcsbFv,
  21. num?: number
  22. };
  23. export class TmFv1D extends React.Component<TmFv1DParams, TmFv1DState> {
  24. protected containerRef = React.createRef<HTMLDivElement>();
  25. public constructor(props: TmFv1DParams) {
  26. super(props);
  27. }
  28. public async componentDidMount(): Promise<void> {
  29. await createRcsbFeatureViewer({
  30. elementId: this.props.elementId,
  31. url: this.props.url,
  32. side1: this.props.side1,
  33. rowTitleWidth: this.props.rowTitleWidth,
  34. trackWidth: this.props.trackWidth
  35. });
  36. if (this.props.autoResize) {
  37. new ResizeObserver(this.resize)
  38. .observe(this.containerRef.current as Element);
  39. }
  40. }
  41. resize(entries: ResizeObserverEntry[], observer: ResizeObserver) {
  42. const currentId = entries[0].target.id;
  43. const observerEntry = entries[0];
  44. const featureViewer = featureViewerRegistry.get(currentId) as RcsbFv;
  45. const boardConfig = featureViewer.getBoardConfig();
  46. boardConfig.trackWidth = observerEntry.contentRect.width - boardConfig.rowTitleWidth!;
  47. resizeBoardConfigTrackWidth(boardConfig, observerEntry.contentRect.width);
  48. featureViewer.updateBoardConfig({ boardConfigData: boardConfig });
  49. }
  50. public render(): JSX.Element {
  51. return (
  52. <div ref={this.containerRef}
  53. id={ this.props.elementId }
  54. style={{ borderWidth: "0px", padding: "0px" }}
  55. />
  56. );
  57. }
  58. }
  59. export class TmFv1DElement extends HTMLElement {
  60. connectedCallback() {
  61. const root = createRoot(this);
  62. setTimeout(() => this.render(root), 1000);
  63. }
  64. render(root: Root) {
  65. let autoResize = true;
  66. if (this.hasAttribute("autoResize")) {
  67. const value = this.getAttribute("autoResize");
  68. autoResize = (value == "true");
  69. }
  70. const elementId = this.getAttribute("elementId")!;
  71. const url = this.getAttribute("url")!;
  72. const side1 = this.getAttribute("side1")!;
  73. const calculatedWidth = calculateViewerWidth(this.parentElement);
  74. const rowTitleWidth = parseInt(this.getAttribute("rowTitleWidth")!)
  75. || defaultRowTitleWidth;
  76. let trackWidth = !autoResize
  77. ? parseInt(this.getAttribute("trackWidth")!) || (defaultViewerWidth - ULTIMATE_GAP_CONSTANT)
  78. : calculatedWidth;
  79. trackWidth -= rowTitleWidth;
  80. console.log('Widths:', [ rowTitleWidth, trackWidth ]);
  81. root.render(
  82. <TmFv1D
  83. elementId={ elementId }
  84. url={ url }
  85. side1={ side1 }
  86. trackWidth={ trackWidth }
  87. rowTitleWidth={ rowTitleWidth }
  88. autoResize={ autoResize }
  89. />
  90. );
  91. }
  92. }
  93. function calculateViewerWidth(parent: HTMLElement|null) {
  94. if (!parent) {
  95. return defaultViewerWidth;
  96. }
  97. const boundingRect = parent.getBoundingClientRect();
  98. const computedStyle = window.getComputedStyle(parent);
  99. const extraWidth = parseInt(computedStyle.paddingLeft)
  100. + parseInt(computedStyle.paddingRight)
  101. + parseInt(computedStyle.borderLeftWidth)
  102. + parseInt(computedStyle.borderRightWidth);
  103. return boundingRect.width - extraWidth - ULTIMATE_GAP_CONSTANT;
  104. }
  105. const featureViewerRegistry = new Map();
  106. /**
  107. * Create an Rcsb Saguaro 1D viewer instance.
  108. *
  109. * @param elementId id of parent element
  110. * @param url source of JSON config
  111. * @param side1 side1 paramter to update site colors, if it is needed
  112. */
  113. export async function createRcsbFeatureViewer(params: {
  114. elementId: string, url: string, side1: string, trackWidth?: number, rowTitleWidth?: number }): Promise<RcsbFv> {
  115. updateSiteColors(params.side1 as any);
  116. const featureTracks = await fetchDescriptor(params.url);
  117. const blockConfig = featureTracks.blockConfig;
  118. const boardConfig = blockConfig[0].featureViewConfig[0].boardConfig;
  119. boardConfig.trackWidth = params.trackWidth ?? 600;
  120. boardConfig.rowTitleWidth = params.rowTitleWidth ?? 160;
  121. const rowConfig = blockConfig[0].featureViewConfig[0].rowConfig;
  122. setRegionColors(rowConfig);
  123. setExternalRowTitleComponent(rowConfig);
  124. const pfv = new RcsbFv({
  125. boardConfigData: boardConfig,
  126. rowConfigData: rowConfig,
  127. elementId: params.elementId
  128. });
  129. featureViewerRegistry.set(params.elementId, pfv);
  130. return pfv;
  131. }
  132. function resizeBoardConfigTrackWidth(boardConfig: RcsbFvBoardConfigInterface, width: number) {
  133. const fullWidth = width - ULTIMATE_GAP_CONSTANT;
  134. boardConfig.trackWidth = fullWidth - boardConfig.rowTitleWidth!;
  135. }