symmetry.ts 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. /**
  2. * Copyright (C) 2022, Protein Bioinformatics Research Group, RCNS
  3. *
  4. * Licensed under CC BY-NC 4.0, see LICENSE file for more info.
  5. *
  6. * @author Gabor Tusnady <tusnady.gabor@ttk.hu>
  7. * @author Csongor Gerdan <gerdan.csongor@ttk.hu>
  8. */
  9. import { MmcifFormat } from '../mol-model-formats/structure/mmcif';
  10. import { Column, Table } from '../mol-data/db';
  11. import { mmCIF_Schema } from '../mol-io/reader/cif/schema/mmcif';
  12. import { Model } from '../mol-model/structure';
  13. import { ModelSymmetry } from '../mol-model-formats/structure/property/symmetry';
  14. import { PDBTMDescriptor } from './types';
  15. import { Mat3, Mat4, Vec3 } from '../mol-math/linear-algebra';
  16. import { transformationForStateTransform } from './transformation';
  17. export function registerTmDetSymmetry(pdbtmDescriptor: PDBTMDescriptor) {
  18. ModelSymmetry.Provider.formatRegistry.remove('mmCIF');
  19. ModelSymmetry.Provider.formatRegistry.add('mmCIF', function(model: Model) {
  20. return tmDetSymmetryFromMmCif(model, pdbtmDescriptor);
  21. });
  22. }
  23. function constructChainListFromOperations(pdbtmDescriptor: PDBTMDescriptor): string[] {
  24. const excludedChains: string[] = [];
  25. // add chain deletes
  26. const biomatrix = pdbtmDescriptor.additional_entry_annotations.biomatrix;
  27. if (biomatrix?.chain_deletes) {
  28. biomatrix.chain_deletes.forEach(
  29. chainId => excludedChains.push(chainId)
  30. );
  31. }
  32. // exclude result of transformations
  33. if (biomatrix?.matrix_list) {
  34. biomatrix.matrix_list.forEach(
  35. matrix => matrix.apply_to_chain_list.forEach(
  36. applyItem => excludedChains.push(applyItem.new_chain_id)
  37. )
  38. );
  39. }
  40. return excludedChains;
  41. }
  42. function tmDetSymmetryFromMmCif(model: Model, pdbtmDescriptor: PDBTMDescriptor) {
  43. if (!MmcifFormat.is(model.sourceData)) return;
  44. let data = model.sourceData.data.db;
  45. let excludedChains: string[] = constructChainListFromOperations(pdbtmDescriptor);
  46. excludedChains = union(
  47. excludedChains,
  48. Array.from(data.pdbx_nonpoly_scheme.asym_id.toArray())
  49. );
  50. const structureOperations: StructureOperation[] = collectOperations(pdbtmDescriptor);
  51. console.log("Structure Operators:", structureOperations);
  52. const updated_pdbx_struct_assembly_gen = createPdbxStructAssemblyGen(
  53. structureOperations,
  54. data.pdbx_struct_assembly_gen,
  55. excludedChains
  56. );
  57. console.log('Non-poly entities:', Table.formatToString(data.pdbx_entity_nonpoly));
  58. console.log('Non-poly chains:', data.pdbx_nonpoly_scheme.asym_id.toArray());
  59. const updated_struct_oper_list = createPdbxStructOperList(structureOperations, data.pdbx_struct_oper_list);
  60. console.log('Orig. struct oper list:', data.pdbx_struct_oper_list);
  61. return ModelSymmetry.fromData({
  62. symmetry: data.symmetry,
  63. cell: data.cell,
  64. struct_ncs_oper: data.struct_ncs_oper,
  65. atom_sites: data.atom_sites,
  66. pdbx_struct_assembly: data.pdbx_struct_assembly,
  67. pdbx_struct_assembly_gen: updated_pdbx_struct_assembly_gen,
  68. pdbx_struct_oper_list: updated_struct_oper_list
  69. });
  70. }
  71. function createPdbxStructAssemblyGen(
  72. structureOperations: StructureOperation[],
  73. pdbx_struct_assembly_gen: Table<mmCIF_Schema['pdbx_struct_assembly_gen']>,
  74. excludedChains: string[]): Table<mmCIF_Schema['pdbx_struct_assembly_gen']> {
  75. const asym_id_list_column = createAsymIdColumn(
  76. pdbx_struct_assembly_gen, excludedChains
  77. );
  78. const assembly_id: string[] = [];
  79. const asym_id_list: string[][] = [];
  80. const oper_expression: string[] = [];
  81. // we create only one assembly - maybe in multiple rows
  82. assembly_id.push("1");
  83. asym_id_list.push(asym_id_list_column);
  84. oper_expression.push("0"); // this is the basic membrane plane transformation
  85. const chainMap = new Map<string, string[]>();
  86. // collect each operator for each chain
  87. structureOperations.slice(1).forEach(function (operation: StructureOperation) {
  88. if (!chainMap.has(operation.chainId)) {
  89. chainMap.set(operation.chainId, [ operation.id.toString() ]);
  90. } else {
  91. chainMap.get(operation.chainId)?.push(operation.id.toString());
  92. }
  93. });
  94. // fill columns of table
  95. chainMap.forEach(function(operators, chain) {
  96. assembly_id.push("1");
  97. asym_id_list.push([ chain ]);
  98. oper_expression.push(operators.join(","));
  99. });
  100. // create table with new column
  101. let updated_pdbx_struct_assembly_gen = Table.ofColumns(
  102. pdbx_struct_assembly_gen._schema,
  103. {
  104. assembly_id: Column.ofStringArray(assembly_id),
  105. asym_id_list: Column.ofStringListArray(asym_id_list),
  106. oper_expression: Column.ofStringArray(oper_expression)
  107. }
  108. );
  109. console.log('Orig. assembly_gen', Table.formatToString(pdbx_struct_assembly_gen));
  110. console.log('Updated assembly_gen', Table.formatToString(updated_pdbx_struct_assembly_gen));
  111. return updated_pdbx_struct_assembly_gen;
  112. }
  113. function createPdbxStructOperList(
  114. structureOperations: StructureOperation[],
  115. pdbx_struct_oper_list: Table<mmCIF_Schema['pdbx_struct_oper_list']>):
  116. Table<mmCIF_Schema['pdbx_struct_oper_list']> {
  117. const id: string[] = [];
  118. const type: string[] = [];
  119. const name: string[] = [];
  120. const symmetry_operation: string[] = [];
  121. const matrix: number[][] = [];
  122. const vector: number[][] = [];
  123. structureOperations.forEach(function (operation: StructureOperation) {
  124. id.push(operation.id.toString());
  125. type.push(".");
  126. const operationName = (operation.id == 0)
  127. ? "Membrane transformation as identity operation"
  128. : `${operation.chainId} -> ${operation.newChainId}`;
  129. name.push(operationName);
  130. symmetry_operation.push(".");
  131. const rotation: number[] = Mat3.toArray(operation.operator.rotation, [], 0);
  132. const translation: number[] = Vec3.toArray(operation.operator.translation, [], 0);;
  133. matrix.push(rotation);
  134. vector.push(translation);
  135. });
  136. let updated_pdbx_struct_oper_list = Table.ofColumns(
  137. pdbx_struct_oper_list._schema,
  138. {
  139. id: Column.ofStringArray(id),
  140. type: Column.ofArray({
  141. array: type,
  142. schema: pdbx_struct_oper_list.type.schema
  143. }),
  144. name: Column.ofStringArray(name),
  145. symmetry_operation: Column.ofStringArray(symmetry_operation),
  146. matrix: Column.ofArray({
  147. array: matrix,
  148. schema: pdbx_struct_oper_list.matrix.schema
  149. }),
  150. vector: Column.ofArray({
  151. array: vector,
  152. schema: pdbx_struct_oper_list.vector.schema
  153. })
  154. }
  155. );
  156. console.log('Orig. pdbx_struct_oper_list', Table.formatToString(pdbx_struct_oper_list));
  157. console.log('Updated pdbx_struct_oper_list', Table.formatToString(updated_pdbx_struct_oper_list));
  158. return updated_pdbx_struct_oper_list;
  159. }
  160. function createAsymIdColumn(pdbx_struct_assembly_gen: Table<mmCIF_Schema['pdbx_struct_assembly_gen']>,
  161. excludedChains: string[]) {
  162. let asym_id_list: string[] = [];
  163. for (let i = 0; i < pdbx_struct_assembly_gen._rowCount; i++) {
  164. const currentAsymIdList = pdbx_struct_assembly_gen.asym_id_list.value(i);
  165. asym_id_list = asym_id_list.concat(currentAsymIdList);
  166. }
  167. asym_id_list = minus(asym_id_list, excludedChains);
  168. console.log('Excluded chains:', excludedChains);
  169. console.log('Included chains:', asym_id_list);
  170. return asym_id_list;
  171. }
  172. // difference of two string arrays (interpreted as sets)
  173. function minus(a: string[], b: string[]): string[] {
  174. const b_set = new Set(b);
  175. const difference = a.filter(x => !b_set.has(x));
  176. return Array.from(new Set(difference).values());
  177. }
  178. // union of two string arrays (interpreted as sets)
  179. function union(a: string[], b: string[]): string[] {
  180. const a_set = new Set(a);
  181. b.forEach(item => a_set.add(item));
  182. return Array.from(a_set.values());
  183. }
  184. //////////////////////////////////////////////////////////////////////
  185. /*
  186. * Construction of struct operators
  187. */
  188. /**
  189. * Type to contain operation properties.
  190. */
  191. type StructureOperation = {
  192. id: number,
  193. chainId: string,
  194. newChainId: string,
  195. operator: {
  196. rotation: Mat3,
  197. translation: Vec3
  198. }
  199. }
  200. function collectOperations(descriptor: PDBTMDescriptor): StructureOperation[] {
  201. const operations: StructureOperation[] = [];
  202. const annotations = descriptor.additional_entry_annotations;
  203. const membraneTransformation = transformationForStateTransform(descriptor.additional_entry_annotations.membrane.transformation_matrix);
  204. operations.push({
  205. id: 0,
  206. chainId: "Any chain with identity operation",
  207. newChainId: ".",
  208. operator: mat4ToRotationAndTranslation(
  209. Mat4.mul(Mat4(), Mat4.rotX90, membraneTransformation)
  210. )
  211. });
  212. let operationId: number = 1;
  213. if (annotations?.biomatrix?.matrix_list) {
  214. annotations.biomatrix.matrix_list.forEach(function(mx) {
  215. mx.apply_to_chain_list.forEach(function(chainPair) {
  216. let id = chainPair.chain_id;
  217. let newId = chainPair.new_chain_id;
  218. if (annotations.biomatrix.chain_deletes?.includes(newId)) {
  219. console.log(`${id} -> ${newId} transformation skipped due to delete rule`);
  220. return;
  221. }
  222. const mtx = transformationForStateTransform(mx.transformation_matrix);
  223. const composedTransformation = Mat4.mul(
  224. Mat4(),
  225. Mat4.rotX90,
  226. Mat4.mul(Mat4(), membraneTransformation, mtx)
  227. );
  228. const operator: StructureOperation = {
  229. id: operationId,
  230. chainId: id,
  231. newChainId: newId,
  232. operator: mat4ToRotationAndTranslation(composedTransformation)
  233. };
  234. operations.push(operator);
  235. operationId++;
  236. });
  237. });
  238. }
  239. return operations;
  240. }
  241. /**
  242. * Convert Mat4 object to array representation of its 3x4 part.
  243. * Rotation matrix 3x3 and translation vector 3x1.
  244. *
  245. * @param mat4 Mat4 object
  246. * @returns Object with rotation matrix and translation vector
  247. */
  248. function mat4ToRotationAndTranslation(mat4: Mat4) {
  249. return {
  250. rotation: Mat3.fromMat4(
  251. Mat3.zero(), Mat4.extractRotation(Mat4.zero(), mat4)
  252. ),
  253. translation: Mat4.getTranslation(Vec3.zero(), mat4)
  254. };
  255. }