index.ts 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  1. /**
  2. * Copyright (c) 2019-2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. * @author Joan Segura <joan.segura@rcsb.org>
  6. * @author Yana Rose <yana.rose@rcsb.org>
  7. * @author Sebastian Bittrich <sebastian.bittrich@rcsb.org>
  8. */
  9. import { BehaviorSubject } from 'rxjs';
  10. import { Plugin } from 'molstar/lib/mol-plugin-ui/plugin';
  11. import { PluginCommands } from 'molstar/lib/mol-plugin/commands';
  12. import { ViewerState, CollapsedState, ModelUrlProvider, LigandUrlProvider, LigandViewerState, LoadParams } from './types';
  13. import { PluginSpec } from 'molstar/lib/mol-plugin/spec';
  14. import { ColorNames } from 'molstar/lib/mol-util/color/names';
  15. import * as React from 'react';
  16. import { ModelLoader } from './helpers/model';
  17. import { PresetProps } from './helpers/preset';
  18. import { ControlsWrapper } from './ui/controls';
  19. import { PluginConfig } from 'molstar/lib/mol-plugin/config';
  20. import { RCSBAssemblySymmetry } from 'molstar/lib/extensions/rcsb/assembly-symmetry/behavior';
  21. import { RCSBValidationReport } from 'molstar/lib/extensions/rcsb/validation-report/behavior';
  22. import { Mat4 } from 'molstar/lib/mol-math/linear-algebra';
  23. import { PluginState } from 'molstar/lib/mol-plugin/state';
  24. import { BuiltInTrajectoryFormat } from 'molstar/lib/mol-plugin-state/formats/trajectory';
  25. import { ObjectKeys } from 'molstar/lib/mol-util/type-helpers';
  26. import { PluginLayoutControlsDisplay } from 'molstar/lib/mol-plugin/layout';
  27. import { SuperposeColorThemeProvider } from './helpers/superpose/color';
  28. import { NakbColorThemeProvider } from './helpers/nakb/color';
  29. import { setFocusFromRange, removeComponent, clearSelection, createComponent, select } from './helpers/viewer';
  30. import { SelectBase, SelectRange, SelectTarget, Target } from './helpers/selection';
  31. import { StructureRepresentationRegistry } from 'molstar/lib/mol-repr/structure/registry';
  32. import { DefaultPluginUISpec, PluginUISpec } from 'molstar/lib/mol-plugin-ui/spec';
  33. import { PluginUIContext } from 'molstar/lib/mol-plugin-ui/context';
  34. import { ANVILMembraneOrientation, MembraneOrientationPreset } from 'molstar/lib/extensions/anvil/behavior';
  35. import { MembraneOrientationRepresentationProvider } from 'molstar/lib/extensions/anvil/representation';
  36. import { PluginContext } from 'molstar/lib/mol-plugin/context';
  37. import { TrajectoryHierarchyPresetProvider } from 'molstar/lib/mol-plugin-state/builder/structure/hierarchy-preset';
  38. import { AnimateStateSnapshots } from 'molstar/lib/mol-plugin-state/animation/built-in/state-snapshots';
  39. import { PluginFeatureDetection } from 'molstar/lib/mol-plugin/features';
  40. import { PresetStructureRepresentations } from 'molstar/lib/mol-plugin-state/builder/structure/representation-preset';
  41. import { MAQualityAssessment } from 'molstar/lib/extensions/model-archive/quality-assessment/behavior';
  42. import { ModelExport } from 'molstar/lib/extensions/model-export';
  43. import { exportHierarchy } from 'molstar/lib/extensions/model-export/export';
  44. import { GeometryExport } from 'molstar/lib/extensions/geo-export';
  45. import { Mp4Export } from 'molstar/lib/extensions/mp4-export';
  46. import { PartialCanvas3DProps } from 'molstar/lib/mol-canvas3d/canvas3d';
  47. import { RSCCScore } from './helpers/rscc/behavior';
  48. import { createRoot } from 'react-dom/client';
  49. import { AssemblySymmetry } from 'molstar/lib/extensions/rcsb/assembly-symmetry/prop';
  50. import { wwPDBChemicalComponentDictionary } from 'molstar/lib/extensions/wwpdb/ccd/behavior';
  51. import { ChemicalCompontentTrajectoryHierarchyPreset } from 'molstar/lib/extensions/wwpdb/ccd/representation';
  52. import { StateTransforms } from 'molstar/lib/mol-plugin-state/transforms';
  53. /** package version, filled in at bundle build time */
  54. declare const __RCSB_MOLSTAR_VERSION__: string;
  55. export const RCSB_MOLSTAR_VERSION = typeof __RCSB_MOLSTAR_VERSION__ != 'undefined' ? __RCSB_MOLSTAR_VERSION__ : 'none';
  56. /** unix time stamp, to be filled in at bundle build time */
  57. declare const __BUILD_TIMESTAMP__: number;
  58. export const BUILD_TIMESTAMP = typeof __BUILD_TIMESTAMP__ != 'undefined' ? __BUILD_TIMESTAMP__ : 'none';
  59. export const BUILD_DATE = new Date(BUILD_TIMESTAMP);
  60. const Extensions = {
  61. 'rcsb-assembly-symmetry': PluginSpec.Behavior(RCSBAssemblySymmetry),
  62. 'rcsb-validation-report': PluginSpec.Behavior(RCSBValidationReport),
  63. 'rscc': PluginSpec.Behavior(RSCCScore),
  64. 'anvil-membrane-orientation': PluginSpec.Behavior(ANVILMembraneOrientation),
  65. 'ma-quality-assessment': PluginSpec.Behavior(MAQualityAssessment),
  66. 'model-export': PluginSpec.Behavior(ModelExport),
  67. 'mp4-export': PluginSpec.Behavior(Mp4Export),
  68. 'geo-export': PluginSpec.Behavior(GeometryExport),
  69. };
  70. const DefaultViewerProps = {
  71. showImportControls: false,
  72. showSessionControls: false,
  73. showStructureSourceControls: true,
  74. showMeasurementsControls: true,
  75. showStrucmotifSubmitControls: true,
  76. showSuperpositionControls: true,
  77. showQuickStylesControls: false,
  78. showStructureComponentControls: true,
  79. showVolumeStreamingControls: true,
  80. showAssemblySymmetryControls: true,
  81. showValidationReportControls: true,
  82. showMembraneOrientationPreset: false,
  83. showNakbColorTheme: false,
  84. /**
  85. * Needed when running outside of sierra. If set to true, the strucmotif UI will use an absolute URL to sierra-prod.
  86. * Otherwise, the link will be relative on the current host.
  87. */
  88. detachedFromSierra: false,
  89. modelUrlProviders: [
  90. (pdbId: string) => ({
  91. url: `https://models.rcsb.org/${pdbId.toLowerCase()}.bcif`,
  92. format: 'mmcif',
  93. isBinary: true
  94. }),
  95. (pdbId: string) => ({
  96. url: `https://files.rcsb.org/download/${pdbId.toLowerCase()}.cif`,
  97. format: 'mmcif',
  98. isBinary: false
  99. })
  100. ] as ModelUrlProvider[],
  101. extensions: ObjectKeys(Extensions),
  102. layoutIsExpanded: false,
  103. layoutShowControls: true,
  104. layoutControlsDisplay: 'reactive' as PluginLayoutControlsDisplay,
  105. layoutShowSequence: true,
  106. layoutShowLog: false,
  107. viewportShowExpand: true,
  108. viewportShowSelectionMode: true,
  109. volumeStreamingServer: 'https://maps.rcsb.org/',
  110. backgroundColor: ColorNames.white,
  111. manualReset: false, // switch to 'true' for 'motif' preset
  112. pickingAlphaThreshold: 0.5, // lower to 0.2 to accommodate 'motif' preset
  113. showWelcomeToast: true,
  114. };
  115. export type ViewerProps = typeof DefaultViewerProps & { canvas3d: PartialCanvas3DProps }
  116. const LigandExtensions = {
  117. 'wwpdb-chemical-component-dictionary': PluginSpec.Behavior(wwPDBChemicalComponentDictionary),
  118. };
  119. const DefaultLigandViewerProps = {
  120. showImportControls: false,
  121. showSessionControls: false,
  122. showStructureSourceControls: true,
  123. showMeasurementsControls: true,
  124. showQuickStylesControls: false,
  125. showStructureComponentControls: true,
  126. ligandUrlProviders: [
  127. (id: string) => ({
  128. url: id.length <= 5 ? `https://files.rcsb.org/ligands/view/${id.toUpperCase()}.cif` : `https://files.rcsb.org/birds/view/${id.toUpperCase()}.cif`,
  129. format: 'mmcif',
  130. isBinary: false
  131. })
  132. ] as LigandUrlProvider[],
  133. extensions: ObjectKeys(LigandExtensions),
  134. layoutIsExpanded: false,
  135. layoutShowControls: false,
  136. layoutControlsDisplay: 'reactive' as PluginLayoutControlsDisplay,
  137. layoutShowSequence: false,
  138. layoutShowLog: false,
  139. viewportShowExpand: true,
  140. viewportShowSelectionMode: true,
  141. volumeStreamingServer: 'https://maps.rcsb.org/',
  142. backgroundColor: ColorNames.white,
  143. manualReset: false,
  144. pickingAlphaThreshold: 0.5,
  145. showWelcomeToast: true,
  146. ignoreHydrogens: true,
  147. showLabels: false,
  148. shownCoordinateType: 'ideal' as const
  149. };
  150. export type LigandViewerProps = typeof DefaultLigandViewerProps & { canvas3d: PartialCanvas3DProps }
  151. export class Viewer {
  152. private readonly _plugin: PluginUIContext;
  153. private readonly modelUrlProviders: ModelUrlProvider[];
  154. private prevExpanded: boolean;
  155. constructor(elementOrId: string | HTMLElement, props: Partial<ViewerProps> = {}) {
  156. const element = typeof elementOrId === 'string' ? document.getElementById(elementOrId)! : elementOrId;
  157. if (!element) throw new Error(`Could not get element with id '${elementOrId}'`);
  158. const o = { ...DefaultViewerProps, ...props };
  159. const defaultSpec = DefaultPluginUISpec();
  160. const spec: PluginUISpec = {
  161. ...defaultSpec,
  162. actions: defaultSpec.actions,
  163. behaviors: [
  164. ...defaultSpec.behaviors,
  165. ...o.extensions.map(e => Extensions[e]),
  166. ],
  167. animations: [...defaultSpec.animations?.filter(a => a.name !== AnimateStateSnapshots.name) || []],
  168. layout: {
  169. initial: {
  170. isExpanded: o.layoutIsExpanded,
  171. showControls: o.layoutShowControls,
  172. controlsDisplay: o.layoutControlsDisplay,
  173. },
  174. },
  175. canvas3d: {
  176. ...defaultSpec.canvas3d,
  177. ...o.canvas3d,
  178. renderer: {
  179. ...defaultSpec.canvas3d?.renderer,
  180. ...o.canvas3d?.renderer,
  181. backgroundColor: o.backgroundColor,
  182. pickingAlphaThreshold: o.pickingAlphaThreshold
  183. },
  184. camera: {
  185. // desirable for alignment view so that the display doesn't "jump around" as more structures get loaded
  186. manualReset: o.manualReset
  187. }
  188. },
  189. components: {
  190. ...defaultSpec.components,
  191. controls: {
  192. ...defaultSpec.components?.controls,
  193. top: o.layoutShowSequence ? undefined : 'none',
  194. bottom: o.layoutShowLog ? undefined : 'none',
  195. left: 'none',
  196. right: ControlsWrapper,
  197. },
  198. remoteState: 'none',
  199. },
  200. config: [
  201. [PluginConfig.Viewport.ShowExpand, o.viewportShowExpand],
  202. [PluginConfig.Viewport.ShowSelectionMode, o.viewportShowSelectionMode],
  203. [PluginConfig.Viewport.ShowAnimation, false],
  204. [PluginConfig.VolumeStreaming.DefaultServer, o.volumeStreamingServer],
  205. [PluginConfig.Download.DefaultPdbProvider, 'rcsb'],
  206. [PluginConfig.Download.DefaultEmdbProvider, 'rcsb'],
  207. [PluginConfig.Structure.DefaultRepresentationPreset, PresetStructureRepresentations.auto.id],
  208. // wboit & webgl1 checks are needed to work properly on recent Safari versions
  209. [PluginConfig.General.EnableWboit, PluginFeatureDetection.preferWebGl1],
  210. [PluginConfig.General.PreferWebGl1, PluginFeatureDetection.preferWebGl1]
  211. ]
  212. };
  213. this._plugin = new PluginUIContext(spec);
  214. this.modelUrlProviders = o.modelUrlProviders;
  215. (this._plugin.customState as ViewerState) = {
  216. showImportControls: o.showImportControls,
  217. showSessionControls: o.showSessionControls,
  218. showStructureSourceControls: o.showStructureSourceControls,
  219. showMeasurementsControls: o.showMeasurementsControls,
  220. showStrucmotifSubmitControls: o.showStrucmotifSubmitControls,
  221. showSuperpositionControls: o.showSuperpositionControls,
  222. showQuickStylesControls: o.showQuickStylesControls,
  223. showStructureComponentControls: o.showStructureComponentControls,
  224. showVolumeStreamingControls: o.showVolumeStreamingControls,
  225. showAssemblySymmetryControls: o.showAssemblySymmetryControls,
  226. showValidationReportControls: o.showValidationReportControls,
  227. modelLoader: new ModelLoader(this._plugin),
  228. collapsed: new BehaviorSubject<CollapsedState>({
  229. selection: true,
  230. measurements: true,
  231. strucmotifSubmit: true,
  232. superposition: true,
  233. quickStyles: false,
  234. component: false,
  235. volume: true,
  236. assemblySymmetry: true,
  237. validationReport: true,
  238. custom: true,
  239. }),
  240. detachedFromSierra: o.detachedFromSierra
  241. };
  242. this._plugin.init()
  243. .then(async () => {
  244. // hide 'Membrane Orientation' preset from UI - has to happen 'before' react render, apparently
  245. // the corresponding behavior must be registered either way, because the 3d-view uses it (even without appearing in the UI)
  246. if (!o.showMembraneOrientationPreset) {
  247. this._plugin.builders.structure.representation.unregisterPreset(MembraneOrientationPreset);
  248. this._plugin.representation.structure.registry.remove(MembraneOrientationRepresentationProvider);
  249. }
  250. // normally, this would be part of CustomStructureControls -- we want to manage its collapsed state individually though
  251. this._plugin.customStructureControls.delete(AssemblySymmetry.Tag.Representation);
  252. const root = createRoot(element);
  253. root.render(React.createElement(Plugin, { plugin: this._plugin }));
  254. this._plugin.representation.structure.themes.colorThemeRegistry.add(SuperposeColorThemeProvider);
  255. if (o.showNakbColorTheme) this._plugin.representation.structure.themes.colorThemeRegistry.add(NakbColorThemeProvider);
  256. if (o.showWelcomeToast) {
  257. await PluginCommands.Toast.Show(this._plugin, {
  258. title: 'Welcome',
  259. message: `RCSB PDB Mol* Viewer ${RCSB_MOLSTAR_VERSION} [${BUILD_DATE.toLocaleString()}]`,
  260. key: 'toast-welcome',
  261. timeoutMs: 5000
  262. });
  263. }
  264. this.prevExpanded = this._plugin.layout.state.isExpanded;
  265. this._plugin.layout.events.updated.subscribe(() => this.toggleControls());
  266. });
  267. }
  268. get plugin() {
  269. return this._plugin;
  270. }
  271. pluginCall(f: (plugin: PluginContext) => void) {
  272. f(this.plugin);
  273. }
  274. private get customState() {
  275. return this._plugin.customState as ViewerState;
  276. }
  277. private toggleControls(): void {
  278. const currExpanded = this._plugin.layout.state.isExpanded;
  279. const expandedChanged = (this.prevExpanded !== currExpanded);
  280. if (!expandedChanged) return;
  281. if (currExpanded && !this._plugin.layout.state.showControls) {
  282. this._plugin.layout.setProps({ showControls: true });
  283. } else if (!currExpanded && this._plugin.layout.state.showControls) {
  284. this._plugin.layout.setProps({ showControls: false });
  285. }
  286. this.prevExpanded = this._plugin.layout.state.isExpanded;
  287. }
  288. resetCamera(durationMs?: number) {
  289. this._plugin.managers.camera.reset(undefined, durationMs);
  290. }
  291. clear() {
  292. const state = this._plugin.state.data;
  293. return PluginCommands.State.RemoveObject(this._plugin, { state, ref: state.tree.root.ref });
  294. }
  295. async loadPdbId<P, S>(pdbId: string, config?: { props?: PresetProps; matrix?: Mat4; reprProvider?: TrajectoryHierarchyPresetProvider<P, S>, params?: P }) {
  296. for (const provider of this.modelUrlProviders) {
  297. try {
  298. const p = provider(pdbId);
  299. return await this.customState.modelLoader.load<P, S>({ fileOrUrl: p.url, format: p.format, isBinary: p.isBinary }, config?.props, config?.matrix, config?.reprProvider, config?.params);
  300. } catch (e) {
  301. console.warn(`loading '${pdbId}' failed with '${e}', trying next model-loader-provider`);
  302. }
  303. }
  304. }
  305. async loadPdbIds<P, S>(args: { pdbId: string, config?: {props?: PresetProps; matrix?: Mat4; reprProvider?: TrajectoryHierarchyPresetProvider<P, S>, params?: P} }[]) {
  306. const out = [];
  307. for (const { pdbId, config } of args) {
  308. out.push(await this.loadPdbId(pdbId, config));
  309. }
  310. if (!this.plugin.spec.canvas3d?.camera?.manualReset) this.resetCamera(0);
  311. return out;
  312. }
  313. loadStructureFromUrl<P, S>(url: string, format: BuiltInTrajectoryFormat, isBinary: boolean, config?: {props?: PresetProps & { dataLabel?: string }; matrix?: Mat4; reprProvider?: TrajectoryHierarchyPresetProvider<P, S>, params?: P}) {
  314. return this.customState.modelLoader.load({ fileOrUrl: url, format, isBinary }, config?.props, config?.matrix, config?.reprProvider, config?.params);
  315. }
  316. loadSnapshotFromUrl(url: string, type: PluginState.SnapshotType) {
  317. return PluginCommands.State.Snapshots.OpenUrl(this._plugin, { url, type });
  318. }
  319. loadStructureFromData<P, S>(data: string | number[], format: BuiltInTrajectoryFormat, isBinary: boolean, config?: {props?: PresetProps & { dataLabel?: string }; matrix?: Mat4; reprProvider?: TrajectoryHierarchyPresetProvider<P, S>, params?: P}) {
  320. return this.customState.modelLoader.parse({ data, format, isBinary }, config?.props, config?.matrix, config?.reprProvider, config?.params);
  321. }
  322. handleResize() {
  323. this._plugin.layout.events.updated.next(void 0);
  324. }
  325. exportLoadedStructures(options?: { format?: 'cif' | 'bcif' }) {
  326. return exportHierarchy(this.plugin, options);
  327. }
  328. setFocus(target: SelectRange) {
  329. setFocusFromRange(this._plugin, target);
  330. }
  331. clearFocus(): void {
  332. this._plugin.managers.structure.focus.clear();
  333. }
  334. select(targets: SelectTarget | SelectTarget[], mode: 'select' | 'hover', modifier: 'add' | 'set') {
  335. select(this._plugin, targets, mode, modifier);
  336. }
  337. clearSelection(mode: 'select' | 'hover', target?: { modelId: string; } & Target) {
  338. clearSelection(this._plugin, mode, target);
  339. }
  340. async createComponent(label: string, targets: SelectBase | SelectTarget | SelectTarget[], representationType: StructureRepresentationRegistry.BuiltIn) {
  341. await createComponent(this._plugin, label, targets, representationType);
  342. }
  343. async removeComponent(componentLabel: string) {
  344. await removeComponent(this._plugin, componentLabel);
  345. }
  346. }
  347. export class LigandViewer {
  348. private readonly _plugin: PluginUIContext;
  349. private readonly ligandUrlProviders: LigandUrlProvider[];
  350. constructor(elementOrId: string | HTMLElement, props: Partial<LigandViewerProps> = {}) {
  351. const element = typeof elementOrId === 'string' ? document.getElementById(elementOrId)! : elementOrId;
  352. if (!element) throw new Error(`Could not get element with id '${elementOrId}'`);
  353. const o = { ...DefaultLigandViewerProps, ...props };
  354. const defaultSpec = DefaultPluginUISpec();
  355. const spec: PluginUISpec = {
  356. ...defaultSpec,
  357. actions: defaultSpec.actions,
  358. behaviors: [
  359. ...defaultSpec.behaviors,
  360. ...o.extensions.map(e => LigandExtensions[e]),
  361. ],
  362. animations: [...defaultSpec.animations?.filter(a => a.name !== AnimateStateSnapshots.name) || []],
  363. layout: {
  364. initial: {
  365. isExpanded: o.layoutIsExpanded,
  366. showControls: o.layoutShowControls,
  367. controlsDisplay: o.layoutControlsDisplay,
  368. },
  369. },
  370. canvas3d: {
  371. ...defaultSpec.canvas3d,
  372. ...o.canvas3d,
  373. renderer: {
  374. ...defaultSpec.canvas3d?.renderer,
  375. ...o.canvas3d?.renderer,
  376. backgroundColor: o.backgroundColor,
  377. pickingAlphaThreshold: o.pickingAlphaThreshold
  378. },
  379. camera: {
  380. // desirable for alignment view so that the display doesn't "jump around" as more structures get loaded
  381. manualReset: o.manualReset,
  382. helper: {
  383. axes: {
  384. name: 'off', params: {}
  385. }
  386. }
  387. }
  388. },
  389. components: {
  390. ...defaultSpec.components,
  391. controls: {
  392. ...defaultSpec.components?.controls,
  393. top: o.layoutShowSequence ? undefined : 'none',
  394. bottom: o.layoutShowLog ? undefined : 'none',
  395. left: 'none',
  396. right: ControlsWrapper,
  397. },
  398. remoteState: 'none',
  399. },
  400. config: [
  401. [PluginConfig.VolumeStreaming.Enabled, false],
  402. [PluginConfig.Viewport.ShowExpand, o.viewportShowExpand],
  403. [PluginConfig.Viewport.ShowSelectionMode, o.viewportShowSelectionMode],
  404. [PluginConfig.Viewport.ShowAnimation, false],
  405. [PluginConfig.Download.DefaultPdbProvider, 'rcsb'],
  406. [PluginConfig.Download.DefaultEmdbProvider, 'rcsb'],
  407. [PluginConfig.Structure.DefaultRepresentationPreset, PresetStructureRepresentations.auto.id],
  408. // wboit & webgl1 checks are needed to work properly on recent Safari versions
  409. [PluginConfig.General.EnableWboit, PluginFeatureDetection.preferWebGl1],
  410. [PluginConfig.General.PreferWebGl1, PluginFeatureDetection.preferWebGl1]
  411. ]
  412. };
  413. this._plugin = new PluginUIContext(spec);
  414. this.ligandUrlProviders = o.ligandUrlProviders;
  415. (this._plugin.customState as LigandViewerState) = {
  416. showMeasurementsControls: o.showMeasurementsControls,
  417. showStructureComponentControls: o.showStructureComponentControls,
  418. modelLoader: new ModelLoader(this._plugin),
  419. collapsed: new BehaviorSubject<CollapsedState>({
  420. selection: true,
  421. measurements: true,
  422. strucmotifSubmit: true,
  423. superposition: true,
  424. quickStyles: true,
  425. component: false,
  426. volume: true,
  427. assemblySymmetry: true,
  428. validationReport: true,
  429. custom: true,
  430. }),
  431. ignoreHydrogens: o.ignoreHydrogens,
  432. showLabels: o.showLabels,
  433. shownCoordinateType: o.shownCoordinateType
  434. };
  435. this._plugin.init()
  436. .then(async () => {
  437. const root = createRoot(element);
  438. root.render(React.createElement(Plugin, { plugin: this._plugin }));
  439. if (o.showWelcomeToast) {
  440. await PluginCommands.Toast.Show(this._plugin, {
  441. title: 'Welcome',
  442. message: `RCSB PDB Mol* Ligand Viewer ${RCSB_MOLSTAR_VERSION} [${BUILD_DATE.toLocaleString()}]`,
  443. key: 'toast-welcome',
  444. timeoutMs: 5000
  445. });
  446. }
  447. // allow picking of individual atoms
  448. this._plugin.managers.interactivity.setProps({ granularity: 'element' });
  449. });
  450. }
  451. private get customState() {
  452. return this._plugin.customState as LigandViewerState;
  453. }
  454. clear() {
  455. const state = this._plugin.state.data;
  456. return PluginCommands.State.RemoveObject(this._plugin, { state, ref: state.tree.root.ref });
  457. }
  458. async loadLigandId(id: string) {
  459. for (const provider of this.ligandUrlProviders) {
  460. try {
  461. const p = provider(id);
  462. await this.load({ fileOrUrl: p.url, format: p.format, isBinary: p.isBinary });
  463. } catch (e) {
  464. console.warn(`loading '${id}' failed with '${e}', trying next ligand-loader-provider`);
  465. }
  466. }
  467. }
  468. private async load(p: LoadParams) {
  469. await this.customState.modelLoader.load<any, any>(p, undefined, undefined, ChemicalCompontentTrajectoryHierarchyPreset, { shownCoordinateType: this.customState.shownCoordinateType });
  470. await this.syncHydrogenState();
  471. for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
  472. for (const c of s.components) {
  473. const isHidden = c.cell.state.isHidden === true || !this.customState.showLabels;
  474. await this._plugin.builders.structure.representation.addRepresentation(c.cell, { type: 'label', typeParams: { level: 'element', fontQuality: 4, borderColor: ColorNames.black, ignoreHydrogens: this.customState.ignoreHydrogens } }, { initialState: { isHidden } });
  475. }
  476. }
  477. }
  478. async toggleHydrogens() {
  479. this.customState.ignoreHydrogens = !this.customState.ignoreHydrogens;
  480. await this.syncHydrogenState();
  481. }
  482. private async syncHydrogenState() {
  483. const update = this._plugin.build();
  484. for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
  485. for (const c of s.components) {
  486. for (const r of c.representations) {
  487. update.to(r.cell).update(StateTransforms.Representation.StructureRepresentation3D, old => {
  488. old.type.params.ignoreHydrogens = this.customState.ignoreHydrogens;
  489. });
  490. }
  491. }
  492. }
  493. await update.commit();
  494. this._plugin.managers.camera.reset(undefined, 100);
  495. }
  496. async toggleLabels() {
  497. this.customState.showLabels = !this.customState.showLabels;
  498. await this.syncLabelState();
  499. }
  500. private async syncLabelState() {
  501. for (const s of this._plugin.managers.structure.hierarchy.current.structures) {
  502. for (const c of s.components) {
  503. if (c.cell.state.isHidden) continue;
  504. for (const r of c.representations) {
  505. if (r.cell.obj?.label !== 'Label') continue;
  506. this._plugin.managers.structure.hierarchy.toggleVisibility([r], this.customState.showLabels ? 'show' : 'hide');
  507. }
  508. }
  509. }
  510. this._plugin.managers.camera.reset(undefined, 100);
  511. }
  512. }