transformation.ts 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. import { MolScriptBuilder as MS } from '../../mol-script/language/builder';
  2. import { PluginUIContext } from '../../mol-plugin-ui/context';
  3. import { Mat4, Vec3 } from '../../mol-math/linear-algebra';
  4. import { PDBTMDescriptor, PDBTMTransformationMatrix, PMS } from './types';
  5. import { createStructureRepresentationParams } from '../../mol-plugin-state/helpers/structure-representation-params';
  6. import { StateTransforms } from '../../mol-plugin-state/transforms';
  7. import { StateObjectRef, StateBuilder } from '../../mol-state';
  8. import { Expression } from '../../mol-script/language/expression';
  9. import { MembraneOrientation } from './prop';
  10. import { TmDetColorThemeProvider } from './tmdet-color-theme';
  11. export async function applyTransformations(plugin: PluginUIContext, pdbtmDescriptor: PDBTMDescriptor) {
  12. const annotations = pdbtmDescriptor.additional_entry_annotations;
  13. console.log('BIOMX', annotations.biomatrix);
  14. if (annotations.biomatrix) {
  15. annotations.biomatrix.matrix_list.forEach(function(mx) {
  16. mx.apply_to_chain_list.forEach(function(chainPair) {
  17. let id = chainPair.chain_id;
  18. chainTransformation(plugin, pdbtmDescriptor, mx.transformation_matrix, id);
  19. });
  20. });
  21. }
  22. // membrane transformation
  23. const membraneTransformation = transformationForStateTransform(pdbtmDescriptor.additional_entry_annotations.membrane.transformation_matrix);
  24. const structure: StateObjectRef<PMS> = plugin.managers.structure.hierarchy.current.models[0].structures[0].cell;
  25. const update: StateBuilder.To<any, any> = plugin.build().to(structure);
  26. update
  27. .apply(StateTransforms.Model.TransformStructureConformation, {
  28. "transform": { name: "matrix", params: { data: membraneTransformation, transpose: false } }
  29. })
  30. .apply(
  31. StateTransforms.Representation.StructureRepresentation3D,
  32. createStructureRepresentationParams(plugin, structure.obj?.data, {
  33. type: 'cartoon',
  34. color: TmDetColorThemeProvider.name as any, colorParams: { pdbtmDescriptor }
  35. })
  36. );
  37. update.commit();
  38. }
  39. /**
  40. * Sets color of specified chain to red.
  41. *
  42. * @param plugin UI context
  43. * @param pdbtmDescriptor TM-specific information
  44. * @param transformationMatrix 4x4 matrix describes transformation and translation
  45. * @param chainId Id of chain to be selected for transformation
  46. */
  47. export function chainTransformation(plugin: PluginUIContext, pdbtmDescriptor: PDBTMDescriptor, transformationMatrix: PDBTMTransformationMatrix, chainId: string): void {
  48. const query: Expression = getChainExpression(chainId);
  49. const transformation = transformationForStateTransform(transformationMatrix);
  50. const structure: StateObjectRef<PMS> = plugin.managers.structure.hierarchy.current.models[0].structures[0].cell;
  51. const update: StateBuilder.To<any, any> = plugin.build().to(structure);
  52. update
  53. .apply(
  54. StateTransforms.Model.StructureSelectionFromExpression,
  55. { label: chainId, expression: query }
  56. )
  57. .apply(StateTransforms.Model.TransformStructureConformation, {
  58. "transform": { name: "matrix", params: { data: transformation, transpose: false } }
  59. })
  60. .apply(
  61. StateTransforms.Representation.StructureRepresentation3D,
  62. createStructureRepresentationParams(plugin, update.selector.data, {
  63. type: 'cartoon',
  64. color: TmDetColorThemeProvider.name as any, colorParams: { pdbtmDescriptor }
  65. })
  66. );
  67. update.commit();
  68. }
  69. function vadd(a: Vec3, b: Vec3): Vec3 {
  70. return Vec3.add(Vec3.zero(), a, b);
  71. }
  72. function vneg(u: Vec3): Vec3 {
  73. return Vec3.negate(Vec3.zero(), u);
  74. }
  75. export function createMembraneOrientation(pdbtmDescriptor: PDBTMDescriptor): MembraneOrientation {
  76. const membrane = pdbtmDescriptor.additional_entry_annotations.membrane;
  77. const membraneNormal: Vec3 = Vec3.fromObj(membrane.normal);
  78. const result: MembraneOrientation = {
  79. planePoint1: membraneNormal,
  80. planePoint2: vneg(membraneNormal),
  81. centroid: Vec3.zero(),
  82. normalVector: membraneNormal,
  83. // (NOTE: the TMDET extension calculates and sets it during applying preset)
  84. radius: membrane.radius
  85. };
  86. return result;
  87. }
  88. export function transformationForStateTransform(tmatrix: PDBTMTransformationMatrix): Mat4 {
  89. // matrix expected in column-major order
  90. const mx: Mat4 = Mat4.fromArray(Mat4.zero(),
  91. [
  92. tmatrix.rowx.x, tmatrix.rowy.x, tmatrix.rowz.x, 0,
  93. tmatrix.rowx.y, tmatrix.rowy.y, tmatrix.rowz.y, 0,
  94. tmatrix.rowx.z, tmatrix.rowy.z, tmatrix.rowz.z, 0,
  95. 0, 0, 0, 1
  96. ], 0
  97. );
  98. Mat4.setTranslation(mx, Vec3.create(
  99. tmatrix.rowx.t,
  100. tmatrix.rowy.t,
  101. tmatrix.rowz.t
  102. ));
  103. // TODO: console.log('MAT4\n', Mat4.makeTable(mx));
  104. return mx;
  105. }
  106. export function getAtomGroupExpression(chainId: string, auth_array: number[]): Expression {
  107. // TODO console.log('auth_array:', auth_array);
  108. const query: Expression =
  109. MS.struct.generator.atomGroups({
  110. 'residue-test': MS.core.set.has([MS.set( ...auth_array ), MS.ammp('auth_seq_id')]),
  111. 'chain-test': MS.core.rel.eq([chainId, MS.ammp('label_asym_id')])
  112. });
  113. return query;
  114. }
  115. export function getChainExpression(chainId: string): Expression {
  116. const query: Expression =
  117. MS.struct.generator.atomGroups({
  118. 'chain-test': MS.core.rel.eq([chainId, MS.ammp('label_asym_id')])
  119. });
  120. return query;
  121. }