@hkdigital/lib-sveltekit 0.1.62 → 0.1.65

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 (209) hide show
  1. package/README.md +135 -135
  2. package/dist/assets/autospuiten/car-paint-picker.js +41 -41
  3. package/dist/assets/autospuiten/labels.js +7 -7
  4. package/dist/classes/data/IterableTree.js +243 -243
  5. package/dist/classes/data/Selector.js +190 -190
  6. package/dist/classes/data/index.js +2 -2
  7. package/dist/classes/index.js +4 -4
  8. package/dist/classes/promise/HkPromise.js +377 -377
  9. package/dist/classes/promise/index.js +1 -1
  10. package/dist/classes/stores/SubscribersCount.js +107 -107
  11. package/dist/classes/stores/index.js +1 -1
  12. package/dist/classes/streams/LogTransformStream.js +19 -19
  13. package/dist/classes/streams/ServerEventsStore.js +110 -110
  14. package/dist/classes/streams/TimeStampSource.js +26 -26
  15. package/dist/classes/streams/index.js +3 -3
  16. package/dist/classes/svelte/audio/AudioLoader.svelte.js +58 -58
  17. package/dist/classes/svelte/audio/AudioScene.svelte.js +295 -295
  18. package/dist/classes/svelte/audio/mocks.js +35 -35
  19. package/dist/classes/svelte/finite-state-machine/FiniteStateMachine.svelte.js +133 -133
  20. package/dist/classes/svelte/finite-state-machine/index.js +1 -1
  21. package/dist/classes/svelte/image/ImageLoader.svelte.js +47 -47
  22. package/dist/classes/svelte/image/ImageScene.svelte.js +253 -253
  23. package/dist/classes/svelte/image/ImageVariantsLoader.svelte.js +152 -152
  24. package/dist/classes/svelte/image/index.js +4 -4
  25. package/dist/classes/svelte/image/mocks.js +35 -35
  26. package/dist/classes/svelte/image/typedef.js +8 -8
  27. package/dist/classes/svelte/index.js +14 -14
  28. package/dist/classes/svelte/loading-state-machine/LoadingStateMachine.svelte.js +109 -109
  29. package/dist/classes/svelte/loading-state-machine/constants.js +16 -16
  30. package/dist/classes/svelte/loading-state-machine/index.js +3 -3
  31. package/dist/classes/svelte/network-loader/NetworkLoader.svelte.js +331 -331
  32. package/dist/classes/svelte/network-loader/constants.js +3 -3
  33. package/dist/classes/svelte/network-loader/index.js +3 -3
  34. package/dist/classes/svelte/network-loader/mocks.js +30 -30
  35. package/dist/classes/svelte/network-loader/typedef.js +8 -8
  36. package/dist/components/area/HkArea.svelte +49 -49
  37. package/dist/components/area/HkGridArea.svelte +77 -77
  38. package/dist/components/area/index.js +2 -2
  39. package/dist/components/buttons/button/Button.svelte +82 -82
  40. package/dist/components/buttons/button-icon-steeze/SteezeIconButton.svelte +30 -30
  41. package/dist/components/buttons/button-text/TextButton.svelte +21 -21
  42. package/dist/components/buttons/index.js +3 -3
  43. package/dist/components/debug/debug-panel-design-scaling/DebugPanelDesignScaling.svelte +146 -146
  44. package/dist/components/debug/index.js +1 -1
  45. package/dist/components/hkdev/blocks/TextBlock.svelte +46 -46
  46. package/dist/components/hkdev/buttons/CheckButton.svelte +62 -62
  47. package/dist/components/icons/HkIcon.svelte +86 -86
  48. package/dist/components/icons/HkTabIcon.svelte +116 -116
  49. package/dist/components/icons/SteezeIcon.svelte +97 -97
  50. package/dist/components/icons/index.js +6 -6
  51. package/dist/components/icons/typedef.js +16 -16
  52. package/dist/components/index.js +2 -2
  53. package/dist/components/inputs/index.js +1 -1
  54. package/dist/components/inputs/text-input/TestTextInput.svelte__ +102 -102
  55. package/dist/components/inputs/text-input/TextInput.svelte +223 -223
  56. package/dist/components/inputs/text-input/TextInput.svelte___ +83 -83
  57. package/dist/components/inputs/text-input/assets/IconInvalid.svelte +14 -14
  58. package/dist/components/inputs/text-input/assets/IconValid.svelte +12 -12
  59. package/dist/components/layout/grid-layers/GridLayers.svelte +167 -167
  60. package/dist/components/layout/index.js +1 -1
  61. package/dist/components/panels/index.js +1 -1
  62. package/dist/components/panels/panel/Panel.svelte +43 -43
  63. package/dist/components/rows/index.js +3 -3
  64. package/dist/components/rows/panel-grid-row/PanelGridRow.svelte +104 -104
  65. package/dist/components/rows/panel-row-2/PanelRow2.svelte +40 -40
  66. package/dist/components/tab-bar/HkTabBar.state.svelte.js +149 -149
  67. package/dist/components/tab-bar/HkTabBar.svelte +74 -74
  68. package/dist/components/tab-bar/HkTabBarSelector.state.svelte.js +93 -93
  69. package/dist/components/tab-bar/HkTabBarSelector.svelte +49 -49
  70. package/dist/components/tab-bar/index.js +17 -17
  71. package/dist/components/tab-bar/typedef.js +8 -8
  72. package/dist/config/imagetools-config.js +189 -189
  73. package/dist/config/imagetools.d.ts +71 -71
  74. package/dist/config/typedef.js +8 -8
  75. package/dist/constants/bases.js +13 -13
  76. package/dist/constants/errors/api.js +9 -9
  77. package/dist/constants/errors/generic.js +5 -5
  78. package/dist/constants/errors/index.js +3 -3
  79. package/dist/constants/errors/jwt.js +5 -5
  80. package/dist/constants/http/headers.js +6 -6
  81. package/dist/constants/http/index.js +2 -2
  82. package/dist/constants/http/methods.js +2 -2
  83. package/dist/constants/index.js +3 -3
  84. package/dist/constants/mime/application.js +5 -5
  85. package/dist/constants/mime/audio.js +13 -13
  86. package/dist/constants/mime/image.js +3 -3
  87. package/dist/constants/mime/index.js +4 -4
  88. package/dist/constants/mime/text.js +2 -2
  89. package/dist/constants/regexp/index.js +31 -31
  90. package/dist/constants/regexp/inspiratie.js__ +95 -95
  91. package/dist/constants/regexp/text.js +49 -49
  92. package/dist/constants/regexp/user.js +32 -32
  93. package/dist/constants/regexp/web.js +3 -3
  94. package/dist/constants/state-labels/input-states.js +11 -11
  95. package/dist/constants/state-labels/submit-states.js +4 -4
  96. package/dist/constants/time.js +28 -28
  97. package/dist/css/utilities.css +43 -43
  98. package/dist/design/design-config.js +73 -73
  99. package/dist/design/tailwind-theme-extend.js +158 -158
  100. package/dist/schemas/index.js +1 -1
  101. package/dist/schemas/validate-url.js +180 -180
  102. package/dist/server/index.js +1 -1
  103. package/dist/server/logger.js +94 -94
  104. package/dist/states/index.js +1 -1
  105. package/dist/states/navigation.svelte.js +55 -55
  106. package/dist/stores/index.js +1 -1
  107. package/dist/stores/theme.js +80 -80
  108. package/dist/themes/hkdev/components/blocks/text-block.css +41 -41
  109. package/dist/themes/hkdev/components/boxes/game-box.css +12 -12
  110. package/dist/themes/hkdev/components/buttons/button-icon-steeze.css +22 -22
  111. package/dist/themes/hkdev/components/buttons/button-text.css +32 -32
  112. package/dist/themes/hkdev/components/buttons/button.css +142 -142
  113. package/dist/themes/hkdev/components/buttons/skip-button.css +6 -6
  114. package/dist/themes/hkdev/components/icons/icon-steeze.css +22 -22
  115. package/dist/themes/hkdev/components/inputs/text-input.css +104 -104
  116. package/dist/themes/hkdev/components/panels/panel.css +27 -27
  117. package/dist/themes/hkdev/components/rows/panel-grid-row.css +6 -6
  118. package/dist/themes/hkdev/components/rows/panel-row-2.css +7 -7
  119. package/dist/themes/hkdev/components.css +47 -47
  120. package/dist/themes/hkdev/debug.css +1 -1
  121. package/dist/themes/hkdev/global/layout.css +39 -39
  122. package/dist/themes/hkdev/global/on-colors.css +53 -53
  123. package/dist/themes/hkdev/globals.css +11 -11
  124. package/dist/themes/hkdev/responsive.css +12 -12
  125. package/dist/themes/hkdev/theme-ext.js +15 -15
  126. package/dist/themes/hkdev/theme.js +235 -235
  127. package/dist/themes/index.js +1 -1
  128. package/dist/util/array/index.js +455 -455
  129. package/dist/util/bases/base58.js +262 -262
  130. package/dist/util/bases/index.js +1 -1
  131. package/dist/util/compare/index.js +247 -247
  132. package/dist/util/css/css-vars.js +83 -83
  133. package/dist/util/css/index.js +1 -1
  134. package/dist/util/design-system/components/states.js +22 -22
  135. package/dist/util/design-system/css/clamp.js +66 -66
  136. package/dist/util/design-system/css/root-design-vars.js +100 -100
  137. package/dist/util/design-system/index.js +5 -5
  138. package/dist/util/design-system/layout/scaling.js +228 -228
  139. package/dist/util/design-system/skeleton.js +208 -208
  140. package/dist/util/design-system/tailwind.js +288 -288
  141. package/dist/util/expect/arrays.js +47 -47
  142. package/dist/util/expect/index.js +259 -259
  143. package/dist/util/expect/primitives.js +55 -55
  144. package/dist/util/expect/url.js +60 -60
  145. package/dist/util/function/index.js +218 -218
  146. package/dist/util/http/errors.js +97 -97
  147. package/dist/util/http/headers.js +45 -45
  148. package/dist/util/http/http-request.js +294 -294
  149. package/dist/util/http/index.js +22 -22
  150. package/dist/util/http/json-request.js +143 -143
  151. package/dist/util/http/mocks.js +65 -65
  152. package/dist/util/http/response.js +241 -241
  153. package/dist/util/http/test-data__/content-length-test-hkdigital-small.V4HfZyBQ.avif +0 -0
  154. package/dist/util/http/url.js +52 -52
  155. package/dist/util/image/index.js +86 -86
  156. package/dist/util/index.js +2 -2
  157. package/dist/util/is/index.js +140 -140
  158. package/dist/util/iterate/index.js +234 -234
  159. package/dist/util/object/index.js +1361 -1361
  160. package/dist/util/singleton/index.js +97 -97
  161. package/dist/util/string/array-path.js +75 -75
  162. package/dist/util/string/convert.js +54 -54
  163. package/dist/util/string/fs.js +226 -226
  164. package/dist/util/string/index.js +5 -5
  165. package/dist/util/string/interpolate.js +61 -61
  166. package/dist/util/string/pad.js +10 -10
  167. package/dist/util/svelte/index.js +4 -4
  168. package/dist/util/svelte/loading/loading-tracker.svelte.js +108 -108
  169. package/dist/util/svelte/observe/index.js +49 -49
  170. package/dist/util/svelte/state-context/index.js +83 -83
  171. package/dist/util/svelte/wait/index.js +38 -38
  172. package/dist/util/sveltekit/index.js +1 -1
  173. package/dist/util/sveltekit/route-folders/index.js +101 -101
  174. package/dist/util/time/index.js +323 -323
  175. package/dist/util/unique/index.js +249 -249
  176. package/dist/valibot/date.js__ +10 -10
  177. package/dist/valibot/index.js +9 -9
  178. package/dist/valibot/url.js +95 -95
  179. package/dist/valibot/user.js +23 -23
  180. package/dist/widgets/button-group/ButtonGroup.svelte +82 -94
  181. package/dist/widgets/button-group/ButtonGroup.svelte.d.ts +0 -2
  182. package/dist/widgets/button-group/typedef.js +10 -10
  183. package/dist/widgets/compare-left-right/CompareLeftRight.svelte +179 -179
  184. package/dist/widgets/compare-left-right/index.js +1 -1
  185. package/dist/widgets/game-box/GameBox.svelte +579 -186
  186. package/dist/widgets/game-box/GameBox.svelte.d.ts +64 -4
  187. package/dist/widgets/game-box/gamebox.util.js +83 -83
  188. package/dist/widgets/hk-app-layout/HkAppLayout.state.svelte.js +25 -25
  189. package/dist/widgets/hk-app-layout/HkAppLayout.svelte +251 -251
  190. package/dist/widgets/image-box/ImageBox.svelte +212 -212
  191. package/dist/widgets/image-box/index.js +5 -5
  192. package/dist/widgets/image-box/typedef.js +32 -32
  193. package/dist/widgets/index.js +23 -23
  194. package/dist/widgets/presenter/(broken) Presenter.state.svelte.js__ +613 -0
  195. package/dist/widgets/presenter/ImageSlide.svelte +64 -64
  196. package/dist/widgets/presenter/Presenter.state.svelte.js +636 -636
  197. package/dist/widgets/presenter/Presenter.svelte +140 -140
  198. package/dist/widgets/presenter/Presenter.svelte__ +125 -0
  199. package/dist/widgets/presenter/constants.js +7 -7
  200. package/dist/widgets/presenter/index.js +10 -10
  201. package/dist/widgets/presenter/typedef.js +106 -106
  202. package/dist/widgets/presenter/util.js +210 -210
  203. package/dist/widgets/virtual-viewport/VirtualViewport.svelte +196 -196
  204. package/dist/zod/all.js +33 -33
  205. package/dist/zod/generic.js +11 -11
  206. package/dist/zod/javascript.js +32 -32
  207. package/dist/zod/user.js +16 -16
  208. package/dist/zod/web.js +52 -52
  209. package/package.json +102 -102
@@ -1,295 +1,295 @@
1
- import * as expect from '../../../util/expect/index.js';
2
-
3
- import {
4
- LoadingStateMachine,
5
- STATE_INITIAL,
6
- STATE_LOADING,
7
- STATE_UNLOADING,
8
- STATE_LOADED,
9
- STATE_CANCELLED,
10
- STATE_ERROR,
11
- LOAD,
12
- // CANCEL,
13
- ERROR,
14
- LOADED,
15
- UNLOAD,
16
- INITIAL
17
- } from '../loading-state-machine/index.js';
18
-
19
- import AudioLoader from './AudioLoader.svelte.js';
20
-
21
- /**
22
- * @typedef {object} SourceConfig
23
- * // property ...
24
- */
25
-
26
- /**
27
- * @typedef {object} MemorySource
28
- * @property {string} label
29
- * @property {AudioLoader} audioLoader
30
- * @property {SourceConfig} [config]
31
- */
32
-
33
- export default class AudioScene {
34
- #state = new LoadingStateMachine();
35
-
36
- // @note this exported state is set by $effect's
37
- state = $state(STATE_INITIAL);
38
-
39
- // @note this exported state is set by $effect's
40
- loaded = $derived.by(() => {
41
- return this.state === STATE_LOADED;
42
- });
43
-
44
- /** @type {AudioContext|null} */
45
- #audioContext = null;
46
-
47
- /** @type {MemorySource[]} */
48
- #memorySources = $state([]);
49
-
50
- #progress = $derived.by(() => {
51
- // console.log('update progress');
52
-
53
- let totalSize = 0;
54
- let totalBytesLoaded = 0;
55
- let sourcesLoaded = 0;
56
-
57
- const sources = this.#memorySources;
58
- const numberOfSources = sources.length;
59
-
60
- for (let j = 0; j < numberOfSources; j++) {
61
- const source = sources[j];
62
- const { audioLoader } = source;
63
-
64
- const { bytesLoaded, size, loaded } = audioLoader.progress;
65
-
66
- totalSize += size;
67
- totalBytesLoaded += bytesLoaded;
68
-
69
- if (loaded) {
70
- sourcesLoaded++;
71
- }
72
- } // end for
73
-
74
- return {
75
- totalBytesLoaded,
76
- totalSize,
77
- sourcesLoaded,
78
- numberOfSources
79
- };
80
- });
81
-
82
- /**
83
- * Construct AudioScene
84
- */
85
- constructor() {
86
- const state = this.#state;
87
-
88
- $effect(() => {
89
- if (state.current === STATE_LOADING) {
90
- // console.log(
91
- // 'progress',
92
- // JSON.stringify($state.snapshot(this.#progress))
93
- // );
94
-
95
- const { sourcesLoaded, numberOfSources } = this.#progress;
96
-
97
- if (sourcesLoaded === numberOfSources) {
98
- // console.debug(`AudioScene: ${numberOfSources} sources loaded`);
99
- this.#state.send(LOADED);
100
- }
101
- }
102
- });
103
-
104
- $effect(() => {
105
- switch (state.current) {
106
- case STATE_LOADING:
107
- {
108
- // console.log('AudioScene:loading');
109
- this.#startLoading();
110
- }
111
- break;
112
-
113
- case STATE_UNLOADING:
114
- {
115
- // console.log('AudioScene:unloading');
116
- // this.#startUnLoading();
117
- }
118
- break;
119
-
120
- case STATE_LOADED:
121
- {
122
- // console.log('AudioScene:loaded');
123
-
124
- // tODO
125
- // this.#abortLoading = null;
126
- }
127
- break;
128
-
129
- case STATE_CANCELLED:
130
- {
131
- // console.log('AudioScene:cancelled');
132
- // TODO
133
- }
134
- break;
135
-
136
- case STATE_ERROR:
137
- {
138
- console.error('AudioScene:error', state.error);
139
- }
140
- break;
141
- } // end switch
142
-
143
- this.state = state.current;
144
- });
145
- }
146
-
147
- destroy() {
148
- // TODO: disconnect all audio sources?
149
- // TODO: Unload AUdioLoaders?
150
- }
151
-
152
- /**
153
- * Add in-memory audio source
154
- * - Uses an AudioLoader instance to load audio data from network
155
- *
156
- * @param {object} _
157
- * @param {string} _.label
158
- * @param {string} _.url
159
- * @param {SourceConfig} [_.config]
160
- */
161
- defineMemorySource({ label, url, config }) {
162
- expect.notEmptyString(label);
163
- expect.notEmptyString(url);
164
-
165
- const audioLoader = new AudioLoader({ url });
166
-
167
- this.#memorySources.push({ label, audioLoader, config });
168
- }
169
-
170
- /**
171
- * Start loading all audio sources
172
- */
173
- load() {
174
- this.#state.send(LOAD);
175
-
176
- // FIXME: in unit test when moved to startloading it hangs!
177
-
178
- for (const { audioLoader } of this.#memorySources) {
179
- audioLoader.load();
180
- }
181
- }
182
-
183
- /**
184
- * Set an audio context to use
185
- *
186
- * @param {AudioContext} [audioContext]
187
- */
188
- setAudioContext( audioContext ) {
189
- this.#audioContext = audioContext;
190
- }
191
-
192
- async #startLoading() {
193
- // console.log('#startLoading');
194
-
195
- // FIXME: in unit test when moved to startloading it hangs!
196
- // for (const { audioLoader } of this.#memorySources) {
197
- // audioLoader.load();
198
- // }
199
- }
200
-
201
- /**
202
- * Get a source that can be used to play the audio once
203
- *
204
- * @param {string} label
205
- *
206
- * @returns {Promise<AudioBufferSourceNode>}
207
- */
208
- async getSourceNode(label) {
209
- // @note Gain setup
210
- // https://stackoverflow.com/questions/46203191/should-i-disconnect-nodes-that-cant-be-used-anymore
211
-
212
- const { audioLoader /*, config */ } = this.#getMemorySource(label);
213
-
214
- if (!audioLoader.loaded) {
215
- throw new Error(`Source [${label}] has not been loaded yet`);
216
- }
217
-
218
- const sourceNode = await audioLoader.getAudioBufferSourceNode(
219
- this.#getAudioContext()
220
- );
221
-
222
- // @ts-ignore
223
- sourceNode.connect(this.#getAudioContext().destination);
224
-
225
- // Clean up
226
- sourceNode.onended = () => {
227
- // console.log(`Source [${label}] ended `);
228
- sourceNode.disconnect();
229
- };
230
-
231
- return sourceNode;
232
- }
233
-
234
- #getAudioContext()
235
- {
236
- if( !this.#audioContext )
237
- {
238
- this.#audioContext = new AudioContext();
239
- }
240
-
241
- return this.#audioContext;
242
- }
243
-
244
- /**
245
- * Get memory source
246
- *
247
- * @param {string} label
248
- *
249
- * @returns {MemorySource}
250
- */
251
- #getMemorySource(label) {
252
- for (const source of this.#memorySources) {
253
- if (label === source.label) {
254
- return source;
255
- }
256
- }
257
-
258
- throw new Error(`Source [${label}] has not been defined`);
259
- }
260
-
261
- // connect
262
- // play
263
-
264
- // source.connect(audioContext.destination);
265
- // source.loop = true;
266
- // source.start();
267
-
268
- // /**
269
- // * Get the source identified by the specified label
270
- // *
271
- // * @param {string} label
272
- // */
273
- // async getBufferSourceNode(label) {
274
- // // expect.notEmptyString( label );
275
-
276
- // for (const source of this.#memorySources) {
277
- // if (label === source.label) {
278
- // if (!source.bufferSourceNode) {
279
- // source.bufferSourceNode =
280
- // await source.AudioLoader.transferToBufferSource(this.#audioContext);
281
- // }
282
-
283
- // return source.bufferSourceNode;
284
- // }
285
- // }
286
- // }
287
-
288
- // async connectSourceToDestination(label) {
289
- // const source = await this.getBufferSourceNode(label);
290
-
291
- // if (source) {
292
- // source.connect(this.#audioContext.destination);
293
- // }
294
- // }
295
- }
1
+ import * as expect from '../../../util/expect/index.js';
2
+
3
+ import {
4
+ LoadingStateMachine,
5
+ STATE_INITIAL,
6
+ STATE_LOADING,
7
+ STATE_UNLOADING,
8
+ STATE_LOADED,
9
+ STATE_CANCELLED,
10
+ STATE_ERROR,
11
+ LOAD,
12
+ // CANCEL,
13
+ ERROR,
14
+ LOADED,
15
+ UNLOAD,
16
+ INITIAL
17
+ } from '../loading-state-machine/index.js';
18
+
19
+ import AudioLoader from './AudioLoader.svelte.js';
20
+
21
+ /**
22
+ * @typedef {object} SourceConfig
23
+ * // property ...
24
+ */
25
+
26
+ /**
27
+ * @typedef {object} MemorySource
28
+ * @property {string} label
29
+ * @property {AudioLoader} audioLoader
30
+ * @property {SourceConfig} [config]
31
+ */
32
+
33
+ export default class AudioScene {
34
+ #state = new LoadingStateMachine();
35
+
36
+ // @note this exported state is set by $effect's
37
+ state = $state(STATE_INITIAL);
38
+
39
+ // @note this exported state is set by $effect's
40
+ loaded = $derived.by(() => {
41
+ return this.state === STATE_LOADED;
42
+ });
43
+
44
+ /** @type {AudioContext|null} */
45
+ #audioContext = null;
46
+
47
+ /** @type {MemorySource[]} */
48
+ #memorySources = $state([]);
49
+
50
+ #progress = $derived.by(() => {
51
+ // console.log('update progress');
52
+
53
+ let totalSize = 0;
54
+ let totalBytesLoaded = 0;
55
+ let sourcesLoaded = 0;
56
+
57
+ const sources = this.#memorySources;
58
+ const numberOfSources = sources.length;
59
+
60
+ for (let j = 0; j < numberOfSources; j++) {
61
+ const source = sources[j];
62
+ const { audioLoader } = source;
63
+
64
+ const { bytesLoaded, size, loaded } = audioLoader.progress;
65
+
66
+ totalSize += size;
67
+ totalBytesLoaded += bytesLoaded;
68
+
69
+ if (loaded) {
70
+ sourcesLoaded++;
71
+ }
72
+ } // end for
73
+
74
+ return {
75
+ totalBytesLoaded,
76
+ totalSize,
77
+ sourcesLoaded,
78
+ numberOfSources
79
+ };
80
+ });
81
+
82
+ /**
83
+ * Construct AudioScene
84
+ */
85
+ constructor() {
86
+ const state = this.#state;
87
+
88
+ $effect(() => {
89
+ if (state.current === STATE_LOADING) {
90
+ // console.log(
91
+ // 'progress',
92
+ // JSON.stringify($state.snapshot(this.#progress))
93
+ // );
94
+
95
+ const { sourcesLoaded, numberOfSources } = this.#progress;
96
+
97
+ if (sourcesLoaded === numberOfSources) {
98
+ // console.debug(`AudioScene: ${numberOfSources} sources loaded`);
99
+ this.#state.send(LOADED);
100
+ }
101
+ }
102
+ });
103
+
104
+ $effect(() => {
105
+ switch (state.current) {
106
+ case STATE_LOADING:
107
+ {
108
+ // console.log('AudioScene:loading');
109
+ this.#startLoading();
110
+ }
111
+ break;
112
+
113
+ case STATE_UNLOADING:
114
+ {
115
+ // console.log('AudioScene:unloading');
116
+ // this.#startUnLoading();
117
+ }
118
+ break;
119
+
120
+ case STATE_LOADED:
121
+ {
122
+ // console.log('AudioScene:loaded');
123
+
124
+ // tODO
125
+ // this.#abortLoading = null;
126
+ }
127
+ break;
128
+
129
+ case STATE_CANCELLED:
130
+ {
131
+ // console.log('AudioScene:cancelled');
132
+ // TODO
133
+ }
134
+ break;
135
+
136
+ case STATE_ERROR:
137
+ {
138
+ console.error('AudioScene:error', state.error);
139
+ }
140
+ break;
141
+ } // end switch
142
+
143
+ this.state = state.current;
144
+ });
145
+ }
146
+
147
+ destroy() {
148
+ // TODO: disconnect all audio sources?
149
+ // TODO: Unload AUdioLoaders?
150
+ }
151
+
152
+ /**
153
+ * Add in-memory audio source
154
+ * - Uses an AudioLoader instance to load audio data from network
155
+ *
156
+ * @param {object} _
157
+ * @param {string} _.label
158
+ * @param {string} _.url
159
+ * @param {SourceConfig} [_.config]
160
+ */
161
+ defineMemorySource({ label, url, config }) {
162
+ expect.notEmptyString(label);
163
+ expect.notEmptyString(url);
164
+
165
+ const audioLoader = new AudioLoader({ url });
166
+
167
+ this.#memorySources.push({ label, audioLoader, config });
168
+ }
169
+
170
+ /**
171
+ * Start loading all audio sources
172
+ */
173
+ load() {
174
+ this.#state.send(LOAD);
175
+
176
+ // FIXME: in unit test when moved to startloading it hangs!
177
+
178
+ for (const { audioLoader } of this.#memorySources) {
179
+ audioLoader.load();
180
+ }
181
+ }
182
+
183
+ /**
184
+ * Set an audio context to use
185
+ *
186
+ * @param {AudioContext} [audioContext]
187
+ */
188
+ setAudioContext( audioContext ) {
189
+ this.#audioContext = audioContext;
190
+ }
191
+
192
+ async #startLoading() {
193
+ // console.log('#startLoading');
194
+
195
+ // FIXME: in unit test when moved to startloading it hangs!
196
+ // for (const { audioLoader } of this.#memorySources) {
197
+ // audioLoader.load();
198
+ // }
199
+ }
200
+
201
+ /**
202
+ * Get a source that can be used to play the audio once
203
+ *
204
+ * @param {string} label
205
+ *
206
+ * @returns {Promise<AudioBufferSourceNode>}
207
+ */
208
+ async getSourceNode(label) {
209
+ // @note Gain setup
210
+ // https://stackoverflow.com/questions/46203191/should-i-disconnect-nodes-that-cant-be-used-anymore
211
+
212
+ const { audioLoader /*, config */ } = this.#getMemorySource(label);
213
+
214
+ if (!audioLoader.loaded) {
215
+ throw new Error(`Source [${label}] has not been loaded yet`);
216
+ }
217
+
218
+ const sourceNode = await audioLoader.getAudioBufferSourceNode(
219
+ this.#getAudioContext()
220
+ );
221
+
222
+ // @ts-ignore
223
+ sourceNode.connect(this.#getAudioContext().destination);
224
+
225
+ // Clean up
226
+ sourceNode.onended = () => {
227
+ // console.log(`Source [${label}] ended `);
228
+ sourceNode.disconnect();
229
+ };
230
+
231
+ return sourceNode;
232
+ }
233
+
234
+ #getAudioContext()
235
+ {
236
+ if( !this.#audioContext )
237
+ {
238
+ this.#audioContext = new AudioContext();
239
+ }
240
+
241
+ return this.#audioContext;
242
+ }
243
+
244
+ /**
245
+ * Get memory source
246
+ *
247
+ * @param {string} label
248
+ *
249
+ * @returns {MemorySource}
250
+ */
251
+ #getMemorySource(label) {
252
+ for (const source of this.#memorySources) {
253
+ if (label === source.label) {
254
+ return source;
255
+ }
256
+ }
257
+
258
+ throw new Error(`Source [${label}] has not been defined`);
259
+ }
260
+
261
+ // connect
262
+ // play
263
+
264
+ // source.connect(audioContext.destination);
265
+ // source.loop = true;
266
+ // source.start();
267
+
268
+ // /**
269
+ // * Get the source identified by the specified label
270
+ // *
271
+ // * @param {string} label
272
+ // */
273
+ // async getBufferSourceNode(label) {
274
+ // // expect.notEmptyString( label );
275
+
276
+ // for (const source of this.#memorySources) {
277
+ // if (label === source.label) {
278
+ // if (!source.bufferSourceNode) {
279
+ // source.bufferSourceNode =
280
+ // await source.AudioLoader.transferToBufferSource(this.#audioContext);
281
+ // }
282
+
283
+ // return source.bufferSourceNode;
284
+ // }
285
+ // }
286
+ // }
287
+
288
+ // async connectSourceToDestination(label) {
289
+ // const source = await this.getBufferSourceNode(label);
290
+
291
+ // if (source) {
292
+ // source.connect(this.#audioContext.destination);
293
+ // }
294
+ // }
295
+ }
@@ -1,35 +1,35 @@
1
- import { CONTENT_TYPE, CONTENT_LENGTH } from '../../../constants/http/index.js';
2
-
3
- import { AUDIO_WAV } from '../../../constants/mime/audio.js';
4
-
5
- // import MockWav from './tiny-silence.wav?raw';
6
-
7
- const BASE64_WAV =
8
- 'UklGRnwAAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YVgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA';
9
-
10
- /**
11
- * Create a response value that can be used by a mocked
12
- * fetch function
13
- *
14
- * @returns {Response}
15
- */
16
- export function createWavResponse(/* data , options */) {
17
- // @note encode as Uint8Array to get the proper byte size of data
18
- // const bytes = new TextEncoder().encode(MockWav);
19
-
20
- const binaryString = atob(BASE64_WAV);
21
- const bytes = new Uint8Array(binaryString.length);
22
-
23
- for (let i = 0; i < binaryString.length; i++) {
24
- bytes[i] = binaryString.charCodeAt(i);
25
- }
26
-
27
- const response = new Response(bytes, {
28
- headers: new Headers({
29
- [CONTENT_TYPE]: AUDIO_WAV,
30
- [CONTENT_LENGTH]: String(bytes.length)
31
- })
32
- });
33
-
34
- return response;
35
- }
1
+ import { CONTENT_TYPE, CONTENT_LENGTH } from '../../../constants/http/index.js';
2
+
3
+ import { AUDIO_WAV } from '../../../constants/mime/audio.js';
4
+
5
+ // import MockWav from './tiny-silence.wav?raw';
6
+
7
+ const BASE64_WAV =
8
+ 'UklGRnwAAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YVgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA';
9
+
10
+ /**
11
+ * Create a response value that can be used by a mocked
12
+ * fetch function
13
+ *
14
+ * @returns {Response}
15
+ */
16
+ export function createWavResponse(/* data , options */) {
17
+ // @note encode as Uint8Array to get the proper byte size of data
18
+ // const bytes = new TextEncoder().encode(MockWav);
19
+
20
+ const binaryString = atob(BASE64_WAV);
21
+ const bytes = new Uint8Array(binaryString.length);
22
+
23
+ for (let i = 0; i < binaryString.length; i++) {
24
+ bytes[i] = binaryString.charCodeAt(i);
25
+ }
26
+
27
+ const response = new Response(bytes, {
28
+ headers: new Headers({
29
+ [CONTENT_TYPE]: AUDIO_WAV,
30
+ [CONTENT_LENGTH]: String(bytes.length)
31
+ })
32
+ });
33
+
34
+ return response;
35
+ }