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