object.ts 8.0 KB

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