bond-inter-unit-cylinder.ts 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. /**
  2. * Copyright (c) 2018-2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { ParamDefinition as PD } from '../../../mol-util/param-definition';
  7. import { VisualContext } from '../../visual';
  8. import { Structure, StructureElement, Bond, Unit } from '../../../mol-model/structure';
  9. import { Theme } from '../../../mol-theme/theme';
  10. import { Mesh } from '../../../mol-geo/geometry/mesh/mesh';
  11. import { Vec3 } from '../../../mol-math/linear-algebra';
  12. import { BitFlags, arrayEqual } from '../../../mol-util';
  13. import { createLinkCylinderMesh, LinkCylinderStyle } from './util/link';
  14. import { ComplexMeshParams, ComplexVisual, ComplexMeshVisual } from '../complex-visual';
  15. import { VisualUpdateState } from '../../util';
  16. import { PickingId } from '../../../mol-geo/geometry/picking';
  17. import { EmptyLoci, Loci } from '../../../mol-model/loci';
  18. import { Interval, OrderedSet } from '../../../mol-data/int';
  19. import { isHydrogen } from './util/common';
  20. import { BondType } from '../../../mol-model/structure/model/types';
  21. import { ignoreBondType, BondCylinderParams, BondIterator } from './util/bond';
  22. const tmpRefPosBondIt = new Bond.ElementBondIterator()
  23. function setRefPosition(pos: Vec3, structure: Structure, unit: Unit.Atomic, index: StructureElement.UnitIndex) {
  24. tmpRefPosBondIt.setElement(structure, unit, index)
  25. while (tmpRefPosBondIt.hasNext) {
  26. const bA = tmpRefPosBondIt.move()
  27. bA.otherUnit.conformation.position(bA.otherUnit.elements[bA.otherIndex], pos)
  28. return pos
  29. }
  30. return null
  31. }
  32. const tmpRef = Vec3()
  33. const tmpLoc = StructureElement.Location.create(void 0)
  34. function createInterUnitBondCylinderMesh(ctx: VisualContext, structure: Structure, theme: Theme, props: PD.Values<InterUnitBondParams>, mesh?: Mesh) {
  35. const bonds = structure.interUnitBonds
  36. const { edgeCount, edges } = bonds
  37. const { sizeFactor, sizeAspectRatio, ignoreHydrogens, includeTypes, excludeTypes } = props
  38. const include = BondType.fromNames(includeTypes)
  39. const exclude = BondType.fromNames(excludeTypes)
  40. const ignoreHydrogen = ignoreHydrogens ? (edgeIndex: number) => {
  41. const b = edges[edgeIndex]
  42. const uA = b.unitA, uB = b.unitB
  43. return isHydrogen(uA, uA.elements[b.indexA]) || isHydrogen(uB, uB.elements[b.indexB])
  44. } : () => false
  45. if (!edgeCount) return Mesh.createEmpty(mesh)
  46. const builderProps = {
  47. linkCount: edgeCount,
  48. referencePosition: (edgeIndex: number) => {
  49. const b = edges[edgeIndex]
  50. let unitA: Unit, unitB: Unit
  51. let indexA: StructureElement.UnitIndex, indexB: StructureElement.UnitIndex
  52. if (b.unitA.id < b.unitB.id) {
  53. unitA = b.unitA, unitB = b.unitB
  54. indexA = b.indexA, indexB = b.indexB
  55. } else if (b.unitA.id > b.unitB.id) {
  56. unitA = b.unitB, unitB = b.unitA
  57. indexA = b.indexB, indexB = b.indexA
  58. } else {
  59. throw new Error('same units in createInterUnitBondCylinderMesh')
  60. }
  61. return setRefPosition(tmpRef, structure, unitA, indexA) || setRefPosition(tmpRef, structure, unitB, indexB)
  62. },
  63. position: (posA: Vec3, posB: Vec3, edgeIndex: number) => {
  64. const b = edges[edgeIndex]
  65. const uA = b.unitA, uB = b.unitB
  66. uA.conformation.position(uA.elements[b.indexA], posA)
  67. uB.conformation.position(uB.elements[b.indexB], posB)
  68. },
  69. style: (edgeIndex: number) => {
  70. const o = edges[edgeIndex].props.order
  71. const f = BitFlags.create(edges[edgeIndex].props.flag)
  72. if (BondType.is(f, BondType.Flag.MetallicCoordination) || BondType.is(f, BondType.Flag.HydrogenBond)) {
  73. // show metall coordinations and hydrogen bonds with dashed cylinders
  74. return LinkCylinderStyle.Dashed
  75. } else if (o === 2) {
  76. return LinkCylinderStyle.Double
  77. } else if (o === 3) {
  78. return LinkCylinderStyle.Triple
  79. } else {
  80. return LinkCylinderStyle.Solid
  81. }
  82. },
  83. radius: (edgeIndex: number) => {
  84. const b = edges[edgeIndex]
  85. tmpLoc.structure = structure
  86. tmpLoc.unit = b.unitA
  87. tmpLoc.element = b.unitA.elements[b.indexA]
  88. const sizeA = theme.size.size(tmpLoc)
  89. tmpLoc.unit = b.unitB
  90. tmpLoc.element = b.unitB.elements[b.indexB]
  91. const sizeB = theme.size.size(tmpLoc)
  92. return Math.min(sizeA, sizeB) * sizeFactor * sizeAspectRatio
  93. },
  94. ignore: (edgeIndex: number) => ignoreHydrogen(edgeIndex) || ignoreBondType(include, exclude, edges[edgeIndex].props.flag)
  95. }
  96. return createLinkCylinderMesh(ctx, builderProps, props, mesh)
  97. }
  98. export const InterUnitBondParams = {
  99. ...ComplexMeshParams,
  100. ...BondCylinderParams,
  101. sizeFactor: PD.Numeric(0.3, { min: 0, max: 10, step: 0.01 }),
  102. sizeAspectRatio: PD.Numeric(2 / 3, { min: 0, max: 3, step: 0.01 }),
  103. ignoreHydrogens: PD.Boolean(false),
  104. }
  105. export type InterUnitBondParams = typeof InterUnitBondParams
  106. export function InterUnitBondVisual(materialId: number): ComplexVisual<InterUnitBondParams> {
  107. return ComplexMeshVisual<InterUnitBondParams>({
  108. defaultProps: PD.getDefaultValues(InterUnitBondParams),
  109. createGeometry: createInterUnitBondCylinderMesh,
  110. createLocationIterator: BondIterator.fromStructure,
  111. getLoci: getBondLoci,
  112. eachLocation: eachBond,
  113. setUpdateState: (state: VisualUpdateState, newProps: PD.Values<InterUnitBondParams>, currentProps: PD.Values<InterUnitBondParams>) => {
  114. state.createGeometry = (
  115. newProps.sizeFactor !== currentProps.sizeFactor ||
  116. newProps.sizeAspectRatio !== currentProps.sizeAspectRatio ||
  117. newProps.radialSegments !== currentProps.radialSegments ||
  118. newProps.linkScale !== currentProps.linkScale ||
  119. newProps.linkSpacing !== currentProps.linkSpacing ||
  120. newProps.ignoreHydrogens !== currentProps.ignoreHydrogens ||
  121. newProps.linkCap !== currentProps.linkCap ||
  122. !arrayEqual(newProps.includeTypes, currentProps.includeTypes) ||
  123. !arrayEqual(newProps.excludeTypes, currentProps.excludeTypes)
  124. )
  125. }
  126. }, materialId)
  127. }
  128. function getBondLoci(pickingId: PickingId, structure: Structure, id: number) {
  129. const { objectId, groupId } = pickingId
  130. if (id === objectId) {
  131. const bond = structure.interUnitBonds.edges[groupId]
  132. return Bond.Loci(structure, [
  133. Bond.Location(
  134. structure, bond.unitA, bond.indexA as StructureElement.UnitIndex,
  135. structure, bond.unitB, bond.indexB as StructureElement.UnitIndex
  136. ),
  137. Bond.Location(
  138. structure, bond.unitB, bond.indexB as StructureElement.UnitIndex,
  139. structure, bond.unitA, bond.indexA as StructureElement.UnitIndex
  140. )
  141. ])
  142. }
  143. return EmptyLoci
  144. }
  145. function eachBond(loci: Loci, structure: Structure, apply: (interval: Interval) => boolean) {
  146. let changed = false
  147. if (Bond.isLoci(loci)) {
  148. if (!Structure.areEquivalent(loci.structure, structure)) return false
  149. for (const b of loci.bonds) {
  150. const idx = structure.interUnitBonds.getBondIndexFromLocation(b)
  151. if (idx !== -1) {
  152. if (apply(Interval.ofSingleton(idx))) changed = true
  153. }
  154. }
  155. } else if (StructureElement.Loci.is(loci)) {
  156. if (!Structure.areEquivalent(loci.structure, structure)) return false
  157. if (loci.elements.length === 1) return false // only a single unit
  158. const map = new Map<number, OrderedSet<StructureElement.UnitIndex>>()
  159. for (const e of loci.elements) map.set(e.unit.id, e.indices)
  160. for (const e of loci.elements) {
  161. const { unit } = e
  162. if (!Unit.isAtomic(unit)) continue
  163. structure.interUnitBonds.getConnectedUnits(unit).forEach(b => {
  164. const otherLociIndices = map.get(b.unitB.id)
  165. if (otherLociIndices) {
  166. OrderedSet.forEach(e.indices, v => {
  167. if (!b.connectedIndices.includes(v)) return
  168. b.getEdges(v).forEach(bi => {
  169. if (OrderedSet.has(otherLociIndices, bi.indexB)) {
  170. const idx = structure.interUnitBonds.getEdgeIndex(v, unit, bi.indexB, b.unitB)
  171. if (apply(Interval.ofSingleton(idx))) changed = true
  172. }
  173. })
  174. })
  175. }
  176. })
  177. }
  178. }
  179. return changed
  180. }