@urso/core 0.8.21 → 0.9.1-dev

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 (134) hide show
  1. package/LICENSE +20 -20
  2. package/README.md +227 -227
  3. package/build/js/index.js +77212 -2
  4. package/package.json +36 -50
  5. package/src/js/app.js +105 -104
  6. package/src/js/components/base/controller.js +76 -78
  7. package/src/js/components/debug/controller.js +39 -39
  8. package/src/js/components/debug/coords.js +23 -23
  9. package/src/js/components/debug/fps.js +36 -36
  10. package/src/js/components/debug/template.js +55 -55
  11. package/src/js/components/debug/timescale.js +60 -60
  12. package/src/js/components/deviceRotate/controller.js +95 -95
  13. package/src/js/components/editor/api.js +127 -127
  14. package/src/js/components/editor/controller.js +13 -13
  15. package/src/js/components/fullscreen/android.js +104 -104
  16. package/src/js/components/fullscreen/controller.js +76 -76
  17. package/src/js/components/fullscreen/desktop.js +49 -49
  18. package/src/js/components/fullscreen/ios.js +115 -115
  19. package/src/js/components/layersSwitcher/config.js +26 -26
  20. package/src/js/components/layersSwitcher/controller.js +36 -36
  21. package/src/js/components/loader/controller.js +66 -66
  22. package/src/js/components/loader/template.js +71 -71
  23. package/src/js/components/soundInitialPopup/controller.js +43 -43
  24. package/src/js/components/soundInitialPopup/template.js +110 -110
  25. package/src/js/components/stateDriven/controller.js +123 -123
  26. package/src/js/config/load.js +325 -194
  27. package/src/js/config/main.js +17 -17
  28. package/src/js/extra/browserEvents.js +57 -57
  29. package/src/js/extra/main.js +23 -25
  30. package/src/js/extra/pixiPatch.js +173 -171
  31. package/src/js/index.js +7 -7
  32. package/src/js/lib/cache.js +199 -125
  33. package/src/js/lib/composition.js +85 -85
  34. package/src/js/lib/device.js +1215 -1215
  35. package/src/js/lib/helper.js +678 -678
  36. package/src/js/lib/loader.js +216 -218
  37. package/src/js/lib/localData.js +29 -29
  38. package/src/js/lib/logger.js +69 -69
  39. package/src/js/lib/math.js +161 -161
  40. package/src/js/lib/objectPool.js +208 -208
  41. package/src/js/lib/time.js +19 -19
  42. package/src/js/lib/tween.js +153 -153
  43. package/src/js/modules/assets/baseModel.js +21 -21
  44. package/src/js/modules/assets/config.js +38 -38
  45. package/src/js/modules/assets/controller.js +65 -65
  46. package/src/js/modules/assets/models/atlas.js +19 -19
  47. package/src/js/modules/assets/models/audiosprite.js +28 -28
  48. package/src/js/modules/assets/models/bitmapFont.js +11 -11
  49. package/src/js/modules/assets/models/container.js +19 -19
  50. package/src/js/modules/assets/models/font.js +11 -11
  51. package/src/js/modules/assets/models/html.js +11 -11
  52. package/src/js/modules/assets/models/image.js +17 -17
  53. package/src/js/modules/assets/models/json.js +11 -11
  54. package/src/js/modules/assets/models/jsonAtlas.js +11 -11
  55. package/src/js/modules/assets/models/sound.js +17 -17
  56. package/src/js/modules/assets/models/spine.js +18 -17
  57. package/src/js/modules/assets/models/spineAtlas.js +11 -0
  58. package/src/js/modules/assets/service.js +564 -553
  59. package/src/js/modules/i18n/config.js +17 -17
  60. package/src/js/modules/i18n/controller.js +71 -71
  61. package/src/js/modules/instances/controller.js +357 -357
  62. package/src/js/modules/logic/config/sounds.js +23 -23
  63. package/src/js/modules/logic/controller.js +52 -52
  64. package/src/js/modules/logic/main.js +8 -8
  65. package/src/js/modules/logic/sounds.js +103 -103
  66. package/src/js/modules/objects/baseModel.js +205 -207
  67. package/src/js/modules/objects/cache.js +99 -99
  68. package/src/js/modules/objects/config.js +10 -10
  69. package/src/js/modules/objects/controller.js +139 -139
  70. package/src/js/modules/objects/find.js +58 -58
  71. package/src/js/modules/objects/models/bitmapText.js +56 -49
  72. package/src/js/modules/objects/models/button.js +208 -209
  73. package/src/js/modules/objects/models/buttonComposite.js +38 -38
  74. package/src/js/modules/objects/models/checkbox.js +100 -100
  75. package/src/js/modules/objects/models/collection.js +56 -54
  76. package/src/js/modules/objects/models/component.js +48 -48
  77. package/src/js/modules/objects/models/container.js +22 -22
  78. package/src/js/modules/objects/models/emitterFx.js +103 -114
  79. package/src/js/modules/objects/models/graphics.js +40 -40
  80. package/src/js/modules/objects/models/group.js +22 -22
  81. package/src/js/modules/objects/models/hitArea.js +187 -187
  82. package/src/js/modules/objects/models/image.js +36 -36
  83. package/src/js/modules/objects/models/imagesAnimation.js +115 -115
  84. package/src/js/modules/objects/models/mask.js +40 -40
  85. package/src/js/modules/objects/models/nineSlicePlane.js +32 -32
  86. package/src/js/modules/objects/models/slider.js +357 -357
  87. package/src/js/modules/objects/models/spine.js +354 -339
  88. package/src/js/modules/objects/models/text.js +70 -70
  89. package/src/js/modules/objects/models/toggle.js +186 -186
  90. package/src/js/modules/objects/models/world.js +22 -22
  91. package/src/js/modules/objects/pool.js +68 -68
  92. package/src/js/modules/objects/propertyAdapter.js +592 -592
  93. package/src/js/modules/objects/proxy.js +299 -298
  94. package/src/js/modules/objects/selector.js +136 -136
  95. package/src/js/modules/objects/service.js +255 -254
  96. package/src/js/modules/objects/styles.js +210 -210
  97. package/src/js/modules/observer/controller.js +168 -168
  98. package/src/js/modules/observer/events.js +51 -56
  99. package/src/js/modules/scenes/controller.js +126 -127
  100. package/src/js/modules/scenes/model.js +28 -28
  101. package/src/js/modules/scenes/pixiWrapper.js +360 -351
  102. package/src/js/modules/scenes/resolutions.js +173 -173
  103. package/src/js/modules/scenes/resolutionsConfig.js +73 -73
  104. package/src/js/modules/scenes/service.js +144 -146
  105. package/src/js/modules/soundManager/controller.js +103 -103
  106. package/src/js/modules/soundManager/soundSprite.js +314 -314
  107. package/src/js/modules/statesManager/action.js +97 -97
  108. package/src/js/modules/statesManager/all.js +23 -23
  109. package/src/js/modules/statesManager/configStates.js +77 -77
  110. package/src/js/modules/statesManager/controller.js +219 -219
  111. package/src/js/modules/statesManager/functionsStorage.js +83 -83
  112. package/src/js/modules/statesManager/helper.js +27 -27
  113. package/src/js/modules/statesManager/race.js +91 -91
  114. package/src/js/modules/statesManager/sequence.js +48 -48
  115. package/src/js/modules/template/controller.js +28 -28
  116. package/src/js/modules/template/model.js +11 -11
  117. package/src/js/modules/template/service.js +137 -137
  118. package/src/js/modules/template/types.js +50 -49
  119. package/src/js/modules/transport/baseConnectionType.js +25 -25
  120. package/src/js/modules/transport/config.js +13 -13
  121. package/src/js/modules/transport/connectionTypes/websocket.js +77 -77
  122. package/src/js/modules/transport/connectionTypes/xhr.js +47 -47
  123. package/src/js/modules/transport/controller.js +48 -48
  124. package/src/js/modules/transport/decorator.js +17 -17
  125. package/src/js/modules/transport/service.js +150 -150
  126. package/vite.config.js +31 -0
  127. package/.babelrc +0 -12
  128. package/build/js/index.js.LICENSE.txt +0 -227
  129. package/src/js/modules/objects/models/atlasImage.js +0 -52
  130. package/src/js/modules/objects/models/dragContainer.js +0 -665
  131. package/src/js/modules/objects/models/emitter.js +0 -69
  132. package/src/js/modules/objects/models/scrollbox.js +0 -67
  133. package/src/js/modules/objects/models/textInput.js +0 -68
  134. package/webpack.config.js +0 -47
@@ -1,553 +1,564 @@
1
- class ModulesAssetsService {
2
-
3
- constructor() {
4
- this.singleton = true;
5
-
6
- this.assets = {};
7
- this._loadCounter = 0;
8
-
9
- this._currentQuality = 'auto';
10
- this._addedAssetsCache = [];
11
- this.lazyLoadProcessStarted = false;
12
- };
13
-
14
- /**
15
- * system; check webP support
16
- */
17
- checkWebPSupport() {
18
- if (Urso.device.webP)
19
- Urso.addInstancesMode('webP');
20
- }
21
-
22
- /**
23
- * get current quality
24
- * @returns {String}
25
- */
26
- getQuality() {
27
- return this._currentQuality;
28
- }
29
-
30
- /**
31
- * Update quality
32
- */
33
- updateQuality() {
34
- this._currentQuality = this._detectQuality();
35
- Urso.addInstancesMode(this._currentQuality + 'Quality');
36
- }
37
-
38
- /**
39
- * sort assets and store to assets space
40
- * @param {Mixed} assets - asset or array of assets
41
- * @returns {Object}
42
- */
43
- sortAssets(assets) {
44
- const assetsSpace = this._createNewAssetsSpace();
45
- assetsSpace[this.getInstance('Config').loadingGroups.initial] = [];
46
-
47
- if (Array.isArray(assets))
48
- for (let asset of assets)
49
- this._addAsset(assetsSpace, asset);
50
- else
51
- this._addAsset(assetsSpace, assets);
52
-
53
- return assetsSpace;
54
- }
55
-
56
- /**
57
- * start load assetsSpace
58
- * @param {Object} assetsSpace
59
- * @param {Function} callback
60
- */
61
- startLoad(assetsSpace, callback, updateCallback, errorCallback, reloadOnError) {
62
- this.loadGroup(
63
- assetsSpace,
64
- this.getInstance('Config').loadingGroups.initial,
65
- (() => { callback(); this._startLazyLoad(errorCallback, reloadOnError); }).bind(this),
66
- updateCallback,
67
- errorCallback,
68
- reloadOnError
69
- )
70
- }
71
-
72
- /**
73
- * load assets group by name
74
- * @param {Object} assetsSpace
75
- * @param {String} group
76
- * @param {Function} callback
77
- * @returns
78
- */
79
- loadGroup(assetsSpace, group, callback, updateCallback, errorCallback, reloadOnError) {
80
- if (!assetsSpace) //global space
81
- assetsSpace = this.assets;
82
-
83
- if (!assetsSpace[group])
84
- return Urso.logger.error('ModulesAssetsService group error, no assets:' + group + ' Check ModulesAssetsConfig please');
85
-
86
- //we need load and parse atlases at first (!)
87
- const loadRestAssetsCallback = () => { this._loadGroupRestAssets(assetsSpace, group, callback, updateCallback, errorCallback, reloadOnError) };
88
-
89
- const loadAtlasesCallback = () => {
90
- this._loadGroupAtlases(assetsSpace, group, loadRestAssetsCallback, errorCallback, reloadOnError, Urso.types.assets.ATLAS);
91
- }
92
-
93
- this._loadGroupAtlases(assetsSpace, group, loadAtlasesCallback, errorCallback, reloadOnError, Urso.types.assets.JSONATLAS);
94
- }
95
-
96
- preloadAllImagesInGPU() {
97
- const world = Urso.findOne('^WORLD')._baseObject;
98
-
99
- Urso.cache.globalAtlas.pages.forEach(({ baseTexture }) => {
100
- const texture = new PIXI.Texture(baseTexture);
101
- const sprite = new PIXI.Sprite(texture);
102
-
103
- sprite.x = -10000;
104
- sprite.y = -10000;
105
-
106
- world.addChild(sprite);
107
-
108
- setTimeout(() => {
109
- world.removeChild(sprite);
110
- sprite.destroy()
111
- }, 1);
112
- });
113
- }
114
-
115
- /**
116
- * create new assets space for loading
117
- * @returns {Object}
118
- */
119
- _createNewAssetsSpace() {
120
- this._loadCounter++;
121
- this.assets[this._loadCounter] = {};
122
- return this.assets[this._loadCounter];
123
- }
124
-
125
- /**
126
- * load atlases from assets group
127
- * @param {Object} assetsSpace
128
- * @param {String} group
129
- * @param {Function} callback
130
- */
131
- _loadGroupAtlases(assetsSpace, group, callback, errorCallback, reloadOnError, atlasType) {
132
- const atlases = assetsSpace[group].filter(assetModel => assetModel.type === atlasType);
133
-
134
- if (!atlases.length)
135
- return callback();
136
-
137
- let loader = Urso.getInstance('Lib.Loader');
138
-
139
- for (let assetModel of atlases)
140
- this._addAssetToLoader(assetModel, loader);
141
-
142
- loader.start(
143
- () => {
144
- if (atlasType === Urso.types.assets.ATLAS)
145
- this._processLoadedAtlases(assetsSpace, group);
146
-
147
- callback();
148
- },
149
- errorCallback,
150
- reloadOnError
151
- );
152
- }
153
-
154
- /**
155
- * load rest assets (atkases is loaded)
156
- * @param {Object} assetsSpace
157
- * @param {String} group
158
- * @param {Function} callback
159
- */
160
- _loadGroupRestAssets(assetsSpace, group, callback, updateCallback, errorCallback, reloadOnError) {
161
- let loader = Urso.getInstance('Lib.Loader');
162
- //load update callback
163
- loader.setOnLoadUpdate((params) => { updateCallback(Math.floor(params.progress)); });
164
- const noAtlasSpines = [];
165
-
166
- for (let assetModel of assetsSpace[group])
167
- if (assetModel.type !== Urso.types.assets.ATLAS && assetModel.type !== Urso.types.assets.JSONATLAS)
168
- if (!Urso.cache.getFile(assetModel.path)) {
169
- //filter noAtlas Spine files
170
- if (assetModel.type === Urso.types.assets.SPINE && assetModel.noAtlas) {
171
- noAtlasSpines.push(assetModel);
172
- } else
173
- this._addAssetToLoader(assetModel, loader);
174
- }
175
-
176
- loader.start(
177
- () => {
178
- this._processLoadedAssets(assetsSpace, group);
179
- this._loadNoAtlasSpines(
180
- noAtlasSpines,
181
- () => {
182
- this.emit(Urso.events.MODULES_ASSETS_GROUP_LOADED, group);
183
- callback();
184
- },
185
- errorCallback,
186
- reloadOnError
187
- );
188
- },
189
- errorCallback,
190
- reloadOnError
191
- );
192
- }
193
-
194
- /**
195
- * load spines without .atlas files
196
- * @param {Array} noAtlasSpines
197
- * @param {Function} callback
198
- * @returns
199
- */
200
- _loadNoAtlasSpines(noAtlasSpines, callback, errorCallback, reloadOnError) {
201
- if (!noAtlasSpines.length)
202
- return callback();
203
-
204
- let loader = Urso.getInstance('Lib.Loader');
205
-
206
- for (let assetModel of noAtlasSpines)
207
- this._addAssetToLoader(assetModel, loader);
208
-
209
- loader.start(callback, errorCallback, reloadOnError);
210
- }
211
-
212
- /**
213
- * process loaded atlases
214
- * @param {Object} assetsSpace
215
- * @param {String} group
216
- */
217
- _processLoadedAtlases(assetsSpace, group) {
218
- const atlases = assetsSpace[group].filter(assetModel => assetModel.type === Urso.types.assets.ATLAS);
219
- const { addFolderPathInAtlasTextureKey } = this.getInstance('Config');
220
-
221
- for (let assetModel of atlases) {
222
- const assetKey = assetModel.key;
223
- let imageData = Urso.cache.getAtlas(assetKey);
224
- const folderPath = imageData.url.split('/').slice(0, -1).join('/');
225
-
226
- for (const name of Object.keys(imageData.spritesheet._frames)) {
227
- let texture = imageData.textures[name];
228
- let newFilename = name;
229
-
230
- if (addFolderPathInAtlasTextureKey && !name.includes('/'))
231
- newFilename = folderPath + '/' + name;
232
-
233
- Urso.cache.addFile(newFilename, texture);
234
-
235
- if (assetModel.cacheTextures) {
236
- const textureKey = newFilename.split('.')[0];
237
- Urso.cache.addTexture(textureKey, texture);
238
- }
239
- }
240
- }
241
- }
242
-
243
- /**
244
- * process loaded assets
245
- * @param {Object} assetsSpace
246
- * @param {String} group
247
- */
248
- _processLoadedAssets(assetsSpace, group) {
249
- for (let assetModel of assetsSpace[group]) {
250
- if (assetModel.type === Urso.types.assets.IMAGE)
251
- this._processLoadedImage(assetModel);
252
-
253
- if (assetModel.type === Urso.types.assets.BITMAPFONT)
254
- this._processLoadedBitmapFont(assetModel);
255
-
256
- if (assetModel.type === Urso.types.assets.FONT)
257
- this._processLoadedFont(assetModel);
258
- }
259
-
260
- delete assetsSpace[group];
261
- }
262
-
263
- /**
264
- * process loaded font
265
- * @param {Object} source
266
- */
267
- _processLoadedFont(source) {
268
- const data = Urso.cache.getFile(source.key);
269
- const font = new FontFace(source.key, data.data);
270
- font.load().then(() => {
271
- document.fonts.add(font);
272
- });
273
- }
274
-
275
- /**
276
- * process loaded bitmap font
277
- * @param {Object} source
278
- */
279
- _processLoadedBitmapFont(assetModel) {
280
- this._updateFontKey(assetModel.key)
281
- }
282
-
283
- /**
284
- * Update font key if cannot use original
285
- * @param {String} fontName
286
- * @returns
287
- */
288
- _updateFontKey(fontName) {
289
- const fontData = Urso.cache.getBitmapFont(fontName);
290
-
291
- if (PIXI.BitmapFont.available[fontName] || !fontData)
292
- return;
293
-
294
- const savedFont = PIXI.BitmapFont.available[fontData.bitmapFont.font];
295
-
296
- if (savedFont) {
297
- Urso.logger.warn(`bitmapFont ${fontData.bitmapFont.font} was rewritten with key ${fontName}`);
298
- PIXI.BitmapFont.available[fontName] = savedFont;
299
- delete PIXI.BitmapFont.available[fontData.bitmapFont.font];
300
- }
301
- }
302
-
303
- /**
304
- * get current assets resolution
305
- * @returns {Number}
306
- */
307
- getCurrentResolution() {
308
- const { qualityFactors, defaultQualityFactor } = this.getInstance('Config');
309
- return qualityFactors[this._currentQuality] || defaultQualityFactor || 1;
310
- }
311
-
312
- /**
313
- * process loaded image
314
- * @param {Object} assetModel
315
- */
316
- _processLoadedImage(assetModel) {
317
- const resolution = this.getCurrentResolution();
318
-
319
- const assetKey = assetModel.key;
320
- //textures cache
321
- let imageData = Urso.cache.getImage(assetKey);
322
-
323
- if (!imageData) {
324
- //from atlas ?!
325
- let texture = Urso.cache.getFile(assetModel.path);
326
-
327
- if (!texture)
328
- return Urso.logger.error('ModulesAssetsService process Loaded Image error: no image ', assetModel);
329
-
330
- Urso.cache.addTexture(assetKey, texture); //TODO change resolution of base texture
331
- } else {
332
- //regular image
333
- const baseTexture = new PIXI.BaseTexture(imageData.data, { resolution });
334
- const texture = new PIXI.Texture(baseTexture);
335
- Urso.cache.addTexture(assetKey, texture);
336
- }
337
-
338
- if (assetModel.preloadGPU) {
339
- let tempOblect = Urso.objects.create({
340
- type: Urso.types.objects.IMAGE,
341
- assetKey: assetKey,
342
- x: -10000, y: -10000
343
- }, false, true, true);
344
-
345
- setTimeout(() => { tempOblect.destroy() }, 1)
346
- }
347
- }
348
-
349
- /**
350
- * add asset
351
- * @param {Object} assetsSpace
352
- * @param {Object} asset
353
- * @param {String} loadingGroup
354
- */
355
- _addAsset(assetsSpace, asset, loadingGroup) {
356
- //cache for all assets. We do not need to load same assets twice or more
357
- if (asset.type !== Urso.types.assets.CONTAINER) {
358
- let addedAssetKey = `${asset.type}_${asset.key}`;
359
-
360
- if (this._addedAssetsCache.includes(addedAssetKey))
361
- return;
362
-
363
- this._addedAssetsCache.push(addedAssetKey);
364
- }
365
-
366
- let model;
367
-
368
- switch (asset.type) {
369
- case Urso.types.assets.ATLAS:
370
- model = this.getInstance('Models.Atlas', asset)
371
- break;
372
- case Urso.types.assets.AUDIOSPRITE:
373
- model = this.getInstance('Models.Audiosprite', asset)
374
- break;
375
- case Urso.types.assets.BITMAPFONT:
376
- model = this.getInstance('Models.BitmapFont', asset)
377
- break;
378
- case Urso.types.assets.CONTAINER:
379
- model = this.getInstance('Models.Container', asset)
380
- break;
381
- case Urso.types.assets.FONT:
382
- model = this.getInstance('Models.Font', asset)
383
- break;
384
- case Urso.types.assets.HTML:
385
- model = this.getInstance('Models.Html', asset)
386
- break;
387
- case Urso.types.assets.IMAGE:
388
- model = this.getInstance('Models.Image', asset)
389
- break;
390
- case Urso.types.assets.JSON:
391
- model = this.getInstance('Models.Json', asset)
392
- break;
393
- case Urso.types.assets.JSONATLAS:
394
- model = this.getInstance('Models.JsonAtlas', asset)
395
- break;
396
- case Urso.types.assets.SOUND:
397
- model = this.getInstance('Models.Sound', asset)
398
- break;
399
- case Urso.types.assets.SPINE:
400
- model = this.getInstance('Models.Spine', asset)
401
- break;
402
- default:
403
- Urso.logger.error('ModulesAssetsService asset type error', asset);
404
- break;
405
- };
406
-
407
- //set loadingGroup
408
- model.loadingGroup = loadingGroup || model.loadingGroup || this.getInstance('Config').loadingGroups.initial;
409
-
410
- //check if container
411
- if (model.contents) {
412
- for (let content of model.contents) {
413
- this._addAsset(assetsSpace, content, model.loadingGroup);
414
- }
415
-
416
- return;
417
- }
418
-
419
- if (model.loadingGroup === this.getInstance('Config').loadingGroups.initial) {
420
- assetsSpace[model.loadingGroup].push(model);
421
- } else {
422
- //add single asset to loading group
423
- if (!this.assets[model.loadingGroup])
424
- this.assets[model.loadingGroup] = [];
425
-
426
- this.assets[model.loadingGroup].push(model);
427
- }
428
- }
429
-
430
- /**
431
- * start lazy load process
432
- */
433
- _startLazyLoad(errorCallback, reloadOnError) {
434
- if (this.lazyLoadProcessStarted)
435
- return;
436
-
437
- this.lazyLoadProcessStarted = true;
438
- this._continueLazyLoad(errorCallback, reloadOnError);
439
- }
440
-
441
- /**
442
- * continue lazy load process (with current step)
443
- * @param {Number} step
444
- */
445
- _continueLazyLoad(errorCallback, reloadOnError, step = 0) {
446
- const lazyLoadGroups = this.getInstance('Config').lazyLoadGroups;
447
-
448
- if (step >= lazyLoadGroups.length) {
449
- this.emit(Urso.events.MODULES_ASSETS_LAZYLOAD_FINISHED);
450
- return;
451
- }
452
-
453
- let groupName = lazyLoadGroups[step];
454
-
455
- if (!groupName)
456
- Urso.logger.error('ModulesAssetsService lazy loading groupName error');
457
-
458
- const callback = () => { this._continueLazyLoad(errorCallback, reloadOnError, step + 1); }
459
- const updateCallback = () => { };
460
- this.loadGroup(null, groupName, callback, updateCallback, errorCallback, reloadOnError);
461
- }
462
-
463
- /**
464
- * quality reducer
465
- * @param {Object} qualityFactors
466
- * @param {Number} widthFactor
467
- * @returns
468
- */
469
- _qualityReducer(qualityFactors, widthFactor) {
470
- return [(acc, val) => {
471
- if (acc === null) {
472
- return val;
473
- }
474
-
475
- const currentQuality = qualityFactors[acc];
476
- const qualityFactor = qualityFactors[val];
477
-
478
- const nextQuality = (currentQuality > qualityFactor && qualityFactor >= widthFactor)
479
- || (qualityFactor >= widthFactor && widthFactor > currentQuality)
480
- || (widthFactor >= qualityFactor && qualityFactor > currentQuality);
481
-
482
- return nextQuality ? val : acc;
483
-
484
- }, null]
485
- }
486
-
487
- /**
488
- * detect assets quality
489
- * @returns {Number}
490
- */
491
- _detectQuality() {
492
- const { qualityFactors } = this.getInstance('Config');
493
- const userQuality = Urso.helper.parseGetParams()['quality'];
494
-
495
- if (userQuality && qualityFactors[userQuality]) {
496
- return userQuality;
497
- }
498
-
499
- return this._calculateQuality(qualityFactors);
500
- }
501
-
502
- /**
503
- * calculate quality
504
- * @param {Object} qualityFactors
505
- */
506
- _calculateQuality(qualityFactors) {
507
- const { android, iOS, iPad, macOS } = Urso.device;
508
- const isMobile = android || iOS || iPad;
509
-
510
- if (macOS && !isMobile) {
511
- return 'high';
512
- }
513
-
514
- if (macOS && iPad) {
515
- return 'medium';
516
- }
517
-
518
- const resCfg = Urso.getInstance('Modules.Scenes.ResolutionsConfig').contents[0];
519
-
520
- const { devicePixelRatio } = window;
521
- let { width, height } = screen;
522
-
523
- if (isMobile) {
524
- width = (width > height) ? width : height;
525
- }
526
-
527
- if (iOS) {
528
- width *= devicePixelRatio;
529
- }
530
-
531
- const widthFactor = width / resCfg.width;
532
-
533
- return Object
534
- .keys(qualityFactors)
535
- .reduce(...this._qualityReducer(qualityFactors, widthFactor));
536
- }
537
-
538
- /**
539
- * add asset to loader instance
540
- * @param {Object} assetModel
541
- * @param {Object} loader
542
- */
543
- _addAssetToLoader(assetModel, loader) {
544
- if (assetModel.path)
545
- loader.addAsset(assetModel);
546
- else if (assetModel.contents) {
547
- //do nothing, its a container
548
- } else
549
- Urso.logger.error('ModulesAssetsService model error', assetModel);
550
- };
551
- }
552
-
553
- module.exports = ModulesAssetsService;
1
+ class ModulesAssetsService {
2
+
3
+ constructor() {
4
+ this.singleton = true;
5
+
6
+ this.assets = {};
7
+ this._loadCounter = 0;
8
+
9
+ this._currentQuality = 'auto';
10
+ this._addedAssetsCache = [];
11
+ this.lazyLoadProcessStarted = false;
12
+ };
13
+
14
+ /**
15
+ * system; check webP support
16
+ */
17
+ checkWebPSupport() {
18
+ if (Urso.device.webP)
19
+ Urso.addInstancesMode('webP');
20
+ }
21
+
22
+ /**
23
+ * get current quality
24
+ * @returns {String}
25
+ */
26
+ getQuality() {
27
+ return this._currentQuality;
28
+ }
29
+
30
+ /**
31
+ * Update quality
32
+ */
33
+ updateQuality() {
34
+ this._currentQuality = this._detectQuality();
35
+ Urso.addInstancesMode(this._currentQuality + 'Quality');
36
+ }
37
+
38
+ /**
39
+ * sort assets and store to assets space
40
+ * @param {Mixed} assets - asset or array of assets
41
+ * @returns {Object}
42
+ */
43
+ sortAssets(assets) {
44
+ const assetsSpace = this._createNewAssetsSpace();
45
+ assetsSpace[this.getInstance('Config').loadingGroups.initial] = [];
46
+
47
+ if (Array.isArray(assets))
48
+ for (let asset of assets)
49
+ this._addAsset(assetsSpace, asset);
50
+ else
51
+ this._addAsset(assetsSpace, assets);
52
+
53
+ return assetsSpace;
54
+ }
55
+
56
+ /**
57
+ * start load assetsSpace
58
+ * @param {Object} assetsSpace
59
+ * @param {Function} callback
60
+ */
61
+ startLoad(assetsSpace, callback, updateCallback) {
62
+ this.loadGroup(
63
+ assetsSpace,
64
+ this.getInstance('Config').loadingGroups.initial,
65
+ (() => { callback(); this._startLazyLoad(); }).bind(this),
66
+ updateCallback
67
+ )
68
+ }
69
+
70
+ /**
71
+ * load assets group by name
72
+ * @param {Object} assetsSpace
73
+ * @param {String} group
74
+ * @param {Function} callback
75
+ * @returns
76
+ */
77
+ loadGroup(assetsSpace, group, callback = () => { }, updateCallback = () => { }) {
78
+ if (!assetsSpace) //global space
79
+ assetsSpace = this.assets;
80
+
81
+ if (!assetsSpace[group])
82
+ return Urso.logger.error('ModulesAssetsService group error, no assets:' + group + ' Check ModulesAssetsConfig please');
83
+
84
+ //we need load and parse atlases at first (!)
85
+ const loadRestAssetsCallback = () => { this._loadGroupRestAssets(assetsSpace, group, callback, updateCallback) };
86
+
87
+ const loadAtlasesCallback = () => {
88
+ this._loadGroupAtlases(assetsSpace, group, loadRestAssetsCallback, Urso.types.assets.ATLAS);
89
+ }
90
+
91
+ this._loadGroupAtlases(assetsSpace, group, loadAtlasesCallback, Urso.types.assets.JSONATLAS);
92
+ }
93
+
94
+ preloadAllImagesInGPU() {
95
+ //FIXME
96
+ // const world = Urso.findOne('^WORLD')._baseObject;
97
+
98
+ // Urso.cache.globalAtlas.pages.forEach(({ baseTexture }) => {
99
+ // const texture = new PIXI.Texture(baseTexture);
100
+ // const sprite = new PIXI.Sprite(texture);
101
+
102
+ // sprite.x = -10000;
103
+ // sprite.y = -10000;
104
+
105
+ // world.addChild(sprite);
106
+
107
+ // setTimeout(() => {
108
+ // world.removeChild(sprite);
109
+ // sprite.destroy()
110
+ // }, 1);
111
+ // });
112
+ }
113
+
114
+ /**
115
+ * create new assets space for loading
116
+ * @returns {Object}
117
+ */
118
+ _createNewAssetsSpace() {
119
+ this._loadCounter++;
120
+ this.assets[this._loadCounter] = {};
121
+ return this.assets[this._loadCounter];
122
+ }
123
+
124
+ /**
125
+ * load atlases from assets group
126
+ * @param {Object} assetsSpace
127
+ * @param {String} group
128
+ * @param {Function} callback
129
+ */
130
+ _loadGroupAtlases(assetsSpace, group, callback, atlasType) {
131
+ const atlases = assetsSpace[group].filter(assetModel => assetModel.type === atlasType);
132
+
133
+ if (!atlases.length)
134
+ return callback();
135
+
136
+ let loader = Urso.getInstance('Lib.Loader');
137
+
138
+ for (let assetModel of atlases)
139
+ this._addAssetToLoader(assetModel, loader);
140
+
141
+ loader.start(() => {
142
+ if (atlasType === Urso.types.assets.ATLAS)
143
+ this._processLoadedAtlases(assetsSpace, group);
144
+
145
+ callback();
146
+ });
147
+ }
148
+
149
+ /**
150
+ * load rest assets (atkases is loaded)
151
+ * @param {Object} assetsSpace
152
+ * @param {String} group
153
+ * @param {Function} callback
154
+ */
155
+ _loadGroupRestAssets(assetsSpace, group, callback, updateCallback) {
156
+ let loader = Urso.getInstance('Lib.Loader');
157
+ //load update callback
158
+ loader.setOnLoadUpdate((params) => { updateCallback(Math.floor(params.progress)); });
159
+ const noAtlasSpines = [];
160
+
161
+ for (let assetModel of assetsSpace[group])
162
+ if (assetModel.type !== Urso.types.assets.ATLAS && assetModel.type !== Urso.types.assets.JSONATLAS)
163
+ if (!Urso.cache.getFile(assetModel.path)) {
164
+ //filter noAtlas Spine files
165
+
166
+ //FIXME remove noAtlasSpine
167
+ if (assetModel.type === Urso.types.assets.SPINE && assetModel.noAtlas) {
168
+ noAtlasSpines.push(assetModel);
169
+ } else
170
+ this._addAssetToLoader(assetModel, loader);
171
+ }
172
+
173
+ loader.start(
174
+ async () => {
175
+ await this._processLoadedAssets(assetsSpace, group);
176
+ this._loadNoAtlasSpines(noAtlasSpines, () => {
177
+ this.emit(Urso.events.MODULES_ASSETS_GROUP_LOADED, group);
178
+ callback();
179
+ });
180
+ }
181
+ );
182
+ }
183
+
184
+ /**
185
+ * load spines without .atlas files
186
+ * @param {Array} noAtlasSpines
187
+ * @param {Function} callback
188
+ * @returns
189
+ */
190
+ _loadNoAtlasSpines(noAtlasSpines, callback) {
191
+ if (!noAtlasSpines.length)
192
+ return callback();
193
+
194
+ let loader = Urso.getInstance('Lib.Loader');
195
+
196
+ for (let assetModel of noAtlasSpines)
197
+ this._addAssetToLoader(assetModel, loader);
198
+
199
+ loader.start(callback);
200
+ }
201
+
202
+ /**
203
+ * process loaded atlases
204
+ * @param {Object} assetsSpace
205
+ * @param {String} group
206
+ */
207
+ _processLoadedAtlases(assetsSpace, group) {
208
+ const atlases = assetsSpace[group].filter(assetModel => assetModel.type === Urso.types.assets.ATLAS);
209
+ const { addFolderPathInAtlasTextureKey } = this.getInstance('Config');
210
+
211
+ for (let assetModel of atlases) {
212
+ const assetKey = assetModel.key;
213
+ let imageData = Urso.cache.getAtlas(assetKey);
214
+ //FIXME
215
+ if(!assetModel.cacheTextures) return;
216
+
217
+ for (let [key, val] of Object.entries(imageData.textures)) {
218
+ Urso.cache.addTexture(key, val);
219
+ }
220
+
221
+ // for (const name of Object.keys(imageData.texture)) {
222
+ // let texture = imageData.textures[name];
223
+ // let newFilename = name;
224
+
225
+ // if (addFolderPathInAtlasTextureKey && !name.includes('/'))
226
+ // newFilename = folderPath + '/' + name;
227
+
228
+ // Urso.cache.addFile(newFilename, texture);
229
+
230
+
231
+ // }
232
+ }
233
+ }
234
+
235
+ /**
236
+ * process loaded assets
237
+ * @param {Object} assetsSpace
238
+ * @param {String} group
239
+ */
240
+ async _processLoadedAssets(assetsSpace, group) {
241
+ //FIXME
242
+ for (let assetModel of assetsSpace[group]) {
243
+ //FIXME
244
+ // if (assetModel.type === Urso.types.assets.IMAGE)
245
+ // this._processLoadedImage(assetModel);
246
+
247
+ //FIXME
248
+ // if (assetModel.type === Urso.types.assets.BITMAPFONT)
249
+ // this._processLoadedBitmapFont(assetModel);
250
+
251
+ //FIXME
252
+ // if (assetModel.type === Urso.types.assets.FONT)
253
+ // await this._processLoadedFont(assetModel);
254
+
255
+ if(assetModel.type === Urso.types.assets.SPINEATLAS) {
256
+ this._processLoadedSpineAtlas(assetModel);
257
+ }
258
+ }
259
+
260
+ delete assetsSpace[group];
261
+ }
262
+
263
+ _processLoadedSpineAtlas(assetModel) {
264
+ const spineAtlas = Urso.cache.getSpineAtlas(assetModel.key);
265
+
266
+ if (!spineAtlas)
267
+ return Urso.logger.error('ModulesAssetsService process Loaded Spine Atlas error: no image ', assetModel);
268
+
269
+ spineAtlas.regions.forEach(({ name, texture: {texture}}) => {
270
+ Urso.cache.addTexture(name, texture);
271
+ });
272
+ }
273
+
274
+ /**
275
+ * process loaded font
276
+ * @param {Object} source
277
+ */
278
+ async _processLoadedFont(source) {
279
+ const data = Urso.cache.getFile(source.key);
280
+ const font = new FontFace(source.key, data);
281
+ await font.load();
282
+ }
283
+
284
+ /**
285
+ * process loaded bitmap font
286
+ * @param {Object} source
287
+ */
288
+ _processLoadedBitmapFont(assetModel) {
289
+ this._updateFontKey(assetModel.key)
290
+ }
291
+
292
+ /**
293
+ * Update font key if cannot use original
294
+ * @param {String} fontName
295
+ * @returns
296
+ */
297
+ _updateFontKey(fontName) {
298
+ const fontData = Urso.cache.getBitmapFont(fontName);
299
+
300
+ if (PIXI.BitmapFont.available[fontName] || !fontData)
301
+ return;
302
+
303
+ const savedFont = PIXI.BitmapFont.available[fontData.bitmapFont.font];
304
+
305
+ if (savedFont) {
306
+ Urso.logger.warn(`bitmapFont ${fontData.bitmapFont.font} was rewritten with key ${fontName}`);
307
+ PIXI.BitmapFont.available[fontName] = savedFont;
308
+ delete PIXI.BitmapFont.available[fontData.bitmapFont.font];
309
+ }
310
+ }
311
+
312
+ /**
313
+ * get current assets resolution
314
+ * @returns {Number}
315
+ */
316
+ getCurrentResolution() {
317
+ const { qualityFactors, defaultQualityFactor } = this.getInstance('Config');
318
+ return qualityFactors[this._currentQuality] || defaultQualityFactor || 1;
319
+ }
320
+
321
+ /**
322
+ * process loaded image
323
+ * @param {Object} assetModel
324
+ */
325
+ _processLoadedImage(assetModel) {
326
+ const resolution = this.getCurrentResolution();
327
+
328
+ const assetKey = assetModel.key;
329
+ //textures cache
330
+ let imageData = Urso.cache.getImage(assetKey);
331
+
332
+ if (!imageData) {
333
+ //from atlas ?!
334
+ let texture = Urso.cache.getFile(assetModel.path);
335
+
336
+ if (!texture)
337
+ return Urso.logger.error('ModulesAssetsService process Loaded Image error: no image ', assetModel);
338
+
339
+ Urso.cache.addTexture(assetKey, texture); //TODO change resolution of base texture
340
+ } else {
341
+ //regular image
342
+ Urso.cache.addTexture(assetKey, imageData);
343
+ }
344
+
345
+ if (assetModel.preloadGPU) {
346
+ let tempOblect = Urso.objects.create({
347
+ type: Urso.types.objects.IMAGE,
348
+ assetKey: assetKey,
349
+ x: -10000, y: -10000
350
+ }, false, true, true);
351
+
352
+ setTimeout(() => { tempOblect.destroy() }, 1)
353
+ }
354
+ }
355
+
356
+ /**
357
+ * add asset
358
+ * @param {Object} assetsSpace
359
+ * @param {Object} asset
360
+ * @param {String} loadingGroup
361
+ */
362
+ _addAsset(assetsSpace, asset, loadingGroup) {
363
+ //cache for all assets. We do not need to load same assets twice or more
364
+ if (asset.type !== Urso.types.assets.CONTAINER) {
365
+ let addedAssetKey = `${asset.type}_${asset.key}`;
366
+
367
+ if (this._addedAssetsCache.includes(addedAssetKey))
368
+ return;
369
+
370
+ this._addedAssetsCache.push(addedAssetKey);
371
+ }
372
+
373
+ let model;
374
+
375
+ switch (asset.type) {
376
+ case Urso.types.assets.ATLAS:
377
+ model = this.getInstance('Models.Atlas', asset)
378
+ break;
379
+ case Urso.types.assets.AUDIOSPRITE:
380
+ model = this.getInstance('Models.Audiosprite', asset)
381
+ break;
382
+ case Urso.types.assets.BITMAPFONT:
383
+ model = this.getInstance('Models.BitmapFont', asset)
384
+ break;
385
+ case Urso.types.assets.CONTAINER:
386
+ model = this.getInstance('Models.Container', asset)
387
+ break;
388
+ case Urso.types.assets.FONT:
389
+ model = this.getInstance('Models.Font', asset)
390
+ break;
391
+ case Urso.types.assets.HTML:
392
+ model = this.getInstance('Models.Html', asset)
393
+ break;
394
+ case Urso.types.assets.IMAGE:
395
+ model = this.getInstance('Models.Image', asset)
396
+ break;
397
+ case Urso.types.assets.JSON:
398
+ model = this.getInstance('Models.Json', asset)
399
+ break;
400
+ case Urso.types.assets.JSONATLAS:
401
+ model = this.getInstance('Models.JsonAtlas', asset)
402
+ break;
403
+ case Urso.types.assets.SOUND:
404
+ model = this.getInstance('Models.Sound', asset)
405
+ break;
406
+ case Urso.types.assets.SPINE:
407
+ model = this.getInstance('Models.Spine', asset)
408
+ break;
409
+ case Urso.types.assets.SPINEATLAS:
410
+ model = this.getInstance('Models.SpineAtlas', asset)
411
+ break;
412
+ default:
413
+ Urso.logger.error('ModulesAssetsService asset type error', asset);
414
+ break;
415
+ };
416
+
417
+ //set loadingGroup
418
+ model.loadingGroup = loadingGroup || model.loadingGroup || this.getInstance('Config').loadingGroups.initial;
419
+
420
+ //check if container
421
+ if (model.contents) {
422
+ for (let content of model.contents) {
423
+ this._addAsset(assetsSpace, content, model.loadingGroup);
424
+ }
425
+
426
+ return;
427
+ }
428
+
429
+ if (model.loadingGroup === this.getInstance('Config').loadingGroups.initial) {
430
+ assetsSpace[model.loadingGroup].push(model);
431
+ } else {
432
+ //add single asset to loading group
433
+ if (!this.assets[model.loadingGroup])
434
+ this.assets[model.loadingGroup] = [];
435
+
436
+ this.assets[model.loadingGroup].push(model);
437
+ }
438
+ }
439
+
440
+ /**
441
+ * start lazy load process
442
+ */
443
+ _startLazyLoad() {
444
+ if (this.lazyLoadProcessStarted)
445
+ return;
446
+
447
+ this.lazyLoadProcessStarted = true;
448
+ this._continueLazyLoad();
449
+ }
450
+
451
+ /**
452
+ * continue lazy load process (with current step)
453
+ * @param {Number} step
454
+ */
455
+ _continueLazyLoad(step) {
456
+ if (!step)
457
+ step = 0;
458
+
459
+ const lazyLoadGroups = this.getInstance('Config').lazyLoadGroups;
460
+
461
+ if (step >= lazyLoadGroups.length) {
462
+ this.emit(Urso.events.MODULES_ASSETS_LAZYLOAD_FINISHED);
463
+ return;
464
+ }
465
+
466
+ let groupName = lazyLoadGroups[step];
467
+
468
+ if (!groupName)
469
+ Urso.logger.error('ModulesAssetsService lazy loading groupName error');
470
+
471
+ this.loadGroup(null, groupName, () => { this._continueLazyLoad(step + 1); })
472
+ }
473
+
474
+ /**
475
+ * quality reducer
476
+ * @param {Object} qualityFactors
477
+ * @param {Number} widthFactor
478
+ * @returns
479
+ */
480
+ _qualityReducer(qualityFactors, widthFactor) {
481
+ return [(acc, val) => {
482
+ if (acc === null) {
483
+ return val;
484
+ }
485
+
486
+ const currentQuality = qualityFactors[acc];
487
+ const qualityFactor = qualityFactors[val];
488
+
489
+ const nextQuality = (currentQuality > qualityFactor && qualityFactor >= widthFactor)
490
+ || (qualityFactor >= widthFactor && widthFactor > currentQuality)
491
+ || (widthFactor >= qualityFactor && qualityFactor > currentQuality);
492
+
493
+ return nextQuality ? val : acc;
494
+
495
+ }, null]
496
+ }
497
+
498
+ /**
499
+ * detect assets quality
500
+ * @returns {Number}
501
+ */
502
+ _detectQuality() {
503
+ const { qualityFactors } = this.getInstance('Config');
504
+ const userQuality = Urso.helper.parseGetParams()['quality'];
505
+
506
+ if (userQuality && qualityFactors[userQuality]) {
507
+ return userQuality;
508
+ }
509
+
510
+ return this._calculateQuality(qualityFactors);
511
+ }
512
+
513
+ /**
514
+ * calculate quality
515
+ * @param {Object} qualityFactors
516
+ */
517
+ _calculateQuality(qualityFactors) {
518
+ const { android, iOS, iPad, macOS } = Urso.device;
519
+ const isMobile = android || iOS || iPad;
520
+
521
+ if (macOS && !isMobile) {
522
+ return 'high';
523
+ }
524
+
525
+ if (macOS && iPad) {
526
+ return 'medium';
527
+ }
528
+
529
+ const resCfg = Urso.getInstance('Modules.Scenes.ResolutionsConfig').contents[0];
530
+
531
+ const { devicePixelRatio } = window;
532
+ let { width, height } = screen;
533
+
534
+ if (isMobile) {
535
+ width = (width > height) ? width : height;
536
+ }
537
+
538
+ if (iOS) {
539
+ width *= devicePixelRatio;
540
+ }
541
+
542
+ const widthFactor = width / resCfg.width;
543
+
544
+ return Object
545
+ .keys(qualityFactors)
546
+ .reduce(...this._qualityReducer(qualityFactors, widthFactor));
547
+ }
548
+
549
+ /**
550
+ * add asset to loader instance
551
+ * @param {Object} assetModel
552
+ * @param {Object} loader
553
+ */
554
+ _addAssetToLoader(assetModel, loader) {
555
+ if (assetModel.path)
556
+ loader.addAsset(assetModel);
557
+ else if (assetModel.contents) {
558
+ //do nothing, its a container
559
+ } else
560
+ Urso.logger.error('ModulesAssetsService model error', assetModel);
561
+ };
562
+ }
563
+
564
+ export default ModulesAssetsService;