element-symbol.ts 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. /**
  2. * Copyright (c) 2018-2022 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { ElementSymbol } from '../../mol-model/structure/model/types';
  7. import { Color, ColorMap } from '../../mol-util/color';
  8. import { StructureElement, Unit, Bond } from '../../mol-model/structure';
  9. import { Location } from '../../mol-model/location';
  10. import { ColorTheme } from '../color';
  11. import { ParamDefinition as PD } from '../../mol-util/param-definition';
  12. import { ThemeDataContext } from '../theme';
  13. import { TableLegend } from '../../mol-util/legend';
  14. import { getAdjustedColorMap } from '../../mol-util/color/color';
  15. import { getColorMapParams } from '../../mol-util/color/params';
  16. import { ChainIdColorTheme, ChainIdColorThemeParams } from './chain-id';
  17. import { OperatorNameColorThemeParams, OperatorNameColorTheme } from './operator-name';
  18. import { EntityIdColorTheme, EntityIdColorThemeParams } from './entity-id';
  19. import { assertUnreachable } from '../../mol-util/type-helpers';
  20. import { EntitySourceColorTheme, EntitySourceColorThemeParams } from './entity-source';
  21. import { ModelIndexColorTheme, ModelIndexColorThemeParams } from './model-index';
  22. // from Jmol http://jmol.sourceforge.net/jscolors/ (or 0xFFFFFF)
  23. export const ElementSymbolColors = ColorMap({
  24. 'H': 0xFFFFFF, 'D': 0xFFFFC0, 'T': 0xFFFFA0, 'HE': 0xD9FFFF, 'LI': 0xCC80FF, 'BE': 0xC2FF00, 'B': 0xFFB5B5, 'C': 0x909090, 'N': 0x3050F8, 'O': 0xFF0D0D, 'F': 0x90E050, 'NE': 0xB3E3F5, 'NA': 0xAB5CF2, 'MG': 0x8AFF00, 'AL': 0xBFA6A6, 'SI': 0xF0C8A0, 'P': 0xFF8000, 'S': 0xFFFF30, 'CL': 0x1FF01F, 'AR': 0x80D1E3, 'K': 0x8F40D4, 'CA': 0x3DFF00, 'SC': 0xE6E6E6, 'TI': 0xBFC2C7, 'V': 0xA6A6AB, 'CR': 0x8A99C7, 'MN': 0x9C7AC7, 'FE': 0xE06633, 'CO': 0xF090A0, 'NI': 0x50D050, 'CU': 0xC88033, 'ZN': 0x7D80B0, 'GA': 0xC28F8F, 'GE': 0x668F8F, 'AS': 0xBD80E3, 'SE': 0xFFA100, 'BR': 0xA62929, 'KR': 0x5CB8D1, 'RB': 0x702EB0, 'SR': 0x00FF00, 'Y': 0x94FFFF, 'ZR': 0x94E0E0, 'NB': 0x73C2C9, 'MO': 0x54B5B5, 'TC': 0x3B9E9E, 'RU': 0x248F8F, 'RH': 0x0A7D8C, 'PD': 0x006985, 'AG': 0xC0C0C0, 'CD': 0xFFD98F, 'IN': 0xA67573, 'SN': 0x668080, 'SB': 0x9E63B5, 'TE': 0xD47A00, 'I': 0x940094, 'XE': 0x940094, 'CS': 0x57178F, 'BA': 0x00C900, 'LA': 0x70D4FF, 'CE': 0xFFFFC7, 'PR': 0xD9FFC7, 'ND': 0xC7FFC7, 'PM': 0xA3FFC7, 'SM': 0x8FFFC7, 'EU': 0x61FFC7, 'GD': 0x45FFC7, 'TB': 0x30FFC7, 'DY': 0x1FFFC7, 'HO': 0x00FF9C, 'ER': 0x00E675, 'TM': 0x00D452, 'YB': 0x00BF38, 'LU': 0x00AB24, 'HF': 0x4DC2FF, 'TA': 0x4DA6FF, 'W': 0x2194D6, 'RE': 0x267DAB, 'OS': 0x266696, 'IR': 0x175487, 'PT': 0xD0D0E0, 'AU': 0xFFD123, 'HG': 0xB8B8D0, 'TL': 0xA6544D, 'PB': 0x575961, 'BI': 0x9E4FB5, 'PO': 0xAB5C00, 'AT': 0x754F45, 'RN': 0x428296, 'FR': 0x420066, 'RA': 0x007D00, 'AC': 0x70ABFA, 'TH': 0x00BAFF, 'PA': 0x00A1FF, 'U': 0x008FFF, 'NP': 0x0080FF, 'PU': 0x006BFF, 'AM': 0x545CF2, 'CM': 0x785CE3, 'BK': 0x8A4FE3, 'CF': 0xA136D4, 'ES': 0xB31FD4, 'FM': 0xB31FBA, 'MD': 0xB30DA6, 'NO': 0xBD0D87, 'LR': 0xC70066, 'RF': 0xCC0059, 'DB': 0xD1004F, 'SG': 0xD90045, 'BH': 0xE00038, 'HS': 0xE6002E, 'MT': 0xEB0026, 'DS': 0xFFFFFF, 'RG': 0xFFFFFF, 'CN': 0xFFFFFF, 'UUT': 0xFFFFFF, 'FL': 0xFFFFFF, 'UUP': 0xFFFFFF, 'LV': 0xFFFFFF, 'UUH': 0xFFFFFF
  25. });
  26. export type ElementSymbolColors = typeof ElementSymbolColors
  27. const DefaultElementSymbolColor = Color(0xFFFFFF);
  28. const Description = 'Assigns a color to every atom according to its chemical element.';
  29. export const ElementSymbolColorThemeParams = {
  30. carbonColor: PD.MappedStatic('chain-id', {
  31. 'chain-id': PD.Group(ChainIdColorThemeParams),
  32. 'entity-id': PD.Group(EntityIdColorThemeParams),
  33. 'entity-source': PD.Group(EntitySourceColorThemeParams),
  34. 'operator-name': PD.Group(OperatorNameColorThemeParams),
  35. 'model-index': PD.Group(ModelIndexColorThemeParams),
  36. 'element-symbol': PD.EmptyGroup()
  37. }, { description: 'Use chain-id coloring for carbon atoms.' }),
  38. saturation: PD.Numeric(0, { min: -6, max: 6, step: 0.1 }),
  39. lightness: PD.Numeric(0.2, { min: -6, max: 6, step: 0.1 }),
  40. colors: PD.MappedStatic('default', {
  41. 'default': PD.EmptyGroup(),
  42. 'custom': PD.Group(getColorMapParams(ElementSymbolColors))
  43. })
  44. };
  45. export type ElementSymbolColorThemeParams = typeof ElementSymbolColorThemeParams
  46. export function getElementSymbolColorThemeParams(ctx: ThemeDataContext) {
  47. return PD.clone(ElementSymbolColorThemeParams);
  48. }
  49. export function elementSymbolColor(colorMap: ElementSymbolColors, element: ElementSymbol): Color {
  50. const c = colorMap[element as keyof ElementSymbolColors];
  51. return c === undefined ? DefaultElementSymbolColor : c;
  52. }
  53. export function ElementSymbolColorTheme(ctx: ThemeDataContext, props: PD.Values<ElementSymbolColorThemeParams>): ColorTheme<ElementSymbolColorThemeParams> {
  54. const colorMap = getAdjustedColorMap(props.colors.name === 'default' ? ElementSymbolColors : props.colors.params, props.saturation, props.lightness);
  55. const pcc = props.carbonColor;
  56. const carbonColor =
  57. pcc.name === 'chain-id' ? ChainIdColorTheme(ctx, pcc.params).color :
  58. pcc.name === 'entity-id' ? EntityIdColorTheme(ctx, pcc.params).color :
  59. pcc.name === 'entity-source' ? EntitySourceColorTheme(ctx, pcc.params).color :
  60. pcc.name === 'operator-name' ? OperatorNameColorTheme(ctx, pcc.params).color :
  61. pcc.name === 'model-index' ? ModelIndexColorTheme(ctx, pcc.params).color :
  62. pcc.name === 'element-symbol' ? undefined :
  63. assertUnreachable(pcc);
  64. function elementColor(element: ElementSymbol, location: Location) {
  65. return (carbonColor && element === 'C')
  66. ? carbonColor(location, false)
  67. : elementSymbolColor(colorMap, element);
  68. }
  69. function color(location: Location): Color {
  70. if (StructureElement.Location.is(location)) {
  71. if (Unit.isAtomic(location.unit)) {
  72. const { type_symbol } = location.unit.model.atomicHierarchy.atoms;
  73. return elementColor(type_symbol.value(location.element), location);
  74. }
  75. } else if (Bond.isLocation(location)) {
  76. if (Unit.isAtomic(location.aUnit)) {
  77. const { type_symbol } = location.aUnit.model.atomicHierarchy.atoms;
  78. const element = type_symbol.value(location.aUnit.elements[location.aIndex]);
  79. return elementColor(element, location);
  80. }
  81. }
  82. return DefaultElementSymbolColor;
  83. }
  84. const granularity = props.carbonColor.name === 'operator-name' ? 'groupInstance' : 'group';
  85. return {
  86. factory: ElementSymbolColorTheme,
  87. granularity,
  88. preferSmoothing: true,
  89. color,
  90. props,
  91. description: Description,
  92. legend: TableLegend(Object.keys(colorMap).map(name => {
  93. return [name, (colorMap as any)[name] as Color] as [string, Color];
  94. }))
  95. };
  96. }
  97. export const ElementSymbolColorThemeProvider: ColorTheme.Provider<ElementSymbolColorThemeParams, 'element-symbol'> = {
  98. name: 'element-symbol',
  99. label: 'Element Symbol',
  100. category: ColorTheme.Category.Atom,
  101. factory: ElementSymbolColorTheme,
  102. getParams: getElementSymbolColorThemeParams,
  103. defaultValues: PD.getDefaultValues(ElementSymbolColorThemeParams),
  104. isApplicable: (ctx: ThemeDataContext) => !!ctx.structure
  105. };