angular-three 2.0.0-beta.9 → 2.1.0

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.
Files changed (160) hide show
  1. package/README.md +221 -4
  2. package/esm2022/index.mjs +13 -14
  3. package/esm2022/lib/canvas.mjs +130 -197
  4. package/esm2022/lib/directives/args.mjs +46 -20
  5. package/esm2022/lib/directives/selection.mjs +65 -0
  6. package/esm2022/lib/dom/events.mjs +2 -2
  7. package/esm2022/lib/events.mjs +33 -31
  8. package/esm2022/lib/html.mjs +40 -0
  9. package/esm2022/lib/instance.mjs +43 -36
  10. package/esm2022/lib/loader.mjs +72 -36
  11. package/esm2022/lib/loop.mjs +29 -26
  12. package/esm2022/lib/pipes/hexify.mjs +67 -0
  13. package/esm2022/lib/portal.mjs +173 -193
  14. package/esm2022/lib/renderer/catalogue.mjs +3 -3
  15. package/esm2022/lib/renderer/constants.mjs +6 -6
  16. package/esm2022/lib/renderer/index.mjs +356 -217
  17. package/esm2022/lib/renderer/state.mjs +54 -0
  18. package/esm2022/lib/renderer/utils.mjs +120 -64
  19. package/esm2022/lib/roots.mjs +86 -62
  20. package/esm2022/lib/routed-scene.mjs +11 -8
  21. package/esm2022/lib/store.mjs +164 -195
  22. package/esm2022/lib/three-types.mjs +1 -1
  23. package/esm2022/lib/types.mjs +1 -1
  24. package/esm2022/lib/utils/apply-props.mjs +31 -28
  25. package/esm2022/lib/utils/attach.mjs +26 -12
  26. package/esm2022/lib/utils/before-render.mjs +12 -0
  27. package/esm2022/lib/utils/is.mjs +6 -5
  28. package/esm2022/lib/utils/make.mjs +19 -17
  29. package/esm2022/lib/utils/non-nullish.mjs +7 -0
  30. package/esm2022/lib/utils/object-events.mjs +91 -0
  31. package/esm2022/lib/utils/parameters.mjs +70 -0
  32. package/esm2022/lib/utils/resolve-ref.mjs +8 -0
  33. package/esm2022/lib/utils/signal-store.mjs +59 -60
  34. package/esm2022/lib/utils/update.mjs +8 -4
  35. package/esm2022/nativescript/angular-three-nativescript.mjs +5 -0
  36. package/esm2022/nativescript/index.mjs +2 -0
  37. package/esm2022/nativescript/lib/canvas.mjs +127 -0
  38. package/esm2022/testing/angular-three-testing.mjs +5 -0
  39. package/esm2022/testing/index.mjs +3 -0
  40. package/esm2022/testing/lib/test-bed.mjs +130 -0
  41. package/esm2022/testing/lib/test-canvas.mjs +45 -0
  42. package/esm2022/testing/lib/utils/mock-canvas.mjs +37 -0
  43. package/esm2022/testing/lib/utils/web-gl-rendering-context.mjs +752 -0
  44. package/fesm2022/angular-three-nativescript.mjs +134 -0
  45. package/fesm2022/angular-three-nativescript.mjs.map +1 -0
  46. package/fesm2022/angular-three-testing.mjs +966 -0
  47. package/fesm2022/angular-three-testing.mjs.map +1 -0
  48. package/fesm2022/angular-three.mjs +2271 -2306
  49. package/fesm2022/angular-three.mjs.map +1 -1
  50. package/index.d.ts +14 -14
  51. package/lib/canvas.d.ts +357 -96
  52. package/lib/directives/args.d.ts +14 -7
  53. package/lib/directives/selection.d.ts +17 -0
  54. package/lib/dom/events.d.ts +2 -3
  55. package/lib/events.d.ts +4 -80
  56. package/lib/html.d.ts +17 -0
  57. package/lib/instance.d.ts +3 -35
  58. package/lib/loader.d.ts +19 -7
  59. package/lib/loop.d.ts +11 -59
  60. package/lib/pipes/hexify.d.ts +20 -0
  61. package/lib/portal.d.ts +54 -48
  62. package/lib/renderer/catalogue.d.ts +7 -3
  63. package/lib/renderer/constants.d.ts +5 -5
  64. package/lib/renderer/index.d.ts +63 -4
  65. package/lib/renderer/state.d.ts +25 -0
  66. package/lib/renderer/utils.d.ts +10 -27
  67. package/lib/roots.d.ts +9 -7
  68. package/lib/store.d.ts +13 -141
  69. package/lib/three-types.d.ts +528 -147
  70. package/lib/types.d.ts +291 -0
  71. package/lib/utils/apply-props.d.ts +2 -3
  72. package/lib/utils/attach.d.ts +4 -6
  73. package/lib/{before-render.d.ts → utils/before-render.d.ts} +1 -1
  74. package/lib/utils/is.d.ts +13 -14
  75. package/lib/utils/make.d.ts +7 -13
  76. package/lib/utils/non-nullish.d.ts +4 -0
  77. package/lib/utils/object-events.d.ts +32 -0
  78. package/lib/utils/parameters.d.ts +20 -0
  79. package/lib/utils/resolve-ref.d.ts +2 -0
  80. package/lib/utils/signal-store.d.ts +13 -4
  81. package/lib/utils/update.d.ts +1 -1
  82. package/metadata.json +1 -1
  83. package/nativescript/README.md +5 -0
  84. package/nativescript/index.d.ts +1 -0
  85. package/nativescript/lib/canvas.d.ts +359 -0
  86. package/nativescript/package.json +3 -0
  87. package/package.json +65 -14
  88. package/plugin/generators.json +8 -30
  89. package/plugin/package.json +3 -3
  90. package/plugin/src/generators/add-soba/compat.js.map +1 -0
  91. package/plugin/src/generators/add-soba/generator.d.ts +3 -0
  92. package/plugin/src/generators/add-soba/generator.js +77 -0
  93. package/plugin/src/generators/add-soba/generator.js.map +1 -0
  94. package/plugin/src/generators/add-soba/schema.json +4 -0
  95. package/plugin/src/generators/init/compat.d.ts +1 -1
  96. package/plugin/src/generators/init/files/experience/experience.component.ts__tmpl__ +29 -0
  97. package/plugin/src/generators/init/generator.d.ts +5 -1
  98. package/plugin/src/generators/init/generator.js +131 -25
  99. package/plugin/src/generators/init/generator.js.map +1 -1
  100. package/plugin/src/generators/init/schema.json +9 -2
  101. package/plugin/src/generators/utils.d.ts +2 -0
  102. package/plugin/src/generators/utils.js +35 -0
  103. package/plugin/src/generators/utils.js.map +1 -0
  104. package/plugin/src/generators/version.d.ts +17 -0
  105. package/plugin/src/generators/version.js +21 -0
  106. package/plugin/src/generators/version.js.map +1 -0
  107. package/plugin/src/index.d.ts +0 -3
  108. package/plugin/src/index.js +0 -9
  109. package/plugin/src/index.js.map +1 -1
  110. package/testing/README.md +3 -0
  111. package/testing/index.d.ts +2 -0
  112. package/testing/lib/test-bed.d.ts +38 -0
  113. package/testing/lib/test-canvas.d.ts +11 -0
  114. package/testing/lib/utils/mock-canvas.d.ts +5 -0
  115. package/testing/lib/utils/web-gl-rendering-context.d.ts +16 -0
  116. package/testing/package.json +3 -0
  117. package/web-types.json +1 -1
  118. package/esm2022/lib/before-render.mjs +0 -13
  119. package/esm2022/lib/directives/common.mjs +0 -41
  120. package/esm2022/lib/directives/key.mjs +0 -29
  121. package/esm2022/lib/directives/parent.mjs +0 -35
  122. package/esm2022/lib/directives/repeat.mjs +0 -17
  123. package/esm2022/lib/ref.mjs +0 -48
  124. package/esm2022/lib/renderer/store.mjs +0 -405
  125. package/esm2022/lib/utils/assert-injection-context.mjs +0 -14
  126. package/esm2022/lib/utils/create-injection-token.mjs +0 -47
  127. package/esm2022/lib/utils/safe-detect-changes.mjs +0 -17
  128. package/lib/directives/common.d.ts +0 -15
  129. package/lib/directives/key.d.ts +0 -10
  130. package/lib/directives/parent.d.ts +0 -11
  131. package/lib/directives/repeat.d.ts +0 -7
  132. package/lib/ref.d.ts +0 -8
  133. package/lib/renderer/store.d.ts +0 -67
  134. package/lib/utils/assert-injection-context.d.ts +0 -2
  135. package/lib/utils/create-injection-token.d.ts +0 -27
  136. package/lib/utils/safe-detect-changes.d.ts +0 -2
  137. package/plugin/src/generators/init-cannon/compat.js.map +0 -1
  138. package/plugin/src/generators/init-cannon/generator.d.ts +0 -2
  139. package/plugin/src/generators/init-cannon/generator.js +0 -22
  140. package/plugin/src/generators/init-cannon/generator.js.map +0 -1
  141. package/plugin/src/generators/init-cannon/schema.json +0 -6
  142. package/plugin/src/generators/init-postprocessing/compat.d.ts +0 -2
  143. package/plugin/src/generators/init-postprocessing/compat.js +0 -6
  144. package/plugin/src/generators/init-postprocessing/compat.js.map +0 -1
  145. package/plugin/src/generators/init-postprocessing/generator.d.ts +0 -2
  146. package/plugin/src/generators/init-postprocessing/generator.js +0 -20
  147. package/plugin/src/generators/init-postprocessing/generator.js.map +0 -1
  148. package/plugin/src/generators/init-postprocessing/schema.json +0 -6
  149. package/plugin/src/generators/init-soba/compat.d.ts +0 -2
  150. package/plugin/src/generators/init-soba/compat.js +0 -6
  151. package/plugin/src/generators/init-soba/compat.js.map +0 -1
  152. package/plugin/src/generators/init-soba/generator.d.ts +0 -2
  153. package/plugin/src/generators/init-soba/generator.js +0 -24
  154. package/plugin/src/generators/init-soba/generator.js.map +0 -1
  155. package/plugin/src/generators/init-soba/schema.json +0 -6
  156. package/plugin/src/generators/versions.d.ts +0 -12
  157. package/plugin/src/generators/versions.js +0 -16
  158. package/plugin/src/generators/versions.js.map +0 -1
  159. /package/plugin/src/generators/{init-cannon → add-soba}/compat.d.ts +0 -0
  160. /package/plugin/src/generators/{init-cannon → add-soba}/compat.js +0 -0
@@ -1,207 +1,176 @@
1
- import { DOCUMENT } from '@angular/common';
2
- import { ChangeDetectorRef, ElementRef, InjectionToken, Injector, Optional, SkipSelf, effect, inject, runInInjectionContext, } from '@angular/core';
1
+ import { ElementRef, InjectionToken, effect, inject } from '@angular/core';
3
2
  import { Subject } from 'rxjs';
4
- import * as THREE from 'three';
5
- import { NGT_LOOP } from './loop';
6
- import { createInjectionToken } from './utils/create-injection-token';
3
+ import { Clock, Vector2, Vector3 } from 'three';
4
+ import { injectLoop } from './loop';
7
5
  import { is } from './utils/is';
8
6
  import { makeDpr } from './utils/make';
9
- import { safeDetectChanges } from './utils/safe-detect-changes';
10
7
  import { signalStore } from './utils/signal-store';
11
8
  import { updateCamera } from './utils/update';
12
- function storeFactory(loop, document, injector, parent) {
13
- return runInInjectionContext(injector, () => {
14
- const window = document.defaultView;
15
- if (!window) {
16
- // TODO: revisit this when we need to support multiple platforms
17
- throw new Error(`[NGT] Window is not available.`);
18
- }
19
- const cdr = inject(ChangeDetectorRef);
20
- // NOTE: using Subject because we do not care about late-subscribers
21
- const pointerMissed$ = new Subject();
22
- const store = signalStore(({ get, set }) => {
23
- const { invalidate, advance } = loop;
24
- const position = new THREE.Vector3();
25
- const defaultTarget = new THREE.Vector3();
26
- const tempTarget = new THREE.Vector3();
27
- function getCurrentViewport(camera = get('camera'), target = defaultTarget, size = get('size')) {
28
- const { width, height, top, left } = size;
29
- const aspect = width / height;
30
- if (target instanceof THREE.Vector3)
31
- tempTarget.copy(target);
32
- else
33
- tempTarget.set(...target);
34
- const distance = camera.getWorldPosition(position).distanceTo(tempTarget);
35
- if (is.orthographicCamera(camera)) {
36
- return {
37
- width: width / camera.zoom,
38
- height: height / camera.zoom,
39
- top,
40
- left,
41
- factor: 1,
42
- distance,
43
- aspect,
44
- };
45
- }
46
- else {
47
- const fov = (camera.fov * Math.PI) / 180; // convert vertical fov to radians
48
- const h = 2 * Math.tan(fov / 2) * distance; // visible height
49
- const w = h * (width / height);
50
- return { width: w, height: h, top, left, factor: width / w, distance, aspect };
51
- }
9
+ function storeFactory(previousStore) {
10
+ const loop = injectLoop();
11
+ // NOTE: using Subject because we do not care about late-subscribers
12
+ const pointerMissed$ = new Subject();
13
+ const store = signalStore(({ get, update }) => {
14
+ const { invalidate, advance } = loop;
15
+ const position = new Vector3();
16
+ const defaultTarget = new Vector3();
17
+ const tempTarget = new Vector3();
18
+ function getCurrentViewport(camera = get('camera'), target = defaultTarget, size = get('size')) {
19
+ const { width, height, top, left } = size;
20
+ const aspect = width / height;
21
+ if (target instanceof Vector3)
22
+ tempTarget.copy(target);
23
+ else
24
+ tempTarget.set(...target);
25
+ const distance = camera.getWorldPosition(position).distanceTo(tempTarget);
26
+ if (is.orthographicCamera(camera)) {
27
+ return { width: width / camera.zoom, height: height / camera.zoom, top, left, factor: 1, distance, aspect };
52
28
  }
53
- let performanceTimeout = undefined;
54
- const setPerformanceCurrent = (current) => set((state) => ({ performance: { ...state.performance, current } }));
55
- const pointer = new THREE.Vector2();
56
- return {
57
- pointerMissed$: pointerMissed$.asObservable(),
58
- events: { priority: 1, enabled: true, connected: false },
59
- invalidate: (frames = 1) => invalidate(store, frames),
60
- advance: (timestamp, runGlobalEffects) => advance(timestamp, runGlobalEffects, store),
61
- legacy: false,
62
- linear: false,
63
- flat: false,
64
- controls: null,
65
- clock: new THREE.Clock(),
66
- pointer,
67
- frameloop: 'always',
68
- performance: {
69
- current: 1,
70
- min: 0.5,
71
- max: 1,
72
- debounce: 200,
73
- regress: () => {
74
- const state = get();
75
- // Clear timeout
76
- if (performanceTimeout)
77
- clearTimeout(performanceTimeout);
78
- // Set lower bound performance
79
- if (state.performance.current !== state.performance.min)
80
- setPerformanceCurrent(state.performance.min);
81
- // Go back to upper bound performance after a while unless something regresses meanwhile
82
- performanceTimeout = setTimeout(() => {
83
- setPerformanceCurrent(get('performance', 'max'));
84
- safeDetectChanges(cdr);
85
- }, state.performance.debounce);
86
- },
87
- },
88
- size: { width: 0, height: 0, top: 0, left: 0, updateStyle: false },
89
- viewport: {
90
- initialDpr: 0,
91
- dpr: 0,
92
- width: 0,
93
- height: 0,
94
- top: 0,
95
- left: 0,
96
- aspect: 0,
97
- distance: 0,
98
- factor: 0,
99
- getCurrentViewport,
100
- },
101
- setEvents: (events) => set((state) => ({ ...state, events: { ...state.events, ...events } })),
102
- setSize: (width, height, top, left) => {
103
- const camera = get('camera');
104
- const size = { width, height, top: top || 0, left: left || 0 };
105
- set((state) => ({
106
- size,
107
- viewport: { ...state.viewport, ...getCurrentViewport(camera, defaultTarget, size) },
108
- }));
29
+ const fov = (camera.fov * Math.PI) / 180; // convert vertical fov to radians
30
+ const h = 2 * Math.tan(fov / 2) * distance; // visible height
31
+ const w = h * (width / height);
32
+ return { width: w, height: h, top, left, factor: width / w, distance, aspect };
33
+ }
34
+ let performanceTimeout = undefined;
35
+ const setPerformanceCurrent = (current) => update((state) => ({ performance: { ...state.performance, current } }));
36
+ const pointer = new Vector2();
37
+ return {
38
+ pointerMissed$: pointerMissed$.asObservable(),
39
+ events: { priority: 1, enabled: true, connected: false },
40
+ invalidate: (frames = 1) => invalidate(store, frames),
41
+ advance: (timestamp, runGlobalEffects) => advance(timestamp, runGlobalEffects, store),
42
+ legacy: false,
43
+ linear: false,
44
+ flat: false,
45
+ controls: null,
46
+ clock: new Clock(),
47
+ pointer,
48
+ frameloop: 'always',
49
+ performance: {
50
+ current: 1,
51
+ min: 0.5,
52
+ max: 1,
53
+ debounce: 200,
54
+ regress: () => {
55
+ const state = get();
56
+ // Clear timeout
57
+ if (performanceTimeout)
58
+ clearTimeout(performanceTimeout);
59
+ // Set lower bound performance
60
+ if (state.performance.current !== state.performance.min)
61
+ setPerformanceCurrent(state.performance.min);
62
+ // Go back to upper bound performance after a while unless something regresses meanwhile
63
+ performanceTimeout = setTimeout(() => {
64
+ setPerformanceCurrent(get('performance', 'max'));
65
+ // safeDetectChanges(cdr);
66
+ }, state.performance.debounce);
109
67
  },
110
- setDpr: (dpr) => set((state) => {
111
- const resolved = makeDpr(dpr, window);
112
- return {
113
- viewport: {
114
- ...state.viewport,
115
- dpr: resolved,
116
- initialDpr: state.viewport.initialDpr || resolved,
117
- },
118
- };
119
- }),
120
- setFrameloop: (frameloop = 'always') => {
121
- const clock = get('clock');
122
- // if frameloop === "never" clock.elapsedTime is updated using advance(timestamp)
123
- clock.stop();
68
+ },
69
+ size: { width: 0, height: 0, top: 0, left: 0, updateStyle: false },
70
+ viewport: {
71
+ initialDpr: 0,
72
+ dpr: 0,
73
+ width: 0,
74
+ height: 0,
75
+ top: 0,
76
+ left: 0,
77
+ aspect: 0,
78
+ distance: 0,
79
+ factor: 0,
80
+ getCurrentViewport,
81
+ },
82
+ setEvents: (events) => update((state) => ({ ...state, events: { ...state.events, ...events } })),
83
+ setSize: (width, height, top, left) => {
84
+ const camera = get('camera');
85
+ const size = { width, height, top: top || 0, left: left || 0 };
86
+ update((state) => ({
87
+ size,
88
+ viewport: { ...state.viewport, ...getCurrentViewport(camera, defaultTarget, size) },
89
+ }));
90
+ },
91
+ setDpr: (dpr) => update((state) => {
92
+ const resolved = makeDpr(dpr, window);
93
+ return { viewport: { ...state.viewport, dpr: resolved, initialDpr: state.viewport.initialDpr || resolved } };
94
+ }),
95
+ setFrameloop: (frameloop = 'always') => {
96
+ const clock = get('clock');
97
+ // if frameloop === "never" clock.elapsedTime is updated using advance(timestamp)
98
+ clock.stop();
99
+ clock.elapsedTime = 0;
100
+ if (frameloop !== 'never') {
101
+ clock.start();
124
102
  clock.elapsedTime = 0;
125
- if (frameloop !== 'never') {
126
- clock.start();
127
- clock.elapsedTime = 0;
128
- }
129
- set(() => ({ frameloop }));
130
- },
131
- previousRoot: parent,
132
- internal: {
133
- active: false,
134
- priority: 0,
135
- frames: 0,
136
- lastEvent: new ElementRef(null),
137
- interaction: [],
138
- hovered: new Map(),
139
- subscribers: [],
140
- initialClick: [0, 0],
141
- initialHits: [],
142
- capturedMap: new Map(),
143
- subscribe: (callback, priority = 0, _store = store) => {
103
+ }
104
+ update(() => ({ frameloop }));
105
+ },
106
+ previousRoot: previousStore,
107
+ internal: {
108
+ active: false,
109
+ priority: 0,
110
+ frames: 0,
111
+ lastEvent: new ElementRef(null),
112
+ interaction: [],
113
+ hovered: new Map(),
114
+ subscribers: [],
115
+ initialClick: [0, 0],
116
+ initialHits: [],
117
+ capturedMap: new Map(),
118
+ subscribe: (callback, priority = 0, _store = store) => {
119
+ const internal = get('internal');
120
+ // If this subscription was given a priority, it takes rendering into its own hands
121
+ // For that reason we switch off automatic rendering and increase the manual flag
122
+ // As long as this flag is positive there can be no internal rendering at all
123
+ // because there could be multiple render subscriptions
124
+ internal.priority = internal.priority + (priority > 0 ? 1 : 0);
125
+ internal.subscribers.push({ callback, priority, store: _store });
126
+ // Register subscriber and sort layers from lowest to highest, meaning,
127
+ // highest priority renders last (on top of the other frames)
128
+ internal.subscribers = internal.subscribers.sort((a, b) => (a.priority || 0) - (b.priority || 0));
129
+ return () => {
144
130
  const internal = get('internal');
145
- // If this subscription was given a priority, it takes rendering into its own hands
146
- // For that reason we switch off automatic rendering and increase the manual flag
147
- // As long as this flag is positive there can be no internal rendering at all
148
- // because there could be multiple render subscriptions
149
- internal.priority = internal.priority + (priority > 0 ? 1 : 0);
150
- internal.subscribers.push({ callback, priority, store });
151
- // Register subscriber and sort layers from lowest to highest, meaning,
152
- // highest priority renders last (on top of the other frames)
153
- internal.subscribers = internal.subscribers.sort((a, b) => (a.priority || 0) - (b.priority || 0));
154
- return () => {
155
- const internal = get('internal');
156
- if (internal?.subscribers) {
157
- // Decrease manual flag if this subscription had a priority
158
- internal.priority = internal.priority - (priority > 0 ? 1 : 0);
159
- // Remove subscriber from list
160
- internal.subscribers = internal.subscribers.filter((s) => s.callback !== callback);
161
- }
162
- };
163
- },
131
+ if (internal?.subscribers) {
132
+ // Decrease manual flag if this subscription had a priority
133
+ internal.priority = internal.priority - (priority > 0 ? 1 : 0);
134
+ // Remove subscriber from list
135
+ internal.subscribers = internal.subscribers.filter((s) => s.callback !== callback);
136
+ }
137
+ };
164
138
  },
165
- };
166
- });
167
- // NOTE: assign pointerMissed$ so we can use it in events
168
- Object.defineProperty(store, 'pointerMissed$', { get: () => pointerMissed$ });
169
- const state = store.get();
170
- let oldSize = state.size;
171
- let oldDpr = state.viewport.dpr;
172
- let oldCamera = state.camera;
173
- const _camera = store.select('camera');
174
- const _size = store.select('size');
175
- const _viewport = store.select('viewport');
176
- effect(() => {
177
- const [camera, size, viewport, gl] = [_camera(), _size(), _viewport(), store.get('gl')];
178
- // Resize camera and renderer on changes to size and pixelratio
179
- if (size !== oldSize || viewport.dpr !== oldDpr) {
180
- oldSize = size;
181
- oldDpr = viewport.dpr;
182
- // Update camera & renderer
183
- updateCamera(camera, size);
184
- gl.setPixelRatio(viewport.dpr);
185
- const updateStyle = typeof HTMLCanvasElement !== 'undefined' && gl.domElement instanceof HTMLCanvasElement;
186
- gl.setSize(size.width, size.height, updateStyle);
187
- }
188
- // Update viewport once the camera changes
189
- if (camera !== oldCamera) {
190
- oldCamera = camera;
191
- updateCamera(camera, size);
192
- // Update viewport
193
- store.set((state) => ({
194
- viewport: { ...state.viewport, ...state.viewport.getCurrentViewport(camera) },
195
- }));
196
- }
197
- });
198
- return store;
139
+ },
140
+ };
199
141
  });
142
+ Object.defineProperty(store, 'pointerMissed$', { get: () => pointerMissed$ });
143
+ let { size: oldSize, viewport: { dpr: oldDpr }, camera: oldCamera, } = store.snapshot;
144
+ effect(() => {
145
+ const { camera: newCamera, size: newSize, viewport: { dpr: newDpr }, gl, } = store.state();
146
+ // Resize camera and renderer on changes to size and pixel-ratio
147
+ if (newSize !== oldSize || newDpr !== oldDpr) {
148
+ oldSize = newSize;
149
+ oldDpr = newDpr;
150
+ // Update camera & renderer
151
+ updateCamera(newCamera, newSize);
152
+ gl.setPixelRatio(newDpr);
153
+ const updateStyle = typeof HTMLCanvasElement !== 'undefined' && gl.domElement instanceof HTMLCanvasElement;
154
+ gl.setSize(newSize.width, newSize.height, updateStyle);
155
+ }
156
+ // Update viewport once the camera changes
157
+ if (newCamera !== oldCamera) {
158
+ oldCamera = newCamera;
159
+ updateCamera(newCamera, newSize);
160
+ // Update viewport
161
+ store.update((state) => ({ viewport: { ...state.viewport, ...state.viewport.getCurrentViewport(newCamera) } }));
162
+ }
163
+ });
164
+ return store;
165
+ }
166
+ export const NGT_STORE = new InjectionToken('NgtStore Token');
167
+ export function provideStore(store) {
168
+ if (store) {
169
+ return { provide: NGT_STORE, useFactory: store };
170
+ }
171
+ return { provide: NGT_STORE, useFactory: storeFactory };
172
+ }
173
+ export function injectStore(options) {
174
+ return inject(NGT_STORE, options);
200
175
  }
201
- export const NGT_STORE = new InjectionToken('NgtStore token');
202
- export const [injectNgtStore, provideNgtStore] = createInjectionToken(storeFactory, {
203
- isRoot: false,
204
- deps: [NGT_LOOP, DOCUMENT, Injector, [new Optional(), new SkipSelf(), NGT_STORE]],
205
- token: NGT_STORE,
206
- });
207
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../../../libs/core/src/lib/store.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n\tChangeDetectorRef,\n\tElementRef,\n\tInjectionToken,\n\tInjector,\n\tOptional,\n\tSkipSelf,\n\teffect,\n\tinject,\n\trunInInjectionContext,\n} from '@angular/core';\nimport { Subject, type Observable } from 'rxjs';\nimport * as THREE from 'three';\nimport type { NgtCamera, NgtDomEvent, NgtEventManager, NgtPointerCaptureTarget, NgtThreeEvent } from './events';\nimport type { NgtInstanceNode } from './instance';\nimport { NGT_LOOP, type NgtLoop } from './loop';\nimport { createInjectionToken } from './utils/create-injection-token';\nimport { is } from './utils/is';\nimport { makeDpr } from './utils/make';\nimport { safeDetectChanges } from './utils/safe-detect-changes';\nimport { signalStore, type NgtSignalStore } from './utils/signal-store';\nimport { updateCamera } from './utils/update';\n\nexport type NgtRenderer = { render: (scene: THREE.Scene, camera: THREE.Camera) => any };\nexport type NgtCameraManual = NgtCamera & { manual?: boolean };\n\nexport type NgtDpr = number | [min: number, max: number];\nexport type NgtSize = {\n\twidth: number;\n\theight: number;\n\ttop: number;\n\tleft: number;\n};\n\nexport type NgtViewport = NgtSize & {\n\t/** The initial pixel ratio */\n\tinitialDpr: number;\n\t/** Current pixel ratio */\n\tdpr: number;\n\t/** size.width / viewport.width */\n\tfactor: number;\n\t/** Camera distance */\n\tdistance: number;\n\t/** Camera aspect ratio: width / height */\n\taspect: number;\n};\n\nexport type NgtPerformance = {\n\t/** Current performance normal, between min and max */\n\tcurrent: number;\n\t/** How low the performance can go, between 0 and max */\n\tmin: number;\n\t/** How high the performance can go, between min and max */\n\tmax: number;\n\t/** Time until current returns to max in ms */\n\tdebounce: number;\n\t/** Sets current to min, puts the system in regression */\n\tregress: () => void;\n};\n\nexport type NgtRenderState = NgtState & { delta: number; frame?: XRFrame };\n\nexport type NgtBeforeRenderEvent<TObject extends NgtInstanceNode = NgtInstanceNode> = {\n\tstate: NgtRenderState;\n\tobject: TObject;\n};\n\nexport type NgtBeforeRenderRecord = {\n\tcallback: (state: NgtRenderState) => void;\n\tstore: NgtSignalStore<NgtState>;\n\tpriority?: number;\n};\n\nexport type NgtInternalState = {\n\tactive: boolean;\n\tpriority: number;\n\tframes: number;\n\tlastEvent: ElementRef<NgtDomEvent | null>;\n\tinteraction: THREE.Object3D[];\n\thovered: Map<string, NgtThreeEvent<NgtDomEvent>>;\n\tcapturedMap: Map<number, Map<THREE.Object3D, NgtPointerCaptureTarget>>;\n\tinitialClick: [x: number, y: number];\n\tinitialHits: THREE.Object3D[];\n\tsubscribers: NgtBeforeRenderRecord[];\n\tsubscribe: (\n\t\tcallback: NgtBeforeRenderRecord['callback'],\n\t\tpriority?: number,\n\t\tstore?: NgtSignalStore<NgtState>,\n\t) => () => void;\n};\n\nexport type NgtState = {\n\t/** when all building blocks are initialized */\n\tready: boolean;\n\t/** The instance of the renderer */\n\tgl: THREE.WebGLRenderer;\n\t/** Default camera */\n\tcamera: NgtCameraManual;\n\t/** Default scene */\n\tscene: THREE.Scene;\n\t/** Default raycaster */\n\traycaster: THREE.Raycaster;\n\t/** Default clock */\n\tclock: THREE.Clock;\n\t/** Event layer interface, contains the event handler and the node they're connected to */\n\tevents: NgtEventManager<any>;\n\t/** XR interface */\n\txr: { connect: () => void; disconnect: () => void };\n\t/** Currently used controls */\n\tcontrols: THREE.EventDispatcher | null;\n\t/** Normalized event coordinates */\n\tpointer: THREE.Vector2;\n\t/* Whether to enable r139's THREE.ColorManagement */\n\tlegacy: boolean;\n\t/** Shortcut to gl.outputColorSpace = THREE.LinearSRGBColorSpace */\n\tlinear: boolean;\n\t/** Shortcut to gl.toneMapping = NoTonemapping */\n\tflat: boolean;\n\t/** Render loop flags */\n\tframeloop: 'always' | 'demand' | 'never';\n\t/** Adaptive performance interface */\n\tperformance: NgtPerformance;\n\t/** Reactive pixel-size of the canvas */\n\tsize: NgtSize;\n\t/** Reactive size of the viewport in threejs units */\n\tviewport: NgtViewport & {\n\t\tgetCurrentViewport: (\n\t\t\tcamera?: NgtCamera,\n\t\t\ttarget?: THREE.Vector3 | Parameters<THREE.Vector3['set']>,\n\t\t\tsize?: NgtSize,\n\t\t) => Omit<NgtViewport, 'dpr' | 'initialDpr'>;\n\t};\n\t/** Flags the canvas for render, but doesn't render in itself */\n\tinvalidate: (frames?: number) => void;\n\t/** Advance (render) one step */\n\tadvance: (timestamp: number, runGlobalEffects?: boolean) => void;\n\t/** Shortcut to setting the event layer */\n\tsetEvents: (events: Partial<NgtEventManager<any>>) => void;\n\t/**\n\t * Shortcut to manual sizing\n\t */\n\tsetSize: (width: number, height: number, top?: number, left?: number) => void;\n\t/** Shortcut to manual setting the pixel ratio */\n\tsetDpr: (dpr: NgtDpr) => void;\n\t/** Shortcut to frameloop flags */\n\tsetFrameloop: (frameloop?: 'always' | 'demand' | 'never') => void;\n\t/** When the canvas was clicked but nothing was hit */\n\t/** PointerMissed Observable */\n\tpointerMissed$: Observable<MouseEvent>;\n\t/** If this state model is layerd (via createPortal) then this contains the previous layer */\n\tpreviousRoot: NgtSignalStore<NgtState> | null;\n\t/** Internals */\n\tinternal: NgtInternalState;\n};\n\nfunction storeFactory(loop: NgtLoop, document: Document, injector: Injector, parent: NgtSignalStore<NgtState> | null) {\n\treturn runInInjectionContext(injector, () => {\n\t\tconst window = document.defaultView;\n\n\t\tif (!window) {\n\t\t\t// TODO: revisit this when we need to support multiple platforms\n\t\t\tthrow new Error(`[NGT] Window is not available.`);\n\t\t}\n\n\t\tconst cdr = inject(ChangeDetectorRef);\n\n\t\t// NOTE: using Subject because we do not care about late-subscribers\n\t\tconst pointerMissed$ = new Subject<MouseEvent>();\n\n\t\tconst store: NgtSignalStore<NgtState> = signalStore<NgtState>(({ get, set }) => {\n\t\t\tconst { invalidate, advance } = loop;\n\n\t\t\tconst position = new THREE.Vector3();\n\t\t\tconst defaultTarget = new THREE.Vector3();\n\t\t\tconst tempTarget = new THREE.Vector3();\n\n\t\t\tfunction getCurrentViewport(\n\t\t\t\tcamera: NgtCamera = get('camera'),\n\t\t\t\ttarget: THREE.Vector3 | Parameters<THREE.Vector3['set']> = defaultTarget,\n\t\t\t\tsize: NgtSize = get('size'),\n\t\t\t): Omit<NgtViewport, 'dpr' | 'initialDpr'> {\n\t\t\t\tconst { width, height, top, left } = size;\n\t\t\t\tconst aspect = width / height;\n\t\t\t\tif (target instanceof THREE.Vector3) tempTarget.copy(target);\n\t\t\t\telse tempTarget.set(...target);\n\t\t\t\tconst distance = camera.getWorldPosition(position).distanceTo(tempTarget);\n\t\t\t\tif (is.orthographicCamera(camera)) {\n\t\t\t\t\treturn {\n\t\t\t\t\t\twidth: width / camera.zoom,\n\t\t\t\t\t\theight: height / camera.zoom,\n\t\t\t\t\t\ttop,\n\t\t\t\t\t\tleft,\n\t\t\t\t\t\tfactor: 1,\n\t\t\t\t\t\tdistance,\n\t\t\t\t\t\taspect,\n\t\t\t\t\t};\n\t\t\t\t} else {\n\t\t\t\t\tconst fov = (camera.fov * Math.PI) / 180; // convert vertical fov to radians\n\t\t\t\t\tconst h = 2 * Math.tan(fov / 2) * distance; // visible height\n\t\t\t\t\tconst w = h * (width / height);\n\t\t\t\t\treturn { width: w, height: h, top, left, factor: width / w, distance, aspect };\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet performanceTimeout: ReturnType<typeof setTimeout> | undefined = undefined;\n\t\t\tconst setPerformanceCurrent = (current: number) =>\n\t\t\t\tset((state) => ({ performance: { ...state.performance, current } }));\n\n\t\t\tconst pointer = new THREE.Vector2();\n\n\t\t\treturn {\n\t\t\t\tpointerMissed$: pointerMissed$.asObservable(),\n\t\t\t\tevents: { priority: 1, enabled: true, connected: false },\n\n\t\t\t\tinvalidate: (frames = 1) => invalidate(store, frames),\n\t\t\t\tadvance: (timestamp: number, runGlobalEffects?: boolean) => advance(timestamp, runGlobalEffects, store),\n\n\t\t\t\tlegacy: false,\n\t\t\t\tlinear: false,\n\t\t\t\tflat: false,\n\n\t\t\t\tcontrols: null,\n\t\t\t\tclock: new THREE.Clock(),\n\t\t\t\tpointer,\n\n\t\t\t\tframeloop: 'always',\n\n\t\t\t\tperformance: {\n\t\t\t\t\tcurrent: 1,\n\t\t\t\t\tmin: 0.5,\n\t\t\t\t\tmax: 1,\n\t\t\t\t\tdebounce: 200,\n\t\t\t\t\tregress: () => {\n\t\t\t\t\t\tconst state = get();\n\t\t\t\t\t\t// Clear timeout\n\t\t\t\t\t\tif (performanceTimeout) clearTimeout(performanceTimeout);\n\t\t\t\t\t\t// Set lower bound performance\n\t\t\t\t\t\tif (state.performance.current !== state.performance.min)\n\t\t\t\t\t\t\tsetPerformanceCurrent(state.performance.min);\n\n\t\t\t\t\t\t// Go back to upper bound performance after a while unless something regresses meanwhile\n\t\t\t\t\t\tperformanceTimeout = setTimeout(() => {\n\t\t\t\t\t\t\tsetPerformanceCurrent(get('performance', 'max'));\n\t\t\t\t\t\t\tsafeDetectChanges(cdr);\n\t\t\t\t\t\t}, state.performance.debounce);\n\t\t\t\t\t},\n\t\t\t\t},\n\n\t\t\t\tsize: { width: 0, height: 0, top: 0, left: 0, updateStyle: false },\n\t\t\t\tviewport: {\n\t\t\t\t\tinitialDpr: 0,\n\t\t\t\t\tdpr: 0,\n\t\t\t\t\twidth: 0,\n\t\t\t\t\theight: 0,\n\t\t\t\t\ttop: 0,\n\t\t\t\t\tleft: 0,\n\t\t\t\t\taspect: 0,\n\t\t\t\t\tdistance: 0,\n\t\t\t\t\tfactor: 0,\n\t\t\t\t\tgetCurrentViewport,\n\t\t\t\t},\n\n\t\t\t\tsetEvents: (events: Partial<NgtEventManager<any>>) =>\n\t\t\t\t\tset((state) => ({ ...state, events: { ...state.events, ...events } })),\n\t\t\t\tsetSize: (width: number, height: number, top?: number, left?: number) => {\n\t\t\t\t\tconst camera = get('camera');\n\t\t\t\t\tconst size = { width, height, top: top || 0, left: left || 0 };\n\t\t\t\t\tset((state) => ({\n\t\t\t\t\t\tsize,\n\t\t\t\t\t\tviewport: { ...state.viewport, ...getCurrentViewport(camera, defaultTarget, size) },\n\t\t\t\t\t}));\n\t\t\t\t},\n\t\t\t\tsetDpr: (dpr: NgtDpr) =>\n\t\t\t\t\tset((state) => {\n\t\t\t\t\t\tconst resolved = makeDpr(dpr, window);\n\t\t\t\t\t\treturn {\n\t\t\t\t\t\t\tviewport: {\n\t\t\t\t\t\t\t\t...state.viewport,\n\t\t\t\t\t\t\t\tdpr: resolved,\n\t\t\t\t\t\t\t\tinitialDpr: state.viewport.initialDpr || resolved,\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t};\n\t\t\t\t\t}),\n\t\t\t\tsetFrameloop: (frameloop: 'always' | 'demand' | 'never' = 'always') => {\n\t\t\t\t\tconst clock = get('clock');\n\n\t\t\t\t\t// if frameloop === \"never\" clock.elapsedTime is updated using advance(timestamp)\n\t\t\t\t\tclock.stop();\n\t\t\t\t\tclock.elapsedTime = 0;\n\n\t\t\t\t\tif (frameloop !== 'never') {\n\t\t\t\t\t\tclock.start();\n\t\t\t\t\t\tclock.elapsedTime = 0;\n\t\t\t\t\t}\n\t\t\t\t\tset(() => ({ frameloop }));\n\t\t\t\t},\n\n\t\t\t\tpreviousRoot: parent,\n\t\t\t\tinternal: {\n\t\t\t\t\tactive: false,\n\t\t\t\t\tpriority: 0,\n\t\t\t\t\tframes: 0,\n\t\t\t\t\tlastEvent: new ElementRef(null),\n\t\t\t\t\tinteraction: [],\n\t\t\t\t\thovered: new Map(),\n\t\t\t\t\tsubscribers: [],\n\t\t\t\t\tinitialClick: [0, 0],\n\t\t\t\t\tinitialHits: [],\n\t\t\t\t\tcapturedMap: new Map(),\n\n\t\t\t\t\tsubscribe: (\n\t\t\t\t\t\tcallback: NgtBeforeRenderRecord['callback'],\n\t\t\t\t\t\tpriority = 0,\n\t\t\t\t\t\t_store: NgtSignalStore<NgtState> = store,\n\t\t\t\t\t) => {\n\t\t\t\t\t\tconst internal = get('internal');\n\t\t\t\t\t\t// If this subscription was given a priority, it takes rendering into its own hands\n\t\t\t\t\t\t// For that reason we switch off automatic rendering and increase the manual flag\n\t\t\t\t\t\t// As long as this flag is positive there can be no internal rendering at all\n\t\t\t\t\t\t// because there could be multiple render subscriptions\n\t\t\t\t\t\tinternal.priority = internal.priority + (priority > 0 ? 1 : 0);\n\t\t\t\t\t\tinternal.subscribers.push({ callback, priority, store });\n\t\t\t\t\t\t// Register subscriber and sort layers from lowest to highest, meaning,\n\t\t\t\t\t\t// highest priority renders last (on top of the other frames)\n\t\t\t\t\t\tinternal.subscribers = internal.subscribers.sort(\n\t\t\t\t\t\t\t(a, b) => (a.priority || 0) - (b.priority || 0),\n\t\t\t\t\t\t);\n\t\t\t\t\t\treturn () => {\n\t\t\t\t\t\t\tconst internal = get('internal');\n\t\t\t\t\t\t\tif (internal?.subscribers) {\n\t\t\t\t\t\t\t\t// Decrease manual flag if this subscription had a priority\n\t\t\t\t\t\t\t\tinternal.priority = internal.priority - (priority > 0 ? 1 : 0);\n\t\t\t\t\t\t\t\t// Remove subscriber from list\n\t\t\t\t\t\t\t\tinternal.subscribers = internal.subscribers.filter((s) => s.callback !== callback);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t};\n\t\t\t\t\t},\n\t\t\t\t},\n\t\t\t};\n\t\t});\n\n\t\t// NOTE: assign pointerMissed$ so we can use it in events\n\t\tObject.defineProperty(store, 'pointerMissed$', { get: () => pointerMissed$ });\n\n\t\tconst state = store.get();\n\n\t\tlet oldSize = state.size;\n\t\tlet oldDpr = state.viewport.dpr;\n\t\tlet oldCamera = state.camera;\n\n\t\tconst _camera = store.select('camera');\n\t\tconst _size = store.select('size');\n\t\tconst _viewport = store.select('viewport');\n\n\t\teffect(() => {\n\t\t\tconst [camera, size, viewport, gl] = [_camera(), _size(), _viewport(), store.get('gl')];\n\n\t\t\t// Resize camera and renderer on changes to size and pixelratio\n\t\t\tif (size !== oldSize || viewport.dpr !== oldDpr) {\n\t\t\t\toldSize = size;\n\t\t\t\toldDpr = viewport.dpr;\n\t\t\t\t// Update camera & renderer\n\t\t\t\tupdateCamera(camera, size);\n\t\t\t\tgl.setPixelRatio(viewport.dpr);\n\n\t\t\t\tconst updateStyle =\n\t\t\t\t\ttypeof HTMLCanvasElement !== 'undefined' && gl.domElement instanceof HTMLCanvasElement;\n\t\t\t\tgl.setSize(size.width, size.height, updateStyle);\n\t\t\t}\n\n\t\t\t// Update viewport once the camera changes\n\t\t\tif (camera !== oldCamera) {\n\t\t\t\toldCamera = camera;\n\t\t\t\tupdateCamera(camera, size);\n\t\t\t\t// Update viewport\n\t\t\t\tstore.set((state) => ({\n\t\t\t\t\tviewport: { ...state.viewport, ...state.viewport.getCurrentViewport(camera) },\n\t\t\t\t}));\n\t\t\t}\n\t\t});\n\n\t\treturn store;\n\t});\n}\n\nexport type NgtStore = ReturnType<typeof storeFactory>;\n\nexport const NGT_STORE = new InjectionToken<NgtStore>('NgtStore token');\nexport const [injectNgtStore, provideNgtStore] = createInjectionToken(storeFactory, {\n\tisRoot: false,\n\tdeps: [NGT_LOOP, DOCUMENT, Injector, [new Optional(), new SkipSelf(), NGT_STORE]],\n\ttoken: NGT_STORE,\n});\n"]}
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../../../libs/core/src/lib/store.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef, InjectOptions, InjectionToken, effect, inject } from '@angular/core';\nimport { Subject } from 'rxjs';\nimport { Clock, Vector2, Vector3 } from 'three';\nimport { injectLoop } from './loop';\nimport { NgtBeforeRenderRecord, NgtCamera, NgtDpr, NgtEventManager, NgtSize, NgtState, NgtViewport } from './types';\nimport { is } from './utils/is';\nimport { makeDpr } from './utils/make';\nimport { NgtSignalStore, signalStore } from './utils/signal-store';\nimport { updateCamera } from './utils/update';\n\nfunction storeFactory(previousStore: NgtSignalStore<NgtState> | null) {\n\tconst loop = injectLoop();\n\n\t// NOTE: using Subject because we do not care about late-subscribers\n\tconst pointerMissed$ = new Subject<MouseEvent>();\n\n\tconst store: NgtSignalStore<NgtState> = signalStore<NgtState>(({ get, update }) => {\n\t\tconst { invalidate, advance } = loop;\n\n\t\tconst position = new Vector3();\n\t\tconst defaultTarget = new Vector3();\n\t\tconst tempTarget = new Vector3();\n\n\t\tfunction getCurrentViewport(\n\t\t\tcamera: NgtCamera = get('camera'),\n\t\t\ttarget: Vector3 | Parameters<Vector3['set']> = defaultTarget,\n\t\t\tsize: NgtSize = get('size'),\n\t\t): Omit<NgtViewport, 'dpr' | 'initialDpr'> {\n\t\t\tconst { width, height, top, left } = size;\n\t\t\tconst aspect = width / height;\n\n\t\t\tif (target instanceof Vector3) tempTarget.copy(target);\n\t\t\telse tempTarget.set(...target);\n\n\t\t\tconst distance = camera.getWorldPosition(position).distanceTo(tempTarget);\n\n\t\t\tif (is.orthographicCamera(camera)) {\n\t\t\t\treturn { width: width / camera.zoom, height: height / camera.zoom, top, left, factor: 1, distance, aspect };\n\t\t\t}\n\n\t\t\tconst fov = (camera.fov * Math.PI) / 180; // convert vertical fov to radians\n\t\t\tconst h = 2 * Math.tan(fov / 2) * distance; // visible height\n\t\t\tconst w = h * (width / height);\n\t\t\treturn { width: w, height: h, top, left, factor: width / w, distance, aspect };\n\t\t}\n\n\t\tlet performanceTimeout: ReturnType<typeof setTimeout> | undefined = undefined;\n\t\tconst setPerformanceCurrent = (current: number) =>\n\t\t\tupdate((state) => ({ performance: { ...state.performance, current } }));\n\n\t\tconst pointer = new Vector2();\n\n\t\treturn {\n\t\t\tpointerMissed$: pointerMissed$.asObservable(),\n\t\t\tevents: { priority: 1, enabled: true, connected: false },\n\n\t\t\tinvalidate: (frames = 1) => invalidate(store, frames),\n\t\t\tadvance: (timestamp: number, runGlobalEffects?: boolean) => advance(timestamp, runGlobalEffects, store),\n\n\t\t\tlegacy: false,\n\t\t\tlinear: false,\n\t\t\tflat: false,\n\n\t\t\tcontrols: null,\n\t\t\tclock: new Clock(),\n\t\t\tpointer,\n\n\t\t\tframeloop: 'always',\n\n\t\t\tperformance: {\n\t\t\t\tcurrent: 1,\n\t\t\t\tmin: 0.5,\n\t\t\t\tmax: 1,\n\t\t\t\tdebounce: 200,\n\t\t\t\tregress: () => {\n\t\t\t\t\tconst state = get();\n\t\t\t\t\t// Clear timeout\n\t\t\t\t\tif (performanceTimeout) clearTimeout(performanceTimeout);\n\t\t\t\t\t// Set lower bound performance\n\t\t\t\t\tif (state.performance.current !== state.performance.min) setPerformanceCurrent(state.performance.min);\n\t\t\t\t\t// Go back to upper bound performance after a while unless something regresses meanwhile\n\t\t\t\t\tperformanceTimeout = setTimeout(() => {\n\t\t\t\t\t\tsetPerformanceCurrent(get('performance', 'max'));\n\t\t\t\t\t\t// safeDetectChanges(cdr);\n\t\t\t\t\t}, state.performance.debounce);\n\t\t\t\t},\n\t\t\t},\n\n\t\t\tsize: { width: 0, height: 0, top: 0, left: 0, updateStyle: false },\n\t\t\tviewport: {\n\t\t\t\tinitialDpr: 0,\n\t\t\t\tdpr: 0,\n\t\t\t\twidth: 0,\n\t\t\t\theight: 0,\n\t\t\t\ttop: 0,\n\t\t\t\tleft: 0,\n\t\t\t\taspect: 0,\n\t\t\t\tdistance: 0,\n\t\t\t\tfactor: 0,\n\t\t\t\tgetCurrentViewport,\n\t\t\t},\n\n\t\t\tsetEvents: (events: Partial<NgtEventManager<any>>) =>\n\t\t\t\tupdate((state) => ({ ...state, events: { ...state.events, ...events } })),\n\t\t\tsetSize: (width: number, height: number, top?: number, left?: number) => {\n\t\t\t\tconst camera = get('camera');\n\t\t\t\tconst size = { width, height, top: top || 0, left: left || 0 };\n\t\t\t\tupdate((state) => ({\n\t\t\t\t\tsize,\n\t\t\t\t\tviewport: { ...state.viewport, ...getCurrentViewport(camera, defaultTarget, size) },\n\t\t\t\t}));\n\t\t\t},\n\t\t\tsetDpr: (dpr: NgtDpr) =>\n\t\t\t\tupdate((state) => {\n\t\t\t\t\tconst resolved = makeDpr(dpr, window);\n\t\t\t\t\treturn { viewport: { ...state.viewport, dpr: resolved, initialDpr: state.viewport.initialDpr || resolved } };\n\t\t\t\t}),\n\t\t\tsetFrameloop: (frameloop: 'always' | 'demand' | 'never' = 'always') => {\n\t\t\t\tconst clock = get('clock');\n\n\t\t\t\t// if frameloop === \"never\" clock.elapsedTime is updated using advance(timestamp)\n\t\t\t\tclock.stop();\n\t\t\t\tclock.elapsedTime = 0;\n\n\t\t\t\tif (frameloop !== 'never') {\n\t\t\t\t\tclock.start();\n\t\t\t\t\tclock.elapsedTime = 0;\n\t\t\t\t}\n\n\t\t\t\tupdate(() => ({ frameloop }));\n\t\t\t},\n\n\t\t\tpreviousRoot: previousStore,\n\t\t\tinternal: {\n\t\t\t\tactive: false,\n\t\t\t\tpriority: 0,\n\t\t\t\tframes: 0,\n\t\t\t\tlastEvent: new ElementRef(null),\n\t\t\t\tinteraction: [],\n\t\t\t\thovered: new Map(),\n\t\t\t\tsubscribers: [],\n\t\t\t\tinitialClick: [0, 0],\n\t\t\t\tinitialHits: [],\n\t\t\t\tcapturedMap: new Map(),\n\t\t\t\tsubscribe: (\n\t\t\t\t\tcallback: NgtBeforeRenderRecord['callback'],\n\t\t\t\t\tpriority = 0,\n\t\t\t\t\t_store: NgtSignalStore<NgtState> = store,\n\t\t\t\t) => {\n\t\t\t\t\tconst internal = get('internal');\n\t\t\t\t\t// If this subscription was given a priority, it takes rendering into its own hands\n\t\t\t\t\t// For that reason we switch off automatic rendering and increase the manual flag\n\t\t\t\t\t// As long as this flag is positive there can be no internal rendering at all\n\t\t\t\t\t// because there could be multiple render subscriptions\n\t\t\t\t\tinternal.priority = internal.priority + (priority > 0 ? 1 : 0);\n\t\t\t\t\tinternal.subscribers.push({ callback, priority, store: _store });\n\t\t\t\t\t// Register subscriber and sort layers from lowest to highest, meaning,\n\t\t\t\t\t// highest priority renders last (on top of the other frames)\n\t\t\t\t\tinternal.subscribers = internal.subscribers.sort((a, b) => (a.priority || 0) - (b.priority || 0));\n\n\t\t\t\t\treturn () => {\n\t\t\t\t\t\tconst internal = get('internal');\n\t\t\t\t\t\tif (internal?.subscribers) {\n\t\t\t\t\t\t\t// Decrease manual flag if this subscription had a priority\n\t\t\t\t\t\t\tinternal.priority = internal.priority - (priority > 0 ? 1 : 0);\n\t\t\t\t\t\t\t// Remove subscriber from list\n\t\t\t\t\t\t\tinternal.subscribers = internal.subscribers.filter((s) => s.callback !== callback);\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t},\n\t\t\t},\n\t\t};\n\t});\n\n\tObject.defineProperty(store, 'pointerMissed$', { get: () => pointerMissed$ });\n\n\tlet {\n\t\tsize: oldSize,\n\t\tviewport: { dpr: oldDpr },\n\t\tcamera: oldCamera,\n\t} = store.snapshot;\n\n\teffect(() => {\n\t\tconst {\n\t\t\tcamera: newCamera,\n\t\t\tsize: newSize,\n\t\t\tviewport: { dpr: newDpr },\n\t\t\tgl,\n\t\t} = store.state();\n\n\t\t// Resize camera and renderer on changes to size and pixel-ratio\n\t\tif (newSize !== oldSize || newDpr !== oldDpr) {\n\t\t\toldSize = newSize;\n\t\t\toldDpr = newDpr;\n\t\t\t// Update camera & renderer\n\t\t\tupdateCamera(newCamera, newSize);\n\t\t\tgl.setPixelRatio(newDpr);\n\n\t\t\tconst updateStyle = typeof HTMLCanvasElement !== 'undefined' && gl.domElement instanceof HTMLCanvasElement;\n\t\t\tgl.setSize(newSize.width, newSize.height, updateStyle);\n\t\t}\n\n\t\t// Update viewport once the camera changes\n\t\tif (newCamera !== oldCamera) {\n\t\t\toldCamera = newCamera;\n\t\t\tupdateCamera(newCamera, newSize);\n\t\t\t// Update viewport\n\t\t\tstore.update((state) => ({ viewport: { ...state.viewport, ...state.viewport.getCurrentViewport(newCamera) } }));\n\t\t}\n\t});\n\n\treturn store;\n}\n\nexport const NGT_STORE = new InjectionToken<NgtSignalStore<NgtState>>('NgtStore Token');\n\nexport function provideStore(store?: () => NgtSignalStore<NgtState>) {\n\tif (store) {\n\t\treturn { provide: NGT_STORE, useFactory: store };\n\t}\n\treturn { provide: NGT_STORE, useFactory: storeFactory };\n}\n\nexport function injectStore(options: InjectOptions & { optional?: false }): NgtSignalStore<NgtState>;\nexport function injectStore(options: InjectOptions): NgtSignalStore<NgtState> | null;\nexport function injectStore(): NgtSignalStore<NgtState>;\nexport function injectStore(options?: InjectOptions) {\n\treturn inject(NGT_STORE, options as InjectOptions);\n}\n"]}