Show:
  1. /**
  2. * @module Player
  3. */
  4.  
  5.  
  6. /**
  7. * I am the type definition of a Videolink.
  8. *
  9. * A video link is classical hyperlink (to either another video or any other resource locatable bei an URL)
  10. * and which has a start and a end time related to the time base of the main video.
  11. *
  12. * Videolinks are managed by the {{#crossLink "VideolinksController"}}VideolinksController{{/crossLink}}.
  13. *
  14. * @class Videolink
  15. * @category TypeDefinition
  16. */
  17.  
  18.  
  19.  
  20. FrameTrail.defineType(
  21.  
  22. 'Videolink',
  23.  
  24. function(data){
  25.  
  26. this.data = data;
  27.  
  28. this.timelineElement = $('<div class="timelineElement"></div>');
  29. this.tileElement = $('<div class="tileElement"></div>');
  30. this.videolinkElement = $('<div class="videolinkElement"></div>');
  31.  
  32.  
  33. },
  34.  
  35. {
  36. /**
  37. * I hold the data object of a Videolink, which is stored in the {{#crossLink "Database"}}Database{{/crossLink}} and saved in the hypervideos's links.json file.
  38. * @attribute data
  39. * @type {}
  40. */
  41. data: {},
  42.  
  43. /**
  44. * I hold the timelineElement (a jquery-enabled HTMLElement), which indicates my start and end time.
  45. * @attribute timelineElement
  46. * @type HTMLElement
  47. */
  48. timelineElement: null,
  49. /**
  50. * I hold the tileElement (a jquery-enabled HTMLElement), which shows a icon for me close to my position in the timeline.
  51. * @attribute tileElement
  52. * @type HTMLElement
  53. */
  54. tileElement: null,
  55. /**
  56. * I hold the videolinkElement (a jquery-enabled HTMLElement), which shows the linked content in an iframe.
  57. * @attribute videolinkElement
  58. * @type HTMLElement
  59. */
  60. videolinkElement: null,
  61.  
  62. /**
  63. * I store my state, wether I am "active" (this is, when my timelineElement and tileElements are highlighted) or not.
  64. * @attribute activeState
  65. * @type Boolean
  66. */
  67. activeState: false,
  68.  
  69. /**
  70. * I store my state, wether I am "in focus" or not. See also:
  71. * * {{#crossLink "Videolink/gotInFocus:method"}}Videolink/gotInFocus(){{/crossLink}}
  72. * * {{#crossLink "Videolink/removedFromFocus:method"}}Videolink/removedFromFocus(){{/crossLink}}
  73. * * {{#crossLink "VideolinksController/videolinkInFocus:attribute"}}VideolinksController/videolinkInFocus{{/crossLink}}
  74. * @attribute permanentFocusState
  75. * @type Boolean
  76. */
  77. permanentFocusState: false,
  78.  
  79. /**
  80. * I render my ({{#crossLink "Videolink/timelineElement:attribute"}}this.timelineElement{{/crossLink}}
  81. * into the DOM.
  82. *
  83. * I am called, when the Videolink is initialized.
  84. *
  85. * The tileElement and videolinkElement however are rendere separately into the DOM (see
  86. * {{#crossLink "Videolink/renderTilesAndContentInDOM:method"}}this.renderTilesAndContentInDOM{{/crossLink}}),
  87. * because their order in the DOM tree has to be sorted by this.data.start after each change.
  88. *
  89. * @method renderTimelineInDOM
  90. */
  91. renderTimelineInDOM: function () {
  92.  
  93. var ViewVideo = FrameTrail.module('ViewVideo');
  94.  
  95. this.timelineElement.unbind('hover');
  96. this.timelineElement.hover(this.brushIn.bind(this), this.brushOut.bind(this));
  97.  
  98. ViewVideo.VideolinkTimeline.append(this.timelineElement);
  99. this.updateTimelineElement();
  100. },
  101.  
  102. /**
  103. * I render my ({{#crossLink "Videolink/tileElement:attribute"}}this.tileElement{{/crossLink}}
  104. * and my {{#crossLink "Videolink/videolinkElement:attribute"}}this.videolinkElement{{/crossLink}} into the DOM.
  105. *
  106. * I am called, when the Videolink is initialized, and also every time, when the global state "editMode" leaves the state
  107. * "links". This is the case, when the user has finished his/her changes to the Videolinks. All tiles and content
  108. * containers have to be re-rendered into the DOM, because they need to sorted by ascending value of this.data.start.
  109. *
  110. * @method renderTilesAndContentInDOM
  111. */
  112. renderTilesAndContentInDOM: function () {
  113.  
  114. var ViewVideo = FrameTrail.module('ViewVideo'),
  115. iFrame = $('<iframe frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen></iframe>'),
  116. linkElem = $('<a class="videolinkAnchor" href="'+ this.data.href +'">Go to Hypervideo</a>');
  117.  
  118. this.videolinkElement.empty();
  119. this.videolinkElement.append(iFrame, linkElem);
  120.  
  121. ViewVideo.VideolinkTileSlider.append(this.tileElement);
  122. ViewVideo.VideolinkContainer.append(this.videolinkElement);
  123.  
  124. this.tileElement.unbind('hover');
  125. this.tileElement.unbind('click')
  126. this.tileElement.hover(this.brushIn.bind(this), this.brushOut.bind(this));
  127. // self = this necessary as self can not be kept in anonymous handler function
  128. var self = this;
  129. this.tileElement.click(function() {
  130. if ( FrameTrail.module('VideolinksController').openedLink == self ) {
  131. self.closeVideolink();
  132. } else {
  133. self.openVideolink();
  134. }
  135. });
  136. },
  137.  
  138.  
  139. /**
  140. * I remove all my elements from the DOM. I am called when a Videolink is to be deleted.
  141. * @method removeFromDOM
  142. */
  143. removeFromDOM: function () {
  144.  
  145. this.timelineElement.remove();
  146. this.tileElement.remove();
  147. this.videolinkElement.remove();
  148.  
  149. },
  150.  
  151. /**
  152. * I update the CSS of the {{#crossLink "Videolink/timelineElement:attribute"}}timelineElement{{/crossLink}}
  153. * to its correct position within the timeline.
  154. *
  155. * @method updateTimelineElement
  156. */
  157. updateTimelineElement: function () {
  158.  
  159. var videoDuration = FrameTrail.module('HypervideoModel').duration,
  160. positionLeft = 100 * (this.data.start / videoDuration),
  161. width = 100 * ((this.data.end - this.data.start) / videoDuration);
  162.  
  163. this.timelineElement.css({
  164. top: '',
  165. left: positionLeft + '%',
  166. right: '',
  167. width: width + '%'
  168. });
  169.  
  170. },
  171.  
  172.  
  173. /**
  174. * When I am scheduled to be displayed, this is the method to be called.
  175. * @method setActive
  176. */
  177. setActive: function () {
  178.  
  179. this.activeState = true;
  180.  
  181. this.timelineElement.addClass('active');
  182. this.tileElement.addClass('active');
  183.  
  184. },
  185.  
  186. /**
  187. * When I am scheduled to disappear, this is the method to be called.
  188. * @method setInactive
  189. */
  190. setInactive: function () {
  191.  
  192. this.activeState = false;
  193.  
  194. this.timelineElement.removeClass('active');
  195. this.tileElement.removeClass('active');
  196.  
  197. },
  198.  
  199.  
  200. /**
  201. * I am called when the mouse pointer is hovering over one of my tile or my timeline element
  202. * @method brushIn
  203. */
  204. brushIn: function () {
  205.  
  206. this.timelineElement.addClass('brushed');
  207. this.tileElement.addClass('brushed');
  208.  
  209. if ( FrameTrail.getState('editMode') == false || FrameTrail.getState('editMode') == 'preview' ) {
  210. clearRaphael();
  211. drawConnections( this.tileElement, this.timelineElement, 10, {stroke: "#6B7884"} );
  212. }
  213.  
  214. },
  215.  
  216. /**
  217. * I am called when the mouse pointer is leaving the hovering area over my tile or my timeline element.
  218. * @method brushOut
  219. */
  220. brushOut: function () {
  221.  
  222. this.timelineElement.removeClass('brushed');
  223. this.tileElement.removeClass('brushed');
  224.  
  225. if ( (FrameTrail.getState('editMode') == false || FrameTrail.getState('editMode') == 'preview') ) {
  226. clearRaphael();
  227. }
  228.  
  229. },
  230.  
  231.  
  232. /**
  233. * A video link can be "opened" and "closed".
  234. *
  235. * When I am called, I open the video link, which means:
  236. * * I set the current play position to my data.start value
  237. * * I tell the {{#crossLink "VideolinksController/openedLink:attribute"}}VideolinksController{{/crossLink}} to set me as the "openedLink".
  238. *
  239. * @method openVideolink
  240. */
  241. openVideolink: function () {
  242.  
  243. var ViewVideo = FrameTrail.module('ViewVideo');
  244.  
  245.  
  246. FrameTrail.module('HypervideoController').currentTime = this.data.start;
  247.  
  248. FrameTrail.module('VideolinksController').openedLink = this;
  249.  
  250. //ViewVideo.ExpandButton.one('click', this.closeVideolink.bind(this));
  251.  
  252.  
  253. },
  254.  
  255. /**
  256. * I tell the {{#crossLink "VideolinksController/openedLink:attribute"}}VideolinksController{{/crossLink}} to set "openedLink" to null.
  257. * @method closeVideolink
  258. * @return
  259. */
  260. closeVideolink: function () {
  261.  
  262. FrameTrail.module('VideolinksController').openedLink = null;
  263.  
  264. },
  265.  
  266.  
  267.  
  268. /**
  269. * I am called when the app switches to the editMode "links".
  270. *
  271. * I make sure
  272. * * that my {{#crossLink "Videolink/timelineElement:attribute"}}timelineElement{{/crossLink}} is resizable and draggable
  273. * * and that it has a click handler for putting myself into focus.
  274. *
  275. * @method startEditing
  276. */
  277. startEditing: function () {
  278.  
  279.  
  280. var self = this,
  281. VideolinksController = FrameTrail.module('VideolinksController');
  282.  
  283. this.makeTimelineElementDraggable();
  284. this.makeTimelineElementResizeable();
  285.  
  286. this.timelineElement.on('click', function(){
  287.  
  288. if (VideolinksController.videolinkInFocus === self){
  289. return VideolinksController.videolinkInFocus = null;
  290. }
  291.  
  292. self.permanentFocusState = true;
  293. VideolinksController.videolinkInFocus = self;
  294.  
  295. FrameTrail.module('HypervideoController').currentTime = self.data.start;
  296.  
  297. });
  298.  
  299. },
  300.  
  301. /**
  302. * When the global editMode leaves the state "links", I am called to
  303. * stop the editing features of the video link.
  304. *
  305. * @method stopEditing
  306. */
  307. stopEditing: function () {
  308.  
  309. this.timelineElement.draggable('destroy');
  310. this.timelineElement.resizable('destroy');
  311.  
  312. this.timelineElement.unbind('click');
  313.  
  314.  
  315. },
  316.  
  317.  
  318. /**
  319. * I make my {{#crossLink "Videolink/timelineElement:attribute"}}timelineElement{{/crossLink}} draggable.
  320. *
  321. * The event handling changes my this.data.start and this.data.end attributes
  322. * accordingly.
  323. *
  324. * @method makeTimelineElementDraggable
  325. */
  326. makeTimelineElementDraggable: function () {
  327.  
  328. var self = this;
  329.  
  330.  
  331. this.timelineElement.draggable({
  332. axis: 'x',
  333. containment: 'parent',
  334. snapTolerance: 10,
  335.  
  336. drag: function(event, ui) {
  337. var closestGridline = FrameTrail.module('ViewVideo').closestToOffset($('.gridline'), {
  338. left: ui.position.left,
  339. top: ui.position.top
  340. }),
  341. snapTolerance = $(this).draggable('option', 'snapTolerance');
  342.  
  343. if (closestGridline) {
  344. $('.gridline').css('background-color', '#ff9900');
  345.  
  346. if ( ui.position.left - snapTolerance < closestGridline.position().left &&
  347. ui.position.left + snapTolerance > closestGridline.position().left ) {
  348.  
  349. ui.position.left = closestGridline.position().left;
  350.  
  351. closestGridline.css('background-color', '#00ff00');
  352.  
  353. }
  354. }
  355.  
  356. var videoDuration = FrameTrail.module('HypervideoModel').duration,
  357. leftPercent = 100 * (ui.helper.position().left / ui.helper.parent().width()),
  358. newStartValue = leftPercent * (videoDuration / 100);
  359.  
  360. FrameTrail.module('HypervideoController').currentTime = newStartValue;
  361. },
  362.  
  363. start: function(event, ui) {
  364.  
  365. if (!self.permanentFocusState) {
  366. FrameTrail.module('VideolinksController').videolinkInFocus = self;
  367. }
  368. },
  369.  
  370. stop: function(event, ui) {
  371.  
  372. if (!self.permanentFocusState) {
  373. FrameTrail.module('VideolinksController').videolinkInFocus = null;
  374. }
  375.  
  376. var videoDuration = FrameTrail.module('HypervideoModel').duration,
  377. leftPercent = 100 * (ui.helper.position().left / ui.helper.parent().width()),
  378. widthPercent = 100 * (ui.helper.width() / ui.helper.parent().width());
  379. self.data.start = leftPercent * (videoDuration / 100);
  380. self.data.end = (leftPercent + widthPercent) * (videoDuration / 100);
  381.  
  382. self.updateTimelineElement();
  383.  
  384. FrameTrail.module('VideolinksController').stackTimelineView();
  385.  
  386. FrameTrail.module('HypervideoModel').newUnsavedChange('links');
  387. }
  388. });
  389.  
  390. },
  391.  
  392. /**
  393. * I make my {{#crossLink "Videolink/timelineElement:attribute"}}timelineElement{{/crossLink}} resizable.
  394. *
  395. * The event handling changes my this.data.start and this.data.end attributes
  396. * accordingly.
  397. *
  398. * @method makeTimelineElementResizeable
  399. */
  400. makeTimelineElementResizeable: function () {
  401.  
  402. var self = this,
  403. endHandleGrabbed;
  404.  
  405.  
  406. this.timelineElement.resizable({
  407. containment: 'parent',
  408. handles: 'e, w',
  409.  
  410. resize: function(event, ui) {
  411. var closestGridline = FrameTrail.module('ViewVideo').closestToOffset($('.gridline'), {
  412. left: (endHandleGrabbed ? (ui.position.left + ui.helper.width()) : ui.position.left),
  413. top: ui.position.top
  414. }),
  415. snapTolerance = $(this).draggable('option', 'snapTolerance');
  416.  
  417. if (closestGridline) {
  418. $('.gridline').css('background-color', '#ff9900');
  419.  
  420. if ( !endHandleGrabbed &&
  421. ui.position.left - snapTolerance < closestGridline.position().left &&
  422. ui.position.left + snapTolerance > closestGridline.position().left ) {
  423.  
  424. ui.position.left = closestGridline.position().left;
  425. ui.size.width = ( ui.helper.width() + ( ui.helper.position().left - ui.position.left ) );
  426.  
  427. closestGridline.css('background-color', '#00ff00');
  428.  
  429. } else if ( endHandleGrabbed &&
  430. ui.position.left + ui.helper.width() - snapTolerance < closestGridline.position().left &&
  431. ui.position.left + ui.helper.width() + snapTolerance > closestGridline.position().left ) {
  432. ui.helper.width(closestGridline.position().left - ui.position.left);
  433.  
  434. closestGridline.css('background-color', '#00ff00');
  435.  
  436. }
  437. }
  438.  
  439.  
  440. var videoDuration = FrameTrail.module('HypervideoModel').duration,
  441. leftPercent = 100 * (ui.position.left / ui.helper.parent().width()),
  442. widthPercent = 100 * (ui.helper.width() / ui.helper.parent().width()),
  443. newValue;
  444.  
  445. if ( endHandleGrabbed ) {
  446.  
  447. newValue = (leftPercent + widthPercent) * (videoDuration / 100);
  448. FrameTrail.module('HypervideoController').currentTime = newValue;
  449.  
  450. } else {
  451.  
  452. newValue = leftPercent * (videoDuration / 100);
  453. FrameTrail.module('HypervideoController').currentTime = newValue;
  454.  
  455. }
  456. },
  457.  
  458. start: function(event, ui) {
  459.  
  460. if (!self.permanentFocusState) {
  461. FrameTrail.module('VideolinksController').videolinkInFocus = self;
  462. }
  463.  
  464. endHandleGrabbed = $(event.originalEvent.target).hasClass('ui-resizable-e')
  465. },
  466.  
  467. stop: function(event, ui) {
  468. if (!self.permanentFocusState) {
  469. FrameTrail.module('VideolinksController').videolinkInFocus = null;
  470. }
  471.  
  472.  
  473. var videoDuration = FrameTrail.module('HypervideoModel').duration,
  474. leftPercent = 100 * (ui.helper.position().left / ui.helper.parent().width()),
  475. widthPercent = 100 * (ui.helper.width() / ui.helper.parent().width());
  476.  
  477. self.data.start = leftPercent * (videoDuration / 100);
  478. self.data.end = (leftPercent + widthPercent) * (videoDuration / 100);
  479.  
  480. FrameTrail.module('VideolinksController').stackTimelineView();
  481.  
  482. FrameTrail.module('HypervideoModel').newUnsavedChange('links');
  483. }
  484. });
  485.  
  486. },
  487.  
  488.  
  489. /**
  490. * When I "got into focus" (which happens, when I become the referenced object in the VideolinksController's
  491. * {{#crossLink "VideolinksController/videolinkInFocus:attribute"}}videolinkInFocus attribute{{/crossLink}}),
  492. * then this method will be called.
  493. *
  494. * @method gotInFocus
  495. */
  496. gotInFocus: function () {
  497.  
  498. var EditPropertiesContainer = FrameTrail.module('ViewVideo').EditPropertiesContainer,
  499. self = this;
  500.  
  501. EditPropertiesContainer.empty();
  502. var propertiesControls = $('<div>'
  503. + ' <div class="propertiesTypeIcon" data-type="videolink"></div>'
  504. + ' <div>Title:</div>'
  505. + ' <div>' + this.data.name + '</div><br>'
  506. + ' <div>Link:</div>'
  507. + ' <div><input id="VideolinkHref" type="text" value="'+ this.data.href +'"></div><br>'
  508. + ' <button id="DeleteVideolink">Delete</button>'
  509. + '</div>');
  510.  
  511. propertiesControls.find('#VideolinkHref').change(function() {
  512.  
  513. self.data.href = $(this).val();
  514. FrameTrail.module('HypervideoModel').newUnsavedChange('links');
  515.  
  516. });
  517.  
  518. propertiesControls.find('#DeleteVideolink').click(function() {
  519.  
  520. FrameTrail.module('VideolinksController').deleteVideolink(self);
  521.  
  522. });
  523.  
  524. EditPropertiesContainer.addClass('active').append(propertiesControls);
  525.  
  526. this.timelineElement.addClass('highlighted');
  527.  
  528.  
  529. },
  530.  
  531.  
  532. /**
  533. * See also: {{#crossLink "Videolink/gotIntoFocus:method"}}this.gotIntoFocus(){{/crossLink}}
  534. *
  535. * When I was "removed from focus" (which happens, when the VideolinksController's
  536. * {{#crossLink "VideolinksController/videolinkInFocus:attribute"}}videolinkInFocus attribute{{/crossLink}}),
  537. * is set either to null or to an other Videolink than myself),
  538. * then this method will be called.
  539. *
  540. * @method removedFromFocus
  541. */
  542. removedFromFocus: function () {
  543.  
  544. FrameTrail.module('ViewVideo').EditPropertiesContainer.removeClass('active').empty();
  545.  
  546. this.timelineElement.removeClass('highlighted');
  547.  
  548.  
  549. }
  550.  
  551.  
  552.  
  553. }
  554.  
  555. );
  556.