@dcl/sdk 7.4.22-8991339646.commit-c3a4d1e → 7.4.22-9005138631.commit-4096037

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/observables.js CHANGED
@@ -2,8 +2,6 @@ import { Observable } from './internal/Observable';
2
2
  import { AvatarBase, AvatarEmoteCommand, AvatarEquippedData, PlayerIdentityData, PointerEventsResult, RealmInfo, engine } from '@dcl/ecs';
3
3
  import { subscribe } from '~system/EngineApi';
4
4
  import players from './players';
5
- // Remove this once new components/observables reaches production.
6
- const __OBSERVABLES_FALLBACK_SUPPORT = true;
7
5
  /**
8
6
  * @internal
9
7
  * This function generates a callback that is passed to the Observable
@@ -11,7 +9,7 @@ const __OBSERVABLES_FALLBACK_SUPPORT = true;
11
9
  */
12
10
  function createSubscriber(eventName) {
13
11
  return () => {
14
- if (eventName === 'comms' || __OBSERVABLES_FALLBACK_SUPPORT) {
12
+ if (eventName === 'comms') {
15
13
  subscribe({ eventId: eventName }).catch(console.error);
16
14
  }
17
15
  else {
@@ -96,47 +94,6 @@ export async function pollEvents(sendBatch) {
96
94
  if (e.generic) {
97
95
  const data = JSON.parse(e.generic.eventData);
98
96
  switch (e.generic.eventId) {
99
- case 'onEnterScene': {
100
- onEnterSceneObservable.notifyObservers(data);
101
- break;
102
- }
103
- case 'onLeaveScene': {
104
- onLeaveSceneObservable.notifyObservers(data);
105
- break;
106
- }
107
- case 'sceneStart': {
108
- onSceneReadyObservable.notifyObservers(data);
109
- break;
110
- }
111
- case 'playerExpression': {
112
- onPlayerExpressionObservable.notifyObservers(data);
113
- break;
114
- }
115
- case 'videoEvent': {
116
- const videoData = data;
117
- onVideoEvent.notifyObservers(videoData);
118
- break;
119
- }
120
- case 'profileChanged': {
121
- onProfileChanged.notifyObservers(data);
122
- break;
123
- }
124
- case 'playerConnected': {
125
- onPlayerConnectedObservable.notifyObservers(data);
126
- break;
127
- }
128
- case 'playerDisconnected': {
129
- onPlayerDisconnectedObservable.notifyObservers(data);
130
- break;
131
- }
132
- case 'onRealmChanged': {
133
- onRealmChangedObservable.notifyObservers(data);
134
- break;
135
- }
136
- case 'playerClicked': {
137
- onPlayerClickedObservable.notifyObservers(data);
138
- break;
139
- }
140
97
  case 'comms': {
141
98
  onCommsMessage.notifyObservers(data);
142
99
  break;
@@ -147,8 +104,6 @@ export async function pollEvents(sendBatch) {
147
104
  }
148
105
  const SDK7ComponentsObservable = processObservables();
149
106
  function processObservables() {
150
- if (__OBSERVABLES_FALLBACK_SUPPORT)
151
- return;
152
107
  const subscriptions = new Set();
153
108
  function subscribe(eventName) {
154
109
  if (subscriptions.has(eventName))
@@ -284,4 +239,4 @@ function processObservables() {
284
239
  engine.addSystem(observableSystem);
285
240
  return { subscribe };
286
241
  }
287
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"observables.js","sourceRoot":"","sources":["src/observables.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable } from './internal/Observable'\nimport {\n  AvatarBase,\n  AvatarEmoteCommand,\n  AvatarEquippedData,\n  Entity,\n  PlayerIdentityData,\n  PointerEventsResult,\n  RealmInfo,\n  Vector3Type,\n  engine\n} from '@dcl/ecs'\nimport { ManyEntityAction, SendBatchResponse, subscribe } from '~system/EngineApi'\nimport players from './players'\n\n/// --- EVENTS ---\n\n/** @public */\nexport type IEventNames = keyof IEvents\n\n/**\n * @public\n * Note: Don't use `on` prefix for IEvents to avoid redundancy with `event.on(\"onEventName\")` syntax.\n */\nexport interface IEvents {\n  playerExpression: {\n    expressionId: string\n  }\n\n  /**\n   * This event gets triggered when the user enters the scene\n   */\n  onEnterScene: {\n    userId: string\n  }\n\n  /**\n   * This event gets triggered when the user leaves the scene\n   */\n  onLeaveScene: {\n    userId: string\n  }\n\n  /**\n   * This event gets triggered after receiving a comms message.\n   */\n  comms: {\n    sender: string\n    message: string\n  }\n\n  /**\n   * This is triggered once the scene should start.\n   */\n  sceneStart: unknown\n\n  /** This is triggered at least for each videoStatus change */\n  videoEvent: {\n    componentId: string\n    videoClipId: string\n    /** Status, can be NONE = 0, ERROR = 1, LOADING = 2, READY = 3, PLAYING = 4,BUFFERING = 5 */\n    videoStatus: number\n    /** Current offset position in seconds */\n    currentOffset: number\n    /** Video length in seconds. Can be -1 */\n    totalVideoLength: number\n  }\n\n  /** This is trigger everytime a profile is changed */\n  profileChanged: {\n    ethAddress: string\n    version: number\n  }\n\n  /** Triggered when peer's avatar is connected and visible */\n  playerConnected: {\n    userId: string\n  }\n\n  /** Triggered when peer disconnect and/or it avatar is set invisible by comms */\n  playerDisconnected: {\n    userId: string\n  }\n\n  /** Triggered when current realm or island changes */\n  onRealmChanged: {\n    domain: string\n    room: string\n    serverName: string\n    displayName: string\n  }\n\n  /** Triggered when other player's avatar is clicked */\n  playerClicked: {\n    userId: string\n    ray: {\n      origin: Vector3Type\n      direction: Vector3Type\n      distance: number\n    }\n  }\n}\n\n// Remove this once new components/observables reaches production.\nconst __OBSERVABLES_FALLBACK_SUPPORT = true\n/**\n * @internal\n * This function generates a callback that is passed to the Observable\n * constructor to subscribe to the events of the DecentralandInterface\n */\nfunction createSubscriber(eventName: keyof IEvents) {\n  return () => {\n    if (eventName === 'comms' || __OBSERVABLES_FALLBACK_SUPPORT) {\n      subscribe({ eventId: eventName }).catch(console.error)\n    } else {\n      SDK7ComponentsObservable?.subscribe(eventName)\n    }\n  }\n}\n\n/**\n * These events are triggered after your character enters the scene.\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onEnterSceneObservable = new Observable<IEvents['onEnterScene']>(createSubscriber('onEnterScene'))\n/** @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed. Use onEnterSceneObservable instead. */\nexport const onEnterScene = onEnterSceneObservable\n\n/**\n * These events are triggered after your character leaves the scene.\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onLeaveSceneObservable = new Observable<IEvents['onLeaveScene']>(createSubscriber('onLeaveScene'))\n\n/** @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed. Use onLeaveSceneObservable instead. */\nexport const onLeaveScene = onLeaveSceneObservable\n\n/**\n * This event is triggered after all the resources of the scene were loaded (models, textures, etc...)\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onSceneReadyObservable = new Observable<IEvents['sceneStart']>(createSubscriber('sceneStart'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerExpressionObservable = new Observable<IEvents['playerExpression']>(\n  createSubscriber('playerExpression')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onVideoEvent = new Observable<IEvents['videoEvent']>(createSubscriber('videoEvent'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onProfileChanged = new Observable<IEvents['profileChanged']>(createSubscriber('profileChanged'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerConnectedObservable = new Observable<IEvents['playerConnected']>(\n  createSubscriber('playerConnected')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerDisconnectedObservable = new Observable<IEvents['playerDisconnected']>(\n  createSubscriber('playerDisconnected')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onRealmChangedObservable = new Observable<IEvents['onRealmChanged']>(createSubscriber('onRealmChanged'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerClickedObservable = new Observable<IEvents['playerClicked']>(createSubscriber('playerClicked'))\n\n/**\n * @internal\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onCommsMessage = new Observable<IEvents['comms']>(createSubscriber('comms'))\n\n/**\n * @internal\n * @deprecated this is an OLD API.\n * This function uses the SDK6 sendBatch to poll events from the renderer\n */\nexport async function pollEvents(sendBatch: (body: ManyEntityAction) => Promise<SendBatchResponse>) {\n  const { events } = await sendBatch({ actions: [] })\n  for (const e of events) {\n    if (e.generic) {\n      const data = JSON.parse(e.generic.eventData)\n      switch (e.generic.eventId) {\n        case 'onEnterScene': {\n          onEnterSceneObservable.notifyObservers(data as IEvents['onEnterScene'])\n          break\n        }\n        case 'onLeaveScene': {\n          onLeaveSceneObservable.notifyObservers(data as IEvents['onLeaveScene'])\n          break\n        }\n        case 'sceneStart': {\n          onSceneReadyObservable.notifyObservers(data as IEvents['sceneStart'])\n          break\n        }\n        case 'playerExpression': {\n          onPlayerExpressionObservable.notifyObservers(data as IEvents['playerExpression'])\n          break\n        }\n        case 'videoEvent': {\n          const videoData = data as IEvents['videoEvent']\n          onVideoEvent.notifyObservers(videoData)\n          break\n        }\n        case 'profileChanged': {\n          onProfileChanged.notifyObservers(data as IEvents['profileChanged'])\n          break\n        }\n        case 'playerConnected': {\n          onPlayerConnectedObservable.notifyObservers(data as IEvents['playerConnected'])\n          break\n        }\n        case 'playerDisconnected': {\n          onPlayerDisconnectedObservable.notifyObservers(data as IEvents['playerDisconnected'])\n          break\n        }\n        case 'onRealmChanged': {\n          onRealmChangedObservable.notifyObservers(data as IEvents['onRealmChanged'])\n          break\n        }\n        case 'playerClicked': {\n          onPlayerClickedObservable.notifyObservers(data as IEvents['playerClicked'])\n          break\n        }\n        case 'comms': {\n          onCommsMessage.notifyObservers(data as IEvents['comms'])\n          break\n        }\n      }\n    }\n  }\n}\n\nconst SDK7ComponentsObservable = processObservables()\nfunction processObservables() {\n  if (__OBSERVABLES_FALLBACK_SUPPORT) return\n  const subscriptions = new Set<keyof IEvents>()\n\n  function subscribe(eventName: keyof IEvents) {\n    if (subscriptions.has(eventName)) return\n    switch (eventName) {\n      case 'playerClicked': {\n        subscribePlayerClick()\n      }\n      case 'onEnterScene':\n      case 'playerConnected': {\n        subscribeEnterScene()\n      }\n      case 'onLeaveScene':\n      case 'playerDisconnected': {\n        subscribeLeaveScene()\n      }\n      case 'onRealmChanged': {\n        subscribeRealmChange()\n      }\n      case 'playerExpression': {\n        subscribePlayerExpression()\n      }\n      case 'profileChanged': {\n        subscribeProfileChange()\n      }\n    }\n    subscriptions.add(eventName)\n  }\n  /**\n   * PLAYER ENTER/CONNECTED observable\n   */\n  function subscribeEnterScene() {\n    players.onEnterScene((player) => {\n      if (subscriptions.has('onEnterScene')) {\n        onEnterSceneObservable.notifyObservers({ userId: player.userId })\n      }\n\n      if (subscriptions.has('playerConnected')) {\n        onPlayerConnectedObservable.notifyObservers({ userId: player.userId })\n      }\n    })\n  }\n  /**\n   * PLAYER LEAVE/DISCONNECTED observable\n   */\n  function subscribeLeaveScene() {\n    players.onLeaveScene((userId) => {\n      if (subscriptions.has('onLeaveScene')) {\n        onLeaveSceneObservable.notifyObservers({ userId })\n      }\n\n      if (subscriptions.has('playerDisconnected')) {\n        onPlayerDisconnectedObservable.notifyObservers({ userId })\n      }\n    })\n  }\n  /**\n   * REALM CHANGE observable\n   */\n  function subscribeRealmChange() {\n    RealmInfo.onChange(engine.RootEntity, (value) => {\n      if (value) {\n        onRealmChangedObservable.notifyObservers({\n          domain: value.baseUrl,\n          displayName: value.realmName,\n          room: value.room ?? '',\n          serverName: value.realmName\n        })\n      }\n    })\n  }\n  /**\n   * PLAYER/AVATAR CLICKED observable\n   */\n  function subscribePlayerClick() {\n    const playerEntities = new Set<Entity>()\n    engine.addSystem(() => {\n      for (const [entity] of engine.getEntitiesWith(PlayerIdentityData)) {\n        if (playerEntities.has(entity)) return\n        playerEntities.add(entity)\n\n        PointerEventsResult.onChange(entity, (data) => {\n          if (data?.hit) {\n            onPlayerClickedObservable.notifyObservers({\n              userId: PlayerIdentityData.getOrNull(entity)?.address ?? '',\n              ray: {\n                direction: data.hit.direction!,\n                distance: data.hit.length,\n                origin: data.hit.globalOrigin!\n              }\n            })\n          }\n        })\n      }\n    })\n  }\n\n  /**\n   * Player expression observable\n   */\n  function subscribePlayerExpression() {\n    AvatarEmoteCommand.onChange(engine.PlayerEntity, (value) => {\n      onPlayerExpressionObservable.notifyObservers({ expressionId: value?.emoteUrn ?? '' })\n    })\n  }\n\n  /**\n   * PROFILE CHANGE observable\n   */\n  function subscribeProfileChange() {\n    AvatarBase.onChange(engine.PlayerEntity, () => {\n      if (!profileAddress) return\n      onProfileChanged.notifyObservers({ ethAddress: profileAddress, version: 0 })\n    })\n\n    AvatarEquippedData.onChange(engine.PlayerEntity, () => {\n      if (!profileAddress) return\n      onProfileChanged.notifyObservers({ ethAddress: profileAddress, version: 0 })\n    })\n  }\n\n  // Flag to call once the scene is initalized.\n  let sceneReady = false\n  let profileAddress: string | undefined\n\n  function observableSystem() {\n    if (sceneReady && profileAddress) {\n      return engine.removeSystem(observableSystem)\n    }\n    if (!sceneReady) {\n      sceneReady = true\n      onSceneReadyObservable.notifyObservers({})\n    }\n    if (profileAddress) return\n    profileAddress = PlayerIdentityData.getOrNull(engine.PlayerEntity)?.address\n  }\n\n  engine.addSystem(observableSystem)\n\n  return { subscribe }\n}\n"]}
242
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"observables.js","sourceRoot":"","sources":["src/observables.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable } from './internal/Observable'\nimport {\n  AvatarBase,\n  AvatarEmoteCommand,\n  AvatarEquippedData,\n  Entity,\n  PlayerIdentityData,\n  PointerEventsResult,\n  RealmInfo,\n  Vector3Type,\n  engine\n} from '@dcl/ecs'\nimport { ManyEntityAction, SendBatchResponse, subscribe } from '~system/EngineApi'\nimport players from './players'\n\n/// --- EVENTS ---\n\n/** @public */\nexport type IEventNames = keyof IEvents\n\n/**\n * @public\n * Note: Don't use `on` prefix for IEvents to avoid redundancy with `event.on(\"onEventName\")` syntax.\n */\nexport interface IEvents {\n  playerExpression: {\n    expressionId: string\n  }\n\n  /**\n   * This event gets triggered when the user enters the scene\n   */\n  onEnterScene: {\n    userId: string\n  }\n\n  /**\n   * This event gets triggered when the user leaves the scene\n   */\n  onLeaveScene: {\n    userId: string\n  }\n\n  /**\n   * This event gets triggered after receiving a comms message.\n   */\n  comms: {\n    sender: string\n    message: string\n  }\n\n  /**\n   * This is triggered once the scene should start.\n   */\n  sceneStart: unknown\n\n  /** This is triggered at least for each videoStatus change */\n  videoEvent: {\n    componentId: string\n    videoClipId: string\n    /** Status, can be NONE = 0, ERROR = 1, LOADING = 2, READY = 3, PLAYING = 4,BUFFERING = 5 */\n    videoStatus: number\n    /** Current offset position in seconds */\n    currentOffset: number\n    /** Video length in seconds. Can be -1 */\n    totalVideoLength: number\n  }\n\n  /** This is trigger everytime a profile is changed */\n  profileChanged: {\n    ethAddress: string\n    version: number\n  }\n\n  /** Triggered when peer's avatar is connected and visible */\n  playerConnected: {\n    userId: string\n  }\n\n  /** Triggered when peer disconnect and/or it avatar is set invisible by comms */\n  playerDisconnected: {\n    userId: string\n  }\n\n  /** Triggered when current realm or island changes */\n  onRealmChanged: {\n    domain: string\n    room: string\n    serverName: string\n    displayName: string\n  }\n\n  /** Triggered when other player's avatar is clicked */\n  playerClicked: {\n    userId: string\n    ray: {\n      origin: Vector3Type\n      direction: Vector3Type\n      distance: number\n    }\n  }\n}\n\n/**\n * @internal\n * This function generates a callback that is passed to the Observable\n * constructor to subscribe to the events of the DecentralandInterface\n */\nfunction createSubscriber(eventName: keyof IEvents) {\n  return () => {\n    if (eventName === 'comms') {\n      subscribe({ eventId: eventName }).catch(console.error)\n    } else {\n      SDK7ComponentsObservable?.subscribe(eventName)\n    }\n  }\n}\n\n/**\n * These events are triggered after your character enters the scene.\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onEnterSceneObservable = new Observable<IEvents['onEnterScene']>(createSubscriber('onEnterScene'))\n/** @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed. Use onEnterSceneObservable instead. */\nexport const onEnterScene = onEnterSceneObservable\n\n/**\n * These events are triggered after your character leaves the scene.\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onLeaveSceneObservable = new Observable<IEvents['onLeaveScene']>(createSubscriber('onLeaveScene'))\n\n/** @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed. Use onLeaveSceneObservable instead. */\nexport const onLeaveScene = onLeaveSceneObservable\n\n/**\n * This event is triggered after all the resources of the scene were loaded (models, textures, etc...)\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onSceneReadyObservable = new Observable<IEvents['sceneStart']>(createSubscriber('sceneStart'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerExpressionObservable = new Observable<IEvents['playerExpression']>(\n  createSubscriber('playerExpression')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onVideoEvent = new Observable<IEvents['videoEvent']>(createSubscriber('videoEvent'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onProfileChanged = new Observable<IEvents['profileChanged']>(createSubscriber('profileChanged'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerConnectedObservable = new Observable<IEvents['playerConnected']>(\n  createSubscriber('playerConnected')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerDisconnectedObservable = new Observable<IEvents['playerDisconnected']>(\n  createSubscriber('playerDisconnected')\n)\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onRealmChangedObservable = new Observable<IEvents['onRealmChanged']>(createSubscriber('onRealmChanged'))\n\n/**\n * @public\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onPlayerClickedObservable = new Observable<IEvents['playerClicked']>(createSubscriber('playerClicked'))\n\n/**\n * @internal\n * @deprecated This function is an inheritance of ECS6, it's here temporary for the feature parity, please read the news and docs to know how handle when it's removed.\n */\nexport const onCommsMessage = new Observable<IEvents['comms']>(createSubscriber('comms'))\n\n/**\n * @internal\n * @deprecated this is an OLD API.\n * This function uses the SDK6 sendBatch to poll events from the renderer\n */\nexport async function pollEvents(sendBatch: (body: ManyEntityAction) => Promise<SendBatchResponse>) {\n  const { events } = await sendBatch({ actions: [] })\n  for (const e of events) {\n    if (e.generic) {\n      const data = JSON.parse(e.generic.eventData)\n      switch (e.generic.eventId) {\n        case 'comms': {\n          onCommsMessage.notifyObservers(data as IEvents['comms'])\n          break\n        }\n      }\n    }\n  }\n}\n\nconst SDK7ComponentsObservable = processObservables()\nfunction processObservables() {\n  const subscriptions = new Set<keyof IEvents>()\n\n  function subscribe(eventName: keyof IEvents) {\n    if (subscriptions.has(eventName)) return\n    switch (eventName) {\n      case 'playerClicked': {\n        subscribePlayerClick()\n      }\n      case 'onEnterScene':\n      case 'playerConnected': {\n        subscribeEnterScene()\n      }\n      case 'onLeaveScene':\n      case 'playerDisconnected': {\n        subscribeLeaveScene()\n      }\n      case 'onRealmChanged': {\n        subscribeRealmChange()\n      }\n      case 'playerExpression': {\n        subscribePlayerExpression()\n      }\n      case 'profileChanged': {\n        subscribeProfileChange()\n      }\n    }\n    subscriptions.add(eventName)\n  }\n  /**\n   * PLAYER ENTER/CONNECTED observable\n   */\n  function subscribeEnterScene() {\n    players.onEnterScene((player) => {\n      if (subscriptions.has('onEnterScene')) {\n        onEnterSceneObservable.notifyObservers({ userId: player.userId })\n      }\n\n      if (subscriptions.has('playerConnected')) {\n        onPlayerConnectedObservable.notifyObservers({ userId: player.userId })\n      }\n    })\n  }\n  /**\n   * PLAYER LEAVE/DISCONNECTED observable\n   */\n  function subscribeLeaveScene() {\n    players.onLeaveScene((userId) => {\n      if (subscriptions.has('onLeaveScene')) {\n        onLeaveSceneObservable.notifyObservers({ userId })\n      }\n\n      if (subscriptions.has('playerDisconnected')) {\n        onPlayerDisconnectedObservable.notifyObservers({ userId })\n      }\n    })\n  }\n  /**\n   * REALM CHANGE observable\n   */\n  function subscribeRealmChange() {\n    RealmInfo.onChange(engine.RootEntity, (value) => {\n      if (value) {\n        onRealmChangedObservable.notifyObservers({\n          domain: value.baseUrl,\n          displayName: value.realmName,\n          room: value.room ?? '',\n          serverName: value.realmName\n        })\n      }\n    })\n  }\n  /**\n   * PLAYER/AVATAR CLICKED observable\n   */\n  function subscribePlayerClick() {\n    const playerEntities = new Set<Entity>()\n    engine.addSystem(() => {\n      for (const [entity] of engine.getEntitiesWith(PlayerIdentityData)) {\n        if (playerEntities.has(entity)) return\n        playerEntities.add(entity)\n\n        PointerEventsResult.onChange(entity, (data) => {\n          if (data?.hit) {\n            onPlayerClickedObservable.notifyObservers({\n              userId: PlayerIdentityData.getOrNull(entity)?.address ?? '',\n              ray: {\n                direction: data.hit.direction!,\n                distance: data.hit.length,\n                origin: data.hit.globalOrigin!\n              }\n            })\n          }\n        })\n      }\n    })\n  }\n\n  /**\n   * Player expression observable\n   */\n  function subscribePlayerExpression() {\n    AvatarEmoteCommand.onChange(engine.PlayerEntity, (value) => {\n      onPlayerExpressionObservable.notifyObservers({ expressionId: value?.emoteUrn ?? '' })\n    })\n  }\n\n  /**\n   * PROFILE CHANGE observable\n   */\n  function subscribeProfileChange() {\n    AvatarBase.onChange(engine.PlayerEntity, () => {\n      if (!profileAddress) return\n      onProfileChanged.notifyObservers({ ethAddress: profileAddress, version: 0 })\n    })\n\n    AvatarEquippedData.onChange(engine.PlayerEntity, () => {\n      if (!profileAddress) return\n      onProfileChanged.notifyObservers({ ethAddress: profileAddress, version: 0 })\n    })\n  }\n\n  // Flag to call once the scene is initalized.\n  let sceneReady = false\n  let profileAddress: string | undefined\n\n  function observableSystem() {\n    if (sceneReady && profileAddress) {\n      return engine.removeSystem(observableSystem)\n    }\n    if (!sceneReady) {\n      sceneReady = true\n      onSceneReadyObservable.notifyObservers({})\n    }\n    if (profileAddress) return\n    profileAddress = PlayerIdentityData.getOrNull(engine.PlayerEntity)?.address\n  }\n\n  engine.addSystem(observableSystem)\n\n  return { subscribe }\n}\n"]}
package/package.json CHANGED
@@ -1,15 +1,15 @@
1
1
  {
2
2
  "name": "@dcl/sdk",
3
3
  "description": "",
4
- "version": "7.4.22-8991339646.commit-c3a4d1e",
4
+ "version": "7.4.22-9005138631.commit-4096037",
5
5
  "author": "Decentraland",
6
6
  "dependencies": {
7
- "@dcl/ecs": "7.4.22-8991339646.commit-c3a4d1e",
7
+ "@dcl/ecs": "7.4.22-9005138631.commit-4096037",
8
8
  "@dcl/ecs-math": "2.0.2",
9
9
  "@dcl/explorer": "1.0.162830-20240429195817.commit-25498a3",
10
- "@dcl/js-runtime": "7.4.22-8991339646.commit-c3a4d1e",
11
- "@dcl/react-ecs": "7.4.22-8991339646.commit-c3a4d1e",
12
- "@dcl/sdk-commands": "7.4.22-8991339646.commit-c3a4d1e",
10
+ "@dcl/js-runtime": "7.4.22-9005138631.commit-4096037",
11
+ "@dcl/react-ecs": "7.4.22-9005138631.commit-4096037",
12
+ "@dcl/sdk-commands": "7.4.22-9005138631.commit-4096037",
13
13
  "text-encoding": "0.7.0"
14
14
  },
15
15
  "keywords": [],
@@ -35,5 +35,5 @@
35
35
  },
36
36
  "types": "./index.d.ts",
37
37
  "typings": "./index.d.ts",
38
- "commit": "c3a4d1ef0b26b2b50a7d746ba086e02154f20e3a"
38
+ "commit": "409603761cfd01cbf21bd665754c8e9b08817a91"
39
39
  }
@@ -101,8 +101,6 @@ export interface IEvents {
101
101
  }
102
102
  }
103
103
 
104
- // Remove this once new components/observables reaches production.
105
- const __OBSERVABLES_FALLBACK_SUPPORT = true
106
104
  /**
107
105
  * @internal
108
106
  * This function generates a callback that is passed to the Observable
@@ -110,7 +108,7 @@ const __OBSERVABLES_FALLBACK_SUPPORT = true
110
108
  */
111
109
  function createSubscriber(eventName: keyof IEvents) {
112
110
  return () => {
113
- if (eventName === 'comms' || __OBSERVABLES_FALLBACK_SUPPORT) {
111
+ if (eventName === 'comms') {
114
112
  subscribe({ eventId: eventName }).catch(console.error)
115
113
  } else {
116
114
  SDK7ComponentsObservable?.subscribe(eventName)
@@ -213,47 +211,6 @@ export async function pollEvents(sendBatch: (body: ManyEntityAction) => Promise<
213
211
  if (e.generic) {
214
212
  const data = JSON.parse(e.generic.eventData)
215
213
  switch (e.generic.eventId) {
216
- case 'onEnterScene': {
217
- onEnterSceneObservable.notifyObservers(data as IEvents['onEnterScene'])
218
- break
219
- }
220
- case 'onLeaveScene': {
221
- onLeaveSceneObservable.notifyObservers(data as IEvents['onLeaveScene'])
222
- break
223
- }
224
- case 'sceneStart': {
225
- onSceneReadyObservable.notifyObservers(data as IEvents['sceneStart'])
226
- break
227
- }
228
- case 'playerExpression': {
229
- onPlayerExpressionObservable.notifyObservers(data as IEvents['playerExpression'])
230
- break
231
- }
232
- case 'videoEvent': {
233
- const videoData = data as IEvents['videoEvent']
234
- onVideoEvent.notifyObservers(videoData)
235
- break
236
- }
237
- case 'profileChanged': {
238
- onProfileChanged.notifyObservers(data as IEvents['profileChanged'])
239
- break
240
- }
241
- case 'playerConnected': {
242
- onPlayerConnectedObservable.notifyObservers(data as IEvents['playerConnected'])
243
- break
244
- }
245
- case 'playerDisconnected': {
246
- onPlayerDisconnectedObservable.notifyObservers(data as IEvents['playerDisconnected'])
247
- break
248
- }
249
- case 'onRealmChanged': {
250
- onRealmChangedObservable.notifyObservers(data as IEvents['onRealmChanged'])
251
- break
252
- }
253
- case 'playerClicked': {
254
- onPlayerClickedObservable.notifyObservers(data as IEvents['playerClicked'])
255
- break
256
- }
257
214
  case 'comms': {
258
215
  onCommsMessage.notifyObservers(data as IEvents['comms'])
259
216
  break
@@ -265,7 +222,6 @@ export async function pollEvents(sendBatch: (body: ManyEntityAction) => Promise<
265
222
 
266
223
  const SDK7ComponentsObservable = processObservables()
267
224
  function processObservables() {
268
- if (__OBSERVABLES_FALLBACK_SUPPORT) return
269
225
  const subscriptions = new Set<keyof IEvents>()
270
226
 
271
227
  function subscribe(eventName: keyof IEvents) {
@@ -50,7 +50,7 @@ export function createTestRuntime(testingModule: TestingModule, engine: IEngine)
50
50
  // continue to run until it reaches a yield point
51
51
  function scheduleValue(value: any, env: RunnerEnvironment) {
52
52
  if (value && typeof value === 'object' && typeof value.then === 'function') {
53
- // console.log('⏱️ yield promise')
53
+ console.log('⏱️ yield promise')
54
54
  // if the value is a promise, schedule it to be awaited after the current frame is finished
55
55
  nextTickFuture.push(async () => {
56
56
  try {
@@ -60,14 +60,14 @@ export function createTestRuntime(testingModule: TestingModule, engine: IEngine)
60
60
  }
61
61
  })
62
62
  } else if (typeof value === 'function') {
63
- // console.log('⏱️ yield function')
63
+ console.log('⏱️ yield function')
64
64
  // if the value is a function, schedule it to be called on the next frame
65
65
  nextTickFuture.push(() => {
66
66
  scheduleValue(value(), env)
67
67
  })
68
68
  return
69
69
  } else if (typeof value === 'undefined' || value === null) {
70
- // console.log('⏱️ yield')
70
+ console.log('⏱️ yield')
71
71
  // if the value is undefined or null, continue processing the generator the next frame
72
72
  nextTickFuture.push(() => {
73
73
  consumeGenerator(env)
@@ -33,7 +33,7 @@ export function createTestRuntime(testingModule, engine) {
33
33
  // continue to run until it reaches a yield point
34
34
  function scheduleValue(value, env) {
35
35
  if (value && typeof value === 'object' && typeof value.then === 'function') {
36
- // console.log('⏱️ yield promise')
36
+ console.log('⏱️ yield promise');
37
37
  // if the value is a promise, schedule it to be awaited after the current frame is finished
38
38
  nextTickFuture.push(async () => {
39
39
  try {
@@ -45,7 +45,7 @@ export function createTestRuntime(testingModule, engine) {
45
45
  });
46
46
  }
47
47
  else if (typeof value === 'function') {
48
- // console.log('⏱️ yield function')
48
+ console.log('⏱️ yield function');
49
49
  // if the value is a function, schedule it to be called on the next frame
50
50
  nextTickFuture.push(() => {
51
51
  scheduleValue(value(), env);
@@ -53,7 +53,7 @@ export function createTestRuntime(testingModule, engine) {
53
53
  return;
54
54
  }
55
55
  else if (typeof value === 'undefined' || value === null) {
56
- // console.log('⏱️ yield')
56
+ console.log('⏱️ yield');
57
57
  // if the value is undefined or null, continue processing the generator the next frame
58
58
  nextTickFuture.push(() => {
59
59
  consumeGenerator(env);
@@ -194,4 +194,4 @@ function globalFail(error) {
194
194
  // for now, the failure is only writing to the console.error.
195
195
  console.error(error);
196
196
  }
197
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime.js","sourceRoot":"","sources":["../src/testing/runtime.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This module provides a createTestRuntime function that returns an object with a test function that can be used to define tests.\n */\n\nimport { IEngine, Transform } from '@dcl/ecs'\nimport { assertEquals } from './assert'\nimport type { TestingModule, TestFunction, TestHelpers } from './types'\n\n// This function creates a test runtime that can be used to define and run tests.\n// It takes a `TestingModule` instance (loaded from require('~system/Testing')) and an `IEngine` instance (from Decentraland's SDK).\n// It returns an object with a `test` function that can be used to define tests.\n/* @__PURE__ */\nexport function createTestRuntime(testingModule: TestingModule, engine: IEngine) {\n  type TestPlanEntry = { name: string; fn: TestFunction }\n  type RunnerEnvironment = {\n    resolve: () => void\n    reject: (error: any) => void\n    helpers: TestHelpers\n    generator: Generator\n  }\n\n  // this flag ensures no tests are added asynchronously\n  let runtimeFrozen = false\n\n  let currentFrameCounter = 0\n  let currentFrameTime = 0\n\n  // array to hold the scheduled tests\n  const scheduledTests: TestPlanEntry[] = []\n\n  // an array of promises that are resolved on the next frame (after the current frame is finished)\n  const nextTickFuture: Array<(dt: number) => void> = []\n\n  // this function returns a promise that resolves on the next frame\n  async function nextTick() {\n    return new Promise<number>((resolve) => {\n      nextTickFuture.push(resolve)\n    })\n  }\n\n  // add a system to the engine that resolves all promises in the `nextTickFuture` array\n  engine.addSystem(function TestingFrameworkCoroutineRunner(dt) {\n    currentFrameCounter++\n    currentFrameTime += dt\n    // resolve all nextTick futures.\n    nextTickFuture.splice(0, nextTickFuture.length).forEach((_) => _(dt))\n  })\n\n  // this function schedules a value to be processed on the next frame, the test runner will\n  // continue to run until it reaches a yield point\n  function scheduleValue(value: any, env: RunnerEnvironment) {\n    if (value && typeof value === 'object' && typeof value.then === 'function') {\n      // console.log('⏱️ yield promise')\n      // if the value is a promise, schedule it to be awaited after the current frame is finished\n      nextTickFuture.push(async () => {\n        try {\n          scheduleValue(await value, env)\n        } catch (err) {\n          env.reject(err)\n        }\n      })\n    } else if (typeof value === 'function') {\n      // console.log('⏱️ yield function')\n      // if the value is a function, schedule it to be called on the next frame\n      nextTickFuture.push(() => {\n        scheduleValue(value(), env)\n      })\n      return\n    } else if (typeof value === 'undefined' || value === null) {\n      // console.log('⏱️ yield')\n      // if the value is undefined or null, continue processing the generator the next frame\n      nextTickFuture.push(() => {\n        consumeGenerator(env)\n      })\n    } else throw new Error(`Unexpected value from test generator: ${value}`)\n  }\n\n  // this function processes a generator function by scheduling its values to be processed on the next frame\n  function consumeGenerator(env: RunnerEnvironment) {\n    try {\n      const ret = env.generator.next()\n      if (!ret.done) {\n        scheduleValue(ret.value, env)\n      } else {\n        env.resolve()\n      }\n    } catch (err) {\n      env.reject(err)\n    }\n  }\n\n  // this function schedules a test run on the next frame\n  function scheduleNextRun() {\n    if (scheduledTests.length) {\n      nextTickFuture.push(runTests)\n    }\n  }\n\n  // this function runs the scheduled tests\n  function runTests() {\n    if (scheduledTests.length) {\n      const entry = scheduledTests.shift()!\n      const initialFrame = currentFrameCounter\n      const startTime = currentFrameTime\n\n      let resolved = false\n\n      // this function should be called only once. it makes the current test pass\n      const resolve = () => {\n        if (resolved) throw new Error('resolved twice')\n        resolved = true\n\n        console.log(`🟢 Test passed ${entry.name}`)\n\n        testingModule\n          .logTestResult({\n            name: entry.name,\n            ok: true,\n            totalFrames: currentFrameCounter - initialFrame,\n            totalTime: currentFrameTime - startTime\n          })\n          .finally(scheduleNextRun)\n      }\n\n      const reject = (err: any) => {\n        if (resolved) throw new Error('resolved twice')\n        resolved = true\n\n        console.log(`🔴 Test failed ${entry.name}`)\n        console.error(err)\n\n        testingModule\n          .logTestResult({\n            name: entry.name,\n            ok: false,\n            error: err.toString(),\n            stack: err && typeof err === 'object' && err.stack,\n            totalFrames: currentFrameCounter - initialFrame,\n            totalTime: currentFrameTime - startTime\n          })\n          .finally(scheduleNextRun)\n      }\n\n      try {\n        console.log(`🧪 Running test ${entry.name}`)\n\n        const testHelpers: TestHelpers = {\n          async setCameraTransform(transform) {\n            await testingModule.setCameraTransform(transform)\n            await nextTick()\n\n            const TransformComponent = engine.getComponent(Transform.componentId) as typeof Transform\n            const actualTransform = TransformComponent.get(engine.CameraEntity)\n\n            assertEquals(actualTransform.position, transform.position, \"positions don't match\")\n            assertEquals(actualTransform.rotation, transform.rotation, \"rotations don't match\")\n          }\n        }\n\n        const returnValue = entry.fn(testHelpers)\n\n        if (returnValue && typeof returnValue === 'object') {\n          if (isGenerator(returnValue)) {\n            const env: RunnerEnvironment = {\n              generator: returnValue,\n              helpers: testHelpers,\n              resolve,\n              reject\n            }\n            consumeGenerator(env)\n          } else if (isPromise(returnValue)) {\n            returnValue.then(resolve).catch(reject)\n          } else {\n            throw new Error(`Unknown test result type: ${returnValue}`)\n          }\n        } else {\n          resolve()\n        }\n      } catch (err: any) {\n        reject(err)\n      }\n    }\n  }\n\n  // schedule the test runner start for the next frame\n  nextTickFuture.push(() => {\n    // once we run the next tick, the test runtime becomes frozen. that means no new\n    // test definitions are accepted\n    runtimeFrozen = true\n\n    if (!scheduledTests.length) return\n\n    // inform the test runner about the plans for this test run\n    testingModule.plan({ tests: scheduledTests }).then(scheduleNextRun).catch(globalFail)\n  })\n\n  // this is the function that is used to plan a test functionn\n  /* @__PURE__ */\n  function test(name: string, fn: TestFunction) {\n    if (runtimeFrozen) throw new Error(\"New tests can't be added at this stage.\")\n\n    if (scheduledTests.some(($) => $.name === name)) throw new Error(`Test with name ${name} already exists`)\n\n    scheduledTests.push({ fn, name })\n  }\n\n  return {\n    test\n  }\n}\n\nfunction isGenerator(t: any): t is Generator {\n  return t && typeof t === 'object' && typeof t[Symbol.iterator] === 'function'\n}\n\nfunction isPromise(t: any): t is Promise<unknown> {\n  return t && typeof t === 'object' && typeof t.then === 'function'\n}\n\nfunction globalFail(error: any) {\n  // for now, the failure is only writing to the console.error.\n  console.error(error)\n}\n"]}
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime.js","sourceRoot":"","sources":["../src/testing/runtime.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This module provides a createTestRuntime function that returns an object with a test function that can be used to define tests.\n */\n\nimport { IEngine, Transform } from '@dcl/ecs'\nimport { assertEquals } from './assert'\nimport type { TestingModule, TestFunction, TestHelpers } from './types'\n\n// This function creates a test runtime that can be used to define and run tests.\n// It takes a `TestingModule` instance (loaded from require('~system/Testing')) and an `IEngine` instance (from Decentraland's SDK).\n// It returns an object with a `test` function that can be used to define tests.\n/* @__PURE__ */\nexport function createTestRuntime(testingModule: TestingModule, engine: IEngine) {\n  type TestPlanEntry = { name: string; fn: TestFunction }\n  type RunnerEnvironment = {\n    resolve: () => void\n    reject: (error: any) => void\n    helpers: TestHelpers\n    generator: Generator\n  }\n\n  // this flag ensures no tests are added asynchronously\n  let runtimeFrozen = false\n\n  let currentFrameCounter = 0\n  let currentFrameTime = 0\n\n  // array to hold the scheduled tests\n  const scheduledTests: TestPlanEntry[] = []\n\n  // an array of promises that are resolved on the next frame (after the current frame is finished)\n  const nextTickFuture: Array<(dt: number) => void> = []\n\n  // this function returns a promise that resolves on the next frame\n  async function nextTick() {\n    return new Promise<number>((resolve) => {\n      nextTickFuture.push(resolve)\n    })\n  }\n\n  // add a system to the engine that resolves all promises in the `nextTickFuture` array\n  engine.addSystem(function TestingFrameworkCoroutineRunner(dt) {\n    currentFrameCounter++\n    currentFrameTime += dt\n    // resolve all nextTick futures.\n    nextTickFuture.splice(0, nextTickFuture.length).forEach((_) => _(dt))\n  })\n\n  // this function schedules a value to be processed on the next frame, the test runner will\n  // continue to run until it reaches a yield point\n  function scheduleValue(value: any, env: RunnerEnvironment) {\n    if (value && typeof value === 'object' && typeof value.then === 'function') {\n      console.log('⏱️ yield promise')\n      // if the value is a promise, schedule it to be awaited after the current frame is finished\n      nextTickFuture.push(async () => {\n        try {\n          scheduleValue(await value, env)\n        } catch (err) {\n          env.reject(err)\n        }\n      })\n    } else if (typeof value === 'function') {\n      console.log('⏱️ yield function')\n      // if the value is a function, schedule it to be called on the next frame\n      nextTickFuture.push(() => {\n        scheduleValue(value(), env)\n      })\n      return\n    } else if (typeof value === 'undefined' || value === null) {\n      console.log('⏱️ yield')\n      // if the value is undefined or null, continue processing the generator the next frame\n      nextTickFuture.push(() => {\n        consumeGenerator(env)\n      })\n    } else throw new Error(`Unexpected value from test generator: ${value}`)\n  }\n\n  // this function processes a generator function by scheduling its values to be processed on the next frame\n  function consumeGenerator(env: RunnerEnvironment) {\n    try {\n      const ret = env.generator.next()\n      if (!ret.done) {\n        scheduleValue(ret.value, env)\n      } else {\n        env.resolve()\n      }\n    } catch (err) {\n      env.reject(err)\n    }\n  }\n\n  // this function schedules a test run on the next frame\n  function scheduleNextRun() {\n    if (scheduledTests.length) {\n      nextTickFuture.push(runTests)\n    }\n  }\n\n  // this function runs the scheduled tests\n  function runTests() {\n    if (scheduledTests.length) {\n      const entry = scheduledTests.shift()!\n      const initialFrame = currentFrameCounter\n      const startTime = currentFrameTime\n\n      let resolved = false\n\n      // this function should be called only once. it makes the current test pass\n      const resolve = () => {\n        if (resolved) throw new Error('resolved twice')\n        resolved = true\n\n        console.log(`🟢 Test passed ${entry.name}`)\n\n        testingModule\n          .logTestResult({\n            name: entry.name,\n            ok: true,\n            totalFrames: currentFrameCounter - initialFrame,\n            totalTime: currentFrameTime - startTime\n          })\n          .finally(scheduleNextRun)\n      }\n\n      const reject = (err: any) => {\n        if (resolved) throw new Error('resolved twice')\n        resolved = true\n\n        console.log(`🔴 Test failed ${entry.name}`)\n        console.error(err)\n\n        testingModule\n          .logTestResult({\n            name: entry.name,\n            ok: false,\n            error: err.toString(),\n            stack: err && typeof err === 'object' && err.stack,\n            totalFrames: currentFrameCounter - initialFrame,\n            totalTime: currentFrameTime - startTime\n          })\n          .finally(scheduleNextRun)\n      }\n\n      try {\n        console.log(`🧪 Running test ${entry.name}`)\n\n        const testHelpers: TestHelpers = {\n          async setCameraTransform(transform) {\n            await testingModule.setCameraTransform(transform)\n            await nextTick()\n\n            const TransformComponent = engine.getComponent(Transform.componentId) as typeof Transform\n            const actualTransform = TransformComponent.get(engine.CameraEntity)\n\n            assertEquals(actualTransform.position, transform.position, \"positions don't match\")\n            assertEquals(actualTransform.rotation, transform.rotation, \"rotations don't match\")\n          }\n        }\n\n        const returnValue = entry.fn(testHelpers)\n\n        if (returnValue && typeof returnValue === 'object') {\n          if (isGenerator(returnValue)) {\n            const env: RunnerEnvironment = {\n              generator: returnValue,\n              helpers: testHelpers,\n              resolve,\n              reject\n            }\n            consumeGenerator(env)\n          } else if (isPromise(returnValue)) {\n            returnValue.then(resolve).catch(reject)\n          } else {\n            throw new Error(`Unknown test result type: ${returnValue}`)\n          }\n        } else {\n          resolve()\n        }\n      } catch (err: any) {\n        reject(err)\n      }\n    }\n  }\n\n  // schedule the test runner start for the next frame\n  nextTickFuture.push(() => {\n    // once we run the next tick, the test runtime becomes frozen. that means no new\n    // test definitions are accepted\n    runtimeFrozen = true\n\n    if (!scheduledTests.length) return\n\n    // inform the test runner about the plans for this test run\n    testingModule.plan({ tests: scheduledTests }).then(scheduleNextRun).catch(globalFail)\n  })\n\n  // this is the function that is used to plan a test functionn\n  /* @__PURE__ */\n  function test(name: string, fn: TestFunction) {\n    if (runtimeFrozen) throw new Error(\"New tests can't be added at this stage.\")\n\n    if (scheduledTests.some(($) => $.name === name)) throw new Error(`Test with name ${name} already exists`)\n\n    scheduledTests.push({ fn, name })\n  }\n\n  return {\n    test\n  }\n}\n\nfunction isGenerator(t: any): t is Generator {\n  return t && typeof t === 'object' && typeof t[Symbol.iterator] === 'function'\n}\n\nfunction isPromise(t: any): t is Promise<unknown> {\n  return t && typeof t === 'object' && typeof t.then === 'function'\n}\n\nfunction globalFail(error: any) {\n  // for now, the failure is only writing to the console.error.\n  console.error(error)\n}\n"]}