structure-view.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. /**
  2. * Copyright (c) 2018 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { Model, Structure } from 'mol-model/structure';
  7. import { CartoonRepresentation } from 'mol-geo/representation/structure/representation/cartoon';
  8. import { BallAndStickRepresentation } from 'mol-geo/representation/structure/representation/ball-and-stick';
  9. import { getStructureFromModel } from './util';
  10. import { AssemblySymmetry } from 'mol-model-props/rcsb/symmetry';
  11. import { ShapeRepresentation, ShapeProps } from 'mol-geo/representation/shape';
  12. import { getAxesShape } from './assembly-symmetry';
  13. import Canvas3D from 'mol-canvas3d/canvas3d';
  14. import { CarbohydrateRepresentation } from 'mol-geo/representation/structure/representation/carbohydrate';
  15. // import { MeshBuilder } from 'mol-geo/mesh/mesh-builder';
  16. // import { addSphere } from 'mol-geo/mesh/builder/sphere';
  17. // import { Shape } from 'mol-model/shape';
  18. // import { Color } from 'mol-util/color';
  19. // import { computeUnitBoundary } from 'mol-model/structure/structure/util/boundary';
  20. // import { addBoundingBox } from 'mol-geo/mesh/builder/bounding-box';
  21. import { PointRepresentation } from 'mol-geo/representation/structure/representation/point';
  22. import { StructureRepresentation } from 'mol-geo/representation/structure';
  23. import { BehaviorSubject } from 'rxjs';
  24. import { SpacefillRepresentation } from 'mol-geo/representation/structure/representation/spacefill';
  25. import { DistanceRestraintRepresentation } from 'mol-geo/representation/structure/representation/distance-restraint';
  26. import { MolecularSurfaceRepresentation } from 'mol-geo/representation/structure/representation/molecular-surface';
  27. import { App } from './app';
  28. export interface StructureView {
  29. readonly app: App
  30. readonly viewer: Canvas3D
  31. readonly label: string
  32. readonly models: ReadonlyArray<Model>
  33. readonly structure: Structure | undefined
  34. readonly assemblySymmetry: AssemblySymmetry | undefined
  35. readonly active: { [k: string]: boolean }
  36. readonly structureRepresentations: { [k: string]: StructureRepresentation<any> }
  37. readonly updated: BehaviorSubject<null>
  38. readonly symmetryAxes: ShapeRepresentation<ShapeProps>
  39. setSymmetryAxes(value: boolean): void
  40. setStructureRepresentation(name: string, value: boolean): void
  41. readonly modelId: number
  42. readonly assemblyId: string
  43. readonly symmetryFeatureId: number
  44. setModel(modelId: number): Promise<void>
  45. getModelIds(): { id: number, label: string }[]
  46. setAssembly(assemblyId: string): Promise<void>
  47. getAssemblyIds(): { id: string, label: string }[]
  48. setSymmetryFeature(symmetryFeatureId: number): Promise<void>
  49. getSymmetryFeatureIds(): { id: number, label: string }[]
  50. destroy: () => void
  51. }
  52. interface StructureViewProps {
  53. assemblyId?: string
  54. symmetryFeatureId?: number
  55. }
  56. export async function StructureView(app: App, viewer: Canvas3D, models: ReadonlyArray<Model>, props: StructureViewProps = {}): Promise<StructureView> {
  57. const active: { [k: string]: boolean } = {
  58. cartoon: true,
  59. point: false,
  60. surface: false,
  61. ballAndStick: false,
  62. carbohydrate: false,
  63. spacefill: false,
  64. distanceRestraint: false,
  65. symmetryAxes: true,
  66. // polymerSphere: false,
  67. }
  68. const structureRepresentations: { [k: string]: StructureRepresentation<any> } = {
  69. cartoon: CartoonRepresentation(),
  70. surface: MolecularSurfaceRepresentation(),
  71. point: PointRepresentation(),
  72. ballAndStick: BallAndStickRepresentation(),
  73. carbohydrate: CarbohydrateRepresentation(),
  74. spacefill: SpacefillRepresentation(),
  75. distanceRestraint: DistanceRestraintRepresentation(),
  76. }
  77. const symmetryAxes = ShapeRepresentation()
  78. const polymerSphere = ShapeRepresentation()
  79. const updated: BehaviorSubject<null> = new BehaviorSubject<null>(null)
  80. let label: string
  81. let model: Model | undefined
  82. let assemblySymmetry: AssemblySymmetry | undefined
  83. let structure: Structure | undefined
  84. let modelId: number
  85. let assemblyId: string
  86. let symmetryFeatureId: number
  87. async function setSymmetryAxes(value: boolean) {
  88. if (!value) {
  89. assemblySymmetry = undefined
  90. } else {
  91. await app.runTask(AssemblySymmetry.attachFromCifOrAPI(models[modelId]), 'Load symmetry annotation')
  92. assemblySymmetry = AssemblySymmetry.get(models[modelId])
  93. }
  94. active.symmetryAxes = value
  95. await setSymmetryFeature()
  96. }
  97. async function setStructureRepresentation(k: string, value: boolean) {
  98. active[k] = value
  99. await createStructureRepr()
  100. }
  101. async function setModel(newModelId: number, newAssemblyId?: string, newSymmetryFeatureId?: number) {
  102. console.log('setModel', newModelId)
  103. modelId = newModelId
  104. model = models[modelId]
  105. if (active.symmetryAxes) {
  106. await AssemblySymmetry.attachFromCifOrAPI(model)
  107. assemblySymmetry = AssemblySymmetry.get(model)
  108. }
  109. await setAssembly(newAssemblyId, newSymmetryFeatureId)
  110. }
  111. function getModelIds() {
  112. const modelIds: { id: number, label: string }[] = []
  113. models.forEach((m, i) => {
  114. modelIds.push({ id: i, label: `${i}: ${m.label} #${m.modelNum}` })
  115. })
  116. return modelIds
  117. }
  118. async function setAssembly(newAssemblyId?: string, newSymmetryFeatureId?: number) {
  119. console.log('setAssembly', newAssemblyId)
  120. if (newAssemblyId !== undefined) {
  121. assemblyId = newAssemblyId
  122. } else if (model && model.symmetry.assemblies.length) {
  123. assemblyId = model.symmetry.assemblies[0].id
  124. } else if (model) {
  125. assemblyId = 'deposited'
  126. } else {
  127. assemblyId = '-1'
  128. }
  129. await getStructure()
  130. await setSymmetryFeature(newSymmetryFeatureId)
  131. }
  132. function getAssemblyIds() {
  133. const assemblyIds: { id: string, label: string }[] = [
  134. { id: 'deposited', label: 'deposited' }
  135. ]
  136. if (model) model.symmetry.assemblies.forEach(a => {
  137. assemblyIds.push({ id: a.id, label: `${a.id}: ${a.details}` })
  138. })
  139. return assemblyIds
  140. }
  141. async function setSymmetryFeature(newSymmetryFeatureId?: number) {
  142. console.log('setSymmetryFeature', newSymmetryFeatureId)
  143. if (newSymmetryFeatureId !== undefined) {
  144. symmetryFeatureId = newSymmetryFeatureId
  145. } else if (assemblySymmetry) {
  146. const s = assemblySymmetry.getSymmetries(assemblyId)
  147. if (s._rowCount) {
  148. symmetryFeatureId = s.id.value(0)
  149. } else {
  150. symmetryFeatureId = -1
  151. }
  152. } else {
  153. symmetryFeatureId = -1
  154. }
  155. await createSymmetryRepr()
  156. }
  157. function getSymmetryFeatureIds() {
  158. const symmetryFeatureIds: { id: number, label: string }[] = []
  159. if (assemblySymmetry) {
  160. const symmetries = assemblySymmetry.getSymmetries(assemblyId)
  161. for (let i = 0, il = symmetries._rowCount; i < il; ++i) {
  162. const id = symmetries.id.value(i)
  163. const kind = symmetries.kind.value(i)
  164. const type = symmetries.type.value(i)
  165. const stoichiometry = symmetries.stoichiometry.value(i)
  166. const label = `${id}: ${kind} ${type} ${stoichiometry}`
  167. symmetryFeatureIds.push({ id, label })
  168. }
  169. }
  170. return symmetryFeatureIds
  171. }
  172. async function getStructure() {
  173. if (model) structure = await app.runTask(getStructureFromModel(model, assemblyId), 'Build structure')
  174. if (model && structure) {
  175. label = `${model.label} - Assembly ${assemblyId}`
  176. } else {
  177. label = ''
  178. }
  179. await createStructureRepr()
  180. }
  181. async function createStructureRepr() {
  182. if (structure) {
  183. console.log('createStructureRepr')
  184. for (const k in structureRepresentations) {
  185. if (active[k]) {
  186. const p = { webgl: viewer.webgl }
  187. await app.runTask(structureRepresentations[k].createOrUpdate(p, structure).run(
  188. progress => app.log(progress)
  189. ), 'Create/update representation')
  190. viewer.add(structureRepresentations[k])
  191. } else {
  192. viewer.remove(structureRepresentations[k])
  193. }
  194. }
  195. viewer.center(structure.boundary.sphere.center)
  196. // const mb = MeshBuilder.create()
  197. // mb.setGroup(0)
  198. // addSphere(mb, structure.boundary.sphere.center, structure.boundary.sphere.radius, 3)
  199. // addBoundingBox(mb, structure.boundary.box, 1, 2, 8)
  200. // for (let i = 0, il = structure.units.length; i < il; ++i) {
  201. // mb.setGroup(1)
  202. // const u = structure.units[i]
  203. // const ci = u.model.atomicHierarchy.chainAtomSegments.index[u.elements[0]]
  204. // const ek = u.model.atomicHierarchy.getEntityKey(ci)
  205. // if (u.model.entities.data.type.value(ek) === 'water') continue
  206. // const boundary = computeUnitBoundary(u)
  207. // addSphere(mb, boundary.sphere.center, boundary.sphere.radius, 3)
  208. // addBoundingBox(mb, boundary.box, 0.5, 2, 8)
  209. // }
  210. // const shape = Shape.create('boundary', mb.getMesh(), [Color(0xCC6633), Color(0x3366CC)], ['sphere boundary'])
  211. // await polymerSphere.createOrUpdate({
  212. // alpha: 0.5,
  213. // doubleSided: false,
  214. // depthMask: false,
  215. // useFog: false // TODO fog not working properly
  216. // }, shape).run()
  217. } else {
  218. for (const k in structureRepresentations) structureRepresentations[k].destroy()
  219. polymerSphere.destroy()
  220. }
  221. viewer.add(polymerSphere)
  222. updated.next(null)
  223. viewer.requestDraw(true)
  224. console.log('stats', viewer.stats)
  225. }
  226. async function createSymmetryRepr() {
  227. if (assemblySymmetry) {
  228. const symmetries = assemblySymmetry.getSymmetries(assemblyId)
  229. if (symmetries._rowCount) {
  230. const axesShape = getAxesShape(symmetryFeatureId, assemblySymmetry)
  231. if (axesShape) {
  232. // const colorTheme = getClusterColorTheme(symmetryFeatureId, assemblySymmetry)
  233. // await structureRepresentations['cartoon'].createOrUpdate({
  234. // colorTheme: 'custom',
  235. // colorFunction: colorTheme.color,
  236. // colorGranularity: colorTheme.granularity,
  237. // }).run()
  238. await symmetryAxes.createOrUpdate({}, axesShape).run()
  239. viewer.add(symmetryAxes)
  240. } else {
  241. viewer.remove(symmetryAxes)
  242. }
  243. } else {
  244. viewer.remove(symmetryAxes)
  245. }
  246. } else {
  247. viewer.remove(symmetryAxes)
  248. }
  249. updated.next(null)
  250. viewer.requestDraw(true)
  251. }
  252. await setModel(0, props.assemblyId, props.symmetryFeatureId)
  253. return {
  254. app,
  255. viewer,
  256. get label() { return label },
  257. models,
  258. get structure() { return structure },
  259. get assemblySymmetry() { return assemblySymmetry },
  260. active,
  261. structureRepresentations,
  262. updated,
  263. symmetryAxes,
  264. setSymmetryAxes,
  265. setStructureRepresentation,
  266. get modelId() { return modelId },
  267. get assemblyId() { return assemblyId },
  268. get symmetryFeatureId() { return symmetryFeatureId },
  269. setModel,
  270. getModelIds,
  271. setAssembly,
  272. getAssemblyIds,
  273. setSymmetryFeature,
  274. getSymmetryFeatureIds,
  275. destroy: () => {
  276. for (const k in structureRepresentations) {
  277. viewer.remove(structureRepresentations[k])
  278. structureRepresentations[k].destroy()
  279. }
  280. viewer.remove(polymerSphere)
  281. viewer.remove(symmetryAxes)
  282. viewer.requestDraw(true)
  283. polymerSphere.destroy()
  284. symmetryAxes.destroy()
  285. }
  286. }
  287. }
  288. // // create new structure via query
  289. // const q1 = Q.generators.atoms({
  290. // residueTest: qtx => SP.residue.label_seq_id(qtx.element) < 7
  291. // });
  292. // const newStructure = StructureSelection.unionStructure(await StructureQuery.run(q1, structure));
  293. // // ball+stick for new structure
  294. // const newBallStickRepr = BallAndStickRepresentation()
  295. // await newBallStickRepr.create(newStructure, {
  296. // colorTheme: { name: 'element-symbol' },
  297. // sizeTheme: { name: 'uniform', value: 0.1 },
  298. // useFog: false // TODO fog not working properly
  299. // }).run()
  300. // viewer.add(newBallStickRepr)
  301. // // create a mesh
  302. // const meshBuilder = MeshBuilder.create(256, 128)
  303. // const colors: Color[] = []
  304. // const labels: string[] = []
  305. // // red sphere
  306. // meshBuilder.setGroup(0)
  307. // colors[0] = Color(0xFF2233)
  308. // labels[0] = 'red sphere'
  309. // addSphere(meshBuilder, Vec3.create(0, 0, 0), 4, 2)
  310. // // green cube
  311. // meshBuilder.setGroup(1)
  312. // colors[1] = Color(0x2233FF)
  313. // labels[1] = 'blue cube'
  314. // const t = Mat4.identity()
  315. // Mat4.fromTranslation(t, Vec3.create(10, 0, 0))
  316. // Mat4.scale(t, t, Vec3.create(3, 3, 3))
  317. // meshBuilder.add(t, Box())
  318. // const mesh = meshBuilder.getMesh()
  319. // const mesh = getObjFromUrl('mesh.obj')
  320. // // create shape from mesh
  321. // const shape = Shape.create('myShape', mesh, colors, labels)
  322. // // add representation from shape
  323. // const customRepr = ShapeRepresentation()
  324. // await customRepr.create(shape, {
  325. // colorTheme: { name: 'shape-group' },
  326. // // colorTheme: { name: 'uniform', value: Color(0xFFCC22) },
  327. // useFog: false // TODO fog not working properly
  328. // }).run()
  329. // viewer.add(customRepr)