element-symbol.ts 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. /**
  2. * Copyright (c) 2018-2020 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 { ChainIdColorTheme, ChainIdColorThemeParams } from './chain-id';
  16. import { OperatorNameColorThemeParams, OperatorNameColorTheme } from './operator-name';
  17. // from Jmol http://jmol.sourceforge.net/jscolors/ (or 0xFFFFFF)
  18. export const ElementSymbolColors = ColorMap({
  19. '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
  20. });
  21. export type ElementSymbolColors = typeof ElementSymbolColors
  22. const DefaultElementSymbolColor = Color(0xFFFFFF);
  23. const Description = 'Assigns a color to every atom according to its chemical element.';
  24. // TODO generalise `carbonColor` param to all themes?
  25. export const ElementSymbolColorThemeParams = {
  26. carbonColor: PD.MappedStatic('chain-id', {
  27. 'chain-id': PD.Group({ ...ChainIdColorThemeParams }),
  28. 'operator-name': PD.Group({ ...OperatorNameColorThemeParams }),
  29. 'element-symbol': PD.Group({})
  30. }, { description: 'Use chain-id coloring for carbon atoms.' }),
  31. saturation: PD.Numeric(0, { min: -6, max: 6, step: 0.1 }),
  32. lightness: PD.Numeric(0.2, { min: -6, max: 6, step: 0.1 })
  33. };
  34. export type ElementSymbolColorThemeParams = typeof ElementSymbolColorThemeParams
  35. export function getElementSymbolColorThemeParams(ctx: ThemeDataContext) {
  36. return ElementSymbolColorThemeParams; // TODO return copy
  37. }
  38. export function elementSymbolColor(colorMap: ElementSymbolColors, element: ElementSymbol): Color {
  39. const c = colorMap[element as keyof ElementSymbolColors];
  40. return c === undefined ? DefaultElementSymbolColor : c;
  41. }
  42. export function ElementSymbolColorTheme(ctx: ThemeDataContext, props: PD.Values<ElementSymbolColorThemeParams>): ColorTheme<ElementSymbolColorThemeParams> {
  43. const colorMap = getAdjustedColorMap(ElementSymbolColors, props.saturation, props.lightness);
  44. const carbonColor = props.carbonColor.name === 'chain-id'
  45. ? ChainIdColorTheme(ctx, props.carbonColor.params).color
  46. : props.carbonColor.name === 'operator-name'
  47. ? OperatorNameColorTheme(ctx, props.carbonColor.params).color
  48. : undefined;
  49. function elementColor(element: ElementSymbol, location: Location) {
  50. return (carbonColor && element === 'C')
  51. ? carbonColor(location, false)
  52. : elementSymbolColor(colorMap, element);
  53. }
  54. function color(location: Location): Color {
  55. if (StructureElement.Location.is(location)) {
  56. if (Unit.isAtomic(location.unit)) {
  57. const { type_symbol } = location.unit.model.atomicHierarchy.atoms;
  58. return elementColor(type_symbol.value(location.element), location);
  59. }
  60. } else if (Bond.isLocation(location)) {
  61. if (Unit.isAtomic(location.aUnit)) {
  62. const { type_symbol } = location.aUnit.model.atomicHierarchy.atoms;
  63. const element = type_symbol.value(location.aUnit.elements[location.aIndex]);
  64. return elementColor(element, location);
  65. }
  66. }
  67. return DefaultElementSymbolColor;
  68. }
  69. return {
  70. factory: ElementSymbolColorTheme,
  71. granularity: 'groupInstance',
  72. color,
  73. props,
  74. description: Description,
  75. legend: TableLegend(Object.keys(ElementSymbolColors).map(name => {
  76. return [name, (ElementSymbolColors as any)[name] as Color] as [string, Color];
  77. }))
  78. };
  79. }
  80. export const ElementSymbolColorThemeProvider: ColorTheme.Provider<ElementSymbolColorThemeParams, 'element-symbol'> = {
  81. name: 'element-symbol',
  82. label: 'Element Symbol',
  83. category: ColorTheme.Category.Atom,
  84. factory: ElementSymbolColorTheme,
  85. getParams: getElementSymbolColorThemeParams,
  86. defaultValues: PD.getDefaultValues(ElementSymbolColorThemeParams),
  87. isApplicable: (ctx: ThemeDataContext) => !!ctx.structure
  88. };