comparison presen/theme/js/slides.js @ 41:8ee3d6448aa6 draft

add css js thema
author Nobuyasu Oshiro <dimolto@cr.ie.u-ryukyu.ac.jp>
date Tue, 28 Feb 2012 22:31:20 +0900
parents
children
comparison
equal deleted inserted replaced
40:ad0ab1378ae6 41:8ee3d6448aa6
1 function main() {
2 // Since we don't have the fallback of attachEvent and
3 // other IE only stuff we won't try to run JS for IE.
4 // It will run though when using Google Chrome Frame
5 if (document.all) { return; }
6
7 var currentSlideNo;
8 var notesOn = false;
9 var expanded = false;
10 var hiddenContext = false;
11 var blanked = false;
12 var slides = document.getElementsByClassName('slide');
13 var touchStartX = 0;
14 var spaces = /\s+/, a1 = [''];
15 var tocOpened = false;
16 var helpOpened = false;
17 var overviewActive = false;
18 var modifierKeyDown = false;
19 var scale = 1;
20 var showingPresenterView = false;
21 var presenterViewWin = null;
22 var isPresenterView = false;
23
24 var str2array = function(s) {
25 if (typeof s == 'string' || s instanceof String) {
26 if (s.indexOf(' ') < 0) {
27 a1[0] = s;
28 return a1;
29 } else {
30 return s.split(spaces);
31 }
32 }
33 return s;
34 };
35
36 var trim = function(str) {
37 return str.replace(/^\s\s*/, '').replace(/\s\s*$/, '');
38 };
39
40 var addClass = function(node, classStr) {
41 classStr = str2array(classStr);
42 var cls = ' ' + node.className + ' ';
43 for (var i = 0, len = classStr.length, c; i < len; ++i) {
44 c = classStr[i];
45 if (c && cls.indexOf(' ' + c + ' ') < 0) {
46 cls += c + ' ';
47 }
48 }
49 node.className = trim(cls);
50 };
51
52 var removeClass = function(node, classStr) {
53 var cls;
54 if (!node) {
55 throw 'no node provided';
56 }
57 if (classStr !== undefined) {
58 classStr = str2array(classStr);
59 cls = ' ' + node.className + ' ';
60 for (var i = 0, len = classStr.length; i < len; ++i) {
61 cls = cls.replace(' ' + classStr[i] + ' ', ' ');
62 }
63 cls = trim(cls);
64 } else {
65 cls = '';
66 }
67 if (node.className != cls) {
68 node.className = cls;
69 }
70 };
71
72 var getSlideEl = function(slideNo) {
73 if (slideNo > 0) {
74 return slides[slideNo - 1];
75 } else {
76 return null;
77 }
78 };
79
80 var getSlideTitle = function(slideNo) {
81 var el = getSlideEl(slideNo);
82 if (el) {
83 var headers = el.getElementsByTagName('header');
84 if (headers.length > 0) {
85 return el.getElementsByTagName('header')[0].innerText;
86 }
87 }
88 return null;
89 };
90
91 var getSlidePresenterNote = function(slideNo) {
92 var el = getSlideEl(slideNo);
93 if (el) {
94 var n = el.getElementsByClassName('presenter_notes');
95 if (n.length > 0) {
96 return n[0];
97 }
98 }
99 return null;
100 };
101
102 var changeSlideElClass = function(slideNo, className) {
103 var el = getSlideEl(slideNo);
104 if (el) {
105 removeClass(el, 'far-past past current future far-future');
106 addClass(el, className);
107 }
108 };
109
110 var updateSlideClasses = function(updateOther) {
111 window.location.hash = (isPresenterView ? "presenter" : "slide") + currentSlideNo;
112
113 for (var i=1; i<currentSlideNo-1; i++) {
114 changeSlideElClass(i, 'far-past');
115 }
116
117 changeSlideElClass(currentSlideNo - 1, 'past');
118 changeSlideElClass(currentSlideNo, 'current');
119 changeSlideElClass(currentSlideNo + 1, 'future');
120
121 for (i=currentSlideNo+2; i<slides.length+1; i++) {
122 changeSlideElClass(i, 'far-future');
123 }
124
125 highlightCurrentTocLink();
126
127 processContext();
128
129 document.getElementsByTagName('title')[0].innerText = getSlideTitle(currentSlideNo);
130
131 updatePresenterNotes();
132
133 if (updateOther) { updateOtherPage(); }
134 };
135
136 var updatePresenterNotes = function() {
137 if (!isPresenterView) { return; }
138
139 var existingNote = document.getElementById('current_presenter_notes');
140 var currentNote = getSlidePresenterNote(currentSlideNo).cloneNode(true);
141 currentNote.setAttribute('id', 'presenter_note');
142
143 existingNote.replaceChild(currentNote, document.getElementById('presenter_note'));
144 };
145
146 var highlightCurrentTocLink = function() {
147 var toc = document.getElementById('toc');
148
149 if (toc) {
150 var tocRows = toc.getElementsByTagName('tr');
151 for (var i=0; i<tocRows.length; i++) {
152 removeClass(tocRows.item(i), 'active');
153 }
154
155 var currentTocRow = document.getElementById('toc-row-' + currentSlideNo);
156 if (currentTocRow) {
157 addClass(currentTocRow, 'active');
158 }
159 }
160 };
161
162 var updateOtherPage = function() {
163 if (!showingPresenterView) { return; }
164
165 var w = isPresenterView ? window.opener : presenterViewWin;
166 w.postMessage('slide#' + currentSlideNo, '*');
167 };
168
169 var nextSlide = function() {
170 if (currentSlideNo < slides.length) {
171 currentSlideNo++;
172 }
173 updateSlideClasses(true);
174 };
175
176 var prevSlide = function() {
177 if (currentSlideNo > 1) {
178 currentSlideNo--;
179 }
180 updateSlideClasses(true);
181 };
182
183 var showNotes = function() {
184 var notes = getSlideEl(currentSlideNo).getElementsByClassName('notes');
185 for (var i = 0, len = notes.length; i < len; i++) {
186 notes.item(i).style.display = (notesOn) ? 'none':'block';
187 }
188 notesOn = !notesOn;
189 };
190
191 var showSlideNumbers = function() {
192 var asides = document.getElementsByClassName('page_number');
193 var hidden = asides[0].style.display != 'block';
194 for (var i = 0; i < asides.length; i++) {
195 asides.item(i).style.display = hidden ? 'block' : 'none';
196 }
197 };
198
199 var showSlideSources = function() {
200 var asides = document.getElementsByClassName('source');
201 var hidden = asides[0].style.display != 'block';
202 for (var i = 0; i < asides.length; i++) {
203 asides.item(i).style.display = hidden ? 'block' : 'none';
204 }
205 };
206
207 var showToc = function() {
208 if (helpOpened) {
209 showHelp();
210 }
211 var toc = document.getElementById('toc');
212 if (toc) {
213 toc.style.marginLeft = tocOpened ? '-' + (toc.clientWidth + 20) + 'px' : '0px';
214 tocOpened = !tocOpened;
215 }
216 updateOverview();
217 };
218
219 var showHelp = function() {
220 if (tocOpened) {
221 showToc();
222 }
223
224 var help = document.getElementById('help');
225
226 if (help) {
227 help.style.marginLeft = helpOpened ? '-' + (help.clientWidth + 20) + 'px' : '0px';
228 helpOpened = !helpOpened;
229 }
230 };
231
232 var showPresenterView = function() {
233 if (isPresenterView) { return; }
234
235 if (showingPresenterView) {
236 presenterViewWin.close();
237 presenterViewWin = null;
238 showingPresenterView = false;
239 } else {
240 presenterViewWin = open(window.location.pathname + "#presenter" + currentSlideNo, 'presenter_notes',
241 'directories=no,location=no,toolbar=no,menubar=no,copyhistory=no');
242 showingPresenterView = true;
243 }
244 };
245
246 var switch3D = function() {
247 if (document.body.className.indexOf('three-d') == -1) {
248 document.getElementsByClassName('presentation')[0].style.webkitPerspective = '1000px';
249 document.body.className += ' three-d';
250 } else {
251 window.setTimeout('document.getElementsByClassName(\'presentation\')[0].style.webkitPerspective = \'0\';', 2000);
252 document.body.className = document.body.className.replace(/three-d/, '');
253 }
254 };
255
256 var toggleOverview = function() {
257 if (!overviewActive) {
258 addClass(document.body, 'expose');
259 overviewActive = true;
260 setScale(1);
261 } else {
262 removeClass(document.body, 'expose');
263 overviewActive = false;
264 if (expanded) {
265 setScale(scale); // restore scale
266 }
267 }
268 processContext();
269 updateOverview();
270 };
271
272 var updateOverview = function() {
273 try {
274 var presentation = document.getElementsByClassName('presentation')[0];
275 } catch (e) {
276 return;
277 }
278
279 if (isPresenterView) {
280 var action = overviewActive ? removeClass : addClass;
281 action(document.body, 'presenter_view');
282 }
283
284 var toc = document.getElementById('toc');
285
286 if (!toc) {
287 return;
288 }
289
290 if (!tocOpened || !overviewActive) {
291 presentation.style.marginLeft = '0px';
292 presentation.style.width = '100%';
293 } else {
294 presentation.style.marginLeft = toc.clientWidth + 'px';
295 presentation.style.width = (presentation.clientWidth - toc.clientWidth) + 'px';
296 }
297 };
298
299 var computeScale = function() {
300 var cSlide = document.getElementsByClassName('current')[0];
301 var sx = cSlide.clientWidth / window.innerWidth;
302 var sy = cSlide.clientHeight / window.innerHeight;
303 return 1 / Math.max(sx, sy);
304 };
305
306 var setScale = function(scale) {
307 var presentation = document.getElementsByClassName('slides')[0];
308 var transform = 'scale(' + scale + ')';
309 presentation.style.MozTransform = transform;
310 presentation.style.WebkitTransform = transform;
311 presentation.style.OTransform = transform;
312 presentation.style.msTransform = transform;
313 presentation.style.transform = transform;
314 };
315
316 var expandSlides = function() {
317 if (overviewActive) {
318 return;
319 }
320 if (expanded) {
321 setScale(1);
322 expanded = false;
323 } else {
324 scale = computeScale();
325 setScale(scale);
326 expanded = true;
327 }
328 };
329
330 var showContext = function() {
331 try {
332 var presentation = document.getElementsByClassName('slides')[0];
333 removeClass(presentation, 'nocontext');
334 } catch (e) {}
335 };
336
337 var hideContext = function() {
338 try {
339 var presentation = document.getElementsByClassName('slides')[0];
340 addClass(presentation, 'nocontext');
341 } catch (e) {}
342 };
343
344 var processContext = function() {
345 if (hiddenContext) {
346 hideContext();
347 } else {
348 showContext();
349 }
350 };
351
352 var toggleContext = function() {
353 hiddenContext = !hiddenContext;
354 processContext();
355 };
356
357 var toggleBlank = function() {
358 blank_elem = document.getElementById('blank');
359
360 blank_elem.style.display = blanked ? 'none' : 'block';
361
362 blanked = !blanked;
363 };
364
365 var isModifierKey = function(keyCode) {
366 switch (keyCode) {
367 case 16: // shift
368 case 17: // ctrl
369 case 18: // alt
370 case 91: // command
371 return true;
372 break;
373 default:
374 return false;
375 break;
376 }
377 };
378
379 var checkModifierKeyUp = function(event) {
380 if (isModifierKey(event.keyCode)) {
381 modifierKeyDown = false;
382 }
383 };
384
385 var checkModifierKeyDown = function(event) {
386 if (isModifierKey(event.keyCode)) {
387 modifierKeyDown = true;
388 }
389 };
390
391 var handleBodyKeyDown = function(event) {
392 switch (event.keyCode) {
393 case 13: // Enter
394 if (overviewActive) {
395 toggleOverview();
396 }
397 break;
398 case 27: // ESC
399 toggleOverview();
400 break;
401 case 37: // left arrow
402 case 33: // page up
403 event.preventDefault();
404 prevSlide();
405 break;
406 case 39: // right arrow
407 case 32: // space
408 case 34: // page down
409 event.preventDefault();
410 nextSlide();
411 break;
412 case 50: // 2
413 if (!modifierKeyDown) {
414 showNotes();
415 }
416 break;
417 case 51: // 3
418 if (!modifierKeyDown && !overviewActive) {
419 switch3D();
420 }
421 break;
422 case 190: // .
423 case 48: // 0
424 case 66: // b
425 if (!modifierKeyDown && !overviewActive) {
426 toggleBlank();
427 }
428 break;
429 case 67: // c
430 if (!modifierKeyDown && !overviewActive) {
431 toggleContext();
432 }
433 break;
434 case 69: // e
435 if (!modifierKeyDown && !overviewActive) {
436 expandSlides();
437 }
438 break;
439 case 72: // h
440 showHelp();
441 break;
442 case 78: // n
443 if (!modifierKeyDown && !overviewActive) {
444 showSlideNumbers();
445 }
446 break;
447 case 80: // p
448 if (!modifierKeyDown && !overviewActive) {
449 showPresenterView();
450 }
451 break;
452 case 83: // s
453 if (!modifierKeyDown && !overviewActive) {
454 showSlideSources();
455 }
456 break;
457 case 84: // t
458 showToc();
459 break;
460 }
461 };
462
463 var handleWheel = function(event) {
464 if (tocOpened || helpOpened || overviewActive) {
465 return;
466 }
467
468 var delta = 0;
469
470 if (!event) {
471 event = window.event;
472 }
473
474 if (event.wheelDelta) {
475 delta = event.wheelDelta/120;
476 if (window.opera) delta = -delta;
477 } else if (event.detail) {
478 delta = -event.detail/3;
479 }
480
481 if (delta && delta <0) {
482 nextSlide();
483 } else if (delta) {
484 prevSlide();
485 }
486 };
487
488 var addSlideClickListeners = function() {
489 for (var i=0; i < slides.length; i++) {
490 var slide = slides.item(i);
491 slide.num = i + 1;
492 slide.addEventListener('click', function(e) {
493 if (overviewActive) {
494 currentSlideNo = this.num;
495 toggleOverview();
496 updateSlideClasses(true);
497 e.preventDefault();
498 }
499 return false;
500 }, true);
501 }
502 };
503
504 var addRemoteWindowControls = function() {
505 window.addEventListener("message", function(e) {
506 if (e.data.indexOf("slide#") != -1) {
507 currentSlideNo = Number(e.data.replace('slide#', ''));
508 updateSlideClasses(false);
509 }
510 }, false);
511 };
512
513 var addTouchListeners = function() {
514 document.addEventListener('touchstart', function(e) {
515 touchStartX = e.touches[0].pageX;
516 }, false);
517 document.addEventListener('touchend', function(e) {
518 var pixelsMoved = touchStartX - e.changedTouches[0].pageX;
519 var SWIPE_SIZE = 150;
520 if (pixelsMoved > SWIPE_SIZE) {
521 nextSlide();
522 }
523 else if (pixelsMoved < -SWIPE_SIZE) {
524 prevSlide();
525 }
526 }, false);
527 };
528
529 var addTocLinksListeners = function() {
530 var toc = document.getElementById('toc');
531 if (toc) {
532 var tocLinks = toc.getElementsByTagName('a');
533 for (var i=0; i < tocLinks.length; i++) {
534 tocLinks.item(i).addEventListener('click', function(e) {
535 currentSlideNo = Number(this.attributes['href'].value.replace('#slide', ''));
536 updateSlideClasses(true);
537 e.preventDefault();
538 }, true);
539 }
540 }
541 };
542
543 // initialize
544
545 (function() {
546 if (window.location.hash == "") {
547 currentSlideNo = 1;
548 } else if (window.location.hash.indexOf("#presenter") != -1) {
549 currentSlideNo = Number(window.location.hash.replace('#presenter', ''));
550 isPresenterView = true;
551 showingPresenterView = true;
552 presenterViewWin = window;
553 addClass(document.body, 'presenter_view');
554 } else {
555 currentSlideNo = Number(window.location.hash.replace('#slide', ''));
556 }
557
558 document.addEventListener('keyup', checkModifierKeyUp, false);
559 document.addEventListener('keydown', handleBodyKeyDown, false);
560 document.addEventListener('keydown', checkModifierKeyDown, false);
561 document.addEventListener('DOMMouseScroll', handleWheel, false);
562
563 window.onmousewheel = document.onmousewheel = handleWheel;
564 window.onresize = expandSlides;
565
566 for (var i = 0, el; el = slides[i]; i++) {
567 addClass(el, 'slide far-future');
568 }
569 updateSlideClasses(false);
570
571 // add support for finger events (filter it by property detection?)
572 addTouchListeners();
573
574 addTocLinksListeners();
575
576 addSlideClickListeners();
577
578 addRemoteWindowControls();
579 })();
580 }