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", "ExportJson"],
 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         this._loadedNumber++;
109         cc.log("cocos2d:Failed loading resource: " + name);
110     },
111 
112     /**
113      * Callback when a resource file loaded.
114      * @example
115      * //example
116      * cc.Loader.getInstance().onResLoaded();
117      */
118     onResLoaded: function () {
119         this._loadedNumber++;
120     },
121 
122     /**
123      * Get loading percentage
124      * @return {Number}
125      * @example
126      * //example
127      * cc.log(cc.Loader.getInstance().getPercentage() + "%");
128      */
129     getPercentage: function () {
130         var percent = 0;
131         if (this._totalNumber == 0) {
132             percent = 100;
133         } else {
134             percent = (0 | (this._loadedNumber / this._totalNumber * 100));
135         }
136         return percent;
137     },
138 
139     /**
140      * release resources from a list
141      * @param resources
142      */
143     releaseResources: function (resources) {
144         if (resources && resources.length > 0) {
145             var sharedTextureCache = cc.TextureCache.getInstance();
146             var sharedEngine = cc.AudioEngine ? cc.AudioEngine.getInstance() : null;
147             var sharedParser = cc.SAXParser.getInstance();
148             var sharedFileUtils = cc.FileUtils.getInstance();
149 
150             var resInfo;
151             for (var i = 0; i < resources.length; i++) {
152                 resInfo = resources[i];
153                 var type = this._getResType(resInfo);
154                 switch (type) {
155                     case "IMAGE":
156                         sharedTextureCache.removeTextureForKey(resInfo.src);
157                         break;
158                     case "SOUND":
159                         if(!sharedEngine) throw "Can not find AudioEngine! Install it, please.";
160                         sharedEngine.unloadEffect(resInfo.src);
161                         break;
162                     case "XML":
163                         sharedParser.unloadPlist(resInfo.src);
164                         break;
165                     case "BINARY":
166                         sharedFileUtils.unloadBinaryFileData(resInfo.src);
167                         break;
168                     case "TEXT":
169                         sharedFileUtils.unloadTextFileData(resInfo.src);
170                         break;
171                     case "FONT":
172                         this._unregisterFaceFont(resInfo);
173                         break;
174                     default:
175                         throw "cocos2d:unknown filename extension: " + type;
176                         break;
177                 }
178             }
179         }
180     },
181 
182     _preload: function () {
183         this._updatePercent();
184         if (this._isAsync) {
185             var frameRate = cc.Director.getInstance()._frameRate;
186             if (frameRate != null && frameRate < 20) {
187                 cc.log("cocos2d: frame rate less than 20 fps, skip frame.");
188                 return;
189             }
190         }
191 
192         if (this._curNumber < this._totalNumber) {
193             this._loadOneResource();
194             this._curNumber++;
195         }
196     },
197 
198     _loadOneResource: function () {
199         var sharedTextureCache = cc.TextureCache.getInstance();
200         var sharedEngine = cc.AudioEngine ? cc.AudioEngine.getInstance() : null;
201         var sharedParser = cc.SAXParser.getInstance();
202         var sharedFileUtils = cc.FileUtils.getInstance();
203 
204         var resInfo = this._resouces[this._curNumber];
205         var type = this._getResType(resInfo);
206         switch (type) {
207             case "IMAGE":
208                 sharedTextureCache.addImage(resInfo.src);
209                 break;
210             case "SOUND":
211                 if(!sharedEngine) throw "Can not find AudioEngine! Install it, please.";
212                 sharedEngine.preloadSound(resInfo.src);
213                 break;
214             case "XML":
215                 sharedParser.preloadPlist(resInfo.src);
216                 break;
217             case "BINARY":
218                 sharedFileUtils.preloadBinaryFileData(resInfo.src);
219                 break;
220             case "TEXT" :
221                 sharedFileUtils.preloadTextFileData(resInfo.src);
222                 break;
223             case "FONT":
224                 this._registerFaceFont(resInfo);
225                 break;
226             default:
227                 throw "cocos2d:unknown filename extension: " + type;
228                 break;
229         }
230     },
231 
232     _schedulePreload: function () {
233         var _self = this;
234         this._interval = setInterval(function () {
235             _self._preload();
236         }, this._animationInterval * 1000);
237     },
238 
239     _unschedulePreload: function () {
240         clearInterval(this._interval);
241     },
242 
243     _getResType: function (resInfo) {
244         var isFont = resInfo.fontName;
245         if (isFont != null) {
246             return cc.RESOURCE_TYPE["FONT"];
247         } else {
248             var src = resInfo.src;
249             var ext = src.substring(src.lastIndexOf(".") + 1, src.length);
250 
251             var index = ext.indexOf("?");
252             if(index > 0) ext = ext.substring(0, index);
253 
254             for (var resType in cc.RESOURCE_TYPE) {
255                 if (cc.RESOURCE_TYPE[resType].indexOf(ext) != -1) {
256                     return resType;
257                 }
258             }
259             return ext;
260         }
261     },
262 
263     _updatePercent: function () {
264         var percent = this.getPercentage();
265 
266         if (percent >= 100) {
267             this._unschedulePreload();
268             this._complete();
269         }
270     },
271 
272     _complete: function () {
273         if (this._target && (typeof(this._selector) == "string")) {
274             this._target[this._selector](this);
275         } else if (this._target && (typeof(this._selector) == "function")) {
276             this._selector.call(this._target, this);
277         } else {
278             this._selector(this);
279         }
280 
281         this._curNumber = 0;
282         this._loadedNumber = 0;
283         this._totalNumber = 0;
284     },
285 
286     _registerFaceFont: function (fontRes) {
287         var srcArr = fontRes.src;
288         var fileUtils = cc.FileUtils.getInstance();
289         if (srcArr && srcArr.length > 0) {
290             var fontStyle = document.createElement("style");
291             fontStyle.type = "text/css";
292             document.body.appendChild(fontStyle);
293 
294             var fontStr = "@font-face { font-family:" + fontRes.fontName + "; src:";
295             for (var i = 0; i < srcArr.length; i++) {
296                 fontStr += "url('" + fileUtils.fullPathForFilename(encodeURI(srcArr[i].src)) + "') format('" + srcArr[i].type + "')";
297                 fontStr += (i == (srcArr.length - 1)) ? ";" : ",";
298             }
299             fontStyle.textContent += fontStr + "};";
300 
301             //preload
302             //<div style="font-family: PressStart;">.</div>
303             var preloadDiv = document.createElement("div");
304             preloadDiv.style.fontFamily = fontRes.fontName;
305             preloadDiv.innerHTML = ".";
306             preloadDiv.style.position = "absolute";
307             preloadDiv.style.left = "-100px";
308             preloadDiv.style.top = "-100px";
309             document.body.appendChild(preloadDiv);
310         }
311         cc.Loader.getInstance().onResLoaded();
312     },
313 
314     _unregisterFaceFont: function (fontRes) {
315         //todo remove style
316     }
317 });
318 
319 /**
320  * Preload resources in the background
321  * @param {Array} resources
322  * @param {Function|String} selector
323  * @param {Object} target
324  * @return {cc.Loader}
325  * @example
326  * //example
327  * var g_mainmenu = [
328  *    {src:"res/hello.png"},
329  *    {src:"res/hello.plist"},
330  *
331  *    {src:"res/logo.png"},
332  *    {src:"res/btn.png"},
333  *
334  *    {src:"res/boom.mp3"},
335  * ]
336  *
337  * var g_level = [
338  *    {src:"res/level01.png"},
339  *    {src:"res/level02.png"},
340  *    {src:"res/level03.png"}
341  * ]
342  *
343  * //load a list of resources
344  * cc.Loader.preload(g_mainmenu, this.startGame, this);
345  *
346  * //load multi lists of resources
347  * cc.Loader.preload([g_mainmenu,g_level], this.startGame, this);
348  */
349 cc.Loader.preload = function (resources, selector, target) {
350     if (!this._instance) {
351         this._instance = new cc.Loader();
352     }
353     this._instance.initWithResources(resources, selector, target);
354     return this._instance;
355 };
356 
357 /**
358  * Preload resources async
359  * @param {Array} resources
360  * @param {Function|String} selector
361  * @param {Object} target
362  * @return {cc.Loader}
363  */
364 cc.Loader.preloadAsync = function (resources, selector, target) {
365     if (!this._instance) {
366         this._instance = new cc.Loader();
367     }
368     this._instance.setAsync(true);
369     this._instance.initWithResources(resources, selector, target);
370     return this._instance;
371 };
372 
373 /**
374  * Release the resources from a list
375  * @param {Array} resources
376  */
377 cc.Loader.purgeCachedData = function (resources) {
378     if (this._instance) {
379         this._instance.releaseResources(resources);
380     }
381 };
382 
383 /**
384  * Returns a shared instance of the loader
385  * @function
386  * @return {cc.Loader}
387  */
388 cc.Loader.getInstance = function () {
389     if (!this._instance) {
390         this._instance = new cc.Loader();
391     }
392     return this._instance;
393 };
394 
395 cc.Loader._instance = null;
396 
397 
398 /**
399  * Used to display the loading screen
400  * @class
401  * @extends cc.Scene
402  */
403 cc.LoaderScene = cc.Scene.extend(/** @lends cc.LoaderScene# */{
404     _logo: null,
405     _logoTexture: null,
406     _texture2d: null,
407     _bgLayer: null,
408     _label: null,
409     _winSize:null,
410 
411     /**
412      * Constructor
413      */
414     ctor: function () {
415         cc.Scene.prototype.ctor.call(this);
416         this._winSize = cc.Director.getInstance().getWinSize();
417     },
418     init:function(){
419         cc.Scene.prototype.init.call(this);
420 
421         //logo
422         var logoWidth = 160;
423         var logoHeight = 200;
424         var centerPos = cc.p(this._winSize.width / 2, this._winSize.height / 2);
425 
426         this._logoTexture = new Image();
427         var _this = this;
428         this._logoTexture.addEventListener("load", function () {
429             _this._initStage(centerPos);
430             this.removeEventListener('load', arguments.callee, false);
431         });
432         this._logoTexture.src = "";
433         this._logoTexture.width = logoWidth;
434         this._logoTexture.height = logoHeight;
435 
436         // bg
437         this._bgLayer = cc.LayerColor.create(cc.c4(32, 32, 32, 255));
438         this._bgLayer.setPosition(0, 0);
439         this.addChild(this._bgLayer, 0);
440 
441         //loading percent
442         this._label = cc.LabelTTF.create("Loading... 0%", "Arial", 14);
443         this._label.setColor(cc.c3(180, 180, 180));
444         this._label.setPosition(cc.pAdd(centerPos, cc.p(0, -logoHeight / 2 - 10)));
445         this._bgLayer.addChild(this._label, 10);
446     },
447 
448     _initStage: function (centerPos) {
449         this._texture2d = new cc.Texture2D();
450         this._texture2d.initWithElement(this._logoTexture);
451         this._texture2d.handleLoadedTexture();
452         this._logo = cc.Sprite.createWithTexture(this._texture2d);
453         this._logo.setScale(cc.CONTENT_SCALE_FACTOR());
454         this._logo.setPosition(centerPos);
455         this._bgLayer.addChild(this._logo, 10);
456     },
457 
458     onEnter: function () {
459         cc.Node.prototype.onEnter.call(this);
460         this.schedule(this._startLoading, 0.3);
461     },
462 
463     onExit: function () {
464         cc.Node.prototype.onExit.call(this);
465         var tmpStr = "Loading... 0%";
466         this._label.setString(tmpStr);
467     },
468 
469     /**
470      * init with resources
471      * @param {Array} resources
472      * @param {Function|String} selector
473      * @param {Object} target
474      */
475     initWithResources: function (resources, selector, target) {
476         this.resources = resources;
477         this.selector = selector;
478         this.target = target;
479     },
480 
481     _startLoading: function () {
482         this.unschedule(this._startLoading);
483         cc.Loader.preload(this.resources, this.selector, this.target);
484         this.schedule(this._updatePercent);
485     },
486 
487     _updatePercent: function () {
488         var percent = cc.Loader.getInstance().getPercentage();
489         var tmpStr = "Loading... " + percent + "%";
490         this._label.setString(tmpStr);
491 
492         if (percent >= 100)
493             this.unschedule(this._updatePercent);
494     }
495 });
496 
497 /**
498  * Preload multi scene resources.
499  * @param {Array} resources
500  * @param {Function|String} selector
501  * @param {Object} target
502  * @return {cc.LoaderScene}
503  * @example
504  * //example
505  * var g_mainmenu = [
506  *    {src:"res/hello.png"},
507  *    {src:"res/hello.plist"},
508  *
509  *    {src:"res/logo.png"},
510  *    {src:"res/btn.png"},
511  *
512  *    {src:"res/boom.mp3"},
513  * ]
514  *
515  * var g_level = [
516  *    {src:"res/level01.png"},
517  *    {src:"res/level02.png"},
518  *    {src:"res/level03.png"}
519  * ]
520  *
521  * //load a list of resources
522  * cc.LoaderScene.preload(g_mainmenu, this.startGame, this);
523  *
524  * //load multi lists of resources
525  * cc.LoaderScene.preload([g_mainmenu,g_level], this.startGame, this);
526  */
527 cc.LoaderScene.preload = function (resources, selector, target) {
528     if (!this._instance) {
529         this._instance = new cc.LoaderScene();
530         this._instance.init();
531     }
532 
533     this._instance.initWithResources(resources, selector, target);
534 
535     var director = cc.Director.getInstance();
536     if (director.getRunningScene()) {
537         director.replaceScene(this._instance);
538     } else {
539         director.runWithScene(this._instance);
540     }
541 
542     return this._instance;
543 };
544