tmdet-color-theme.ts 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  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 { StructureElement, Unit } from 'molstar/lib/mol-model/structure';
  10. import { ColorTheme } from 'molstar/lib/mol-theme/color';
  11. import { ThemeDataContext } from 'molstar/lib/mol-theme/theme';
  12. import { Color } from 'molstar/lib/mol-util/color';
  13. import { ColorNames } from 'molstar/lib/mol-util/color/names';
  14. import { ParamDefinition as PD } from 'molstar/lib/mol-util/param-definition';
  15. import { Location } from 'molstar/lib/mol-model/location';
  16. import { ChainList, getResidue, PDBTMChain } from './types';
  17. import { TmDetChainListCache, TmDetDescriptorCache } from './prop';
  18. export type TmDetColorThemeParams = {
  19. pdbtmDescriptor: any
  20. };
  21. export function TmDetColorTheme(
  22. ctx: ThemeDataContext,
  23. props: PD.Values<TmDetColorThemeParams>
  24. ): ColorTheme<TmDetColorThemeParams> {
  25. let descriptorChains: PDBTMChain[] = [];
  26. let pdbtmDescriptor = props.pdbtmDescriptor;
  27. const pdbId = ctx.structure?.model.entryId;
  28. // If it is not given as parameter,
  29. // but there is one in the cache for this structure.
  30. if (!pdbtmDescriptor) {
  31. if (pdbId) {
  32. pdbtmDescriptor = TmDetDescriptorCache.get(pdbId);
  33. }
  34. }
  35. if (pdbtmDescriptor) {
  36. descriptorChains = pdbtmDescriptor.chains;
  37. }
  38. const tmType = pdbtmDescriptor.additional_entry_annotations.tm_type;
  39. const chainList = createResidueListsPerChain(descriptorChains, pdbtmDescriptor.side1, tmType);
  40. if (pdbId) {
  41. TmDetChainListCache.set(pdbId, chainList);
  42. }
  43. return {
  44. factory: TmDetColorTheme,
  45. granularity: 'group', //'residue' as any,
  46. color: (location: Location) => getColor(location, chainList),
  47. props: props,
  48. description: 'TMDet...',
  49. };
  50. }
  51. const DefaultResidueColor = ColorNames.lightgrey;
  52. enum SiteIndexes {
  53. Side1 = 0,
  54. Side2 = 1,
  55. TmAlpha = 2,
  56. TmBeta = 3,
  57. TmReentrantLoop = 4,
  58. InterfacialHelix = 5,
  59. UnknownLocalization = 6,
  60. MembraneInside = 7,
  61. Periplasm = 8
  62. };
  63. // Old default values - it is overwritten by ult_* CSS classes
  64. // See below updateSiteColors().
  65. const siteColors = [
  66. Color.fromArray([255,100,100], 0), // Side1
  67. Color.fromArray([100,100,255], 0), // Side2
  68. Color.fromArray([255,255, 0], 0), // TM alpha
  69. Color.fromArray([255,255, 0], 0), // TM beta
  70. Color.fromArray([255,127, 0], 0), // TM re-entrant loop
  71. Color.fromArray([0,255, 0], 0), // Interfacial Helix
  72. Color.fromArray([196,196,196], 0), // Unknow localization
  73. Color.fromArray([0,255, 0], 0), // Membrane Inside
  74. Color.fromArray([255, 0, 255], 0) // Periplasm
  75. ];
  76. const siteCssNames = [
  77. "ult_side1",
  78. "ult_side2",
  79. "ult_alpha",
  80. "ult_beta",
  81. "ult_reentrant",
  82. "ult_ifh",
  83. "ult_unknown",
  84. "ult_membins",
  85. "ult_periplasm"
  86. ];
  87. const regionColorMapFromCss = new Map();
  88. // set default values
  89. siteCssNames.forEach((className, index) => {
  90. regionColorMapFromCss.set(className, siteColors[index]);
  91. });
  92. function getColor(location: Location, chains: ChainList): Color {
  93. let color = DefaultResidueColor;
  94. // TODO: How to solve cases when chain operation occurs?
  95. if (StructureElement.Location.is(location) && Unit.isAtomic(location.unit)) {
  96. const { chainId, residueId } = getChainAndResidueIds(location);
  97. color = residueColor(chains, chainId!, residueId!);
  98. }
  99. return color;
  100. }
  101. export function getChainAndResidueIds(location: Location) {
  102. let chainId = undefined;
  103. let residueId = undefined;
  104. if (StructureElement.Location.is(location) && Unit.isAtomic(location.unit)) {
  105. const atomicHierarchy = location.unit.model.atomicHierarchy;
  106. const residueIdx = StructureElement.residueIndex(location);
  107. const chainIdx = StructureElement.Location.chainIndex(location);
  108. residueId = atomicHierarchy.residues.label_seq_id.value(residueIdx).toString();
  109. chainId = atomicHierarchy.chains.label_asym_id.value(chainIdx).toString();
  110. }
  111. return {
  112. chainId: chainId,
  113. residueId: residueId
  114. };
  115. }
  116. export function createResidueListsPerChain(chains: PDBTMChain[], side1: string|null, tmType: string) {
  117. const chainList: ChainList = [];
  118. const hasBeta = chains.some(chain => chain.additional_chain_annotations.type == "beta");
  119. chains.forEach((chain: PDBTMChain) => {
  120. const chainType = chain.additional_chain_annotations.type;
  121. chainList.push({ chainId: chain.chain_label, type: chainType, residues: [] });
  122. let annotationErrorLogged = false;
  123. chain.residues.forEach((residue) => {
  124. let siteId = residue.site_data![0].site_id_ref - 1;
  125. let siteColorId = siteId;
  126. if (tmType == "Tm_Alpha" && hasBeta && side1 == "Periplasm") {
  127. if (siteId == SiteIndexes.Side1) {
  128. siteColorId = SiteIndexes.Periplasm;
  129. } else if (siteId == SiteIndexes.Side2) {
  130. if (chainType == "non_tm" || chainType == "alpha") {
  131. siteColorId = SiteIndexes.Side1;
  132. } else if (chainType == "beta") {
  133. siteColorId = SiteIndexes.Side2;
  134. }
  135. }
  136. } else if (chainType == "beta") {
  137. if (side1 == "Periplasm" && siteColorId == SiteIndexes.Side1
  138. || side1 == "Outside" && siteColorId == SiteIndexes.Side2) {
  139. siteColorId = SiteIndexes.Periplasm;
  140. } else if (!annotationErrorLogged && side1 == "Inside" && siteColorId == SiteIndexes.Side1) {
  141. console.error(`Annotation error: beta chain has inside region ${chain.chain_label}:${residue.pdb_res_label}`);
  142. annotationErrorLogged = true;
  143. }
  144. }
  145. chainList[chainList.length - 1].residues.push({
  146. authId: residue.pdb_res_label,
  147. siteId: siteId,
  148. siteColorId: siteColorId
  149. });
  150. });
  151. });
  152. return chainList;
  153. }
  154. function residueColor(chains: ChainList, chainId: string, residueId: string): Color {
  155. let color = DefaultResidueColor;
  156. const residue = getResidue(chains, chainId, residueId);
  157. if (residue) {
  158. color = siteColors[residue.siteColorId];
  159. }
  160. return color;
  161. }
  162. // Provider
  163. export const TmDetColorThemeProvider: ColorTheme.Provider<TmDetColorThemeParams, 'tmdet-custom-color-theme'> = {
  164. name: 'tmdet-custom-color-theme',
  165. label: 'TMDet Topology Theme',
  166. category: 'TMDet',
  167. factory: TmDetColorTheme,
  168. getParams: () => ({ pdbtmDescriptor: { isOptional: true } }),
  169. defaultValues: { pdbtmDescriptor: undefined },
  170. isApplicable: () => true,
  171. };
  172. // Colors from CSS rules
  173. function loadRegionColorsFromStyleSheets(prefix: string = 'ult_'): void {
  174. const sheets: CSSStyleSheet[] = Array.from(document.styleSheets);
  175. sheets.forEach((sheet: CSSStyleSheet) => {
  176. const rules: CSSRule[] = Array.from(sheet.cssRules);
  177. rules.forEach((rule: CSSRule) => fetchRule(rule, prefix));
  178. });
  179. }
  180. export function updateSiteColors(side1: "Inside"|"Outside"|null): void {
  181. if (!side1) {
  182. return;
  183. }
  184. loadRegionColorsFromStyleSheets();
  185. if (regionColorMapFromCss.size == 0) {
  186. console.warn('Cannot read any region-related color rules');
  187. }
  188. siteCssNames.forEach((ultClassName, index) => {
  189. const color = regionColorMapFromCss.get(ultClassName);
  190. if (color != null) {
  191. siteColors[index] = color;
  192. }
  193. });
  194. const inside = regionColorMapFromCss.get("ult_inside");
  195. const outside = regionColorMapFromCss.get("ult_outside");
  196. if (side1 == "Inside") {
  197. siteColors[SiteIndexes.Side1] = inside;
  198. siteColors[SiteIndexes.Side2] = outside;
  199. } else if (side1 == "Outside") {
  200. siteColors[SiteIndexes.Side1] = outside;
  201. siteColors[SiteIndexes.Side2] = inside;
  202. }
  203. }
  204. function fetchRule(rule: CSSRule, prefix: string) {
  205. let styleRule = rule as CSSStyleRule;
  206. if (styleRule.selectorText?.startsWith('.' + prefix)) {
  207. const value = styleRule.style.getPropertyValue('fill');
  208. const color: Color = getStyleColor(value);
  209. const key = styleRule.selectorText.slice(1);
  210. regionColorMapFromCss.set(key, color);
  211. }
  212. }
  213. function getStyleColor(cssColorText: string): Color {
  214. const values = cssColorText?.match(/\d+/g);
  215. let intValues = values?.map(value => parseInt(value));
  216. if (!intValues) {
  217. intValues = [ 0, 0, 0 ];
  218. }
  219. return Color.fromArray(intValues, 0);
  220. }