PageRenderTime 54ms CodeModel.GetById 21ms RepoModel.GetById 0ms app.codeStats 1ms

/flowplayer/branches/coverImage/src/actionscript/org/flowplayer/view/Launcher.as

http://flowplayer-core.googlecode.com/
ActionScript | 950 lines | 761 code | 139 blank | 50 comment | 145 complexity | 35fa20ba2bab55a353a53ba4fca2e12b MD5 | raw file
Possible License(s): GPL-3.0, AGPL-1.0
  1. /*
  2. * Copyright (c) 2008-2011 Flowplayer Oy *
  3. * This file is part of Flowplayer.
  4. *
  5. * Flowplayer is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * Flowplayer is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with Flowplayer. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. package org.flowplayer.view {
  19. import flash.external.ExternalInterface;
  20. import org.flowplayer.config.Config;
  21. import org.flowplayer.config.ConfigParser;
  22. import org.flowplayer.config.ExternalInterfaceHelper;
  23. import org.flowplayer.config.VersionInfo;
  24. import org.flowplayer.controller.PlayListController;
  25. import org.flowplayer.controller.ResourceLoader;
  26. import org.flowplayer.controller.ResourceLoaderImpl;
  27. import org.flowplayer.flow_internal;
  28. import org.flowplayer.model.Callable;
  29. import org.flowplayer.model.Clip;
  30. import org.flowplayer.model.ClipEvent;
  31. import org.flowplayer.model.ClipEventType;
  32. import org.flowplayer.model.DisplayPluginModel;
  33. import org.flowplayer.model.DisplayProperties;
  34. import org.flowplayer.model.DisplayPropertiesImpl;
  35. import org.flowplayer.model.ErrorCode;
  36. import org.flowplayer.model.EventDispatcher;
  37. import org.flowplayer.model.Loadable;
  38. import org.flowplayer.model.Logo;
  39. import org.flowplayer.model.PlayButtonOverlay;
  40. import org.flowplayer.model.PlayerError;
  41. import org.flowplayer.model.PlayerEvent;
  42. import org.flowplayer.model.Playlist;
  43. import org.flowplayer.model.Plugin;
  44. import org.flowplayer.model.PluginError;
  45. import org.flowplayer.model.PluginEvent;
  46. import org.flowplayer.model.PluginModel;
  47. import org.flowplayer.model.ProviderModel;
  48. import org.flowplayer.model.State;
  49. import org.flowplayer.util.Arrange;
  50. import org.flowplayer.util.Log;
  51. import org.flowplayer.util.TextUtil;
  52. import org.flowplayer.util.URLUtil;
  53. import org.flowplayer.view.Panel;
  54. import org.flowplayer.view.PluginLoader;
  55. import org.flowplayer.view.Screen;
  56. import org.flowplayer.view.KeyboardHandler;
  57. import org.osflash.thunderbolt.Logger;
  58. import flash.display.DisplayObject;
  59. import flash.display.DisplayObjectContainer;
  60. import flash.display.Sprite;
  61. import flash.display.BlendMode;
  62. import flash.events.Event;
  63. import flash.events.MouseEvent;
  64. import flash.events.TimerEvent;
  65. import flash.net.URLRequest;
  66. import flash.net.navigateToURL;
  67. import flash.system.Capabilities;
  68. import flash.system.Security;
  69. import flash.text.TextField;
  70. import flash.text.TextFieldAutoSize;
  71. import flash.utils.*;
  72. CONFIG::FLASH_10_1 {
  73. import flash.media.StageVideo;
  74. }
  75. use namespace flow_internal;
  76. public class Launcher extends StyleableSprite implements ErrorHandler {
  77. private var _panel:Panel;
  78. private var _screen:Screen;
  79. private var _config:Config;
  80. private var _flowplayer:Flowplayer;
  81. private var _pluginRegistry:PluginRegistry;
  82. private var _animationEngine:AnimationEngine;
  83. private var _playButtonOverlay:PlayButtonOverlay;
  84. private var _controlsModel:DisplayPluginModel;
  85. private var _providers:Dictionary = new Dictionary();
  86. private var _canvasLogo:Sprite;
  87. private var _pluginLoader:PluginLoader;
  88. private var _error:TextField;
  89. private var _pluginsInitialized:Number = 0;
  90. private var _enteringFullscreen:Boolean;
  91. private var _copyrightNotice:TextField;
  92. private var _playlistLoader:ResourceLoader;
  93. private var _fullscreenManager:FullscreenManager;
  94. private var _screenArrangeCount:int = 0;
  95. private var _clickCount:int;
  96. private var _clickTimer:Timer = new Timer(200, 1);
  97. private var _clickEvent:MouseEvent;
  98. private var _screenMask:Sprite;
  99. [Frame(factoryClass="org.flowplayer.view.Preloader")]
  100. public function Launcher() {
  101. addEventListener(Event.ADDED_TO_STAGE, function(e:Event):void {
  102. URLUtil.loaderInfo = loaderInfo;
  103. trace("Launcher added to stage");
  104. callAndHandleError(createFlashVarsConfig, PlayerError.INIT_FAILED);
  105. });
  106. super("#canvas", this);
  107. }
  108. private function initPhase1():void {
  109. if (_flowplayer) {
  110. log.debug("already initialized, returning");
  111. return;
  112. }
  113. Log.configure(_config.getLogConfiguration());
  114. trace("created log configuration, tracing enabled? " + Log.traceEnabled)
  115. initCustomClipEvents();
  116. if (_config.playerId) {
  117. Security.allowDomain(URLUtil.pageLocation);
  118. }
  119. loader = createNewLoader();
  120. rootStyle = _config.canvas.style;
  121. stage.addEventListener(Event.RESIZE, onStageResize);
  122. stage.addEventListener(Event.RESIZE, arrangeScreen);
  123. setSize(Arrange.parentWidth, Arrange.parentHeight);
  124. if (! VersionInfo.commercial) {
  125. log.debug("Adding logo to canvas");
  126. createLogoForCanvas();
  127. }
  128. log = new Log(this);
  129. EventDispatcher.playerId = _config.playerId;
  130. log.debug("security sandbox type: " + Security.sandboxType);
  131. log.info(VersionInfo.versionInfo());
  132. trace(VersionInfo.versionInfo());
  133. log.debug("creating Panel");
  134. createPanel();
  135. _pluginRegistry = new PluginRegistry(_panel);
  136. log.debug("Creating animation engine");
  137. createAnimationEngine(_pluginRegistry);
  138. log.debug("creating play button overlay");
  139. createPlayButtonOverlay();
  140. log.debug("creating Flowplayer API");
  141. createFlowplayer();
  142. // keyboard handler must be present for plugins.
  143. //
  144. loadPlaylistFeed();
  145. }
  146. private function initPhase2(event:Event = null):void {
  147. log.info("initPhase2");
  148. _flowplayer.keyboardHandler = new KeyboardHandler(stage, function():Boolean { return enteringFullscreen });
  149. loadPlugins();
  150. }
  151. private function initPhase3(event:Event = null):void {
  152. log.debug("initPhase3, all plugins loaded");
  153. createScreen();
  154. _config.getPlaylist().onBeforeBegin(function(event:ClipEvent):void { hideErrorMessage(); });
  155. if (_playButtonOverlay) {
  156. PlayButtonOverlayView(_playButtonOverlay.getDisplayObject()).playlist = _config.getPlaylist();
  157. }
  158. log.debug("creating PlayListController");
  159. _providers = _pluginLoader.providers;
  160. var playListController:PlayListController = createPlayListController();
  161. addPlayListListeners();
  162. createFullscreenManager(playListController.playlist);
  163. addScreenToPanel();
  164. if (!validateLicenseKey()) {
  165. createLogoForCanvas();
  166. resizeCanvasLogo();
  167. }
  168. log.debug("creating logo");
  169. createLogo();
  170. contextMenu = new ContextMenuBuilder(_config.playerId, _config.contextMenu).build();
  171. log.debug("initializing ExternalInterface");
  172. if (useExternalInterface()) {
  173. _flowplayer.initExternalInterface();
  174. }
  175. log.debug("calling onLoad to plugins");
  176. _pluginRegistry.onLoad(_flowplayer);
  177. if (countPlugins() == 0) {
  178. log.debug("no loadable plugins, calling initPhase4");
  179. initPhase4();
  180. }
  181. }
  182. private function initPhase4(event:Event = null):void {
  183. log.info("initPhase4, all plugins initialized");
  184. log.debug("Adding visible plugins to panel");
  185. addPluginsToPanel(_pluginRegistry);
  186. log.debug("dispatching onLoad");
  187. _flowplayer.dispatchEvent(PlayerEvent.load("player"));
  188. log.debug("starting configured streams");
  189. startStreams();
  190. createScreenMask();
  191. arrangeScreen();
  192. addListeners();
  193. // _controlsModel.onPluginEvent(function(event:PluginEvent):void {
  194. // log.debug("received plugin event " + event.id);
  195. // var model:DisplayPluginModel = event.target as DisplayPluginModel;
  196. // log.debug("controls y-pos now is " + model.getDisplayObject().y);
  197. // });
  198. //
  199. // _controlsModel.onBeforePluginEvent(function(event:PluginEvent):void {
  200. // log.debug("received before plugin event " + event.id);
  201. // var model:DisplayPluginModel = event.target as DisplayPluginModel;
  202. // log.debug("controls y-pos now is " + model.getDisplayObject().y);
  203. // event.preventDefault();
  204. // });
  205. // lookupSlowMotionPlugin(_flowplayer);
  206. }
  207. private function createScreenMask():void {
  208. blendMode = BlendMode.LAYER;
  209. _screenMask = new Sprite();
  210. _screenMask.graphics.beginFill(0xff0000);
  211. _screenMask.graphics.drawRect(0, 0, 1, 1);
  212. _screenMask.blendMode = BlendMode.ERASE;
  213. _screenMask.x = 0;
  214. _screenMask.y = 0;
  215. _screenMask.width = 100;
  216. _screenMask.height = 100;
  217. }
  218. private function resizeCanvasLogo():void {
  219. _canvasLogo.alpha = 1;
  220. _canvasLogo.width = 150;
  221. _canvasLogo.scaleY = _canvasLogo.scaleX;
  222. arrangeCanvasLogo();
  223. }
  224. private function useExternalInterface():Boolean {
  225. log.debug("useExternalInteface: " + (_config.playerId != null));
  226. return _config.playerId != null;
  227. }
  228. private function onStageResize(event:Event = null):void {
  229. setSize(Arrange.parentWidth, Arrange.parentHeight);
  230. arrangeCanvasLogo();
  231. }
  232. private function arrangeCanvasLogo():void {
  233. if (!_canvasLogo) return;
  234. _canvasLogo.x = 15;
  235. _canvasLogo.y = Arrange.parentHeight - (_controlsModel ? _controlsModel.dimensions.height.toPx(Arrange.parentHeight) + 10 : 10) - _canvasLogo.height - _copyrightNotice.height;
  236. _copyrightNotice.x = 12;
  237. _copyrightNotice.y = _canvasLogo.y + _canvasLogo.height;
  238. }
  239. private function loadPlugins():void {
  240. var plugins:Array = _config.getLoadables();
  241. log.debug("will load following plugins: ");
  242. logPluginInfo(plugins);
  243. _pluginLoader = new PluginLoader(URLUtil.playerBaseUrl, _pluginRegistry, this, useExternalInterface());
  244. _pluginLoader.addEventListener(Event.COMPLETE, pluginLoadListener);
  245. _flowplayer.pluginLoader = _pluginLoader;
  246. if (plugins.length == 0) {
  247. log.debug("configuration has no plugins");
  248. initPhase3();
  249. } else {
  250. // _builtInPlugins = _config.createLoadables(BuiltInConfig.config.plugins);
  251. // log.debug("following built-in plugins will be instantiated");
  252. // trace("builtIn plugins: ");
  253. // logPluginInfo(_builtInPlugins, true);
  254. _pluginLoader.load(plugins, onPluginLoad, onPluginLoadError);
  255. }
  256. }
  257. private function logPluginInfo(plugins:Array, doTrace:Boolean = false):void {
  258. for (var i:Number = 0; i < plugins.length; i++) {
  259. log.info("" + plugins[i]);
  260. if (doTrace) {
  261. trace("" + plugins[i]);
  262. }
  263. }
  264. }
  265. private function pluginLoadListener(event:Event = null):void {
  266. _pluginLoader.removeEventListener(Event.COMPLETE, pluginLoadListener);
  267. callAndHandleError(initPhase3, PlayerError.INIT_FAILED);
  268. }
  269. private function loadPlaylistFeed():void {
  270. var playlistFeed:String = _config.playlistFeed;
  271. if (! playlistFeed) {
  272. callAndHandleError(initPhase2, PlayerError.INIT_FAILED);
  273. return;
  274. }
  275. log.info("loading playlist from " + playlistFeed);
  276. _playlistLoader = _flowplayer.createLoader();
  277. _playlistLoader.addTextResourceUrl(playlistFeed);
  278. _playlistLoader.load(null,
  279. function(loader:ResourceLoader):void {
  280. log.info("received playlist feed");
  281. _config.playlistDocument = loader.getContent() as String;
  282. _config.getPlaylist().dispatchPlaylistReplace();
  283. callAndHandleError(initPhase2, PlayerError.INIT_FAILED);
  284. });
  285. }
  286. private function onPluginLoad(event:PluginEvent):void {
  287. var plugin:PluginModel = event.target as PluginModel;
  288. log.info("plugin " + plugin + " initialized");
  289. checkPluginsInitialized();
  290. }
  291. private function onPluginLoadError(event:PluginEvent):void {
  292. if (event.target is Loadable) {
  293. handleError(PlayerError.PLUGIN_LOAD_FAILED, "unable to load plugin '" + Loadable(event.target).name + "', url: '" + Loadable(event.target).url + "'");
  294. // throw new Error("unable to load plugin '" + Loadable(event.target).name + "', url: '" + Loadable(event.target).url + "'");
  295. } else {
  296. var plugin:PluginModel = event.target as PluginModel;
  297. _pluginRegistry.removePlugin(plugin);
  298. handleError(PlayerError.PLUGIN_LOAD_FAILED, "load/init error on " + plugin);
  299. }
  300. }
  301. private function checkPluginsInitialized():void {
  302. var numPlugins:int = countPlugins();
  303. if (++_pluginsInitialized == numPlugins) {
  304. log.info("all plugins initialized");
  305. callAndHandleError(initPhase4, PlayerError.INIT_FAILED);
  306. }
  307. log.info(_pluginsInitialized + " out of " + numPlugins + " plugins initialized");
  308. }
  309. private function countPlugins():int {
  310. var count:Number = 0;
  311. var loadables:Array = _config.getLoadables();
  312. for (var i:Number = 0; i < loadables.length; i++) {
  313. var plugin:PluginModel = Loadable(loadables[i]).plugin;
  314. if (! plugin) {
  315. handleError(PlayerError.PLUGIN_LOAD_FAILED, "Unable to load plugin, url " + Loadable(loadables[i]).url + ", name " + Loadable(loadables[i]).name);
  316. // throw new Error("Plugin " + loadables[i] + " not available");
  317. }
  318. else
  319. {
  320. var isNonAdHocPlugin:Boolean = plugin.pluginObject is Plugin;
  321. // var isNonAdHocPlugin:Boolean = (plugin is DisplayPluginModel && DisplayPluginModel(plugin).getDisplayObject() is Plugin) ||
  322. // plugin is ProviderModel && ProviderModel(plugin).pluginObject is Plugin;
  323. if (Loadable(loadables[i]).loadFailed) {
  324. log.debug("load failed for " + loadables[i]);
  325. count++;
  326. } else if (! plugin) {
  327. log.debug("this plugin is not loaded yet");
  328. count++;
  329. } else if (isNonAdHocPlugin) {
  330. log.debug("will wait for onLoad from plugin " + plugin);
  331. count++;
  332. } else {
  333. log.debug("will NOT wait for onLoad from plugin " + Loadable(loadables[i]).plugin);
  334. }
  335. }
  336. }
  337. // +1 comes from the playbuttonoverlay
  338. return count + (_playButtonOverlay ? 1 : 0);
  339. }
  340. private function validateLicenseKey():Boolean {
  341. try {
  342. return LicenseKey.validate(root.loaderInfo.url, _flowplayer.version, _config.licenseKey, useExternalInterface());
  343. } catch (e:Error) {
  344. log.warn("License key not accepted, will show flowplayer logo");
  345. }
  346. return false;
  347. }
  348. private function createFullscreenManager(playlist:Playlist):void {
  349. _fullscreenManager = new FullscreenManager(stage, playlist, _panel, _pluginRegistry, _animationEngine);
  350. _flowplayer.fullscreenManager = _fullscreenManager;
  351. }
  352. public function showError(message:String):void {
  353. if (! _panel) return;
  354. if (! _config.showErrors) return;
  355. if (_error && _error.parent == this) {
  356. removeChild(_error);
  357. }
  358. _error = TextUtil.createTextField(false);
  359. _error.background = true;
  360. _error.backgroundColor = 0;
  361. _error.textColor = 0xffffff;
  362. _error.autoSize = TextFieldAutoSize.CENTER;
  363. _error.multiline = true;
  364. _error.wordWrap = true;
  365. _error.text = message;
  366. _error.selectable = true;
  367. _error.width = Arrange.parentWidth - 40;
  368. Arrange.center(_error, Arrange.parentWidth, Arrange.parentHeight);
  369. addChild(_error);
  370. createErrorMessageHideTimer();
  371. }
  372. private function createErrorMessageHideTimer():void {
  373. var errorHideTimer:Timer = new Timer(10000, 1);
  374. errorHideTimer.addEventListener(TimerEvent.TIMER_COMPLETE, hideErrorMessage);
  375. errorHideTimer.start();
  376. }
  377. private function hideErrorMessage(event:TimerEvent = null):void {
  378. if (_error && _error.parent == this) {
  379. if (_animationEngine) {
  380. _animationEngine.fadeOut(_error, 1000, function():void { removeChild(_error); });
  381. } else {
  382. removeChild(_error);
  383. }
  384. }
  385. }
  386. public function handleError(error:ErrorCode, info:Object = null, throwError:Boolean = true):void {
  387. if (_flowplayer) {
  388. _flowplayer.dispatchError(error, info);
  389. } else {
  390. // initialization is not complete, create a dispatcher just to dispatch this error
  391. new PlayerEventDispatcher().dispatchError(error, info);
  392. }
  393. var stack:String = "";
  394. if ( CONFIG::debug && info is Error && info.getStackTrace() )
  395. stack = info.getStackTrace();
  396. doHandleError(error.code + ": " + error.message + ( info ? ": " + info + (stack ? " - Stack: "+ stack : "") : ""), throwError);
  397. }
  398. private function doHandleError(message:String, throwError:Boolean = true):void {
  399. if (_config && _config.playerId) {
  400. Logger.error(message);
  401. }
  402. showError(message);
  403. if (throwError && Capabilities.isDebugger && _config.showErrors) {
  404. throw new Error(message);
  405. }
  406. }
  407. private function createAnimationEngine(pluginRegistry:PluginRegistry):void {
  408. _animationEngine = new AnimationEngine(_panel, pluginRegistry);
  409. }
  410. private function addPluginsToPanel(_pluginRegistry:PluginRegistry):void {
  411. for each (var pluginObj:Object in _pluginRegistry.plugins) {
  412. if (pluginObj is DisplayPluginModel) {
  413. var model:DisplayPluginModel = pluginObj as DisplayPluginModel;
  414. log.debug("adding plugin '"+ model.name +"' to panel: " + model.visible + ", plugin object is " + model.getDisplayObject());
  415. if (model.visible) {
  416. if (model.zIndex == -1) {
  417. model.zIndex = 100;
  418. }
  419. _panel.addView(model.getDisplayObject(), undefined, model);
  420. }
  421. if (model.name == "controls") {
  422. _controlsModel = model;
  423. }
  424. }
  425. }
  426. if (_controlsModel) {
  427. arrangeCanvasLogo();
  428. }
  429. }
  430. private function addScreenToPanel():void {
  431. // if controls visible and screen was not explicitly configured --> place screen on top of controls
  432. var screen:DisplayProperties = _pluginRegistry.getPlugin("screen") as DisplayProperties;
  433. screen.display = "none";
  434. screen.getDisplayObject().visible = false;
  435. _panel.addView(screen.getDisplayObject(), null, screen);
  436. }
  437. private function arrangeScreen(event:Event = null):void {
  438. log.debug("arrangeScreen(), already arranged " + _screenArrangeCount);
  439. if (_screenArrangeCount > 1) return;
  440. if (! _pluginRegistry) return;
  441. var screen:DisplayProperties = _pluginRegistry.getPlugin("screen") as DisplayProperties;
  442. if (! screen) return;
  443. if (_controlsModel && _controlsModel.visible) {
  444. if (isControlsAlwaysAutoHide() || (_controlsModel.position.bottom.px > 0)) {
  445. log.debug("controls is autoHide or it's in a non-default vertical position, configuring screen to take all available space");
  446. setScreenBottomAndHeight(screen, 100, 0);
  447. } else {
  448. var controlsHeight:Number = _controlsModel.getDisplayObject().height;
  449. var occupiedHeight:Number = screenTopOrBottomConfigured() ? getScreenTopOrBottomPx(screen) : controlsHeight;
  450. log.debug("occupied by controls or screen's configured bottom/top is " + occupiedHeight);
  451. var heightPct:Number = 0;
  452. if (screenTopOrBottomConfigured() && (screen.position.top.pct >= 0 || screen.position.bottom.pct >= 0)) {
  453. heightPct = 100 - Math.abs(50 - (screen.position.top.pct >= 0 ? screen.position.top.pct : screen.position.bottom.pct))*2;
  454. setScreenBottomAndHeight(screen, heightPct, controlsHeight);
  455. } else {
  456. heightPct = ((Arrange.parentHeight - occupiedHeight) / Arrange.parentHeight) * 100;
  457. setScreenBottomAndHeight(screen, heightPct, controlsHeight);
  458. }
  459. }
  460. }
  461. log.debug("arrangeScreen(): arranging screen to pos " + screen.position);
  462. screen.display = "block";
  463. screen.alpha = 1;
  464. screen.getDisplayObject().visible = true;
  465. _pluginRegistry.updateDisplayProperties(screen, true);
  466. _panel.update(screen.getDisplayObject(), screen);
  467. _panel.draw(screen.getDisplayObject());
  468. _screenArrangeCount++;
  469. }
  470. private function getScreenTopOrBottomPx(screen:DisplayProperties):Number {
  471. var screenConf:Object = _config.getObject("screen");
  472. if (screenConf.hasOwnProperty("top")) return screen.position.top.toPx(Arrange.parentHeight);
  473. if (screenConf.hasOwnProperty("bottom")) return screen.position.bottom.toPx(Arrange.parentHeight);
  474. return 0;
  475. }
  476. private function setScreenBottomAndHeight(screen:DisplayProperties, heightPct:Number, bottom:Number = 0):void {
  477. if (! screenTopOrBottomConfigured()) {
  478. log.debug("screen vertical pos not configured, setting bottom to value " + bottom);
  479. screen.bottom = bottom;
  480. } else {
  481. log.debug("using configured top/bottom for screen");
  482. }
  483. var heightConfigured:Boolean = _config.getObject("screen") && _config.getObject("screen").hasOwnProperty("height");
  484. if (! heightConfigured) {
  485. log.debug("screen height not configured, setting it to value " + heightPct + "%");
  486. screen.height = heightPct + "%";
  487. } else {
  488. log.debug("using configured height for screen");
  489. }
  490. }
  491. private function screenTopOrBottomConfigured():Boolean {
  492. var screen:Object = _config.getObject("screen");
  493. if (! screen) return false;
  494. if (screen.hasOwnProperty("top")) return true;
  495. if (screen.hasOwnProperty("bottom")) return true;
  496. return false;
  497. }
  498. private function isControlsAlwaysAutoHide():Boolean {
  499. if (!_controlsModel) return false;
  500. var controls:DisplayObject = _controlsModel.getDisplayObject();
  501. log.debug("controls.auotoHide " + controls["getAutoHide"]());
  502. return ! controls["getAutoHide"]()["fullscreenOnly"];
  503. }
  504. private function createFlowplayer():void {
  505. _flowplayer = new Flowplayer(stage, _pluginRegistry, _panel,
  506. _animationEngine, this, this, _config, URLUtil.playerBaseUrl);
  507. _flowplayer.onBeforeFullscreen(onFullscreen);
  508. // _flowplayer.onFullscreenExit(onFullscreen);
  509. }
  510. private function onFullscreen(event:PlayerEvent):void {
  511. log.debug("entering fullscreen, disabling display clicks");
  512. _screenArrangeCount = 100;
  513. stage.removeEventListener(Event.RESIZE, arrangeScreen);
  514. _enteringFullscreen = true;
  515. var delay:Timer = new Timer(1000, 1);
  516. delay.addEventListener(TimerEvent.TIMER_COMPLETE, onTimerComplete);
  517. delay.start();
  518. }
  519. private function onTimerComplete(event:TimerEvent):void {
  520. log.debug("fullscreen wait delay complete, display clicks are enabled again");
  521. _enteringFullscreen = false;
  522. }
  523. private function createFlashVarsConfig():void {
  524. log.debug("createFlashVarsConfig()");
  525. if (! root.loaderInfo.parameters) {
  526. return;
  527. }
  528. var configStr:String = Preloader(root).injectedConfig || root.loaderInfo.parameters["config"];
  529. var configObj:Object = configStr && configStr.indexOf("{") == 0 ? ConfigParser.parse(configStr) : {};
  530. if (! configStr || (configStr && configStr.indexOf("{") == 0 && ! configObj.hasOwnProperty("url"))) {
  531. _config = ConfigParser.parseConfig(configObj, BuiltInConfig.config, loaderInfo.url, VersionInfo.controlsVersion, VersionInfo.audioVersion);
  532. callAndHandleError(initPhase1, PlayerError.INIT_FAILED);
  533. } else {
  534. ConfigParser.loadConfig(configObj.hasOwnProperty("url") ? String(configObj["url"]) : configStr, BuiltInConfig.config, function(config:Config):void {
  535. _config = config;
  536. callAndHandleError(initPhase1, PlayerError.INIT_FAILED);
  537. }, new ResourceLoaderImpl(null, this), loaderInfo.url, VersionInfo.controlsVersion, VersionInfo.audioVersion);
  538. }
  539. }
  540. private function createPlayListController():PlayListController {
  541. createHttpProviders();
  542. var playListController:PlayListController = new PlayListController(_config.getPlaylist(), _providers, _config, createNewLoader());
  543. playListController.playerEventDispatcher = _flowplayer;
  544. _flowplayer.playlistController = playListController;
  545. return playListController;
  546. }
  547. private function createHttpProviders():void {
  548. if (! _providers) {
  549. _providers = new Dictionary();
  550. }
  551. _providers["http"] = createProvider("http");
  552. _providers["httpInstream"] = createProvider("httpInstream");
  553. }
  554. private function createProvider(name:String):Object {
  555. log.debug("creating provider with name " + name);
  556. var httpProvider:ProviderModel = _config.createHttpProvider(name);
  557. _pluginRegistry.registerProvider(httpProvider);
  558. return httpProvider.pluginObject;
  559. }
  560. private function get hasHttpChildClip():Boolean {
  561. var children:Array = _config.getPlaylist().childClips;
  562. // log.debug("configuration has child clips", children);
  563. for (var i:int = 0; i < children.length; i++) {
  564. if (Clip(children[i]).provider == "httpInstream") {
  565. log.info("child clip with http provider found");
  566. return true;
  567. }
  568. }
  569. return false;
  570. }
  571. private function createScreen():void {
  572. _screen = new Screen(_config.getPlaylist(), _animationEngine, _playButtonOverlay, _pluginRegistry);
  573. var screenModel:DisplayProperties = _config.getScreenProperties();
  574. initView(_screen, screenModel, null, false);
  575. if (_playButtonOverlay) {
  576. PlayButtonOverlayView(_playButtonOverlay.getDisplayObject()).setScreen(_screen, hasClip && _config.useBufferingAnimation);
  577. }
  578. // addViewLiteners(_screen);
  579. }
  580. private function createPlayButtonOverlay():void {
  581. _playButtonOverlay = _config.getPlayButtonOverlay();
  582. if (! _playButtonOverlay) return;
  583. _playButtonOverlay.onLoad(onPluginLoad);
  584. _playButtonOverlay.onError(onPluginLoadError);
  585. var overlay:PlayButtonOverlayView = new PlayButtonOverlayView(! playButtonOverlayWidthDefined(), _playButtonOverlay, _pluginRegistry);
  586. initView(overlay, _playButtonOverlay, null, false);
  587. }
  588. private function playButtonOverlayWidthDefined():Boolean {
  589. if (! _config.getObject("play")) return false;
  590. return _config.getObject("play").hasOwnProperty("width");
  591. }
  592. private function get hasClip():Boolean {
  593. var firstClip:Clip = _config.getPlaylist().current;
  594. var hasClip:Boolean = ! firstClip.isNullClip && (firstClip.url || firstClip.provider != 'http');
  595. return hasClip;
  596. }
  597. private function createLogo():void {
  598. var logoView:LogoView = new LogoView(_panel, _flowplayer);
  599. var logo:Logo = _config.getLogo(logoView) || new Logo(logoView, "logo");
  600. // do not show it initially
  601. logo.visible = false;
  602. logoView.model = logo;
  603. initView(logoView, logo, logoView.draw, false);
  604. }
  605. private function initView(view:DisplayObject, props:DisplayProperties, resizeListener:Function = null, addToPanel:Boolean = true):void {
  606. if (props.name != "logo" || VersionInfo.commercial) {
  607. _pluginRegistry.registerDisplayPlugin(props, view);
  608. }
  609. if (addToPanel) {
  610. _panel.addView(view, resizeListener, props);
  611. }
  612. if (props is Callable) {
  613. ExternalInterfaceHelper.initializeInterface(props as Callable, view);
  614. }
  615. }
  616. private function addListeners():void {
  617. _clickTimer.addEventListener(TimerEvent.TIMER, onClickTimer);
  618. doubleClickEnabled = true;
  619. addEventListener(MouseEvent.DOUBLE_CLICK, onDoubleClick);
  620. _screen.addEventListener(MouseEvent.CLICK, onClickEvent);
  621. if (_playButtonOverlay) {
  622. _playButtonOverlay.getDisplayObject().addEventListener(MouseEvent.CLICK, onClickEvent);
  623. }
  624. addEventListener(MouseEvent.ROLL_OVER, onMouseOver);
  625. addEventListener(MouseEvent.ROLL_OUT, onMouseOut);
  626. // add some color so that the ROLL_OVER/ROLL_OUT events are always triggered
  627. graphics.beginFill(0, 0);
  628. graphics.drawRect(0, 0, Arrange.parentWidth, Arrange.parentHeight);
  629. graphics.endFill();
  630. CONFIG::FLASH_10_1 {
  631. _flowplayer.playlist.onStageVideoStateChange(onStageVideoStateChange);
  632. }
  633. }
  634. private function onMouseOut(event:MouseEvent):void {
  635. _flowplayer.dispatchEvent(PlayerEvent.mouseOut());
  636. }
  637. private function onMouseOver(event:MouseEvent):void {
  638. _flowplayer.dispatchEvent(PlayerEvent.mouseOver());
  639. }
  640. CONFIG::FLASH_10_1 {
  641. private function onStageVideoStateChange(event:ClipEvent):void {
  642. var stageVideo:StageVideo = event.info as StageVideo;
  643. log.info("stage video state changed " + stageVideo);
  644. if ( stageVideo ) {
  645. _screenMask.width = stageVideo.viewPort.width;
  646. _screenMask.height = stageVideo.viewPort.height;
  647. _screenMask.x = stageVideo.viewPort.x;
  648. _screenMask.y = stageVideo.viewPort.y;
  649. log.debug("mask dimensions " + _screenMask.width + " x " + _screenMask.height);
  650. log.debug("mask pos " + _screenMask.x + ", " + _screenMask.y);
  651. if ( ! contains(_screenMask) ) {
  652. addChildAt(_screenMask, _canvasLogo ? getChildIndex(_canvasLogo) + 1 : 1);
  653. log.debug("adding mask");
  654. }
  655. } else {
  656. if ( contains(_screenMask) ) {
  657. log.debug("removing mask")
  658. removeChild(_screenMask);
  659. }
  660. }
  661. }
  662. }
  663. private function createPanel():void {
  664. _panel = new Panel();
  665. addChild(_panel);
  666. }
  667. private function startStreams():void {
  668. var canStart:Boolean = true;
  669. if (_flowplayer.state != State.WAITING) {
  670. log.debug("streams have been started in player.onLoad(), will not start streams here.");
  671. canStart = false;
  672. }
  673. if (! hasClip) {
  674. log.info("Configuration has no clips to play.");
  675. canStart = false;
  676. }
  677. var playButton:PlayButtonOverlayView = _playButtonOverlay ? PlayButtonOverlayView(_playButtonOverlay.getDisplayObject()) : null;
  678. if (canStart) {
  679. if (_flowplayer.currentClip.autoPlay) {
  680. log.debug("clip is autoPlay");
  681. _flowplayer.play();
  682. } else if (_flowplayer.currentClip.autoBuffering) {
  683. log.debug("clip is autoBuffering");
  684. _flowplayer.startBuffering();
  685. } else {
  686. if (playButton) {
  687. playButton.stopBuffering();
  688. playButton.showButton();
  689. }
  690. }
  691. } else {
  692. // cannot start playing here, stop buffering indicator, don't show the button
  693. if (playButton) {
  694. playButton.stopBuffering();
  695. }
  696. }
  697. }
  698. private function addPlayListListeners():void {
  699. var playlist:Playlist = _config.getPlaylist();
  700. playlist.onError(onClipError);
  701. playlist.onBegin(onBegin);
  702. }
  703. private function onBegin(event:ClipEvent):void {
  704. this.buttonMode = Boolean(Clip(event.target).linkUrl);
  705. }
  706. private function onClipError(event:ClipEvent):void {
  707. if (event.isDefaultPrevented()) return;
  708. doHandleError(event.error.code + ", " + event.error.message + ", " + event.info2 + ", clip: '" + Clip(event.target) + "'");
  709. }
  710. private function onClickTimer(event:TimerEvent):void {
  711. if (_clickCount == 1) {
  712. onSingleClick(_clickEvent);
  713. }
  714. _clickCount = 0;
  715. }
  716. private function onDoubleClick(event:MouseEvent = null):void {
  717. log.debug("onDoubleClick");
  718. _flowplayer.toggleFullscreen();
  719. }
  720. private function onSingleClick(event:MouseEvent):void {
  721. if (isParent(DisplayObject(event.target), _screen)) {
  722. log.debug("screen clicked");
  723. _flowplayer.toggle();
  724. }
  725. }
  726. private function onClickEvent(event:MouseEvent):void {
  727. if (_enteringFullscreen) return;
  728. log.debug("onViewClicked, target " + event.target + ", current target " + event.currentTarget);
  729. event.stopPropagation();
  730. if (_playButtonOverlay && isParent(DisplayObject(event.target), _playButtonOverlay.getDisplayObject())) {
  731. _flowplayer.toggle();
  732. return;
  733. } else {
  734. // if using linkUrl, no doubleclick to fullscreen
  735. var clip:Clip = _flowplayer.playlist.current;
  736. if (clip.linkUrl) {
  737. log.debug("opening linked page " + clip.linkUrl);
  738. _flowplayer.pause();
  739. URLUtil.openPage(clip.linkUrl, clip.linkWindow);
  740. return;
  741. }
  742. }
  743. if (++_clickCount == 2) {
  744. onDoubleClick(event);
  745. } else {
  746. _clickEvent = event;
  747. _clickTimer.start();
  748. }
  749. }
  750. private function isParent(child:DisplayObject, parent:DisplayObject):Boolean {
  751. try {
  752. if (DisplayObject(child).parent == parent) return true;
  753. if (! (parent is DisplayObjectContainer)) return false;
  754. for (var i:Number = 0;i < DisplayObjectContainer(parent).numChildren; i++) {
  755. var curChild:DisplayObject = DisplayObjectContainer(parent).getChildAt(i);
  756. if (isParent(child, curChild)) {
  757. return true;
  758. }
  759. }
  760. } catch (e:SecurityError) {
  761. return true;
  762. }
  763. return false;
  764. }
  765. override protected function onRedraw():void {
  766. if (bgImageHolder && getChildIndex(bgImageHolder) > getChildIndex(_panel)) {
  767. swapChildren(bgImageHolder, _panel);
  768. }
  769. }
  770. private function createLogoForCanvas():void {
  771. if (_canvasLogo) return;
  772. _copyrightNotice = LogoUtil.createCopyrightNotice(8);
  773. addChild(_copyrightNotice);
  774. _canvasLogo = new CanvasLogo();
  775. _canvasLogo.width = 85;
  776. _canvasLogo.scaleY = _canvasLogo.scaleX;
  777. _canvasLogo.alpha = .4;
  778. _canvasLogo.addEventListener(MouseEvent.CLICK,
  779. function(event:MouseEvent):void { navigateToURL(new URLRequest("http://flowplayer.org"), "_self"); });
  780. _canvasLogo.buttonMode = true;
  781. log.debug("adding logo to display list");
  782. addChild(_canvasLogo);
  783. onStageResize();
  784. }
  785. private function createNewLoader():ResourceLoader {
  786. return new ResourceLoaderImpl(_config.playerId ? null : URLUtil.playerBaseUrl, this);
  787. }
  788. private function initCustomClipEvents():void {
  789. createCustomClipEvents(_config.connectionCallbacks);
  790. createCustomClipEvents(_config.streamCallbacks);
  791. }
  792. private function createCustomClipEvents(callbacks:Array):void {
  793. if (! callbacks) return;
  794. for (var i:int = 0; i < callbacks.length; i++) {
  795. log.debug("creating custom event type " + callbacks[i]);
  796. new ClipEventType(callbacks[i], true);
  797. }
  798. }
  799. private function callAndHandleError(func:Function, error:PlayerError):void {
  800. try {
  801. func();
  802. } catch (e:Error) {
  803. handleError(error, e, false);
  804. throw e;
  805. }
  806. }
  807. internal function get enteringFullscreen():Boolean {
  808. return _enteringFullscreen;
  809. }
  810. }
  811. }