custom-element-property.ts 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. /**
  2. * Copyright (c) 2018 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author David Sehnal <david.sehnal@gmail.com>
  5. */
  6. import { ElementIndex, Model, ModelPropertyDescriptor } from 'mol-model/structure';
  7. import { StructureElement } from 'mol-model/structure/structure';
  8. import { Location } from 'mol-model/location';
  9. import { CustomPropertyRegistry } from './custom-property-registry';
  10. import { Task } from 'mol-task';
  11. import { ThemeDataContext } from 'mol-theme/theme';
  12. import { ColorTheme, LocationColor } from 'mol-theme/color';
  13. import { Color } from 'mol-util/color';
  14. import { TableLegend } from 'mol-util/color/tables';
  15. import { Loci } from 'mol-model/loci';
  16. import { OrderedSet } from 'mol-data/int';
  17. export { CustomElementProperty };
  18. namespace CustomElementProperty {
  19. export interface CreateParams<S, T> {
  20. isStatic: boolean,
  21. name: string,
  22. autoAttach?: boolean,
  23. display: string,
  24. attachableTo: (model: Model) => boolean,
  25. getData(model: Model): Map<ElementIndex, T> | Promise<Map<ElementIndex, T>>,
  26. format?(e: T): string,
  27. coloring?: {
  28. getColor: (e: T) => Color,
  29. defaultColor: Color
  30. }
  31. }
  32. export function create<S, T>(params: CreateParams<S, T>) {
  33. const name = params.name;
  34. const Descriptor = ModelPropertyDescriptor({
  35. isStatic: params.isStatic,
  36. name: params.name,
  37. });
  38. function attach(model: Model) {
  39. return Task.create(`Attach ${params.display}`, async () => {
  40. const data = await params.getData(model);
  41. if (params.isStatic) {
  42. model._staticPropertyData[name] = data;
  43. } else {
  44. model._dynamicPropertyData[name] = data;
  45. }
  46. return true;
  47. })
  48. }
  49. function getStatic(e: StructureElement) { return e.unit.model._staticPropertyData[name].get(e.element); }
  50. function getDynamic(e: StructureElement) { return e.unit.model._staticPropertyData[name].get(e.element); }
  51. const provider: CustomPropertyRegistry.Provider = {
  52. option: [name, params.display],
  53. descriptor: Descriptor,
  54. defaultSelected: !!params.autoAttach,
  55. attachableTo: params.attachableTo || (() => true),
  56. attach
  57. };
  58. const get = params.isStatic ? getStatic : getDynamic;
  59. function Coloring(ctx: ThemeDataContext, props: {}): ColorTheme<{}> {
  60. let color: LocationColor;
  61. const getColor = params.coloring!.getColor;
  62. const defaultColor = params.coloring!.defaultColor;
  63. if (ctx.structure && !ctx.structure.isEmpty && ctx.structure.models[0].customProperties.has(Descriptor)) {
  64. color = (location: Location) => {
  65. if (StructureElement.isLocation(location)) {
  66. const e = get(location);
  67. if (typeof e !== 'undefined') return getColor(e);
  68. }
  69. return defaultColor;
  70. }
  71. } else {
  72. color = () => defaultColor;
  73. }
  74. return {
  75. factory: Coloring,
  76. granularity: 'group',
  77. color: color,
  78. props: props,
  79. description: 'Assign element colors based on the provided data.',
  80. legend: TableLegend([])
  81. };
  82. }
  83. function LabelProvider(loci: Loci): string | undefined {
  84. if (loci.kind === 'element-loci') {
  85. const e = loci.elements[0];
  86. return params.format!(get(StructureElement.create(e.unit, e.unit.elements[OrderedSet.getAt(e.indices, 0)])));
  87. }
  88. return void 0;
  89. }
  90. return {
  91. Descriptor,
  92. attach,
  93. get,
  94. provider,
  95. colorTheme: params.coloring ? Coloring : void 0,
  96. labelProvider: params.format ? LabelProvider : ((loci: Loci) => void 0)
  97. };
  98. }
  99. }