jee.js 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999
  1. /*
  2. _ _
  3. |_|___ _ _ ___ ___ ___ ___| |_ ___ ___ ___ ___ ___ ___ ___
  4. | | -_| | | | -_|_ -| | -_| _| | -_| | . | .'| . | -_|_ -|
  5. _| |___|___|_|_|___|___| |___|_| |___|_|_|_ |__,|_ |___|___|
  6. |___| |___| |___|
  7. Bachir Soussi Chiadmi
  8. 2015
  9. */
  10. jQuery(document).ready(function($) {
  11. // console.log('Hello Jee');
  12. // http://stackoverflow.com/questions/2890898/preventing-mouse-emulation-events-ie-click-from-touch-events-in-mobile-safari
  13. // FastClick.attach(document.body);
  14. var _debug = false,
  15. _avgDelay = 1,
  16. _lastDraw = new Date,
  17. _fps,
  18. _$nav_cursor;
  19. var _$body = $('body'),
  20. _$header = $("#header"),
  21. _$chapitres = $('.node-chapitre', '#main'),
  22. _chapitres_len = _$chapitres.length,
  23. _chapters = [],
  24. _$container = $('#main'),
  25. _container = {
  26. w:_$container.width(),
  27. h:_$container.height()
  28. },
  29. _center = {x:_container.w/2,y:_container.h/2},
  30. _nav_pos = {x:0, y:0},
  31. _mouse_down_pos = {x:0,y:0},
  32. _prev_mouse_pos = {x:0,y:0},
  33. _fps = 1000/12,
  34. _dragging = false, _timeout_dragging,
  35. _previewed_chapter = false;
  36. var _$chapter_wrapper = $('<div>').attr('id', 'chapter-wrapper').appendTo(_$container),
  37. _loaded_chapter = false;
  38. var _bubbles = new Array(),
  39. _stars = new Array();
  40. var _$footer = $('#footer'),
  41. _$static_wrapper = $('<div><div class="inner"></div></div>').attr('id', 'static-wrapper').appendTo(_$container),
  42. _$close_static = $('<div>').addClass('close').appendTo(_$static_wrapper);
  43. _loaded_static = false;
  44. var _$loader = $('<div>').attr('id','loader').appendTo(_$body);
  45. var _touch_started = false;
  46. function init(){
  47. if(_debug)
  48. initDebug();
  49. initChapterWrapper();
  50. initChapters();
  51. initStaticLinks();
  52. launchNav();
  53. // FULL SCREEN
  54. var $fullscreenBtn = $('<div>Plein écran</div>').attr('id','fullscreen-btn').appendTo("#root");
  55. $fullscreenBtn.on('click', onFullScreen);
  56. // create bubbles
  57. for (var i = 2; i > 0; i--) {
  58. _bubbles.push($('<div>').addClass('bubble bubble-'+i).prependTo(_$container));
  59. };
  60. // create stars
  61. // for (var i = 20; i > 0; i--) {
  62. // _stars.push(
  63. // $('<div>')
  64. // .addClass('star star-'+i)
  65. // .css({
  66. // left:randB(-400, _container.w+400),
  67. // top:randB(-400, _container.h+400)
  68. // })
  69. // .prependTo(_$container)
  70. // );
  71. // };
  72. };
  73. function initDebug(){
  74. $('<p>').attr('id', 'fps').appendTo(_$container);
  75. _fps = document.getElementById('fps');
  76. requestAnimationFrame(displayFps);
  77. _$nav_cursor = $('<div id="nav-cursor"></div>').appendTo(_$container);
  78. moveDebugCursor();
  79. };
  80. function displayFps(){
  81. requestAnimationFrame(displayFps);
  82. var now = new Date;
  83. var delay = now - _lastDraw;
  84. _avgDelay += (delay - _avgDelay) / 10;
  85. _lastDraw = now;
  86. _fps.innerHTML = (1000/_avgDelay).toFixed(1) + " fps";
  87. };
  88. function moveDebugCursor(){
  89. _$nav_cursor.css({
  90. left:_nav_pos.x+_center.x+"px",
  91. top:_nav_pos.y+_center.y+"px"
  92. });
  93. };
  94. function initChapterWrapper(){
  95. $('<div id="home-btn">')
  96. .on("click", closeChapterWrapper)
  97. .appendTo(_$chapter_wrapper);
  98. };
  99. function closeChapterWrapper(e){
  100. _loaded_chapter = false;
  101. _$chapter_wrapper.removeClass('visible');
  102. _$body.removeClass('chapter-displayed');
  103. };
  104. function initChapters(){
  105. // Place each chapters on the ellipse contained on the screen
  106. var base_a = Math.random() *360;
  107. _$chapitres.each(function(i, e) {
  108. // Lets create the chapter object and place him self
  109. _chapters.push(new Chapter(i, e, base_a));
  110. });
  111. };
  112. function launchNav(){
  113. $("#block-system-main")
  114. /*
  115. ____ _____ _____ _____ _____ _____ _____ _____ _____ _____ _____ _____ _____
  116. | \| __| __| | |_ _| | _ | | __| | | __| | |_ _| __|
  117. | | | __|__ | -| | | | | | __| | __| | | __| | | | | | |__ |
  118. |____/|_____|_____|__|__| |_| |_____|__| |_____|\___/|_____|_|___| |_| |_____|
  119. */
  120. .on('mousemove', function(e){
  121. e.preventDefault();
  122. // prevent firing mousemove with touch events
  123. if(_touch_started)
  124. return false;
  125. console.log('document mousemove');
  126. clearTimeout(_timeout_dragging);
  127. // if(_previewed_chapter || _loaded_chapter || _loaded_static)
  128. // return false;
  129. // activate dragging if already activated
  130. if(!_dragging){
  131. _dragging = true;
  132. startMoveNav();
  133. }else{
  134. updateNavPos(e.clientX, e.clientY, false, false);
  135. }
  136. })
  137. .on('mouseup', function(e){
  138. e.preventDefault();
  139. console.log('document mouseup');
  140. if(_loaded_chapter || _loaded_static)
  141. return false;
  142. for (var i = _chapitres_len - 1; i >= 0; i--)
  143. _chapters[i].closePreview().unMitigate();
  144. })
  145. /*
  146. _____ _____ _____ _____ _____ _____ _____ _____ _____ _____ _____
  147. |_ _| | | | | | | | __| | | __| | |_ _| __|
  148. | | | | | | | --| | | __| | | __| | | | | | |__ |
  149. |_| |_____|_____|_____|__|__| |_____|\___/|_____|_|___| |_| |_____|
  150. */
  151. .on('touchstart', function(e){
  152. // e.preventDefault();
  153. console.log('document touchstart');
  154. _touch_started = true;
  155. clearTimeout(_timeout_dragging);
  156. if(_loaded_chapter || _loaded_static)
  157. return false;
  158. // set initial pos
  159. updateNavPos(e.originalEvent.touches[0].clientX, e.originalEvent.touches[0].clientY, true, true);
  160. // activate dragging if already activated
  161. if(!_dragging){
  162. _dragging = true;
  163. startMoveNav();
  164. }
  165. })
  166. .on('touchmove', function(e){
  167. // e.preventDefault();
  168. console.log('document touchmove');
  169. if(_loaded_chapter || _loaded_static)
  170. return false;
  171. updateNavPos(e.originalEvent.touches[0].clientX, e.originalEvent.touches[0].clientY, true, false);
  172. })
  173. .on('touchend', function(e){
  174. // e.preventDefault();
  175. console.log("document touchend");
  176. if(_loaded_chapter || _loaded_static)
  177. return false;
  178. stopMoveNav();
  179. setTimeout(function(){_touch_started = false;}, 500);
  180. });
  181. // TODO : nav on scroll events
  182. };
  183. function updateNavPos(x,y,touch,init){
  184. if(touch){
  185. if(!init){
  186. _nav_pos.x += x - _prev_mouse_pos.x;
  187. _nav_pos.y += y - _prev_mouse_pos.y;
  188. // constrain nav pos on container
  189. _nav_pos.x = _nav_pos.x < -_center.x*2
  190. ? -_center.x*2
  191. : _nav_pos.x > _center.x*2
  192. ? _center.x*2
  193. : _nav_pos.x;
  194. _nav_pos.y = _nav_pos.y < -_center.y*2
  195. ? -_center.y*2
  196. : _nav_pos.y > _center.y*2
  197. ? _center.y*2
  198. : _nav_pos.y;
  199. }
  200. }else{
  201. _nav_pos.x = (_center.x -x)*0.5;
  202. _nav_pos.y = (_center.y -y)*0.5;
  203. }
  204. _prev_mouse_pos.x = x;
  205. _prev_mouse_pos.y = y;
  206. // debuging cursor
  207. if(_debug)
  208. moveDebugCursor();
  209. };
  210. function startMoveNav(){
  211. window.requestAnimationFrame(moveNav);
  212. };
  213. function moveNav(){
  214. // console.log("moveNav");
  215. if(!_dragging) return;
  216. window.requestAnimationFrame(moveNav);
  217. // move chapters
  218. for (var i = _chapitres_len - 1; i >= 0; i--)
  219. _chapters[i].move();
  220. // move header
  221. // _$header.css({
  222. // transform:"translate3d("+(_nav_pos.x)*0.2+"px, "+(_nav_pos.y)*0.2+"px,0)"
  223. // });
  224. _$header.translate3d({x:_nav_pos.x*0.2,y:_nav_pos.y*0.2});
  225. // bubbles and stars dont move so smoothly on tablette ...
  226. // should keep it only on desktop
  227. // move bubbles
  228. // for (var i = _bubbles.length - 1; i >= 0; i--) {
  229. // _bubbles[i].translate3d({x:_nav_pos.x*0.4,y:_nav_pos.y*0.4});
  230. // };
  231. // move stars
  232. // for (var i = _stars.length - 1; i >= 0; i--) {
  233. // _stars[i].translate3d({x:_nav_pos.x*-0.3,y:_nav_pos.y*-0.3});
  234. // };
  235. };
  236. function stopMoveNav(){
  237. console.log('stopMoveNav');
  238. clearTimeout(_timeout_dragging);
  239. _timeout_dragging = setTimeout(function(){
  240. console.log("dragging stoped");
  241. _dragging = false;
  242. },3000);
  243. };
  244. function onFullScreen(e){
  245. if(_$body.is('fullscreen')){
  246. exitFullscreen();
  247. _$body.removeClass('fullscreen');
  248. }else{
  249. launchIntoFullscreen(document.getElementById("root"));
  250. _$body.addClass('fullscreen');
  251. }
  252. if(_loaded_chapter)
  253. setTimeout(_loaded_chapter.buildVideos(), 100);
  254. };
  255. function initStaticLinks(){
  256. $('ul.menu a', _$footer).on('click', onClickStaticLink);
  257. _$close_static.on('click', onCloseStatic);
  258. };
  259. function onClickStaticLink(e){
  260. e.preventDefault();
  261. _$static_wrapper.removeClass('visible');
  262. loadStaticContent($(this).attr("href"));
  263. _$body.addClass('loading');
  264. _loaded_static = true;
  265. return false;
  266. };
  267. function loadStaticContent(h){
  268. $.getJSON(
  269. '/jee/static',
  270. {"href":h},
  271. onStaticLoaded
  272. );
  273. };
  274. function onStaticLoaded(data, textstatus){
  275. console.log('data',data);
  276. displayStatic(data.node);
  277. };
  278. function displayStatic(node){
  279. _$static_wrapper.addClass('visible').find('.inner').html(node);
  280. _$body.removeClass('loading');
  281. };
  282. function onCloseStatic(e){
  283. _$static_wrapper.removeClass('visible');
  284. _loaded_static = false;
  285. };
  286. /*
  287. ________ __
  288. / ____/ /_ ____ _____ / /____ _____
  289. / / / __ \/ __ `/ __ \/ __/ _ \/ ___/
  290. / /___/ / / / /_/ / /_/ / /_/ __/ /
  291. \____/_/ /_/\__,_/ .___/\__/\___/_/
  292. /_/
  293. */
  294. function Chapter(i, e, base_a){
  295. // $e.obj = this;
  296. this.i = i;
  297. this.e = e;
  298. this.$e = $(e);
  299. this.nid = this.$e.attr("id").match(/^node-(\d+)/)[1];
  300. this.geom = {
  301. base_a:base_a,
  302. a:0,
  303. r:0,
  304. w:this.$e.outerWidth(true),
  305. h:this.$e.outerHeight(true)
  306. }
  307. this.pos = {x:0,y:0};
  308. this.trans = {x:0, y:0,z:0};
  309. this.trans_previewed = {x:0, y:0,z:0};
  310. this.ease = randB(0.05, 0.2);
  311. //drifting
  312. this.$title = $('h2.node-title', this.$e);
  313. this.$content = $('.content:first', this.$e);
  314. this.title_x = 0;
  315. this.content_x = 0;
  316. this.drifting_direction = Math.random()-0.5 > 0 ? 1 : -1;
  317. this.drifting_time = null;
  318. //preview
  319. this.is_previewed = false;
  320. //mitigate
  321. this.is_mitigated = false;
  322. //parties
  323. this.$parties = $('.field-name-field-partie', e);
  324. this.parts_pos = {xs:new Array(), ys:new Array()};
  325. this.lines = new Array();
  326. this.displayPreviewAnimeStartTime = 0,
  327. this.displayPreviewAnimeDuration = 4000, // milli sec
  328. this.are_lines_animated = false;
  329. // chapter
  330. this.$n = false;
  331. this.$blocks = false;
  332. this.$vids_container = false;
  333. this.$vids = false;
  334. this.texts_pos = shuffleArray([1,2,3]);
  335. this.dimvideo = {w:0, h:0};
  336. this.$slider = null;
  337. this.cur_vid_playing = 0;
  338. // prototypes
  339. if (typeof Chapter.initialized == "undefined") {
  340. Chapter.prototype.init = function(){
  341. this.setInitPos();
  342. this.drawLines();
  343. this.setEvents();
  344. // this.initDrifiting();
  345. // enable nodes after intro anime
  346. setTimeout(
  347. (function(t){
  348. return function(){
  349. t.$e.addClass('enabled');
  350. }
  351. }(this)),
  352. randB(5000, 8000));
  353. };
  354. Chapter.prototype.setInitPos = function(){
  355. // distribute elements arround the center
  356. this.geom.a = (360/_chapitres_len*this.i+this.geom.base_a)*Math.PI/180;
  357. // console.log("Chapter :: setInitPos", this.$e);
  358. this.geom.c = Math.cos(this.geom.a);
  359. this.geom.s = Math.sin(this.geom.a);
  360. this.geom.abs_c = Math.abs(this.geom.c);
  361. this.geom.abs_s = Math.abs(this.geom.s);
  362. if (this.geom.abs_c * _container.h > this.geom.abs_s * _container.w) {
  363. // It crosses left or right side
  364. this.geom.r = (_center.x / this.geom.abs_c)*0.5;
  365. }else {
  366. // Top or bottom side
  367. this.geom.r = (_center.y / this.geom.abs_s)*0.5;
  368. }
  369. // change randomly radius
  370. if(this.i%2){
  371. this.geom.r = randB(this.geom.r*1.5, this.geom.r*2);
  372. }else{
  373. this.geom.r = randB(this.geom.r*0.8, this.geom.r*1.5);
  374. }
  375. this.pos.x = Math.round(_center.x+this.geom.r * this.geom.c) - this.geom.w/2;
  376. this.pos.y = Math.round(_center.y+this.geom.r * -this.geom.s) - this.geom.h/2;
  377. // calculate the translation needed for the previewed position (center top)
  378. this.trans_previewed.x = -(this.pos.x - _center.x) - this.geom.w*0.5;
  379. this.trans_previewed.y = -this.pos.y+this.$e.find('.node-title').height()*1.3;
  380. this.$e.css({
  381. left:this.pos.x,
  382. top:this.pos.y
  383. });
  384. };
  385. Chapter.prototype.setEvents = function(){
  386. //http://technify.me/user-experience/javascript/jquery/trigger-custom-events-with-jquery/
  387. // click to preview chapter
  388. // $('h2.node-title, .field-name-field-partie:first>.field-name-field-vignette', this.$e)
  389. this.$e
  390. .on('click', this, function(e){
  391. // e.stopImmediatePropagation();
  392. console.log('click on chapter');
  393. e.stopPropagation();
  394. e.preventDefault();
  395. e.data.preview(e);
  396. return false;
  397. });
  398. $('.links a', this.$e)
  399. .on('click', this, function(e){
  400. e.stopPropagation();
  401. e.preventDefault();
  402. e.data.loadNode();
  403. return false;
  404. });
  405. };
  406. Chapter.prototype.initDrifiting = function(){
  407. // an other option could be to drift the whole page with the same engine than draging
  408. requestAnimationFrame(this.drift.bind(this));
  409. };
  410. Chapter.prototype.drift = function(timestamp){
  411. requestAnimationFrame(this.drift.bind(this));
  412. var now = new Date().getTime(),
  413. dt = now - (this.drifting_time || now);
  414. this.drifting_time = now;
  415. if(!this.is_previewed){
  416. this.title_x += (10/1000)*dt*this.drifting_direction;
  417. this.content_x += (6/1000)*dt*this.drifting_direction;
  418. this.drifting_direction =
  419. this.content_x > randB(100,130)
  420. ? -1
  421. : this.content_x < -randB(100,130)
  422. ? 1
  423. : this.drifting_direction;
  424. this.$title.translate3d({x:this.title_x});
  425. this.$content.translate3d({x:this.content_x});
  426. }
  427. };
  428. Chapter.prototype.move = function(){
  429. if(this.is_previewed)
  430. return false;
  431. this.trans.x += (_nav_pos.x - this.trans.x)*this.ease;
  432. this.trans.y += (_nav_pos.y - this.trans.y)*this.ease;
  433. // this.trans.z = Math.floor(Math.sqrt(
  434. // Math.pow(
  435. // _center.x-(this.pos.x+this.trans.x)
  436. // ,2
  437. // )
  438. // +
  439. // Math.pow(
  440. // _center.y-(this.pos.y+this.trans.y)
  441. // ,2
  442. // )
  443. // ));
  444. // if(this.i === 0)
  445. // console.log(this.i+" this.trans.z", this.trans.z);
  446. this.$e.translate3d({x:this.trans.x,y:this.trans.y});
  447. };
  448. Chapter.prototype.preview = function(){
  449. // don't relaunch preview more that one time
  450. if(this.is_previewed) return false;
  451. _previewed_chapter = this;
  452. console.log('preview', this.i);
  453. this.unMitigate();
  454. this.is_previewed = true;
  455. // close other chapters
  456. for (var i = _chapitres_len - 1; i >= 0; i--)
  457. if(i !== this.i)
  458. _chapters[i].closePreview().mitigate();
  459. this.displayPreview();
  460. // moveToChapter(this);
  461. };
  462. Chapter.prototype.displayPreview = function(e){
  463. // define randomly position of parties
  464. this.resetPartsPos();
  465. this.are_lines_animated = false;
  466. requestAnimationFrame(this.animePreviewDisplay.bind(this));
  467. };
  468. Chapter.prototype.resetPartsPos = function(){
  469. this.parts_pos = {xs:new Array(), ys:new Array()};
  470. var h = _container.h * 0.6;
  471. for (var i = 0; i < 2; i++) {
  472. switch(i){
  473. case 0:
  474. this.parts_pos.xs.push(randB(180,280));
  475. this.parts_pos.ys.push(randB(h*0.55,h*0.6));
  476. break;
  477. case 1:
  478. this.parts_pos.xs.push(randB(-280,-180));
  479. this.parts_pos.ys.push(randB(h*0.65,h));
  480. break;
  481. }
  482. };
  483. this.parts_pos.xs = shuffleArray(this.parts_pos.xs);
  484. this.parts_pos.ys = shuffleArray(this.parts_pos.ys);
  485. // always place the first circle on top center
  486. this.parts_pos.xs.unshift(randB(-30,30));
  487. this.parts_pos.ys.unshift(120);
  488. };
  489. Chapter.prototype.closePreview = function(){
  490. if(!this.is_previewed) return this;
  491. this.$e.removeClass('previewed')
  492. .find('.field-name-field-partie')
  493. .css({transform:"none"});
  494. this.is_previewed = false;
  495. _previewed_chapter = false;
  496. requestAnimationFrame(this.animePreviewDisplay.bind(this));
  497. return this;
  498. };
  499. Chapter.prototype.mitigate = function(){
  500. if(this.is_mitigated) return this;
  501. this.$e.addClass('mitigated');
  502. return this;
  503. };
  504. Chapter.prototype.unMitigate = function(){
  505. if(this.is_mitigated) return this;
  506. this.$e.removeClass('mitigated');
  507. return this;
  508. };
  509. Chapter.prototype.drawLines = function(){
  510. for (var i = 0; i < 2; i++) {
  511. this.lines.push({
  512. $line:$("<div>").addClass('line', 'line-'+i).prependTo(this.$parties[i])
  513. });
  514. };
  515. };
  516. Chapter.prototype.animePreviewDisplay = function(timestamp){
  517. // console.log("anime line "+this.nid);
  518. // get the time on first anime launch
  519. if(this.displayPreviewAnimeStartTime === 0)
  520. this.displayPreviewAnimeStartTime = timestamp;
  521. // limit the animation time
  522. if(timestamp - this.displayPreviewAnimeStartTime < this.displayPreviewAnimeDuration){
  523. requestAnimationFrame(this.animePreviewDisplay.bind(this));
  524. }else{
  525. this.displayPreviewAnimeStartTime = 0;
  526. }
  527. // first move chapter to previewed pos (center top screen)
  528. if(this.is_previewed && !this.are_lines_animated){
  529. this.trans.x = this.trans.x + (this.trans_previewed.x - this.trans.x)*0.4;
  530. this.trans.y = this.trans.y + (this.trans_previewed.y - this.trans.y)*0.4;
  531. this.$e.translate3d({x:this.trans.x,y:this.trans.y});
  532. // append just once when positioning is done
  533. if(Math.abs(this.trans_previewed.x - this.trans.x) < 5){
  534. // apply new position to parties
  535. var that = this;
  536. this.$parties.each(function(i, e) {
  537. setTimeout(
  538. (function(i, e, xs, ys){
  539. return function(){
  540. $(e).translate3d({x:xs[i],y:ys[i]});
  541. }
  542. }(i, e, that.parts_pos.xs, that.parts_pos.ys)),
  543. 10);
  544. }); // each $parties
  545. //activate lines anime for next frame
  546. this.are_lines_animated = true;
  547. this.$e.addClass('previewed');
  548. }
  549. }
  550. // after positiong juste anime lines
  551. if(this.are_lines_animated){
  552. // get the lines length
  553. var l, a, pos1, pos2;
  554. for (var i = 0; i < this.lines.length; i++) {
  555. pos1 = this.$parties.eq(i).position();
  556. pos2 = this.$parties.eq(i+1).position();
  557. l = Math.sqrt(
  558. Math.pow(
  559. pos2.left - pos1.left
  560. ,2
  561. )
  562. +
  563. Math.pow(
  564. pos2.top - pos1.top
  565. ,2
  566. )
  567. );
  568. // get the rotation
  569. a = 180 / 3.14 * Math.acos((pos2.top - pos1.top) / l);
  570. if(pos2.left > pos1.left)
  571. a *= -1;
  572. // console.log("a = "+a);
  573. this.lines[i].$line.css({
  574. 'height':l,
  575. transform:"rotate3d(0,0,1,"+a+"deg)"
  576. });
  577. };
  578. }
  579. };
  580. // _____ _____ _____ _____ _____ _____ _____ _____ _____ _____ _____
  581. // | | _ | __| | | | | | | _ | _ |_ _| __| __ |
  582. // | | | __| __| | | | | --| | | __| | | | __| -|
  583. // |_____|__| |_____|_|___| |_____|__|__|__|__|__| |_| |_____|__|__|
  584. Chapter.prototype.loadNode = function(e){
  585. // console.log("Chapter :: open : nid", this.nid);
  586. _$body.addClass('loading');
  587. $.getJSON(
  588. '/jee/chapter/'+this.nid,
  589. {},
  590. this.nodeLoaded.bind(this)
  591. );
  592. _$body.addClass('chapter-displayed');
  593. };
  594. Chapter.prototype.nodeLoaded = function(json, textstatus){
  595. console.log('Chapter :: nodeLoaded '+this.nid+' : json', json);
  596. // remove previous loaded nodes
  597. _$chapter_wrapper.find('.node').remove();
  598. // insert ajax loaded into dom
  599. _$chapter_wrapper.append(json.node);
  600. // record some usefull data
  601. this.$n = $('.node-chapitre', _$chapter_wrapper);
  602. this.$blocks = $('.field-type-text-long', this.$n);
  603. this.$vids_container = $('.field-name-field-partie', this.$n);
  604. this.$vids = $('iframe', this.$vids_container);
  605. // record the current loaded chapter
  606. // this will stop first interface to run
  607. _loaded_chapter = this;
  608. // wait to build correct display of chapter
  609. setTimeout(this.displayNode.bind(this), 100);
  610. };
  611. Chapter.prototype.displayNode = function(){
  612. console.log('Chapter :: displayNode '+this.nid);
  613. // place text blocks
  614. this.texts_pos = shuffleArray([1,2,3]);
  615. this.$blocks
  616. .each(this.placeText.bind(this))
  617. .pep({
  618. allowDragEventPropagation:false,
  619. disableSelect:true,
  620. velocityMultiplier:1
  621. });
  622. // build video player
  623. this.buildVideos();
  624. // show the whole thing
  625. _$chapter_wrapper.addClass('visible');
  626. // this.$vids.eq(this.cur_vid_playing).vimeo('play');
  627. _$body.removeClass('loading');
  628. };
  629. Chapter.prototype.placeText = function(i, e){
  630. // console.log("Chapter :: placeText", e);
  631. switch(this.texts_pos[i]){
  632. case 1: // top right
  633. $(e)
  634. .css({
  635. top:randB(_container.h*0.05-$(e).height(),_container.h*0.15-$(e).height()),
  636. left:randB(_container.w*0.8-$(e).width(),_container.w*0.9-$(e).width())
  637. })
  638. // .pep({debug:true, startPos:{bottom:randB(80,90),right:randB(80,90)}})
  639. .find('.field-label')
  640. .appendTo(e);
  641. break;
  642. case 2: // bottom left
  643. $(e).css({
  644. top:randB(_container.h*0.8,_container.h*0.9),
  645. left:randB(_container.w*0.1,_container.w*0.2)
  646. });
  647. // $(e).pep({debug:true, startPos:{top:randB(80,90),left:randB(10,30)}});
  648. break;
  649. case 3: // bottom right
  650. $(e).css({
  651. top:randB(_container.h*0.8,_container.h*0.9),
  652. left:randB(_container.w*0.7,_container.w*0.9-$(e).width())
  653. });
  654. // $(e).pep({debug:true, startPos:{top:randB(80,90),right:randB(10,30)}});
  655. break;
  656. }
  657. };
  658. Chapter.prototype.buildVideos = function(){
  659. console.log('Chapter :: buildVideos');
  660. this.dimvideo.h = this.$vids_container.height();
  661. // redim each iframe to fit
  662. // add a mask on top of each iframe to avoid bad interaction with vimeo
  663. this.$vids
  664. .after('<div class="mask"></div>')
  665. .each(this.redimVideo.bind(this));
  666. this.$vids_container.css({
  667. width:this.dimvideo.w*1.2,
  668. height:this.dimvideo.h,
  669. marginLeft:(_container.w-this.dimvideo.w*1.2)/2
  670. });
  671. // create the slider with peppermint
  672. this.$slider = $('.field-items', this.$vids_container).Peppermint({
  673. onSlideChange:this.onSlideChange.bind(this)
  674. });
  675. };
  676. Chapter.prototype.redimVideo = function(i,e){
  677. // compute proportional video width ragarding the parent height
  678. this.dimvideo.w = (this.dimvideo.h * parseInt($(e).attr("width")))/parseInt($(e).attr("height"));
  679. // apply thees sizes
  680. $(e)
  681. .css({ width:this.dimvideo.w, height:this.dimvideo.h })
  682. .attr({ width:this.dimvideo.w, height:this.dimvideo.h })
  683. // add some paading to parent for slider display
  684. .parent()
  685. .css({
  686. paddingLeft:this.dimvideo.w*0.15,
  687. paddingright:this.dimvideo.w*0.15
  688. })
  689. .on("click", this.onClickVid.bind(this));
  690. };
  691. Chapter.prototype.onClickVid = function(e){
  692. e.stopPropagation();
  693. e.preventDefault();
  694. console.log('Chapter :: onClickVid '+this.nid, e);
  695. var $vid = $('iframe', e.currentTarget);
  696. $vid.vimeo('paused', function(data){
  697. console.log('paused : ', data);
  698. if(data){
  699. $vid.vimeo('play');
  700. }else{
  701. $vid.vimeo('pause');
  702. }
  703. });
  704. return false;
  705. };
  706. Chapter.prototype.onSlideChange = function(){
  707. console.log('onSlideChange '+this.nid, this.$slider.data('Peppermint').getCurrentPos());
  708. //stop current video playing
  709. this.$vids.eq(this.cur_vid_playing).vimeo('pause');
  710. // start new current video
  711. this.cur_vid_playing = this.$slider.data('Peppermint').getCurrentPos();
  712. this.$vids.eq(this.cur_vid_playing).vimeo('play');
  713. };
  714. Node.initialized = true;
  715. };
  716. this.init();
  717. };//Chapter
  718. /*
  719. __ __________ ____ __________ _____
  720. / / / / ____/ / / __ \/ ____/ __ \/ ___/
  721. / /_/ / __/ / / / /_/ / __/ / /_/ /\__ \
  722. / __ / /___/ /___/ ____/ /___/ _, _/___/ /
  723. /_/ /_/_____/_____/_/ /_____/_/ |_|/____/
  724. */
  725. function randB(min, max){
  726. return Math.random() * (max - min) + min;
  727. };
  728. //+ Jonas Raoni Soares Silva
  729. //@ http://jsfromhell.com/array/shuffle [v1.0]
  730. function shuffleArray(o){ //v1.0
  731. for(var j, x, i = o.length; i; j = Math.floor(Math.random() * i), x = o[--i], o[i] = o[j], o[j] = x);
  732. return o;
  733. };
  734. if (!Date.now) {
  735. Date.now = function now() {
  736. return new Date().getTime();
  737. };
  738. }
  739. // http://davidwalsh.name/fullscreen
  740. // http://www.sitepoint.com/html5-full-screen-api/
  741. function launchIntoFullscreen(element) {
  742. // Find the right method, call on correct element
  743. if(element.requestFullscreen) {
  744. element.requestFullscreen();
  745. } else if(element.mozRequestFullScreen) {
  746. element.mozRequestFullScreen();
  747. } else if(element.webkitRequestFullscreen) {
  748. element.webkitRequestFullscreen();
  749. } else if(element.msRequestFullscreen) {
  750. element.msRequestFullscreen();
  751. }
  752. }
  753. // Whack fullscreen
  754. function exitFullscreen() {
  755. if(document.exitFullscreen) {
  756. document.exitFullscreen();
  757. } else if(document.mozCancelFullScreen) {
  758. document.mozCancelFullScreen();
  759. } else if(document.webkitExitFullscreen) {
  760. document.webkitExitFullscreen();
  761. }
  762. }
  763. /*
  764. _ _ __
  765. (_)___ (_) /_
  766. / / __ \/ / __/
  767. / / / / / / /_
  768. /_/_/ /_/_/\__/
  769. */
  770. init();
  771. });
  772. /*
  773. ____ __________ __ ___________________ ___ _ ________ ______ ______________ _ __ __________ ___ __ _________
  774. / __ \/ ____/ __ \/ / / / ____/ ___/_ __/ / | / | / / _/ |/ / |/_ __/ _/ __ \/ | / / / ____/ __ \/ | / |/ / ____/
  775. / /_/ / __/ / / / / / / / __/ \__ \ / / / /| | / |/ // // /|_/ / /| | / / / // / / / |/ / / /_ / /_/ / /| | / /|_/ / __/
  776. / _, _/ /___/ /_/ / /_/ / /___ ___/ // / / ___ |/ /| // // / / / ___ |/ / _/ // /_/ / /| / / __/ / _, _/ ___ |/ / / / /___
  777. /_/ |_/_____/\___\_\____/_____//____//_/ /_/ |_/_/ |_/___/_/ /_/_/ |_/_/ /___/\____/_/ |_/ /_/ /_/ |_/_/ |_/_/ /_/_____/
  778. */
  779. (function() {
  780. var lastTime = 0;
  781. var vendors = ['ms', 'moz', 'webkit', 'o'];
  782. for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
  783. window.requestAnimationFrame = window[vendors[x]+'RequestAnimationFrame'];
  784. window.cancelAnimationFrame = window[vendors[x]+'CancelAnimationFrame']
  785. || window[vendors[x]+'CancelRequestAnimationFrame'];
  786. }
  787. if (!window.requestAnimationFrame)
  788. window.requestAnimationFrame = function(callback, element) {
  789. var currTime = new Date().getTime();
  790. var timeToCall = Math.max(0, 16 - (currTime - lastTime));
  791. var id = window.setTimeout(function() { callback(currTime + timeToCall); },
  792. timeToCall);
  793. lastTime = currTime + timeToCall;
  794. return id;
  795. };
  796. if (!window.cancelAnimationFrame)
  797. window.cancelAnimationFrame = function(id) {
  798. clearTimeout(id);
  799. };
  800. }());
  801. /*
  802. ____ __ __ _____________ _______
  803. / __ \/ / / / / / ____/ _/ | / / ___/
  804. / /_/ / / / / / / / __ / // |/ /\__ \
  805. / ____/ /___/ /_/ / /_/ // // /| /___/ /
  806. /_/ /_____/\____/\____/___/_/ |_//____/
  807. */
  808. // translate3d
  809. (function($) {
  810. $.fn.translate3d = function(opt) {
  811. opt = $.extend({x:0,y:0,z:0}, opt);
  812. var t = "translate3d("+opt.x+"px, "+opt.y+"px,"+opt.z+"px)";
  813. this.css({
  814. "-webkit-transform": t,
  815. "-moz-transform": t,
  816. "-ms-transform": t,
  817. "-o-transform": t,
  818. "transform": t
  819. });
  820. };
  821. })(jQuery);