1 /****************************************************************************
  2  Copyright (c) 2010-2012 cocos2d-x.org
  3  Copyright (c) 2008-2010 Ricardo Quesada
  4  Copyright (c) 2011      Zynga Inc.
  5 
  6  http://www.cocos2d-x.org
  7 
  8  Permission is hereby granted, free of charge, to any person obtaining a copy
  9  of this software and associated documentation files (the "Software"), to deal
 10  in the Software without restriction, including without limitation the rights
 11  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 12  copies of the Software, and to permit persons to whom the Software is
 13  furnished to do so, subject to the following conditions:
 14 
 15  The above copyright notice and this permission notice shall be included in
 16  all copies or substantial portions of the Software.
 17 
 18  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 19  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 20  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 21  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 22  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 23  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 24  THE SOFTWARE.
 25  ****************************************************************************/
 26 
 27 /**
 28  * resource type
 29  * @constant
 30  * @type Object
 31  */
 32 cc.RESOURCE_TYPE = {
 33     "IMAGE": ["png", "jpg", "bmp","jpeg","gif"],
 34     "SOUND": ["mp3", "ogg", "wav", "mp4", "m4a"],
 35     "XML": ["plist", "xml", "fnt", "tmx", "tsx"],
 36     "BINARY": ["ccbi"],
 37     "FONT": "FONT",
 38     "TEXT":["txt", "vsh", "fsh","json"],
 39     "UNKNOW": []
 40 };
 41 
 42 /**
 43  * A class to pre-load resources before engine start game main loop.
 44  * @class
 45  * @extends cc.Scene
 46  */
 47 cc.Loader = cc.Class.extend(/** @lends cc.Loader# */{
 48     _curNumber: 0,
 49     _totalNumber: 0,
 50     _loadedNumber: 0,
 51     _resouces: null,
 52     _animationInterval: 1 / 60,
 53     _interval: null,
 54     _isAsync: false,
 55 
 56     /**
 57      * Constructor
 58      */
 59     ctor: function () {
 60         this._resouces = [];
 61     },
 62 
 63     /**
 64      * init with resources
 65      * @param {Array} resources
 66      * @param {Function|String} selector
 67      * @param {Object} target
 68      */
 69     initWithResources: function (resources, selector, target) {
 70         if(!resources){
 71             console.log("resources should not null");
 72             return;
 73         }
 74 
 75         if (selector) {
 76             this._selector = selector;
 77             this._target = target;
 78         }
 79 
 80         if ((resources != this._resouces) || (this._curNumber == 0)) {
 81             this._curNumber = 0;
 82             this._loadedNumber = 0;
 83             if (resources[0] instanceof Array) {
 84                 for (var i = 0; i < resources.length; i++) {
 85                     var each = resources[i];
 86                     this._resouces = this._resouces.concat(each);
 87                 }
 88             } else
 89                 this._resouces = resources;
 90             this._totalNumber = this._resouces.length;
 91         }
 92 
 93         //load resources
 94         this._schedulePreload();
 95     },
 96 
 97     setAsync: function (isAsync) {
 98         this._isAsync = isAsync;
 99     },
100 
101     /**
102      * Callback when a resource file load failed.
103      * @example
104      * //example
105      * cc.Loader.getInstance().onResLoaded();
106      */
107     onResLoadingErr: function (name) {
108         cc.log("cocos2d:Failed loading resource: " + name);
109     },
110 
111     /**
112      * Callback when a resource file loaded.
113      * @example
114      * //example
115      * cc.Loader.getInstance().onResLoaded();
116      */
117     onResLoaded: function () {
118         this._loadedNumber++;
119     },
120 
121     /**
122      * Get loading percentage
123      * @return {Number}
124      * @example
125      * //example
126      * cc.log(cc.Loader.getInstance().getPercentage() + "%");
127      */
128     getPercentage: function () {
129         var percent = 0;
130         if (this._totalNumber == 0) {
131             percent = 100;
132         } else {
133             percent = (0 | (this._loadedNumber / this._totalNumber * 100));
134         }
135         return percent;
136     },
137 
138     /**
139      * release resources from a list
140      * @param resources
141      */
142     releaseResources: function (resources) {
143         if (resources && resources.length > 0) {
144             var sharedTextureCache = cc.TextureCache.getInstance();
145             var sharedEngine = cc.AudioEngine.getInstance();
146             var sharedParser = cc.SAXParser.getInstance();
147             var sharedFileUtils = cc.FileUtils.getInstance();
148 
149             var resInfo;
150             for (var i = 0; i < resources.length; i++) {
151                 resInfo = resources[i];
152                 var type = this._getResType(resInfo);
153                 switch (type) {
154                     case "IMAGE":
155                         sharedTextureCache.removeTextureForKey(resInfo.src);
156                         break;
157                     case "SOUND":
158                         sharedEngine.unloadEffect(resInfo.src);
159                         break;
160                     case "XML":
161                         sharedParser.unloadPlist(resInfo.src);
162                         break;
163                     case "BINARY":
164                         sharedFileUtils.unloadBinaryFileData(resInfo.src);
165                         break;
166                     case "TEXT":
167                         sharedFileUtils.unloadTextFileData(resInfo.src);
168                         break;
169                     case "FONT":
170                         this._unregisterFaceFont(resInfo);
171                         break;
172                     default:
173                         throw "cocos2d:unknown filename extension: " + type;
174                         break;
175                 }
176             }
177         }
178     },
179 
180     _preload: function () {
181         this._updatePercent();
182         if (this._isAsync) {
183             var frameRate = cc.Director.getInstance()._frameRate;
184             if (frameRate != null && frameRate < 20) {
185                 cc.log("cocos2d: frame rate less than 20 fps, skip frame.");
186                 return;
187             }
188         }
189 
190         if (this._curNumber < this._totalNumber) {
191             this._loadOneResource();
192             this._curNumber++;
193         }
194     },
195 
196     _loadOneResource: function () {
197         var sharedTextureCache = cc.TextureCache.getInstance();
198         var sharedEngine = cc.AudioEngine.getInstance();
199         var sharedParser = cc.SAXParser.getInstance();
200         var sharedFileUtils = cc.FileUtils.getInstance();
201 
202         var resInfo = this._resouces[this._curNumber];
203         var type = this._getResType(resInfo);
204         switch (type) {
205             case "IMAGE":
206                 sharedTextureCache.addImage(resInfo.src);
207                 break;
208             case "SOUND":
209                 sharedEngine.preloadSound(resInfo.src);
210                 break;
211             case "XML":
212                 sharedParser.preloadPlist(resInfo.src);
213                 break;
214             case "BINARY":
215                 sharedFileUtils.preloadBinaryFileData(resInfo.src);
216                 break;
217             case "TEXT" :
218                 sharedFileUtils.preloadTextFileData(resInfo.src);
219                 break;
220             case "FONT":
221                 this._registerFaceFont(resInfo);
222                 break;
223             default:
224                 throw "cocos2d:unknown filename extension: " + type;
225                 break;
226         }
227     },
228 
229     _schedulePreload: function () {
230         var _self = this;
231         this._interval = setInterval(function () {
232             _self._preload();
233         }, this._animationInterval * 1000);
234     },
235 
236     _unschedulePreload: function () {
237         clearInterval(this._interval);
238     },
239 
240     _getResType: function (resInfo) {
241         var isFont = resInfo.fontName;
242         if (isFont != null) {
243             return cc.RESOURCE_TYPE["FONT"];
244         } else {
245             var src = resInfo.src;
246             var ext = src.substring(src.lastIndexOf(".") + 1, src.length);
247 
248             var index = ext.indexOf("?");
249             if(index > 0) ext = ext.substring(0, index);
250 
251             for (var resType in cc.RESOURCE_TYPE) {
252                 if (cc.RESOURCE_TYPE[resType].indexOf(ext) != -1) {
253                     return resType;
254                 }
255             }
256             return ext;
257         }
258     },
259 
260     _updatePercent: function () {
261         var percent = this.getPercentage();
262 
263         if (percent >= 100) {
264             this._unschedulePreload();
265             this._complete();
266         }
267     },
268 
269     _complete: function () {
270         if (this._target && (typeof(this._selector) == "string")) {
271             this._target[this._selector](this);
272         } else if (this._target && (typeof(this._selector) == "function")) {
273             this._selector.call(this._target, this);
274         } else {
275             this._selector(this);
276         }
277 
278         this._curNumber = 0;
279         this._loadedNumber = 0;
280         this._totalNumber = 0;
281     },
282 
283     _registerFaceFont: function (fontRes) {
284         var srcArr = fontRes.src;
285         var fileUtils = cc.FileUtils.getInstance();
286         if (srcArr && srcArr.length > 0) {
287             var fontStyle = document.createElement("style");
288             fontStyle.type = "text/css";
289             document.body.appendChild(fontStyle);
290 
291             var fontStr = "@font-face { font-family:" + fontRes.fontName + "; src:";
292             for (var i = 0; i < srcArr.length; i++) {
293                 fontStr += "url('" + fileUtils.fullPathForFilename(encodeURI(srcArr[i].src)) + "') format('" + srcArr[i].type + "')";
294                 fontStr += (i == (srcArr.length - 1)) ? ";" : ",";
295             }
296             fontStyle.textContent += fontStr + "};";
297 
298             //preload
299             //<div style="font-family: PressStart;">.</div>
300             var preloadDiv = document.createElement("div");
301             preloadDiv.style.fontFamily = fontRes.fontName;
302             preloadDiv.innerHTML = ".";
303             preloadDiv.style.position = "absolute";
304             preloadDiv.style.left = "-100px";
305             preloadDiv.style.top = "-100px";
306             document.body.appendChild(preloadDiv);
307         }
308         cc.Loader.getInstance().onResLoaded();
309     },
310 
311     _unregisterFaceFont: function (fontRes) {
312         //todo remove style
313     }
314 });
315 
316 /**
317  * Preload resources in the background
318  * @param {Array} resources
319  * @param {Function|String} selector
320  * @param {Object} target
321  * @return {cc.Loader}
322  * @example
323  * //example
324  * var g_mainmenu = [
325  *    {src:"res/hello.png"},
326  *    {src:"res/hello.plist"},
327  *
328  *    {src:"res/logo.png"},
329  *    {src:"res/btn.png"},
330  *
331  *    {src:"res/boom.mp3"},
332  * ]
333  *
334  * var g_level = [
335  *    {src:"res/level01.png"},
336  *    {src:"res/level02.png"},
337  *    {src:"res/level03.png"}
338  * ]
339  *
340  * //load a list of resources
341  * cc.Loader.preload(g_mainmenu, this.startGame, this);
342  *
343  * //load multi lists of resources
344  * cc.Loader.preload([g_mainmenu,g_level], this.startGame, this);
345  */
346 cc.Loader.preload = function (resources, selector, target) {
347     if (!this._instance) {
348         this._instance = new cc.Loader();
349     }
350     this._instance.initWithResources(resources, selector, target);
351     return this._instance;
352 };
353 
354 /**
355  * Preload resources async
356  * @param {Array} resources
357  * @param {Function|String} selector
358  * @param {Object} target
359  * @return {cc.Loader}
360  */
361 cc.Loader.preloadAsync = function (resources, selector, target) {
362     if (!this._instance) {
363         this._instance = new cc.Loader();
364     }
365     this._instance.setAsync(true);
366     this._instance.initWithResources(resources, selector, target);
367     return this._instance;
368 };
369 
370 /**
371  * Release the resources from a list
372  * @param {Array} resources
373  */
374 cc.Loader.purgeCachedData = function (resources) {
375     if (this._instance) {
376         this._instance.releaseResources(resources);
377     }
378 };
379 
380 /**
381  * Returns a shared instance of the loader
382  * @function
383  * @return {cc.Loader}
384  */
385 cc.Loader.getInstance = function () {
386     if (!this._instance) {
387         this._instance = new cc.Loader();
388     }
389     return this._instance;
390 };
391 
392 cc.Loader._instance = null;
393 
394 
395 /**
396  * Used to display the loading screen
397  * @class
398  * @extends cc.Scene
399  */
400 cc.LoaderScene = cc.Scene.extend(/** @lends cc.LoaderScene# */{
401     _logo: null,
402     _logoTexture: null,
403     _texture2d: null,
404     _bgLayer: null,
405     _label: null,
406     _winSize:null,
407 
408     /**
409      * Constructor
410      */
411     ctor: function () {
412         cc.Scene.prototype.ctor.call(this);
413         this._winSize = cc.Director.getInstance().getWinSize();
414     },
415     init:function(){
416         cc.Scene.prototype.init.call(this);
417 
418         //logo
419         var logoWidth = 160;
420         var logoHeight = 200;
421         var centerPos = cc.p(this._winSize.width / 2, this._winSize.height / 2);
422 
423         this._logoTexture = new Image();
424         var _this = this;
425         this._logoTexture.addEventListener("load", function () {
426             _this._initStage(centerPos);
427             this.removeEventListener('load', arguments.callee, false);
428         });
429         this._logoTexture.src = "";
430         this._logoTexture.width = logoWidth;
431         this._logoTexture.height = logoHeight;
432 
433         // bg
434         this._bgLayer = cc.LayerColor.create(cc.c4(32, 32, 32, 255));
435         this._bgLayer.setPosition(0, 0);
436         this.addChild(this._bgLayer, 0);
437 
438         //loading percent
439         this._label = cc.LabelTTF.create("Loading... 0%", "Arial", 14);
440         this._label.setColor(cc.c3(180, 180, 180));
441         this._label.setOpacity(0);
442         this._label.setPosition(cc.pAdd(centerPos, cc.p(0, -logoHeight / 2 - 10)));
443         this._bgLayer.addChild(this._label, 10);
444     },
445 
446     _initStage: function (centerPos) {
447         this._texture2d = new cc.Texture2D();
448         this._texture2d.initWithElement(this._logoTexture);
449         this._texture2d.handleLoadedTexture();
450         this._logo = cc.Sprite.createWithTexture(this._texture2d);
451 
452         this._logo.setPosition(centerPos);
453         this._bgLayer.addChild(this._logo, 10);
454 
455         //load resources
456         this._logoFadeIn();
457     },
458 
459     onEnter: function () {
460         cc.Node.prototype.onEnter.call(this);
461         this.schedule(this._startLoading, 0.3);
462     },
463 
464     onExit: function () {
465         cc.Node.prototype.onExit.call(this);
466         var tmpStr = "Loading... 0%";
467         this._label.setString(tmpStr);
468     },
469 
470     /**
471      * init with resources
472      * @param {Array} resources
473      * @param {Function|String} selector
474      * @param {Object} target
475      */
476     initWithResources: function (resources, selector, target) {
477         this.resources = resources;
478         this.selector = selector;
479         this.target = target;
480     },
481 
482     _startLoading: function () {
483         this.unschedule(this._startLoading);
484         cc.Loader.preload(this.resources, this.selector, this.target);
485         this.schedule(this._updatePercent);
486     },
487 
488     _logoFadeIn: function () {
489         var logoAction = cc.Spawn.create(
490             cc.EaseBounce.create(cc.MoveBy.create(0.25, cc.p(0, 10))),
491             cc.FadeIn.create(0.5));
492 
493         var labelAction = cc.Sequence.create(
494             cc.DelayTime.create(0.15),
495             logoAction.clone());
496 
497         this._logo.runAction(logoAction);
498         this._label.runAction(labelAction);
499     },
500 
501     _updatePercent: function () {
502         var percent = cc.Loader.getInstance().getPercentage();
503         var tmpStr = "Loading... " + percent + "%";
504         this._label.setString(tmpStr);
505 
506         if (percent >= 100)
507             this.unschedule(this._updatePercent);
508     }
509 });
510 
511 /**
512  * Preload multi scene resources.
513  * @param {Array} resources
514  * @param {Function|String} selector
515  * @param {Object} target
516  * @return {cc.LoaderScene}
517  * @example
518  * //example
519  * var g_mainmenu = [
520  *    {src:"res/hello.png"},
521  *    {src:"res/hello.plist"},
522  *
523  *    {src:"res/logo.png"},
524  *    {src:"res/btn.png"},
525  *
526  *    {src:"res/boom.mp3"},
527  * ]
528  *
529  * var g_level = [
530  *    {src:"res/level01.png"},
531  *    {src:"res/level02.png"},
532  *    {src:"res/level03.png"}
533  * ]
534  *
535  * //load a list of resources
536  * cc.LoaderScene.preload(g_mainmenu, this.startGame, this);
537  *
538  * //load multi lists of resources
539  * cc.LoaderScene.preload([g_mainmenu,g_level], this.startGame, this);
540  */
541 cc.LoaderScene.preload = function (resources, selector, target) {
542     if (!this._instance) {
543         this._instance = new cc.LoaderScene();
544         this._instance.init();
545     }
546 
547     this._instance.initWithResources(resources, selector, target);
548 
549     var director = cc.Director.getInstance();
550     if (director.getRunningScene()) {
551         director.replaceScene(this._instance);
552     } else {
553         director.runWithScene(this._instance);
554     }
555 
556     return this._instance;
557 };
558