diff options
author | Hakim El Hattab <hakim.elhattab@gmail.com> | 2011-06-07 21:10:59 +0200 |
---|---|---|
committer | Hakim El Hattab <hakim.elhattab@gmail.com> | 2011-06-07 21:10:59 +0200 |
commit | fdd6b36f01b34cdf8c978ce2144784baf995fc57 (patch) | |
tree | ce53dc64b56ab4b0fe6526c4aaffa736e86a0bf3 | |
download | fosdem-2018-presentation-fdd6b36f01b34cdf8c978ce2144784baf995fc57.tar fosdem-2018-presentation-fdd6b36f01b34cdf8c978ce2144784baf995fc57.tar.gz |
initial commit
-rw-r--r-- | .gitignore | 4 | ||||
-rw-r--r-- | LICENSE | 19 | ||||
-rw-r--r-- | README.md | 1 | ||||
-rw-r--r-- | assets/fonts/leaguegothic/league_gothic-webfont.ttf | bin | 0 -> 42324 bytes | |||
-rw-r--r-- | assets/images/breakdom.jpg | bin | 0 -> 56714 bytes | |||
-rw-r--r-- | css/main.css | 218 | ||||
-rw-r--r-- | css/reset.css | 57 | ||||
-rw-r--r-- | index.html | 151 | ||||
-rw-r--r-- | js/slideshow.js | 282 |
9 files changed, 732 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..dec0ea4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +.DS_Store +.svn +log/*.log +tmp/** @@ -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 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 Binary files differnew file mode 100644 index 0000000..29f896a --- /dev/null +++ b/assets/fonts/leaguegothic/league_gothic-webfont.ttf diff --git a/assets/images/breakdom.jpg b/assets/images/breakdom.jpg Binary files differnew file mode 100644 index 0000000..64dc3f3 --- /dev/null +++ b/assets/images/breakdom.jpg diff --git a/css/main.css b/css/main.css new file mode 100644 index 0000000..494cfc2 --- /dev/null +++ b/css/main.css @@ -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 index 0000000..68f227a --- /dev/null +++ b/css/reset.css @@ -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 index 0000000..5c71250 --- /dev/null +++ b/index.html @@ -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 index 0000000..8165a40 --- /dev/null +++ b/js/slideshow.js @@ -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 + }; + +})(); + |