multilayer-color-theme.ts 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /**
  2. * Copyright (c) 2023-2024 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Adam Midlik <midlik@gmail.com>
  5. */
  6. import { Location } from '../../../mol-model/location';
  7. import { Bond, Structure, StructureElement } from '../../../mol-model/structure';
  8. import { ColorTheme, LocationColor } from '../../../mol-theme/color';
  9. import { ThemeDataContext } from '../../../mol-theme/theme';
  10. import { Color } from '../../../mol-util/color';
  11. import { ColorNames } from '../../../mol-util/color/names';
  12. import { ParamDefinition as PD } from '../../../mol-util/param-definition';
  13. import { stringToWords } from '../../../mol-util/string';
  14. import { isMVSStructure } from './is-mvs-model-prop';
  15. import { ElementSet, SelectorParams, isSelectorAll } from './selector';
  16. /** Special value that can be used as color with null-like semantic (i.e. "no color provided").
  17. * By some lucky coincidence, Mol* treats -1 as white. */
  18. export const NoColor = Color(-1);
  19. /** Return true if `color` is a real color, false if it is `NoColor`. */
  20. function isValidColor(color: Color): boolean {
  21. return color >= 0;
  22. }
  23. const DefaultBackgroundColor = ColorNames.white;
  24. /** Parameter definition for color theme "Multilayer" */
  25. export function makeMultilayerColorThemeParams(colorThemeRegistry: ColorTheme.Registry, ctx: ThemeDataContext) {
  26. const colorThemeInfo = {
  27. help: (value: { name: string, params: {} }) => {
  28. const { name, params } = value;
  29. const p = colorThemeRegistry.get(name);
  30. const ct = p.factory({}, params);
  31. return { description: ct.description, legend: ct.legend };
  32. }
  33. };
  34. const nestedThemeTypes = colorThemeRegistry.types.filter(([name, label, category]) => name !== MultilayerColorThemeName && colorThemeRegistry.get(name).isApplicable(ctx)); // Adding 'multilayer' theme itself would cause infinite recursion
  35. return {
  36. layers: PD.ObjectList(
  37. {
  38. theme: PD.Mapped<any>(
  39. 'uniform',
  40. nestedThemeTypes,
  41. name => PD.Group<any>(colorThemeRegistry.get(name).getParams({ structure: Structure.Empty })),
  42. colorThemeInfo),
  43. selection: SelectorParams,
  44. },
  45. obj => stringToWords(obj.theme.name),
  46. { description: 'A list of layers, each defining a color theme. The last listed layer is the top layer (applies first). If the top layer does not provide color for a location or its selection does not cover the location, the underneath layers will apply.' }),
  47. background: PD.Color(DefaultBackgroundColor, { description: 'Color for elements where no layer applies' }),
  48. };
  49. }
  50. /** Parameter definition for color theme "Multilayer" */
  51. export type MultilayerColorThemeParams = ReturnType<typeof makeMultilayerColorThemeParams>
  52. /** Parameter values for color theme "Multilayer" */
  53. export type MultilayerColorThemeProps = PD.Values<MultilayerColorThemeParams>
  54. /** Default values for `MultilayerColorThemeProps` */
  55. export const DefaultMultilayerColorThemeProps: MultilayerColorThemeProps = { layers: [], background: DefaultBackgroundColor };
  56. /** Return color theme that assigns colors based on a list of nested color themes (layers).
  57. * The last layer in the list whose selection covers the given location
  58. * and which provides a valid (non-negative) color value will be used.
  59. * If a nested theme provider has `ensureCustomProperties` methods, these will not be called automatically
  60. * (the caller must ensure that any required custom properties be attached). */
  61. function makeMultilayerColorTheme(ctx: ThemeDataContext, props: MultilayerColorThemeProps, colorThemeRegistry: ColorTheme.Registry): ColorTheme<MultilayerColorThemeParams> {
  62. const colorLayers: { color: LocationColor, elementSet: ElementSet | undefined }[] = []; // undefined elementSet means 'all'
  63. for (let i = props.layers.length - 1; i >= 0; i--) { // iterate from end to get top layer first, bottom layer last
  64. const layer = props.layers[i];
  65. const themeProvider = colorThemeRegistry.get(layer.theme.name);
  66. if (!themeProvider) {
  67. console.warn(`Skipping color theme '${layer.theme.name}', cannot find it in registry.`);
  68. continue;
  69. }
  70. if (themeProvider.ensureCustomProperties?.attach) {
  71. console.warn(`Multilayer color theme: layer "${themeProvider.name}" has ensureCustomProperties.attach method, but Multilayer color theme does not call it. If the layer does not work, make sure you call ensureCustomProperties.attach somewhere.`);
  72. }
  73. const theme = themeProvider.factory(ctx, layer.theme.params);
  74. switch (theme.granularity) {
  75. case 'uniform':
  76. case 'instance':
  77. case 'group':
  78. case 'groupInstance':
  79. case 'vertex':
  80. case 'vertexInstance':
  81. const elementSet = isSelectorAll(layer.selection) ? undefined : ElementSet.fromSelector(ctx.structure, layer.selection); // treating 'all' specially for performance reasons (it's expected to be used most often)
  82. colorLayers.push({ color: theme.color, elementSet });
  83. break;
  84. default:
  85. console.warn(`Skipping color theme '${layer.theme.name}', cannot process granularity '${theme.granularity}'`);
  86. }
  87. };
  88. function structureElementColor(loc: StructureElement.Location, isSecondary: boolean): Color {
  89. for (const layer of colorLayers) {
  90. const matches = !layer.elementSet || ElementSet.has(layer.elementSet, loc);
  91. if (!matches) continue;
  92. const color = layer.color(loc, isSecondary);
  93. if (!isValidColor(color)) continue;
  94. return color;
  95. }
  96. return props.background;
  97. }
  98. const auxLocation = StructureElement.Location.create(ctx.structure);
  99. const color: LocationColor = (location: Location, isSecondary: boolean) => {
  100. if (StructureElement.Location.is(location)) {
  101. return structureElementColor(location, isSecondary);
  102. } else if (Bond.isLocation(location)) {
  103. // this will be applied for each bond twice, to get color of each half (a* refers to the adjacent atom, b* to the opposite atom)
  104. auxLocation.unit = location.aUnit;
  105. auxLocation.element = location.aUnit.elements[location.aIndex];
  106. return structureElementColor(auxLocation, isSecondary);
  107. }
  108. return props.background;
  109. };
  110. return {
  111. factory: (ctx_, props_) => makeMultilayerColorTheme(ctx_, props_, colorThemeRegistry),
  112. granularity: 'group',
  113. preferSmoothing: true,
  114. color: color,
  115. props: props,
  116. description: 'Combines colors from multiple color themes.',
  117. };
  118. }
  119. /** Unique name for "Multilayer" color theme */
  120. export const MultilayerColorThemeName = 'mvs-multilayer';
  121. /** A thingy that is needed to register color theme "Multilayer" */
  122. export function makeMultilayerColorThemeProvider(colorThemeRegistry: ColorTheme.Registry): ColorTheme.Provider<MultilayerColorThemeParams, typeof MultilayerColorThemeName> {
  123. return {
  124. name: MultilayerColorThemeName,
  125. label: 'MVS Multi-layer',
  126. category: ColorTheme.Category.Misc,
  127. factory: (ctx, props) => makeMultilayerColorTheme(ctx, props, colorThemeRegistry),
  128. getParams: (ctx: ThemeDataContext) => makeMultilayerColorThemeParams(colorThemeRegistry, ctx),
  129. defaultValues: DefaultMultilayerColorThemeProps,
  130. isApplicable: (ctx: ThemeDataContext) => !!ctx.structure && isMVSStructure(ctx.structure),
  131. };
  132. }