jee.js 32 KB

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