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