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 * A class to pre-load resources before engine start game main loop. 29 * @class 30 * @extends cc.Class 31 */ 32 cc.Loader = cc.Class.extend(/** @lends cc.Loader# */{ 33 resourceCount:0, 34 loadedResourceCount:0, 35 timer:0, 36 37 /** 38 * Check the loading status 39 */ 40 isLoadedComplete:function () { 41 var loaderCache = cc.Loader.getInstance(); 42 if (loaderCache.loadedResourceCount == loaderCache.resourceCount) { 43 if (loaderCache.onload) { 44 loaderCache.timer = setTimeout(loaderCache.onload, 16); 45 } else { 46 cc.Assert(0, "cocos2d:no load callback defined"); 47 } 48 } else { 49 if (loaderCache.onloading) { 50 loaderCache.timer = setTimeout(loaderCache.onloading, 16); 51 } 52 else { 53 cc.LoaderScene.getInstance().draw(); 54 } 55 loaderCache.timer = setTimeout(loaderCache.isLoadedComplete, 16); 56 } 57 }, 58 59 /** 60 * Callback when loading resource error 61 * @param {String} name 62 * @example 63 * //example 64 * cc.Loader.getInstance().onResLoadingErr(name); 65 */ 66 onResLoadingErr:function (name) { 67 cc.log("cocos2d:Failed loading resource: " + name); 68 }, 69 70 /** 71 *Callback when a resource file loaded. 72 * @example 73 * //example 74 * cc.Loader.getInstance().onResLoaded(); 75 */ 76 onResLoaded:function () { 77 this.loadedResourceCount++; 78 }, 79 80 /** 81 * For loading percentage 82 * You can use this method to create a custom loading screen. 83 * @return {Number} 84 * @example 85 * //example 86 * cc.log(cc.Loader.getInstance().getProgressBar() + "%"); 87 */ 88 getProgressBar:function () { 89 var per = this.loadedResourceCount / this.resourceCount; 90 per = 0 | (per * 100); 91 return per; 92 }, 93 94 /** 95 * status when resources loading success 96 * @example 97 * //example 98 * cc.Loader.getInstance().onload = function () { 99 * cc.AppController.shareAppController().didFinishLaunchingWithOptions(); 100 * }; 101 */ 102 onload:undefined, 103 104 /** 105 * status when res loading error 106 * @example 107 * //example 108 * cc.Loader.getInstance().onerror = function () { 109 * //do something 110 * }; 111 */ 112 onerror:undefined, 113 114 /** 115 * status when res loading 116 * @example 117 * //example 118 * cc.Loader.getInstance().onloading = function () { 119 * cc.LoaderScene.getInstance().draw(); 120 * }; 121 */ 122 onloading:undefined, 123 124 _registerFaceFont:function (fontRes) { 125 var srcArr = fontRes.srcArr; 126 if (fontRes.srcArr && srcArr.length > 0) { 127 var fontStyle = document.createElement("style"); 128 fontStyle.type = "text/css"; 129 document.body.appendChild(fontStyle); 130 131 var fontStr = "@font-face { font-family:" + fontRes.fontName + "; src:"; 132 for (var i = 0; i < srcArr.length; i++) { 133 fontStr += "url('" + encodeURI(srcArr[i].src) + "') format('" + srcArr[i].type + "')"; 134 fontStr += (i == (srcArr.length - 1)) ? ";" : ","; 135 } 136 fontStyle.textContent += fontStr + "};"; 137 } 138 cc.Loader.getInstance().onResLoaded(); 139 }, 140 141 /** 142 * Pre-load the resources before engine start game main loop. 143 * There will be some error without pre-loading resources. 144 * @param {object} res 145 * @example 146 * //example 147 * var res = [ 148 * {type:"image", src:"hello.png"}, 149 * {type:"tmx", src:"hello.tmx"} 150 * ] 151 * cc.Loader.getInstance().preload(res); 152 */ 153 preload:function (res) { 154 var sharedTextureCache = cc.TextureCache.getInstance(); 155 var sharedEngine = cc.AudioEngine.getInstance(); 156 var sharedParser = cc.SAXParser.getInstance(); 157 var sharedFileUtils = cc.FileUtils.getInstance(); 158 159 this.loadedResourceCount = 0; 160 this.resourceCount = res.length; 161 for (var i = 0; i < res.length; i++) { 162 switch (res[i].type) { 163 case "image": 164 sharedTextureCache.addImage(res[i].src); 165 break; 166 case "sound": 167 sharedEngine.preloadSound(res[i].src); 168 break; 169 case "plist": 170 case "tmx": 171 case "fnt": 172 sharedParser.preloadPlist(res[i].src); 173 break; 174 case "tga": 175 //cc.log("cocos2d:not implemented yet") 176 break; 177 case "ccbi": 178 case "binary": 179 sharedFileUtils.preloadBinaryFileData(res[i].src); 180 break; 181 case "face-font": 182 this._registerFaceFont(res[i]); 183 break; 184 default: 185 throw "cocos2d:unknow type : " + res[i].type; 186 break; 187 } 188 } 189 this.isLoadedComplete(); 190 } 191 }); 192 193 /** 194 * Share Loader 195 * @return {cc.Loader} 196 */ 197 cc.Loader.getInstance = function () { 198 if (!this._instance) { 199 this._instance = new cc.Loader(); 200 } 201 return this._instance; 202 }; 203 cc.Loader._instance = null; 204 205 /** 206 * Default loading screen, you can customize the loading screen. 207 * @class 208 * @extends cc.Class 209 */ 210 cc.LoaderScene = cc.Class.extend(/** @lends cc.LoaderScene# */{ 211 _logo:new Image(), 212 213 /** 214 * Constructor 215 */ 216 ctor:function () { 217 this._logo.src = ""; 218 219 this._logo.width = 160; 220 this._logo.height = 200; 221 }, 222 223 /** 224 * Draw loading screen 225 */ 226 draw:function () { 227 var logoWidth = (cc.canvas.width - this._logo.width) / 2; 228 var logoHeight = (cc.canvas.height - this._logo.height) / 2; 229 cc.renderContext.clearRect(0, -cc.canvas.height, cc.canvas.width, cc.canvas.height); 230 cc.renderContext.fillStyle = "#202020"; 231 cc.renderContext.fillRect(0, -cc.canvas.height, cc.canvas.width, cc.canvas.height); 232 cc.drawingUtil.drawImage(this._logo, cc.p(logoWidth, logoHeight)); 233 cc.renderContext.fillStyle = "#b2b4b3"; 234 cc.renderContext.font = 'Bold 12px Verdana'; 235 cc.renderContext.textAlign = 'left'; 236 cc.drawingUtil.fillText("Loading " + cc.Loader.getInstance().getProgressBar() + "%", logoWidth + 30, logoHeight - 15); 237 } 238 }); 239 240 /** 241 * Shared loader scene 242 * @return {cc.LoaderScene} 243 */ 244 cc.LoaderScene.getInstance = function () { 245 if (!this._instance) { 246 this._instance = new cc.LoaderScene(); 247 } 248 return this._instance; 249 }; 250 251 cc.LoaderScene._instance = null;