initial commit
authorHakim El Hattab <hakim.elhattab@gmail.com>
Tue, 7 Jun 2011 19:10:59 +0000 (21:10 +0200)
committerHakim El Hattab <hakim.elhattab@gmail.com>
Tue, 7 Jun 2011 19:10:59 +0000 (21:10 +0200)
.gitignore [new file with mode: 0644]
LICENSE [new file with mode: 0644]
README.md [new file with mode: 0644]
assets/fonts/leaguegothic/league_gothic-webfont.ttf [new file with mode: 0644]
assets/images/breakdom.jpg [new file with mode: 0644]
css/main.css [new file with mode: 0644]
css/reset.css [new file with mode: 0644]
index.html [new file with mode: 0644]
js/slideshow.js [new file with mode: 0644]

diff --git a/.gitignore b/.gitignore
new file mode 100644 (file)
index 0000000..dec0ea4
--- /dev/null
@@ -0,0 +1,4 @@
+.DS_Store
+.svn
+log/*.log
+tmp/**
diff --git a/LICENSE b/LICENSE
new file mode 100644 (file)
index 0000000..28cfd7d
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,19 @@
+Copyright (C) 2011 Hakim El Hattab, http://hakim.se
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
\ No newline at end of file
diff --git a/README.md b/README.md
new file mode 100644 (file)
index 0000000..923ddb8
--- /dev/null
+++ b/README.md
@@ -0,0 +1 @@
+# CSS 3D Slideshow
diff --git a/assets/fonts/leaguegothic/league_gothic-webfont.ttf b/assets/fonts/leaguegothic/league_gothic-webfont.ttf
new file mode 100644 (file)
index 0000000..29f896a
Binary files /dev/null and b/assets/fonts/leaguegothic/league_gothic-webfont.ttf differ
diff --git a/assets/images/breakdom.jpg b/assets/images/breakdom.jpg
new file mode 100644 (file)
index 0000000..64dc3f3
Binary files /dev/null and b/assets/images/breakdom.jpg differ
diff --git a/css/main.css b/css/main.css
new file mode 100644 (file)
index 0000000..494cfc2
--- /dev/null
@@ -0,0 +1,218 @@
+/**
+ * @author Hakim El Hattab
+ */
+
+
+/*********************************************
+ * FONT-FACE DEFINITIONS
+ *********************************************/
+
+@font-face {
+       font-family: 'League Gothic';
+       src: url('../assets/fonts/leaguegothic/league_gothic-webfont.ttf') format('truetype');
+       font-weight: normal;
+       font-style: normal;
+}
+
+
+/*********************************************
+ * GLOBAL STYLES
+ *********************************************/
+
+html, body {
+       padding: 0;
+       margin: 0;
+       overflow: hidden;
+       
+       font-family: 'Crimson Text', Times, 'Times New Roman', serif;
+       font-size: 36px;
+       
+       background: #fff;
+       color: #222;
+       
+       width: 100%;
+       height: 100%;
+       
+       background-image: -webkit-gradient(
+               radial, 
+               50% 50%, 0, 
+               50% 50%, 1000, 
+               from(rgba(245,245,245,1.0)), 
+               to(rgba(100,100,100,1.0))
+       );
+
+       background-image: -moz-radial-gradient(
+               50% 50% 90deg,
+               rgba(245,245,245,1.0) 0%, 
+               rgba(100,100,100,1.0) 100%
+       );
+       
+}
+
+
+/*********************************************
+ * HEADERS
+ *********************************************/
+h1, h2, h3, h4 {
+       margin: 0 0 20px 0;
+       font-family: 'League Gothic', Arial, Helvetica, sans-serif;
+       line-height: 0.9em;
+       letter-spacing: 0.02em;
+       text-transform: uppercase;
+       color: #222;
+       text-shadow: 0px 0px 2px #fff, 0px 0px 4px #bbb;
+}
+
+h1 { font-size: 136px;         }
+h2 { font-size: 76px;  }
+h3 { font-size: 56px;  }
+h4 { font-size: 36px;  }
+
+h1.inverted,
+h2.inverted,
+h3.inverted,
+h4.inverted {
+       color: #fff;
+       text-shadow: 0px 0px 2px #fff, 0px 0px 2px #888;
+}
+
+
+/*********************************************
+ * SLIDES
+ *********************************************/
+#main {
+       position: absolute;
+       width: 800px;
+       height: 600px;
+       
+       left: 50%;
+       top: 50%;
+       margin-left: -400px;
+       margin-top: -320px;
+       
+       text-align: center;
+       
+       -webkit-perspective: 600px;
+       -webkit-perspective-origin: 50% 25%;
+}
+
+#main>section,
+#main>section>section {
+       display: none;
+       
+       position: absolute;
+       width: 100%;
+       min-height: 600px;
+       
+       -webkit-transform-style: preserve-3d;
+       
+       -webkit-transition: all 800ms cubic-bezier(0.260, 0.860, 0.440, 0.985); 
+          -moz-transition: all 800ms cubic-bezier(0.260, 0.860, 0.440, 0.985); 
+            -o-transition: all 800ms cubic-bezier(0.260, 0.860, 0.440, 0.985); 
+               transition: all 800ms cubic-bezier(0.260, 0.860, 0.440, 0.985);
+}
+
+#main section.past {
+       display: block;
+       opacity: 0;
+       
+       -webkit-transform:      translate3d(-100%, 0, 0) 
+                                               rotateY(-90deg) 
+                                               translate3d(-100%, 0, 0);
+}
+
+#main section.present {
+       display: block;
+}
+
+#main section.future {
+       display: block;
+       opacity: 0;
+       
+       -webkit-transform:      translate3d(100%, 0, 0) 
+                                               rotateY(90deg) 
+                                               translate3d(100%, 0, 0);
+}
+
+#main section>section.past {
+       display: block;
+       opacity: 0;
+       
+       -webkit-transform:      translate3d(0, -50%, 0) 
+                                               rotateX(70deg) 
+                                               translate3d(0, -50%, 0);
+}
+#main section>section.future {
+       display: block;
+       opacity: 0;
+       
+       -webkit-transform:      translate3d(0, 50%, 0) 
+                                               rotateX(-70deg) 
+                                               translate3d(0, 50%, 0);
+}
+
+
+/*********************************************
+ * DEFAULT ELEMENT STYLES
+ *********************************************/
+
+#main>section {
+       line-height: 1.2em;
+       text-shadow: 0px 0px 2px #fff, 0px 0px 4px #bbb;
+       font-weight: 600;
+}
+
+ol {
+       list-style: decimal;
+       list-style-position: inside;
+}
+
+li, p {
+       margin-bottom: 10px;
+}
+
+a:not(.image) {
+       color: #1b6263;
+       text-decoration: none;
+       border-bottom: 1px dashed rgba(0,0,0,0.3);
+       padding: 1px 3px;
+}
+       
+       a:not(.image):hover {
+               color: #fff;
+               background: #2fa794;
+               text-shadow: none;
+               border: none;
+       }
+
+img {
+       margin: 30px 0 0 0;
+       background: rgba(255,255,255,0.12);
+       border: 4px solid #eee;
+       
+       -webkit-box-shadow: 0 0 10px rgba(0, 0, 0, 0.15);
+          -moz-box-shadow: 0 0 10px rgba(0, 0, 0, 0.15);
+               box-shadow: 0 0 10px rgba(0, 0, 0, 0.15);
+       
+       -webkit-transition: all .11s linear;
+          -moz-transition: all .11s linear;
+            -o-transition: all .11s linear;
+               transition: all .11s linear;
+}
+
+       a.image:hover img {
+               background: rgba(255,255,255,0.2);
+               
+               -webkit-box-shadow: 0 0 20px rgba(0, 0, 0, 0.25);
+                  -moz-box-shadow: 0 0 20px rgba(0, 0, 0, 0.25);
+                       box-shadow: 0 0 20px rgba(0, 0, 0, 0.25);
+       }
+
+
+
+
+
+
+
+
+
diff --git a/css/reset.css b/css/reset.css
new file mode 100644 (file)
index 0000000..68f227a
--- /dev/null
@@ -0,0 +1,57 @@
+/* http://meyerweb.com/eric/tools/css/reset/ 
+   v2.0 | 20110126
+   License: none (public domain)
+*/
+
+html, body, div, span, applet, object, iframe,
+h1, h2, h3, h4, h5, h6, p, blockquote, pre,
+a, abbr, acronym, address, big, cite, code,
+del, dfn, em, img, ins, kbd, q, s, samp,
+small, strike, strong, sub, sup, tt, var,
+b, u, i, center,
+dl, dt, dd, ol, ul, li,
+fieldset, form, label, legend,
+table, caption, tbody, tfoot, thead, tr, th, td,
+article, aside, canvas, details, embed, 
+figure, figcaption, footer, header, hgroup, 
+menu, nav, output, ruby, section, summary,
+time, mark, audio, video {
+       margin: 0;
+       padding: 0;
+       border: 0;
+       font-size: 100%;
+       font: inherit;
+       vertical-align: baseline;
+}
+/* HTML5 display-role reset for older browsers */
+article, aside, details, figcaption, figure, 
+footer, header, hgroup, menu, nav, section {
+       display: block;
+}
+body {
+       line-height: 1;
+}
+ol, ul {
+       list-style: none;
+}
+blockquote, q {
+       quotes: none;
+}
+blockquote:before, blockquote:after,
+q:before, q:after {
+       content: '';
+       content: none;
+}
+table {
+       border-collapse: collapse;
+       border-spacing: 0;
+}
+
+
+/* HTML5BP:
+   These selection declarations have to be separate.
+   No text-shadow: twitter.com/miketaylr/status/12228805301
+   Also: hot pink. */
+::-moz-selection{ background: #FF5E99; color:#fff; text-shadow: none; }
+::selection { background:#FF5E99; color:#fff; text-shadow: none; }
+
diff --git a/index.html b/index.html
new file mode 100644 (file)
index 0000000..5c71250
--- /dev/null
@@ -0,0 +1,151 @@
+<!doctype html>  
+<html lang="en">
+       
+       <head>
+               <meta charset="utf-8">
+               
+               <title>CSS 3D Slideshow</title>
+               
+               <link href='http://fonts.googleapis.com/css?family=Crimson+Text:regular,600,bold' rel='stylesheet' type='text/css'>
+               
+               <link rel="stylesheet" href="css/reset.css">
+               <link rel="stylesheet" href="css/main.css">
+       </head>
+       
+       <body>
+               
+               <!-- Any section element inside of this container is displayed as a slide -->
+               <div id="main">
+                       
+                       <section>
+                               <h1>Slideshow</h1>
+                               <h3 class="inverted">With 3D effects. And stuff.</h3>
+                               <script>
+                                       // Delicously hacky. Look away.
+                                       var message = navigator.userAgent.match( /(iPhone|iPad|iPod|Android)/i ) ? 'Tap to navigate' : 'Navigate via keyboard';
+                                       document.write( '<p style="color: rgba(0,0,0,0.1); text-shadow: none;">('+message+')</p>' );
+                               </script>
+                       </section>
+                       
+                       <section>
+                               <h2>Heads Up</h2>
+                               <p>
+                                       This requires a browser with support for CSS3 3D transforms, such as Mobile Safari.
+                               </p>
+                       </section>
+                       
+                       <!-- Example of nested vertical slides -->
+                       <section>
+                               <section>
+                                       <h2>Vertical Slides</h2>
+                                       <p>
+                                               Slides can be nested inside of other slides,<br/>
+                                               try pressing <a href="#/2/1">down</a>.
+                                       </p>
+                                       <a href="#/2/1" class="image">
+                                               <img src="">
+                                       </a>
+                               </section>
+                               <section>
+                                       <h2>Basement Level 1</h2>
+                                       <p>Press down or up to navigate.</p>
+                               </section>
+                               <section>
+                                       <h2>Basement Level 2</h2>
+                                       <p>This is totally the Google logo:</p>
+                                       <img src="%3D%3D">
+                               </section>
+                               <section>
+                                       <h2>Basement Level 3</h2>
+                                       <p>That's it, time to go back up.</p>
+                                       <a href="#/2" class="image">
+                                               <img style="-webkit-transform: rotate(180deg);" src="">
+                                       </a>
+                               </section>
+                       </section>
+                               
+                       <section>
+                               <h2>Marvelous Unordered List</h2>
+                               <ul>
+                                       <li>No order here</li>
+                                       <li>Or here</li>
+                                       <li>Or here</li>
+                                       <li>Or here</li>
+                               </ul>
+                       </section>
+                       
+                       <section>
+                               <h2>Fantastic Ordered List</h2>
+                               <ol>
+                                       <li>One is smaller than...</li>
+                                       <li>Two is smaller than...</li>
+                                       <li>Three!</li>
+                               </ol>
+                       </section>
+                       
+                       <section>
+                               <h2>Intergalactic Interconnections</h2>
+                               <p>
+                                       You can link between slides internally,<br/>
+                                       <a href="#/2/3">like this</a>.
+                               </p>
+                       </section>
+                       
+                       <section>
+                               <h2>Spectacular image!</h2>
+                               <a class="image" href="http://hakim.se/experiments/html5/breakdom/" target="_blank">
+                                       <img src="assets/images/breakdom.jpg">
+                               </a>
+                       </section>
+                       
+                       <section>
+                               <h2>Stellar Links</h2>
+                               <ul>
+                                       <li><a href="hakim-slideshow-0.3.zip">Download this slideshow</a></li>
+                                       <li><a href="http://hakim.se/experiments/css3-3d-slideshow">Read more on my site</a></li>
+                                       <li><a href="http://twitter.com/hakimel">Follow me on Twitter</a></li>
+                               </ul>
+                       </section>
+                       
+                       <section>
+                               <h1>THE END</h1>
+                               <h3 class="inverted">BY Hakim El Hattab / hakim.se</h3>
+                       </section>
+                       
+               </div>
+               
+               <script src="js/slideshow.js"></script>
+               
+               
+               
+               <!-- Everything below this point is unrelated to the slideshow -->
+               
+               <div class="tweet-button" style="position: absolute; bottom: 10px; left: 50%; margin-left: -25px">
+                       <a href="http://twitter.com/share" class="twitter-share-button" data-text="A CSS 3D slideshow by @hakimel." data-url="http://hakim.se/experiments/css3-3d-slideshow" data-count="none" data-related="hakimel"></a>
+                       <script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script>
+               </div>
+               
+               <script>
+               var _gaq = [['_setAccount', 'UA-15240703-1'], ['_trackPageview']];
+               (function(d, t) {
+               var g = d.createElement(t),
+                   s = d.getElementsByTagName(t)[0];
+               g.async = true;
+               g.src = ('https:' == location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
+               s.parentNode.insertBefore(g, s);
+               })(document, 'script');
+               </script>
+               
+               <style>a[href="http://www.w3counter.com"] { display: none!important; }</style>
+               <script src="http://www.w3counter.com/tracker.js"></script>
+               <script>
+                       w3counter(49720);
+                       var ps = document.createElement('script');
+                               ps.type = 'text/javascript';
+                               ps.async = true;
+                               ps.src = '//pulse.w3counter.com/pulse.js?id=49720';
+                       (document.getElementsByTagName('head')[0]||document.getElementsByTagName('body')[0]).appendChild(ps);
+               </script>
+               
+       </body>
+</html>
\ No newline at end of file
diff --git a/js/slideshow.js b/js/slideshow.js
new file mode 100644 (file)
index 0000000..8165a40
--- /dev/null
@@ -0,0 +1,282 @@
+/**
+ * Copyright (C) 2011 Hakim El Hattab, http://hakim.se
+ * 
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ * 
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ * 
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+/**
+ * Handles the very minimal navigation logic involved in the 
+ * slideshow. Including keyboard navigation, touch interaction 
+ * and URL history behavior.
+ * 
+ * Slides are given unique hash based URL's so that they can be 
+ * opened directly. I didn't use the HTML5 History API for this 
+ * as it would have required the addition of server side rewrite 
+ * rules and hence require more effort for anyone to set up.
+ * 
+ * This component can be called from other scripts via a tiny API:
+ * - Slideshow.navigateTo( indexh, indexv );
+ * - Slideshow.navigateLeft();
+ * - Slideshow.navigateRight();
+ * - Slideshow.navigateUp();
+ * - Slideshow.navigateDown();
+ * 
+ * 
+ * version 0.1:
+ * - First release
+ * 
+ * version 0.2:                
+ * - Refactored code and added inline documentation
+ * - Slides now have unique URL's
+ * - A basic API to invoke navigation was added
+ * 
+ * version 0.3:                
+ * - Added licensing terms
+ * 
+ *     
+ * @author Hakim El Hattab
+ * @version 0.3
+ */
+var Slideshow = (function(){
+       
+       var indexh = 0,
+               indexv = 0;
+       
+       /**
+        * Activates the main program logic.
+        */
+       function initialize() {
+               document.addEventListener('keydown', onDocumentKeyDown, false);
+               document.addEventListener('touchstart', onDocumentTouchStart, false);
+               window.addEventListener('hashchange', onWindowHashChange, false);
+               
+               // Read the initial state of the URL (hash)
+               readURL();
+       }
+       
+       /**
+        * Handler for the document level 'keydown' event.
+        * 
+        * @param {Object} event
+        */
+       function onDocumentKeyDown( event ) {
+               
+               if( event.keyCode >= 37 && event.keyCode <= 40 ) {
+                       
+                       switch( event.keyCode ) {
+                               case 37: navigateLeft(); break; // left
+                               case 39: navigateRight(); break; // right
+                               case 38: navigateUp(); break; // up
+                               case 40: navigateDown(); break; // down
+                       }
+                       
+                       slide();
+                       
+                       event.preventDefault();
+                       
+               }
+       }
+       
+       /**
+        * Handler for the document level 'touchstart' event.
+        * 
+        * This enables very basic tap interaction for touch
+        * devices. Added mainly for performance testing of 3D
+        * transforms on iOS but was so happily surprised with
+        * how smoothly it runs so I left it in here. Apple +1
+        * 
+        * @param {Object} event
+        */
+       function onDocumentTouchStart( event ) {
+               
+               // We're only interested in one point taps
+               if (event.touches.length == 1) {
+                       event.preventDefault();
+                       
+                       var point = {
+                               x: event.touches[0].clientX,
+                               y: event.touches[0].clientY
+                       };
+                       
+                       // Define the extent of the areas that may be tapped
+                       // to navigate
+                       var wt = window.innerWidth * 0.3;
+                       var ht = window.innerHeight * 0.3;
+                       
+                       if( point.x < wt ) {
+                               navigateLeft();
+                       }
+                       else if( point.x > window.innerWidth - wt ) {
+                               navigateRight();
+                       }
+                       else if( point.y < ht ) {
+                               navigateUp();
+                       }
+                       else if( point.y > window.innerHeight - ht ) {
+                               navigateDown();
+                       }
+                       
+                       slide();
+                       
+               }
+       }
+       
+       
+       /**
+        * Handler for the window level 'hashchange' event.
+        * 
+        * @param {Object} event
+        */
+       function onWindowHashChange( event ) {
+               readURL();
+       }
+       
+       /**
+        * Updates one dimension of slides by showing the slide
+        * with the specified index.
+        * 
+        * @param {String} selector A CSS selector that will fetch
+        * the group of slides we are working with
+        * @param {Number} index The index of the slide that should be
+        * shown
+        * 
+        * @return {Number} The index of the slide that is now shown,
+        * might differ from the passed in index if it was out of 
+        * bounds.
+        */
+       function updateSlides( selector, index ) {
+               
+               // Select all slides and convert the NodeList result to
+               // an array
+               var slides = Array.prototype.slice.call( document.querySelectorAll( selector ) );
+               
+               if( slides.length ) {
+                       // Enforce max and minimum index bounds
+                       index = Math.max(Math.min(index, slides.length - 1), 0);
+                       
+                       slides[index].setAttribute('class', 'present');
+                       
+                       // Any element previous to index is given the 'past' class
+                       slides.slice(0, index).map(function(element){
+                               element.setAttribute('class', 'past');
+                       });
+                       
+                       // Any element subsequent to index is given the 'future' class
+                       slides.slice(index + 1).map(function(element){
+                               element.setAttribute('class', 'future');
+                       });
+               }
+               else {
+                       // Since there are no slides we can't be anywhere beyond the 
+                       // zeroth index
+                       index = 0;
+               }
+               
+               return index;
+               
+       }
+       
+       /**
+        * Updates the visual slides to represent the currently
+        * set indices. 
+        */
+       function slide() {
+               indexh = updateSlides( '#main>section', indexh );
+               indexv = updateSlides( 'section.present>section', indexv );
+               
+               writeURL();
+       }
+       
+       /**
+        * Reads the current URL (hash) and navigates accordingly.
+        */
+       function readURL() {
+               // Break the hash down to separate components
+               var bits = window.location.hash.slice(2).split('/');
+               
+               // Read the index components of the hash
+               indexh = bits[0] ? parseInt( bits[0] ) : 0;
+               indexv = bits[1] ? parseInt( bits[1] ) : 0;
+               
+               navigateTo( indexh, indexv );
+       }
+       
+       /**
+        * Updates the page URL (hash) to reflect the current
+        * navigational state. 
+        */
+       function writeURL() {
+               var url = '/';
+               
+               // Only include the minimum possible number of components in
+               // the URL
+               if( indexh > 0 || indexv > 0 ) url += indexh
+               if( indexv > 0 ) url += '/' + indexv
+               
+               window.location.hash = url;
+       }
+       
+       /**
+        * Triggers a navigation to the specified indices.
+        * 
+        * @param {Number} h The horizontal index of the slide to show
+        * @param {Number} v The vertical index of the slide to show
+        */
+       function navigateTo( h, v ) {
+               indexh = h === undefined ? indexh : h;
+               indexv = v === undefined ? indexv : v;
+               
+               slide();
+       }
+       
+       function navigateLeft() {
+               indexh --;
+               indexv = 0;
+               slide();
+       }
+       function navigateRight() {
+               indexh ++;
+               indexv = 0;
+               slide();
+       }
+       function navigateUp() {
+               indexv --;
+               slide();
+       }
+       function navigateDown() {
+               indexv ++;
+               slide();
+       }
+       
+       // Initialize the program. Done right before returning to ensure
+       // that any inline variable definitions are available to all
+       // functions 
+       initialize();
+       
+       // Expose some methods publicly
+       return {
+               navigateTo: navigateTo,
+               navigateLeft: navigateLeft,
+               navigateRight: navigateRight,
+               navigateUp: navigateUp,
+               navigateDown: navigateDown
+       };
+       
+})();
+