sifts-mapping.ts 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. /**
  2. * Copyright (c) 2021 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author David Sehnal <david.sehnal@gmail.com>
  5. */
  6. import { Column } from '../../mol-data/db';
  7. import { MmcifFormat } from '../../mol-model-formats/structure/mmcif';
  8. import { CustomPropertyDescriptor } from '../../mol-model/custom-property';
  9. import { Model } from '../../mol-model/structure';
  10. import { StructureElement } from '../../mol-model/structure/structure';
  11. import { CustomModelProperty } from '../common/custom-model-property';
  12. export { SIFTSMapping as SIFTSMapping };
  13. interface SIFTSMappingMapping {
  14. readonly dbName: string[],
  15. readonly accession: string[],
  16. readonly num: string[],
  17. readonly residue: string[]
  18. }
  19. namespace SIFTSMapping {
  20. export const Provider: CustomModelProperty.Provider<{}, SIFTSMappingMapping> = CustomModelProperty.createProvider({
  21. label: 'SIFTS Mapping',
  22. descriptor: CustomPropertyDescriptor({
  23. name: 'sifts_sequence_mapping'
  24. }),
  25. type: 'static',
  26. defaultParams: {},
  27. getParams: () => ({}),
  28. isApplicable: (data: Model) => isAvailable(data),
  29. obtain: async (ctx, data) => {
  30. return { value: fromCif(data) };
  31. }
  32. });
  33. export function isAvailable(model: Model) {
  34. if (!MmcifFormat.is(model.sourceData)) return false;
  35. const {
  36. pdbx_sifts_xref_db_name: db_name,
  37. pdbx_sifts_xref_db_acc: db_acc,
  38. pdbx_sifts_xref_db_num: db_num,
  39. pdbx_sifts_xref_db_res: db_res
  40. } = model.sourceData.data.db.atom_site;
  41. return db_name.isDefined && db_acc.isDefined && db_num.isDefined && db_res.isDefined;
  42. }
  43. export function getKey(loc: StructureElement.Location) {
  44. const model = loc.unit.model;
  45. const data = Provider.get(model).value;
  46. if (!data) return '';
  47. const eI = loc.unit.elements[loc.element];
  48. const rI = model.atomicHierarchy.residueAtomSegments.index[eI];
  49. return data.accession[rI];
  50. }
  51. export function getLabel(loc: StructureElement.Location) {
  52. const model = loc.unit.model;
  53. const data = Provider.get(model).value;
  54. if (!data) return;
  55. const eI = loc.unit.elements[loc.element];
  56. const rI = model.atomicHierarchy.residueAtomSegments.index[eI];
  57. const dbName = data.dbName[rI];
  58. if (!dbName) return;
  59. return `${dbName} ${data.accession[rI]} ${data.num[rI]} ${data.residue[rI]}`;
  60. }
  61. function fromCif(model: Model): SIFTSMappingMapping | undefined {
  62. if (!MmcifFormat.is(model.sourceData)) return;
  63. const {
  64. pdbx_sifts_xref_db_name: db_name,
  65. pdbx_sifts_xref_db_acc: db_acc,
  66. pdbx_sifts_xref_db_num: db_num,
  67. pdbx_sifts_xref_db_res: db_res
  68. } = model.sourceData.data.db.atom_site;
  69. if (!db_name.isDefined || !db_acc.isDefined || !db_num.isDefined || !db_res.isDefined) return;
  70. const { atomSourceIndex } = model.atomicHierarchy;
  71. const { count, offsets: residueOffsets } = model.atomicHierarchy.residueAtomSegments;
  72. const dbName = new Array<string>(count);
  73. const accession = new Array<string>(count);
  74. const num = new Array<string>(count);
  75. const residue = new Array<string>(count);
  76. for (let i = 0; i < count; i++) {
  77. const row = atomSourceIndex.value(residueOffsets[i]);
  78. if (db_name.valueKind(row) !== Column.ValueKind.Present) {
  79. dbName[i] = '';
  80. accession[i] = '';
  81. num[i] = '';
  82. residue[i] = '';
  83. continue;
  84. }
  85. dbName[i] = db_name.value(row);
  86. accession[i] = db_acc.value(row);
  87. num[i] = db_num.value(row);
  88. residue[i] = db_res.value(row);
  89. }
  90. return { dbName, accession, num, residue };
  91. }
  92. }