object.ts 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  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 { UUID } from '../mol-util';
  7. import { StateTransform } from './transform';
  8. import { ParamDefinition } from '../mol-util/param-definition';
  9. import { State } from './state';
  10. import { StateSelection, StateTransformer } from '../mol-state';
  11. export { StateObject, StateObjectCell }
  12. interface StateObject<D = any, T extends StateObject.Type = StateObject.Type<any>> {
  13. readonly id: UUID,
  14. readonly type: T,
  15. readonly data: D,
  16. readonly label: string,
  17. readonly description?: string,
  18. // assigned by reconciler to be StateTransform.props.tag
  19. readonly tags?: string[]
  20. }
  21. namespace StateObject {
  22. export function factory<T extends Type>() {
  23. return <D = { }>(type: T) => create<D, T>(type);
  24. }
  25. export type Type<Cls extends string = string> = { name: string, typeClass: Cls }
  26. export type Ctor<T extends StateObject = StateObject> = { new(...args: any[]): T, type: any }
  27. export type From<C extends Ctor> = C extends Ctor<infer T> ? T : never
  28. export function create<Data, T extends Type>(type: T) {
  29. return class O implements StateObject<Data, T> {
  30. static type = type;
  31. static is(obj?: StateObject): obj is O { return !!obj && type === obj.type; }
  32. id = UUID.create22();
  33. type = type;
  34. label: string;
  35. description?: string;
  36. constructor(public data: Data, props?: { label: string, description?: string }) {
  37. this.label = props && props.label || type.name;
  38. this.description = props && props.description;
  39. }
  40. }
  41. }
  42. /** A special object indicating a transformer result has no value. */
  43. export const Null: StateObject<any, any> = {
  44. id: UUID.create22(),
  45. type: { name: 'Null', typeClass: 'Null' },
  46. data: void 0,
  47. label: 'Null'
  48. };
  49. }
  50. interface StateObjectCell<T extends StateObject = StateObject, F extends StateTransform = StateTransform> {
  51. parent: State,
  52. transform: F,
  53. // Which object was used as a parent to create data in this cell
  54. sourceRef: StateTransform.Ref | undefined,
  55. status: StateObjectCell.Status,
  56. state: StateTransform.State,
  57. params: {
  58. definition: ParamDefinition.Params,
  59. values: any
  60. } | undefined,
  61. dependencies: {
  62. dependentBy: StateObjectCell[],
  63. dependsOn: StateObjectCell[]
  64. },
  65. errorText?: string,
  66. obj?: T,
  67. cache: unknown | undefined
  68. }
  69. namespace StateObjectCell {
  70. export type Status = 'ok' | 'error' | 'pending' | 'processing'
  71. export type Obj<C extends StateObjectCell> = C extends StateObjectCell<infer T> ? T : never
  72. export type Transform<C extends StateObjectCell> = C extends StateObjectCell<any, infer T> ? T : never
  73. export type Transformer<C extends StateObjectCell> = C extends StateObjectCell<any, StateTransform<infer T>> ? T : never
  74. export function is(o: any): o is StateObjectCell {
  75. const c: StateObjectCell = o;
  76. return !!c && !!c.transform && !!c.parent && !!c.status;
  77. }
  78. export type Ref = StateTransform.Ref | StateObjectCell | StateObjectSelector
  79. export function resolve(state: State, refOrCellOrSelector: StateTransform.Ref | StateObjectCell | StateObjectSelector) {
  80. const ref = typeof refOrCellOrSelector === 'string'
  81. ? refOrCellOrSelector
  82. : StateObjectCell.is(refOrCellOrSelector)
  83. ? refOrCellOrSelector.transform.ref
  84. : refOrCellOrSelector.ref;
  85. return state.cells.get(ref);
  86. }
  87. }
  88. // TODO: improve the API?
  89. export class StateObjectTracker<T extends StateObject> {
  90. private query: StateSelection.Query;
  91. private version: string = '';
  92. cell: StateObjectCell | undefined;
  93. data: T['data'] | undefined;
  94. setQuery(sel: StateSelection.Selector) {
  95. this.query = StateSelection.compile(sel);
  96. }
  97. update() {
  98. const cell = this.state.select(this.query)[0];
  99. const version = cell ? cell.transform.version : void 0;
  100. const changed = this.cell !== cell || this.version !== version;
  101. this.cell = cell;
  102. this.version = version || '';
  103. this.data = cell && cell.obj ? cell.obj.data as T : void 0 as any;
  104. return changed;
  105. }
  106. constructor(private state: State) { }
  107. }
  108. export class StateObjectSelector<S extends StateObject = StateObject, T extends StateTransformer = StateTransformer> {
  109. get cell(): StateObjectCell<S, StateTransform<T>> | undefined {
  110. return this.state?.cells.get(this.ref) as StateObjectCell<S, StateTransform<T>> | undefined;
  111. }
  112. get obj(): S | undefined {
  113. return this.state?.cells.get(this.ref)?.obj as S | undefined;
  114. }
  115. get data(): S['data'] | undefined {
  116. return this.obj?.data;
  117. }
  118. /** Checks if the object exists. If not throw an error. */
  119. checkValid() {
  120. if (!this.state) {
  121. throw new Error('Unassigned State.');
  122. }
  123. const cell = this.cell;
  124. if (!cell) {
  125. throw new Error(`Not created at all. Did you await/then the corresponding state update?`);
  126. }
  127. if (cell.status === 'ok') return true;
  128. if (cell.status === 'error') throw new Error(cell.errorText);
  129. throw new Error(`Unresolved. Did you await/then the corresponding state update?`);
  130. }
  131. get isOk() {
  132. const cell = this.cell;
  133. return cell && cell.status === 'ok';
  134. }
  135. constructor(public ref: StateTransform.Ref, public state?: State) {
  136. }
  137. }
  138. export namespace StateObjectSelector {
  139. export type Obj<S extends StateObjectSelector> = S extends StateObjectSelector<infer A> ? A : never
  140. export type Transformer<S extends StateObjectSelector> = S extends StateObjectSelector<any, infer T> ? T : never
  141. }
  142. export type StateObjectRef<S extends StateObject = StateObject> = StateObjectSelector<S> | StateObjectCell<S> | StateTransform.Ref
  143. export namespace StateObjectRef {
  144. export function resolveRef<S extends StateObject>(state: State, ref?: StateObjectRef<S>): StateTransform.Ref | undefined {
  145. if (!ref) return;
  146. if (typeof ref === 'string') return ref;
  147. if (StateObjectCell.is(ref)) return ref.transform.ref;
  148. return ref.cell?.transform.ref;
  149. }
  150. export function resolve<S extends StateObject>(state: State, ref?: StateObjectRef<S>): StateObjectCell<S> | undefined {
  151. if (!ref) return;
  152. if (StateObjectCell.is(ref)) return ref;
  153. if (typeof ref === 'string') return state.cells.get(ref) as StateObjectCell<S> | undefined;
  154. return ref.cell;
  155. }
  156. export function resolveAndCheck<S extends StateObject>(state: State, ref?: StateObjectRef<S>): StateObjectCell<S> | undefined {
  157. const cell = resolve(state, ref);
  158. if (!cell || !cell.obj || cell.status !== 'ok') return;
  159. return cell;
  160. }
  161. }