labeling.ts 3.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  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 { Loci } from 'molstar/lib/mol-model/loci';
  10. import { StructureElement } from 'molstar/lib/mol-model/structure';
  11. import { LociLabel, LociLabelProvider } from 'molstar/lib/mol-plugin-state/manager/loci-label';
  12. import { TmDetChainListCache, TmDetDescriptorCache } from './prop';
  13. import { createResidueListsPerChain, getChainAndResidueIds } from './tmdet-color-theme';
  14. import { ChainList, getResidue, ResidueItem } from './types';
  15. const siteLabels = [
  16. "Side1",
  17. "Side2",
  18. "TM alpha",
  19. "TM beta",
  20. "TM re-entrant loop",
  21. "Interfacial Helix",
  22. "Unknown localization",
  23. "Membrane Inside"
  24. ];
  25. const DefaultResidueLabel = 6; // Unknown localization
  26. export const TmDetLabelProvider: LociLabelProvider = {
  27. label: (loci: Loci): LociLabel => {
  28. let labelText = siteLabels[DefaultResidueLabel];
  29. if (loci.kind == 'element-loci') {
  30. const unit = loci.elements[0].unit;
  31. const pdbId = unit.model.entryId;
  32. const descriptor = TmDetDescriptorCache.get(pdbId);
  33. if (!descriptor) {
  34. return labelText;
  35. }
  36. let chainList = TmDetChainListCache.get(pdbId);
  37. if (!chainList) {
  38. const tmType = descriptor.additional_entry_annotations.tm_type;
  39. chainList = createResidueListsPerChain(descriptor.chains, descriptor.side1, tmType);
  40. TmDetChainListCache.set(pdbId, chainList);
  41. }
  42. const location = StructureElement.Loci.getFirstLocation(loci);
  43. const { chainId, residueId } = getChainAndResidueIds(location!);
  44. const residue = getResidue(chainList, chainId!, residueId!);
  45. if (residue) {
  46. labelText = siteLabels[residue?.siteId];
  47. let regionText = getRegionText(chainList, chainId!, residue);
  48. labelText = `${labelText}: ${regionText}`
  49. }
  50. }
  51. return labelText;
  52. }
  53. }
  54. function getRegionText(chainList: ChainList, chainId: string, residue: ResidueItem): string {
  55. let value = "Unknown region range";
  56. const chain = chainList.filter((chain) => chain.chainId === chainId)[0];
  57. if (chain) {
  58. // find start of region
  59. const residues = chain.residues;
  60. const authId = parseInt(residue.authId!)
  61. let previous = residues[residues.length-1];
  62. for (let i = residues.length-1; i > 0; i--) {
  63. const current = residues[i];
  64. const currentId = parseInt(current.authId!);
  65. // cancel loop when siteId changes
  66. if (currentId < authId && current.siteId != residue.siteId) {
  67. break;
  68. }
  69. previous = current;
  70. }
  71. value = `[ ${previous.authId}`;
  72. // find end of region
  73. previous = residues[0];
  74. for (let current of residues) {
  75. const currentId = parseInt(current.authId!);
  76. // cancel loop when siteId changes
  77. if (authId < currentId && current.siteId !== residue.siteId) {
  78. break;
  79. }
  80. previous = current;
  81. }
  82. value = `${value}-${previous.authId} ]`;
  83. }
  84. return value;
  85. }