PageRenderTime 34ms CodeModel.GetById 14ms RepoModel.GetById 1ms app.codeStats 0ms

/src/com/greensock/loading/core/DisplayObjectLoader.as

https://github.com/playpower/flashEditor
ActionScript | 249 lines | 172 code | 23 blank | 54 comment | 52 complexity | fbff0b8c6f82ca2ec70662a672b3b297 MD5 | raw file
  1. /**
  2. * VERSION: 1.83
  3. * DATE: 2011-02-15
  4. * AS3
  5. * UPDATES AND DOCS AT: http://www.greensock.com/loadermax/
  6. **/
  7. package com.greensock.loading.core {
  8. import com.greensock.events.LoaderEvent;
  9. import com.greensock.loading.LoaderMax;
  10. import com.greensock.loading.LoaderStatus;
  11. import com.greensock.loading.display.ContentDisplay;
  12. import flash.display.DisplayObject;
  13. import flash.display.Loader;
  14. import flash.display.Sprite;
  15. import flash.events.ErrorEvent;
  16. import flash.events.Event;
  17. import flash.events.ProgressEvent;
  18. import flash.net.LocalConnection;
  19. import flash.system.ApplicationDomain;
  20. import flash.system.Capabilities;
  21. import flash.system.LoaderContext;
  22. import flash.system.Security;
  23. import flash.system.SecurityDomain;
  24. /**
  25. * Serves as the base class for SWFLoader and ImageLoader. There is no reason to use this class on its own.
  26. * Please refer to the documentation for the other classes.
  27. * <br /><br />
  28. *
  29. * <b>Copyright 2011, GreenSock. All rights reserved.</b> This work is subject to the terms in <a href="http://www.greensock.com/terms_of_use.html">http://www.greensock.com/terms_of_use.html</a> or for corporate Club GreenSock members, the software agreement that was issued with the corporate membership.
  30. *
  31. * @author Jack Doyle, jack@greensock.com
  32. */
  33. public class DisplayObjectLoader extends LoaderItem {
  34. /** @private the Sprite to which the EVENT_LISTENER was attached for forcing garbage collection after 1 frame (improves performance especially when multiple loaders are disposed at one time). **/
  35. protected static var _gcDispatcher:Sprite;
  36. /** @private **/
  37. protected static var _gcCycles:uint = 0;
  38. /** @private **/
  39. protected var _loader:Loader;
  40. /** @private **/
  41. protected var _sprite:Sprite;
  42. /** @private **/
  43. protected var _context:LoaderContext;
  44. /** @private **/
  45. protected var _initted:Boolean;
  46. /** @private used by SWFLoader when the loader is canceled before the SWF ever had a chance to init which causes garbage collection issues. We slip into stealthMode at that point, wait for it to init, and then cancel the _loader's loading.**/
  47. protected var _stealthMode:Boolean;
  48. /**
  49. * Constructor
  50. *
  51. * @param urlOrRequest The url (<code>String</code>) or <code>URLRequest</code> from which the loader should get its content
  52. * @param vars An object containing optional parameters like <code>estimatedBytes, name, autoDispose, onComplete, onProgress, onError</code>, etc. For example, <code>{estimatedBytes:2400, name:"myImage1", onComplete:completeHandler}</code>.
  53. */
  54. public function DisplayObjectLoader(urlOrRequest:*, vars:Object=null) {
  55. super(urlOrRequest, vars);
  56. _refreshLoader(false);
  57. if (LoaderMax.contentDisplayClass is Class) {
  58. _sprite = new LoaderMax.contentDisplayClass(this);
  59. if (!_sprite.hasOwnProperty("rawContent")) {
  60. throw new Error("LoaderMax.contentDisplayClass must be set to a class with a 'rawContent' property, like com.greensock.loading.display.ContentDisplay");
  61. }
  62. } else {
  63. _sprite = new ContentDisplay(this);
  64. }
  65. }
  66. /** @private Set inside ContentDisplay's or FlexContentDisplay's "loader" setter. **/
  67. public function setContentDisplay(contentDisplay:Sprite):void {
  68. _sprite = contentDisplay;
  69. }
  70. /** @private **/
  71. override protected function _load():void {
  72. _prepRequest();
  73. if (this.vars.context is LoaderContext) {
  74. _context = this.vars.context;
  75. } else if (_context == null) {
  76. if (LoaderMax.defaultContext != null) {
  77. _context = LoaderMax.defaultContext;
  78. if (_isLocal) {
  79. _context.securityDomain = null;
  80. }
  81. } else if (!_isLocal) {
  82. _context = new LoaderContext(true, new ApplicationDomain(ApplicationDomain.currentDomain), SecurityDomain.currentDomain); //avoids some security sandbox headaches that plague many users.
  83. }
  84. }
  85. if (Capabilities.playerType != "Desktop") { //AIR apps will choke on Security.allowDomain()
  86. Security.allowDomain(_url);
  87. }
  88. _loader.load(_request, _context);
  89. }
  90. /** @private **/
  91. protected function _refreshLoader(unloadContent:Boolean=true):void {
  92. if (_loader != null) {
  93. //to avoid gc issues and get around a bug in Flash that incorrectly reports progress values on Loaders that were closed before completing, we must force gc and recreate the Loader altogether...
  94. if (_status == LoaderStatus.LOADING) {
  95. try {
  96. _loader.close();
  97. } catch (error:Error) {
  98. }
  99. }
  100. _loader.contentLoaderInfo.removeEventListener(ProgressEvent.PROGRESS, _progressHandler);
  101. _loader.contentLoaderInfo.removeEventListener(Event.COMPLETE, _completeHandler);
  102. _loader.contentLoaderInfo.removeEventListener("ioError", _failHandler);
  103. _loader.contentLoaderInfo.removeEventListener("securityError", _securityErrorHandler);
  104. _loader.contentLoaderInfo.removeEventListener("httpStatus", _httpStatusHandler);
  105. _loader.contentLoaderInfo.removeEventListener(Event.INIT, _initHandler);
  106. if (unloadContent) {
  107. try {
  108. if (_loader.parent == null && _sprite != null) {
  109. _sprite.addChild(_loader); //adding the _loader to the display list BEFORE calling unloadAndStop() and then removing it will greatly improve its ability to gc correctly if event listeners were added to the stage from within a subloaded swf without specifying "true" for the weak parameter of addEventListener(). The order here is critical.
  110. }
  111. if (_loader.hasOwnProperty("unloadAndStop")) { //Flash Player 10 and later only
  112. (_loader as Object).unloadAndStop();
  113. } else {
  114. _loader.unload();
  115. }
  116. } catch (error:Error) {
  117. }
  118. if (_loader.parent) {
  119. _loader.parent.removeChild(_loader);
  120. }
  121. }
  122. forceGC((this.hasOwnProperty("getClass")) ? 3 : 1);
  123. }
  124. _initted = false;
  125. _loader = new Loader();
  126. _loader.contentLoaderInfo.addEventListener(ProgressEvent.PROGRESS, _progressHandler, false, 0, true);
  127. _loader.contentLoaderInfo.addEventListener(Event.COMPLETE, _completeHandler, false, 0, true);
  128. _loader.contentLoaderInfo.addEventListener("ioError", _failHandler, false, 0, true);
  129. _loader.contentLoaderInfo.addEventListener("securityError", _securityErrorHandler, false, 0, true);
  130. _loader.contentLoaderInfo.addEventListener("httpStatus", _httpStatusHandler, false, 0, true);
  131. _loader.contentLoaderInfo.addEventListener(Event.INIT, _initHandler, false, 0, true);
  132. }
  133. /** @private works around bug in Flash Player that prevents SWFs from properly being garbage collected after being unloaded - for certain types of objects like swfs, this needs to be run more than once (spread out over several frames) to force Flash to properly garbage collect everything. **/
  134. public static function forceGC(cycles:uint=1):void {
  135. if (_gcCycles < cycles) {
  136. _gcCycles = cycles;
  137. if (_gcDispatcher == null) {
  138. _gcDispatcher = new Sprite();
  139. _gcDispatcher.addEventListener(Event.ENTER_FRAME, _forceGCHandler, false, 0, true);
  140. }
  141. }
  142. }
  143. /** @private **/
  144. protected static function _forceGCHandler(event:Event):void {
  145. if (_gcCycles == 0) {
  146. _gcDispatcher.removeEventListener(Event.ENTER_FRAME, _forceGCHandler);
  147. _gcDispatcher = null;
  148. } else {
  149. _gcCycles--;
  150. }
  151. try {
  152. new LocalConnection().connect("FORCE_GC");
  153. new LocalConnection().connect("FORCE_GC");
  154. } catch (error:Error) {
  155. }
  156. }
  157. /** @private scrubLevel: 0 = cancel, 1 = unload, 2 = dispose, 3 = flush **/
  158. override protected function _dump(scrubLevel:int=0, newStatus:int=LoaderStatus.READY, suppressEvents:Boolean=false):void {
  159. if (!_stealthMode) {
  160. _refreshLoader(Boolean(scrubLevel != 2));
  161. }
  162. if (scrubLevel == 1) { //unload
  163. (_sprite as Object).rawContent = null;
  164. } else if (scrubLevel == 2) { //dispose
  165. (_sprite as Object).loader = null;
  166. } else if (scrubLevel == 3) { //unload and dispose
  167. (_sprite as Object).dispose(false, false); //makes sure the ContentDisplay is removed from its parent as well.
  168. }
  169. super._dump(scrubLevel, newStatus, suppressEvents);
  170. }
  171. /** @private **/
  172. protected function _determineScriptAccess():void {
  173. if (!_scriptAccessDenied) {
  174. if (!_loader.contentLoaderInfo.childAllowsParent) {
  175. _scriptAccessDenied = true;
  176. dispatchEvent(new LoaderEvent(LoaderEvent.SCRIPT_ACCESS_DENIED, this, "Error #2123: Security sandbox violation: " + this + ". No policy files granted access."));
  177. }
  178. }
  179. }
  180. //---- EVENT HANDLERS ------------------------------------------------------------------------------------
  181. /** @private **/
  182. protected function _securityErrorHandler(event:ErrorEvent):void {
  183. //If a security error is thrown because of a missing crossdomain.xml file for example and the user didn't define a specific LoaderContext, we'll try again without checking the policy file, accepting the restrictions that come along with it because typically people would rather have the content show up on the screen rather than just error out (and they can always check the scriptAccessDenied property if they need to figure out whether it's safe to do BitmapData stuff on it, etc.)
  184. if (_context != null && _context.checkPolicyFile && !(this.vars.context is LoaderContext)) {
  185. _context = new LoaderContext(false);
  186. _scriptAccessDenied = true;
  187. dispatchEvent(new LoaderEvent(LoaderEvent.SCRIPT_ACCESS_DENIED, this, event.text));
  188. _errorHandler(event);
  189. _load();
  190. } else {
  191. _failHandler(event);
  192. }
  193. }
  194. /** @private **/
  195. protected function _initHandler(event:Event):void {
  196. if (!_initted) {
  197. _initted = true;
  198. if (_content == null) { //_content is set in ImageLoader or SWFLoader (subclasses), but we put this here just in case someone wants to use DisplayObjectLoader on its own as a lighter weight alternative without the bells & whistles of SWFLoader/ImageLoader.
  199. _content = (_scriptAccessDenied) ? _loader : _loader.content;
  200. }
  201. (_sprite as Object).rawContent = (_content as DisplayObject);
  202. dispatchEvent(new LoaderEvent(LoaderEvent.INIT, this));
  203. }
  204. }
  205. //---- GETTERS / SETTERS -------------------------------------------------------------------------
  206. /** A ContentDisplay object (a Sprite) that will contain the remote content as soon as the <code>INIT</code> event has been dispatched. This ContentDisplay can be accessed immediately; you do not need to wait for the content to load. **/
  207. override public function get content():* {
  208. return _sprite;
  209. }
  210. /**
  211. * The raw content that was successfully loaded <strong>into</strong> the <code>content</code> ContentDisplay
  212. * Sprite which varies depending on the type of loader and whether or not script access was denied while
  213. * attempting to load the file:
  214. *
  215. * <ul>
  216. * <li>ImageLoader with script access granted: <code>flash.display.Bitmap</code></li>
  217. * <li>ImageLoader with script access denied: <code>flash.display.Loader</code></li>
  218. * <li>SWFLoader with script access granted: <code>flash.display.DisplayObject</code> (the swf's <code>root</code>)</li>
  219. * <li>SWFLoader with script access denied: <code>flash.display.Loader</code> (the swf's <code>root</code> cannot be accessed because it would generate a security error)</li>
  220. * </ul>
  221. **/
  222. public function get rawContent():* {
  223. return _content;
  224. }
  225. }
  226. }