scateu.me-jekyll-theme 0.1.1 → 0.1.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (112) hide show
  1. checksums.yaml +4 -4
  2. data/css/main.scss +56 -0
  3. data/feed.xml +30 -0
  4. data/index.html +22 -0
  5. data/js/.DS_Store +0 -0
  6. data/js/flowchart-latest.js +991 -0
  7. data/js/jquery-3.1.1.min.js +4 -0
  8. data/js/qrcode.min.js +1 -0
  9. data/js/raphael-min.js +11 -0
  10. data/js/reveal.js/.gitignore +13 -0
  11. data/js/reveal.js/.travis.yml +5 -0
  12. data/js/reveal.js/CONTRIBUTING.md +23 -0
  13. data/js/reveal.js/Gruntfile.js +176 -0
  14. data/js/reveal.js/LICENSE +19 -0
  15. data/js/reveal.js/README.md +1148 -0
  16. data/js/reveal.js/bower.json +27 -0
  17. data/js/reveal.js/css/print/paper.css +202 -0
  18. data/js/reveal.js/css/print/pdf.css +160 -0
  19. data/js/reveal.js/css/reveal.css +1344 -0
  20. data/js/reveal.js/css/reveal.scss +1412 -0
  21. data/js/reveal.js/css/theme/README.md +21 -0
  22. data/js/reveal.js/css/theme/beige.css +291 -0
  23. data/js/reveal.js/css/theme/black.css +287 -0
  24. data/js/reveal.js/css/theme/blood.css +310 -0
  25. data/js/reveal.js/css/theme/league.css +293 -0
  26. data/js/reveal.js/css/theme/moon.css +291 -0
  27. data/js/reveal.js/css/theme/night.css +285 -0
  28. data/js/reveal.js/css/theme/serif.css +287 -0
  29. data/js/reveal.js/css/theme/simple.css +287 -0
  30. data/js/reveal.js/css/theme/sky.css +294 -0
  31. data/js/reveal.js/css/theme/solarized.css +291 -0
  32. data/js/reveal.js/css/theme/source/beige.scss +39 -0
  33. data/js/reveal.js/css/theme/source/black.scss +49 -0
  34. data/js/reveal.js/css/theme/source/blood.scss +79 -0
  35. data/js/reveal.js/css/theme/source/league.scss +34 -0
  36. data/js/reveal.js/css/theme/source/moon.scss +57 -0
  37. data/js/reveal.js/css/theme/source/night.scss +35 -0
  38. data/js/reveal.js/css/theme/source/serif.scss +35 -0
  39. data/js/reveal.js/css/theme/source/simple.scss +38 -0
  40. data/js/reveal.js/css/theme/source/sky.scss +46 -0
  41. data/js/reveal.js/css/theme/source/solarized.scss +63 -0
  42. data/js/reveal.js/css/theme/source/white.scss +49 -0
  43. data/js/reveal.js/css/theme/template/mixins.scss +29 -0
  44. data/js/reveal.js/css/theme/template/settings.scss +43 -0
  45. data/js/reveal.js/css/theme/template/theme.scss +346 -0
  46. data/js/reveal.js/css/theme/white.css +287 -0
  47. data/js/reveal.js/demo.html +410 -0
  48. data/js/reveal.js/index.html +50 -0
  49. data/js/reveal.js/js/reveal.js +4744 -0
  50. data/js/reveal.js/lib/css/zenburn.css +80 -0
  51. data/js/reveal.js/lib/font/league-gothic/LICENSE +2 -0
  52. data/js/reveal.js/lib/font/league-gothic/league-gothic.css +10 -0
  53. data/js/reveal.js/lib/font/league-gothic/league-gothic.eot +0 -0
  54. data/js/reveal.js/lib/font/league-gothic/league-gothic.ttf +0 -0
  55. data/js/reveal.js/lib/font/league-gothic/league-gothic.woff +0 -0
  56. data/js/reveal.js/lib/font/source-sans-pro/LICENSE +45 -0
  57. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-italic.eot +0 -0
  58. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-italic.ttf +0 -0
  59. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-italic.woff +0 -0
  60. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-regular.eot +0 -0
  61. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-regular.ttf +0 -0
  62. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-regular.woff +0 -0
  63. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibold.eot +0 -0
  64. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibold.ttf +0 -0
  65. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibold.woff +0 -0
  66. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibolditalic.eot +0 -0
  67. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibolditalic.ttf +0 -0
  68. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro-semibolditalic.woff +0 -0
  69. data/js/reveal.js/lib/font/source-sans-pro/source-sans-pro.css +39 -0
  70. data/js/reveal.js/lib/js/classList.js +2 -0
  71. data/js/reveal.js/lib/js/head.min.js +9 -0
  72. data/js/reveal.js/lib/js/html5shiv.js +7 -0
  73. data/js/reveal.js/package.json +44 -0
  74. data/js/reveal.js/plugin/highlight/highlight.js +31 -0
  75. data/js/reveal.js/plugin/markdown/example.html +129 -0
  76. data/js/reveal.js/plugin/markdown/markdown.js +405 -0
  77. data/js/reveal.js/plugin/markdown/marked.js +6 -0
  78. data/js/reveal.js/plugin/math/math.js +67 -0
  79. data/js/reveal.js/plugin/multiplex/client.js +13 -0
  80. data/js/reveal.js/plugin/multiplex/index.js +64 -0
  81. data/js/reveal.js/plugin/multiplex/master.js +31 -0
  82. data/js/reveal.js/plugin/multiplex/package.json +19 -0
  83. data/js/reveal.js/plugin/notes-server/client.js +65 -0
  84. data/js/reveal.js/plugin/notes-server/index.js +69 -0
  85. data/js/reveal.js/plugin/notes-server/notes.html +407 -0
  86. data/js/reveal.js/plugin/notes/notes.html +414 -0
  87. data/js/reveal.js/plugin/notes/notes.js +136 -0
  88. data/js/reveal.js/plugin/print-pdf/print-pdf.js +48 -0
  89. data/js/reveal.js/plugin/search/search.js +196 -0
  90. data/js/reveal.js/plugin/zoom-js/zoom.js +278 -0
  91. data/js/reveal.js/test/examples/assets/image1.png +0 -0
  92. data/js/reveal.js/test/examples/assets/image2.png +0 -0
  93. data/js/reveal.js/test/examples/barebones.html +41 -0
  94. data/js/reveal.js/test/examples/embedded-media.html +49 -0
  95. data/js/reveal.js/test/examples/math.html +185 -0
  96. data/js/reveal.js/test/examples/slide-backgrounds.html +144 -0
  97. data/js/reveal.js/test/examples/slide-transitions.html +101 -0
  98. data/js/reveal.js/test/qunit-1.12.0.css +244 -0
  99. data/js/reveal.js/test/qunit-1.12.0.js +2212 -0
  100. data/js/reveal.js/test/test-markdown-element-attributes.html +134 -0
  101. data/js/reveal.js/test/test-markdown-element-attributes.js +46 -0
  102. data/js/reveal.js/test/test-markdown-slide-attributes.html +128 -0
  103. data/js/reveal.js/test/test-markdown-slide-attributes.js +47 -0
  104. data/js/reveal.js/test/test-markdown.html +52 -0
  105. data/js/reveal.js/test/test-markdown.js +15 -0
  106. data/js/reveal.js/test/test-pdf.html +83 -0
  107. data/js/reveal.js/test/test-pdf.js +15 -0
  108. data/js/reveal.js/test/test.html +86 -0
  109. data/js/reveal.js/test/test.js +597 -0
  110. data/js/sequence-diagram-min.js +8 -0
  111. data/js/underscore-min.js +1 -0
  112. metadata +111 -1
@@ -0,0 +1,48 @@
1
+ /**
2
+ * phantomjs script for printing presentations to PDF.
3
+ *
4
+ * Example:
5
+ * phantomjs print-pdf.js "http://lab.hakim.se/reveal-js?print-pdf" reveal-demo.pdf
6
+ *
7
+ * By Manuel Bieh (https://github.com/manuelbieh)
8
+ */
9
+
10
+ // html2pdf.js
11
+ var page = new WebPage();
12
+ var system = require( 'system' );
13
+
14
+ var slideWidth = system.args[3] ? system.args[3].split( 'x' )[0] : 960;
15
+ var slideHeight = system.args[3] ? system.args[3].split( 'x' )[1] : 700;
16
+
17
+ page.viewportSize = {
18
+ width: slideWidth,
19
+ height: slideHeight
20
+ };
21
+
22
+ // TODO
23
+ // Something is wrong with these config values. An input
24
+ // paper width of 1920px actually results in a 756px wide
25
+ // PDF.
26
+ page.paperSize = {
27
+ width: Math.round( slideWidth * 2 ),
28
+ height: Math.round( slideHeight * 2 ),
29
+ border: 0
30
+ };
31
+
32
+ var inputFile = system.args[1] || 'index.html?print-pdf';
33
+ var outputFile = system.args[2] || 'slides.pdf';
34
+
35
+ if( outputFile.match( /\.pdf$/gi ) === null ) {
36
+ outputFile += '.pdf';
37
+ }
38
+
39
+ console.log( 'Printing PDF (Paper size: '+ page.paperSize.width + 'x' + page.paperSize.height +')' );
40
+
41
+ page.open( inputFile, function( status ) {
42
+ window.setTimeout( function() {
43
+ console.log( 'Printed successfully' );
44
+ page.render( outputFile );
45
+ phantom.exit();
46
+ }, 1000 );
47
+ } );
48
+
@@ -0,0 +1,196 @@
1
+ /*
2
+ * Handles finding a text string anywhere in the slides and showing the next occurrence to the user
3
+ * by navigatating to that slide and highlighting it.
4
+ *
5
+ * By Jon Snyder <snyder.jon@gmail.com>, February 2013
6
+ */
7
+
8
+ var RevealSearch = (function() {
9
+
10
+ var matchedSlides;
11
+ var currentMatchedIndex;
12
+ var searchboxDirty;
13
+ var myHilitor;
14
+
15
+ // Original JavaScript code by Chirp Internet: www.chirp.com.au
16
+ // Please acknowledge use of this code by including this header.
17
+ // 2/2013 jon: modified regex to display any match, not restricted to word boundaries.
18
+
19
+ function Hilitor(id, tag)
20
+ {
21
+
22
+ var targetNode = document.getElementById(id) || document.body;
23
+ var hiliteTag = tag || "EM";
24
+ var skipTags = new RegExp("^(?:" + hiliteTag + "|SCRIPT|FORM|SPAN)$");
25
+ var colors = ["#ff6", "#a0ffff", "#9f9", "#f99", "#f6f"];
26
+ var wordColor = [];
27
+ var colorIdx = 0;
28
+ var matchRegex = "";
29
+ var matchingSlides = [];
30
+
31
+ this.setRegex = function(input)
32
+ {
33
+ input = input.replace(/^[^\w]+|[^\w]+$/g, "").replace(/[^\w'-]+/g, "|");
34
+ matchRegex = new RegExp("(" + input + ")","i");
35
+ }
36
+
37
+ this.getRegex = function()
38
+ {
39
+ return matchRegex.toString().replace(/^\/\\b\(|\)\\b\/i$/g, "").replace(/\|/g, " ");
40
+ }
41
+
42
+ // recursively apply word highlighting
43
+ this.hiliteWords = function(node)
44
+ {
45
+ if(node == undefined || !node) return;
46
+ if(!matchRegex) return;
47
+ if(skipTags.test(node.nodeName)) return;
48
+
49
+ if(node.hasChildNodes()) {
50
+ for(var i=0; i < node.childNodes.length; i++)
51
+ this.hiliteWords(node.childNodes[i]);
52
+ }
53
+ if(node.nodeType == 3) { // NODE_TEXT
54
+ if((nv = node.nodeValue) && (regs = matchRegex.exec(nv))) {
55
+ //find the slide's section element and save it in our list of matching slides
56
+ var secnode = node.parentNode;
57
+ while (secnode.nodeName != 'SECTION') {
58
+ secnode = secnode.parentNode;
59
+ }
60
+
61
+ var slideIndex = Reveal.getIndices(secnode);
62
+ var slidelen = matchingSlides.length;
63
+ var alreadyAdded = false;
64
+ for (var i=0; i < slidelen; i++) {
65
+ if ( (matchingSlides[i].h === slideIndex.h) && (matchingSlides[i].v === slideIndex.v) ) {
66
+ alreadyAdded = true;
67
+ }
68
+ }
69
+ if (! alreadyAdded) {
70
+ matchingSlides.push(slideIndex);
71
+ }
72
+
73
+ if(!wordColor[regs[0].toLowerCase()]) {
74
+ wordColor[regs[0].toLowerCase()] = colors[colorIdx++ % colors.length];
75
+ }
76
+
77
+ var match = document.createElement(hiliteTag);
78
+ match.appendChild(document.createTextNode(regs[0]));
79
+ match.style.backgroundColor = wordColor[regs[0].toLowerCase()];
80
+ match.style.fontStyle = "inherit";
81
+ match.style.color = "#000";
82
+
83
+ var after = node.splitText(regs.index);
84
+ after.nodeValue = after.nodeValue.substring(regs[0].length);
85
+ node.parentNode.insertBefore(match, after);
86
+ }
87
+ }
88
+ };
89
+
90
+ // remove highlighting
91
+ this.remove = function()
92
+ {
93
+ var arr = document.getElementsByTagName(hiliteTag);
94
+ while(arr.length && (el = arr[0])) {
95
+ el.parentNode.replaceChild(el.firstChild, el);
96
+ }
97
+ };
98
+
99
+ // start highlighting at target node
100
+ this.apply = function(input)
101
+ {
102
+ if(input == undefined || !input) return;
103
+ this.remove();
104
+ this.setRegex(input);
105
+ this.hiliteWords(targetNode);
106
+ return matchingSlides;
107
+ };
108
+
109
+ }
110
+
111
+ function openSearch() {
112
+ //ensure the search term input dialog is visible and has focus:
113
+ var inputbox = document.getElementById("searchinput");
114
+ inputbox.style.display = "inline";
115
+ inputbox.focus();
116
+ inputbox.select();
117
+ }
118
+
119
+ function toggleSearch() {
120
+ var inputbox = document.getElementById("searchinput");
121
+ if (inputbox.style.display !== "inline") {
122
+ openSearch();
123
+ }
124
+ else {
125
+ inputbox.style.display = "none";
126
+ myHilitor.remove();
127
+ }
128
+ }
129
+
130
+ function doSearch() {
131
+ //if there's been a change in the search term, perform a new search:
132
+ if (searchboxDirty) {
133
+ var searchstring = document.getElementById("searchinput").value;
134
+
135
+ //find the keyword amongst the slides
136
+ myHilitor = new Hilitor("slidecontent");
137
+ matchedSlides = myHilitor.apply(searchstring);
138
+ currentMatchedIndex = 0;
139
+ }
140
+
141
+ //navigate to the next slide that has the keyword, wrapping to the first if necessary
142
+ if (matchedSlides.length && (matchedSlides.length <= currentMatchedIndex)) {
143
+ currentMatchedIndex = 0;
144
+ }
145
+ if (matchedSlides.length > currentMatchedIndex) {
146
+ Reveal.slide(matchedSlides[currentMatchedIndex].h, matchedSlides[currentMatchedIndex].v);
147
+ currentMatchedIndex++;
148
+ }
149
+ }
150
+
151
+ var dom = {};
152
+ dom.wrapper = document.querySelector( '.reveal' );
153
+
154
+ if( !dom.wrapper.querySelector( '.searchbox' ) ) {
155
+ var searchElement = document.createElement( 'div' );
156
+ searchElement.id = "searchinputdiv";
157
+ searchElement.classList.add( 'searchdiv' );
158
+ searchElement.style.position = 'absolute';
159
+ searchElement.style.top = '10px';
160
+ searchElement.style.left = '10px';
161
+ //embedded base64 search icon Designed by Sketchdock - http://www.sketchdock.com/:
162
+ searchElement.innerHTML = '<span><input type="search" id="searchinput" class="searchinput" style="vertical-align: top;"/><img src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABgAAAAYCAYAAADgdz34AAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAJiSURBVHjatFZNaxNBGH5md+Mmu92NVdKDRipSAyqCghgQD4L4cRe86UUtAQ+eFCxoa4/25EXBFi8eBE+eRPoDhB6KgiiixdAPCEkx2pjvTXadd9yNsflwuyUDD/O+u8PzzDPvzOwyx3EwyCZhwG3gAkp7MnpjgbopjsltcD4gjuXZZKeAR348MYLYTm3LzOs/y3j3JTfZxgXWXmTuwPHIc4VmoOmv5IrI53+AO2DdHLjkDWQ3GoEEVFXtXQOvkSnPWcyUceviLhwbDYv8/XIVj97kse7TodLvZXxYxrPUHkQ1ufXs3FEdybEIxucySOesoNvUgWU1cP3MkCBfTFdw9fGaAMVmRELq7LBw2Q3/FaAxxWIRpw+ZIr/7IouPqzUBiqmdHAv7EuhRAwf1er2Vy4x1jW3b2d5Jfvu5IPp7l2LYbcgCFFNb+FoJ7oBqEAqFMPNqFcmEgVMJDfMT+1tvN0pNjERlMS6QA5pFOKxiKVPFhakPeL3It+WGJUDxt2wFR+JhzI7v5ctkd8DXOZAkCYYxhO+lKm4+Xfqz/rIixBuNBl7eOYzkQQNzqX249mRl6zUgEcYkaJrGhUwBinVdh6IouPzwE6/DL5w4oLkH8y981aDf+uq6hlKpJESiUdNfDZi7/ehG9K6KfiA3pml0PLcsq+cSMTj2NL9ukc4UOmz7AZ3+crkC4mHujFvXNaMFB3bEr8xPS6p5O+jXxq4VZtaen7/PwzrntjcLUE0iHPS1Ud1cdiEJl/8WivZk0wXd7zWOMkeF8s0CcAmkNrC2nvXZDbbbN73ccYnZoH9bfgswAFzAe9/h3dbKAAAAAElFTkSuQmCC" id="searchbutton" class="searchicon" style="vertical-align: top; margin-top: -1px;"/></span>';
163
+ dom.wrapper.appendChild( searchElement );
164
+ }
165
+
166
+ document.getElementById("searchbutton").addEventListener( 'click', function(event) {
167
+ doSearch();
168
+ }, false );
169
+
170
+ document.getElementById("searchinput").addEventListener( 'keyup', function( event ) {
171
+ switch (event.keyCode) {
172
+ case 13:
173
+ event.preventDefault();
174
+ doSearch();
175
+ searchboxDirty = false;
176
+ break;
177
+ default:
178
+ searchboxDirty = true;
179
+ }
180
+ }, false );
181
+
182
+ // Open the search when the 's' key is hit (yes, this conflicts with the notes plugin, disabling for now)
183
+ /*
184
+ document.addEventListener( 'keydown', function( event ) {
185
+ // Disregard the event if the target is editable or a
186
+ // modifier is present
187
+ if ( document.querySelector( ':focus' ) !== null || event.shiftKey || event.altKey || event.ctrlKey || event.metaKey ) return;
188
+
189
+ if( event.keyCode === 83 ) {
190
+ event.preventDefault();
191
+ openSearch();
192
+ }
193
+ }, false );
194
+ */
195
+ return { open: openSearch };
196
+ })();
@@ -0,0 +1,278 @@
1
+ // Custom reveal.js integration
2
+ (function(){
3
+ var isEnabled = true;
4
+
5
+ document.querySelector( '.reveal .slides' ).addEventListener( 'mousedown', function( event ) {
6
+ var modifier = ( Reveal.getConfig().zoomKey ? Reveal.getConfig().zoomKey : 'alt' ) + 'Key';
7
+
8
+ var zoomPadding = 20;
9
+ var revealScale = Reveal.getScale();
10
+
11
+ if( event[ modifier ] && isEnabled ) {
12
+ event.preventDefault();
13
+
14
+ var bounds = event.target.getBoundingClientRect();
15
+
16
+ zoom.to({
17
+ x: ( bounds.left * revealScale ) - zoomPadding,
18
+ y: ( bounds.top * revealScale ) - zoomPadding,
19
+ width: ( bounds.width * revealScale ) + ( zoomPadding * 2 ),
20
+ height: ( bounds.height * revealScale ) + ( zoomPadding * 2 ),
21
+ pan: false
22
+ });
23
+ }
24
+ } );
25
+
26
+ Reveal.addEventListener( 'overviewshown', function() { isEnabled = false; } );
27
+ Reveal.addEventListener( 'overviewhidden', function() { isEnabled = true; } );
28
+ })();
29
+
30
+ /*!
31
+ * zoom.js 0.3 (modified for use with reveal.js)
32
+ * http://lab.hakim.se/zoom-js
33
+ * MIT licensed
34
+ *
35
+ * Copyright (C) 2011-2014 Hakim El Hattab, http://hakim.se
36
+ */
37
+ var zoom = (function(){
38
+
39
+ // The current zoom level (scale)
40
+ var level = 1;
41
+
42
+ // The current mouse position, used for panning
43
+ var mouseX = 0,
44
+ mouseY = 0;
45
+
46
+ // Timeout before pan is activated
47
+ var panEngageTimeout = -1,
48
+ panUpdateInterval = -1;
49
+
50
+ // Check for transform support so that we can fallback otherwise
51
+ var supportsTransforms = 'WebkitTransform' in document.body.style ||
52
+ 'MozTransform' in document.body.style ||
53
+ 'msTransform' in document.body.style ||
54
+ 'OTransform' in document.body.style ||
55
+ 'transform' in document.body.style;
56
+
57
+ if( supportsTransforms ) {
58
+ // The easing that will be applied when we zoom in/out
59
+ document.body.style.transition = 'transform 0.8s ease';
60
+ document.body.style.OTransition = '-o-transform 0.8s ease';
61
+ document.body.style.msTransition = '-ms-transform 0.8s ease';
62
+ document.body.style.MozTransition = '-moz-transform 0.8s ease';
63
+ document.body.style.WebkitTransition = '-webkit-transform 0.8s ease';
64
+ }
65
+
66
+ // Zoom out if the user hits escape
67
+ document.addEventListener( 'keyup', function( event ) {
68
+ if( level !== 1 && event.keyCode === 27 ) {
69
+ zoom.out();
70
+ }
71
+ } );
72
+
73
+ // Monitor mouse movement for panning
74
+ document.addEventListener( 'mousemove', function( event ) {
75
+ if( level !== 1 ) {
76
+ mouseX = event.clientX;
77
+ mouseY = event.clientY;
78
+ }
79
+ } );
80
+
81
+ /**
82
+ * Applies the CSS required to zoom in, prefers the use of CSS3
83
+ * transforms but falls back on zoom for IE.
84
+ *
85
+ * @param {Object} rect
86
+ * @param {Number} scale
87
+ */
88
+ function magnify( rect, scale ) {
89
+
90
+ var scrollOffset = getScrollOffset();
91
+
92
+ // Ensure a width/height is set
93
+ rect.width = rect.width || 1;
94
+ rect.height = rect.height || 1;
95
+
96
+ // Center the rect within the zoomed viewport
97
+ rect.x -= ( window.innerWidth - ( rect.width * scale ) ) / 2;
98
+ rect.y -= ( window.innerHeight - ( rect.height * scale ) ) / 2;
99
+
100
+ if( supportsTransforms ) {
101
+ // Reset
102
+ if( scale === 1 ) {
103
+ document.body.style.transform = '';
104
+ document.body.style.OTransform = '';
105
+ document.body.style.msTransform = '';
106
+ document.body.style.MozTransform = '';
107
+ document.body.style.WebkitTransform = '';
108
+ }
109
+ // Scale
110
+ else {
111
+ var origin = scrollOffset.x +'px '+ scrollOffset.y +'px',
112
+ transform = 'translate('+ -rect.x +'px,'+ -rect.y +'px) scale('+ scale +')';
113
+
114
+ document.body.style.transformOrigin = origin;
115
+ document.body.style.OTransformOrigin = origin;
116
+ document.body.style.msTransformOrigin = origin;
117
+ document.body.style.MozTransformOrigin = origin;
118
+ document.body.style.WebkitTransformOrigin = origin;
119
+
120
+ document.body.style.transform = transform;
121
+ document.body.style.OTransform = transform;
122
+ document.body.style.msTransform = transform;
123
+ document.body.style.MozTransform = transform;
124
+ document.body.style.WebkitTransform = transform;
125
+ }
126
+ }
127
+ else {
128
+ // Reset
129
+ if( scale === 1 ) {
130
+ document.body.style.position = '';
131
+ document.body.style.left = '';
132
+ document.body.style.top = '';
133
+ document.body.style.width = '';
134
+ document.body.style.height = '';
135
+ document.body.style.zoom = '';
136
+ }
137
+ // Scale
138
+ else {
139
+ document.body.style.position = 'relative';
140
+ document.body.style.left = ( - ( scrollOffset.x + rect.x ) / scale ) + 'px';
141
+ document.body.style.top = ( - ( scrollOffset.y + rect.y ) / scale ) + 'px';
142
+ document.body.style.width = ( scale * 100 ) + '%';
143
+ document.body.style.height = ( scale * 100 ) + '%';
144
+ document.body.style.zoom = scale;
145
+ }
146
+ }
147
+
148
+ level = scale;
149
+
150
+ if( document.documentElement.classList ) {
151
+ if( level !== 1 ) {
152
+ document.documentElement.classList.add( 'zoomed' );
153
+ }
154
+ else {
155
+ document.documentElement.classList.remove( 'zoomed' );
156
+ }
157
+ }
158
+ }
159
+
160
+ /**
161
+ * Pan the document when the mosue cursor approaches the edges
162
+ * of the window.
163
+ */
164
+ function pan() {
165
+ var range = 0.12,
166
+ rangeX = window.innerWidth * range,
167
+ rangeY = window.innerHeight * range,
168
+ scrollOffset = getScrollOffset();
169
+
170
+ // Up
171
+ if( mouseY < rangeY ) {
172
+ window.scroll( scrollOffset.x, scrollOffset.y - ( 1 - ( mouseY / rangeY ) ) * ( 14 / level ) );
173
+ }
174
+ // Down
175
+ else if( mouseY > window.innerHeight - rangeY ) {
176
+ window.scroll( scrollOffset.x, scrollOffset.y + ( 1 - ( window.innerHeight - mouseY ) / rangeY ) * ( 14 / level ) );
177
+ }
178
+
179
+ // Left
180
+ if( mouseX < rangeX ) {
181
+ window.scroll( scrollOffset.x - ( 1 - ( mouseX / rangeX ) ) * ( 14 / level ), scrollOffset.y );
182
+ }
183
+ // Right
184
+ else if( mouseX > window.innerWidth - rangeX ) {
185
+ window.scroll( scrollOffset.x + ( 1 - ( window.innerWidth - mouseX ) / rangeX ) * ( 14 / level ), scrollOffset.y );
186
+ }
187
+ }
188
+
189
+ function getScrollOffset() {
190
+ return {
191
+ x: window.scrollX !== undefined ? window.scrollX : window.pageXOffset,
192
+ y: window.scrollY !== undefined ? window.scrollY : window.pageYOffset
193
+ }
194
+ }
195
+
196
+ return {
197
+ /**
198
+ * Zooms in on either a rectangle or HTML element.
199
+ *
200
+ * @param {Object} options
201
+ * - element: HTML element to zoom in on
202
+ * OR
203
+ * - x/y: coordinates in non-transformed space to zoom in on
204
+ * - width/height: the portion of the screen to zoom in on
205
+ * - scale: can be used instead of width/height to explicitly set scale
206
+ */
207
+ to: function( options ) {
208
+
209
+ // Due to an implementation limitation we can't zoom in
210
+ // to another element without zooming out first
211
+ if( level !== 1 ) {
212
+ zoom.out();
213
+ }
214
+ else {
215
+ options.x = options.x || 0;
216
+ options.y = options.y || 0;
217
+
218
+ // If an element is set, that takes precedence
219
+ if( !!options.element ) {
220
+ // Space around the zoomed in element to leave on screen
221
+ var padding = 20;
222
+ var bounds = options.element.getBoundingClientRect();
223
+
224
+ options.x = bounds.left - padding;
225
+ options.y = bounds.top - padding;
226
+ options.width = bounds.width + ( padding * 2 );
227
+ options.height = bounds.height + ( padding * 2 );
228
+ }
229
+
230
+ // If width/height values are set, calculate scale from those values
231
+ if( options.width !== undefined && options.height !== undefined ) {
232
+ options.scale = Math.max( Math.min( window.innerWidth / options.width, window.innerHeight / options.height ), 1 );
233
+ }
234
+
235
+ if( options.scale > 1 ) {
236
+ options.x *= options.scale;
237
+ options.y *= options.scale;
238
+
239
+ magnify( options, options.scale );
240
+
241
+ if( options.pan !== false ) {
242
+
243
+ // Wait with engaging panning as it may conflict with the
244
+ // zoom transition
245
+ panEngageTimeout = setTimeout( function() {
246
+ panUpdateInterval = setInterval( pan, 1000 / 60 );
247
+ }, 800 );
248
+
249
+ }
250
+ }
251
+ }
252
+ },
253
+
254
+ /**
255
+ * Resets the document zoom state to its default.
256
+ */
257
+ out: function() {
258
+ clearTimeout( panEngageTimeout );
259
+ clearInterval( panUpdateInterval );
260
+
261
+ magnify( { x: 0, y: 0 }, 1 );
262
+
263
+ level = 1;
264
+ },
265
+
266
+ // Alias
267
+ magnify: function( options ) { this.to( options ) },
268
+ reset: function() { this.out() },
269
+
270
+ zoomLevel: function() {
271
+ return level;
272
+ }
273
+ }
274
+
275
+ })();
276
+
277
+
278
+