-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathjsonmanager.js
172 lines (154 loc) · 5.37 KB
/
jsonmanager.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
// @ts-check
/*
* GDevelop JS Platform
* Copyright 2013-present Florian Rival (Florian.Rival@gmail.com). All rights reserved.
* This project is released under the MIT License.
*/
/**
* JsonManager loads json files (using XMLHttpRequest), using the "json" resources
* registered in the game resources.
*
* Contrary to audio/fonts, json files are loaded asynchronously, when requested.
* You should properly handle errors, and give the developer/player a way to know
* that loading failed.
*
* @class JsonManager
* @memberof gdjs
* @param {ResourceData[]} resources The resources data of the game.
*/
gdjs.JsonManager = function (resources) {
this._resources = resources;
/** @type Object.<string, Object> */
this._loadedJsons = {};
};
/**
* Update the resources data of the game. Useful for hot-reloading, should not be used otherwise.
*
* @param {ResourceData[]} resources The resources data of the game.
*/
gdjs.JsonManager.prototype.setResources = function (resources) {
this._resources = resources;
};
/**
* The callback called when a json is preloaded
* @callback JsonManagerOnProgressCallback
* @param {number} loaded The number of json files loaded so far
* @param {number} total The total number to be loaded
* @returns {void} Nothing
*/
/**
* The callback called when all jsons are preloaded
* @callback JsonManagerOnCompleteCallback
* @param {number} total The total number to be loaded
* @returns {void} Nothing
*/
/**
* Request all the json resources to be preloaded (unless they are marked as not preloaded).
*
* Note that even if a JSON is already loaded, it will be reloaded (useful for hot-reloading,
* as JSON files can have been modified without the editor knowing).
*
* @param {JsonManagerOnProgressCallback} onProgress The function called after each json is loaded.
* @param {JsonManagerOnCompleteCallback} onComplete The function called when all jsons are loaded.
*/
gdjs.JsonManager.prototype.preloadJsons = function (onProgress, onComplete) {
var resources = this._resources;
var jsonResources = resources.filter(function (resource) {
return resource.kind === 'json' && !resource.disablePreload;
});
if (jsonResources.length === 0) return onComplete(jsonResources.length);
var loaded = 0;
/** @type JsonManagerRequestCallback */
var onLoad = function (error) {
if (error) {
console.error('Error while preloading a json resource:' + error);
}
loaded++;
if (loaded === jsonResources.length) {
onComplete(jsonResources.length);
} else {
onProgress(loaded, jsonResources.length);
}
};
for (var i = 0; i < jsonResources.length; ++i) {
this.loadJson(jsonResources[i].name, onLoad);
}
};
/**
* The callback called when a json that was requested is loaded (or an error occured).
* @callback JsonManagerRequestCallback
* @param {?Error} error The error, if any. `null` otherwise.
* @param {?Object} jsonContent The content of the json file (or null if an error occured).
* @returns {void} Nothing
*/
/**
* Request the json file from the given resource name.
* This method is asynchronous. When loaded, the `callback` is called with the error
* (null if none) and the loaded json (a JS Object).
*
* @param {string} resourceName The resource pointing to the json file to load.
* @param {JsonManagerRequestCallback} callback The callback function called when json is loaded (or an error occured).
*/
gdjs.JsonManager.prototype.loadJson = function (resourceName, callback) {
// @ts-ignore - find is not ES5
var resource = this._resources.find(function (resource) {
return resource.kind === 'json' && resource.name === resourceName;
});
if (!resource) {
callback(
new Error(
'Can\'t find resource with name: "' +
resourceName +
'" (or is not a json resource).'
),
null
);
return;
}
// Don't fetch again an object that is already in memory
if (this._loadedJsons[resourceName]) {
callback(null, this._loadedJsons[resourceName]);
return;
}
var that = this;
var xhr = new XMLHttpRequest();
xhr.responseType = 'json';
xhr.open('GET', resource.file);
xhr.onload = function () {
if (xhr.status !== 200) {
callback(
new Error('HTTP error: ' + xhr.status + '(' + xhr.statusText + ')'),
null
);
return;
}
// Cache the result
that._loadedJsons[resourceName] = xhr.response;
callback(null, xhr.response);
};
xhr.onerror = function () {
callback(new Error('Network error'), null);
};
xhr.onabort = function () {
callback(new Error('Request aborted'), null);
};
xhr.send();
};
/**
* Check if the given json resource was loaded (preloaded or loaded with `loadJson`).
* @param {string} resourceName The name of the json resource.
* @returns {boolean} true if the content of the json resource is loaded. false otherwise.
*/
gdjs.JsonManager.prototype.isJsonLoaded = function (resourceName) {
return !!this._loadedJsons[resourceName];
};
/**
* Get the object for the given resource that is already loaded (preloaded or loaded with `loadJson`).
* If the resource is not loaded, `null` will be returned.
*
* @param {string} resourceName The name of the json resource.
* @returns {?Object} the content of the json resource, if loaded. `null` otherwise.
*/
gdjs.JsonManager.prototype.getLoadedJson = function (resourceName) {
return this._loadedJsons[resourceName] || null;
};