├── README.md ├── index.html ├── js ├── makisu.min.js └── makisu.js └── css └── style.css /README.md: -------------------------------------------------------------------------------- 1 | ### Makisu 2 | 3 | An experimental CSS 3D dropdown concept, wrapped up in a [jQuery](http://jquery.com/) plugin. 4 | 5 | Check out the [demo](http://soulwire.github.com/Makisu/) _(you'll need a CSS 3D capable browser, such as [Chrome](www.google.com/chrome).)_ 6 | 7 | A current list of supported browsers can be found [here](http://caniuse.com/#search=3d%20transform). 8 | 9 | #### Example usage 10 | 11 | Use it like any regular jQuery plugin: 12 | 13 | $( '.list' ).makisu({ 14 | selector: 'li', 15 | overlap: 0.2, 16 | speed: 0.8 17 | }); 18 | 19 | The options available are: 20 | 21 | - `selector` Children matching this selector will be _folded_ into the Makisu 22 | - `speed` The animation duration (in _seconds_) for each folding item 23 | - `overlap` Fraction of `speed ` by which folding items overlap (`0` to `1`) 24 | - `shading` Default shading colour (`null` for no shading) 25 | - `perspective` Perspective to apply to 3D transformed objects 26 | 27 | #### API 28 | 29 | Once an element has been extended as in the example above, you can `open`, `close` and `toggle` it. 30 | 31 | $( '.list' ).makisu( 'open' ); 32 | 33 | #### Trivia 34 | 35 | The name comes from the [object](http://en.wikipedia.org/wiki/Makisu) that inspired it. 36 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 |
4 |
73 |
74 |
75 |
76 |
128 |
129 |
--------------------------------------------------------------------------------
/js/makisu.min.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (C) 2012 by Justin Windle
3 | *
4 | * Permission is hereby granted, free of charge, to any person obtaining a copy
5 | * of this software and associated documentation files (the "Software"), to deal
6 | * in the Software without restriction, including without limitation the rights
7 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8 | * copies of the Software, and to permit persons to whom the Software is
9 | * furnished to do so, subject to the following conditions:
10 | *
11 | * The above copyright notice and this permission notice shall be included in
12 | * all copies or substantial portions of the Software.
13 | *
14 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17 | * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20 | * THE SOFTWARE.
21 | */(function(e){var t=!1,n=document.createElement("div"),r=/^(Moz|(w|W)ebkit|O|ms)(?=[A-Z])/,i=function(){for(var e in n.style)if(r.test(e))return e.match(r)[0]}()||"",s=i+"Perspective"in n.style,o="-"+i.toLowerCase()+"-",u,a,f,l,c,h,p,d,v,m,g,y={toggle:function(){u=e(this),u.makisu(u.hasClass("open")?"close":"open")},open:function(t,n,r){u=e(this),a=u.find(".root"),l=u.find(".node").not(a),t=b.resolve(u,"speed",t),r=b.resolve(u,"easing",r),n=b.resolve(u,"overlap",n),l.each(function(i,s){m="unfold"+(i?"":"-first"),g=i===l.length-1,time=t*(1-n),v=i*time,h=e(s),p=h.find(".over"),h.css(b.prefix({transform:"rotateX(180deg)",animation:m+" "+t+"s "+r+" "+v+"s 1 normal forwards"})),g||(v=(i+1)*time),p.css(b.prefix({animation:"unfold-over "+t*.45+"s "+r+" "+v+"s 1 normal forwards"}))}),a.css(b.prefix({animation:"swing-out "+l.length*time*1.4+"s ease-in-out 0s 1 normal forwards"})),u.addClass("open")},close:function(t,n,r){u=e(this),a=u.find(".root"),l=u.find(".node").not(a),t=b.resolve(u,"speed",t)*.66,r=b.resolve(u,"easing",r),n=b.resolve(u,"overlap",n),l.each(function(i,s){m="fold"+(i?"":"-first"),g=i===0,time=t*(1-n),v=(l.length-i-1)*time,h=e(s),p=h.find(".over"),h.css(b.prefix({transform:"rotateX(0deg)",animation:m+" "+t+"s "+r+" "+v+"s 1 normal forwards"})),g||(v=(l.length-i-2)*time+t*.35),p.css(b.prefix({animation:"fold-over "+t*.45+"s "+r+" "+v+"s 1 normal forwards"}))}),a.css(b.prefix({animation:"swing-in "+l.length*time*1+"s ease-in-out 0s 1 normal forwards"})),u.removeClass("open")}},b={resolve:function(e,t,n){return typeof n=="undefined"?e.data(t):n},prefix:function(e){for(var t in e)e[o+t]=e[t];return e},inject:function(e){try{var t=document.createElement("style");t.innerHTML=e,document.getElementsByTagName("head")[0].appendChild(t)}catch(n){}}},w={node:'',back:'',over:''};e.fn.makisu=function(n){if(!s){var r="Failed to detect CSS 3D support";console&&console.warn&&(console.warn(r),this.each(function(){e(this).trigger("error",r)}));return}t||(t=!0,b.inject("@"+o+"keyframes unfold {"+"0% {"+o+"transform: rotateX(180deg); }"+"50% {"+o+"transform: rotateX(-30deg); }"+"100% {"+o+"transform: rotateX(0deg); }"+"}"),b.inject("@"+o+"keyframes unfold-first {"+"0% {"+o+"transform: rotateX(-90deg); }"+"50% {"+o+"transform: rotateX(60deg); }"+"100% {"+o+"transform: rotateX(0deg); }"+"}"),b.inject("@"+o+"keyframes fold {"+"0% {"+o+"transform: rotateX(0deg); }"+"100% {"+o+"transform: rotateX(180deg); }"+"}"),b.inject("@"+o+"keyframes fold-first {"+"0% {"+o+"transform: rotateX(0deg); }"+"100% {"+o+"transform: rotateX(-180deg); }"+"}"),b.inject("@"+o+"keyframes swing-out {"+"0% {"+o+"transform: rotateX(0deg); }"+"30% {"+o+"transform: rotateX(-30deg); }"+"60% {"+o+"transform: rotateX(15deg); }"+"100% {"+o+"transform: rotateX(0deg); }"+"}"),b.inject("@"+o+"keyframes swing-in {"+"0% {"+o+"transform: rotateX(0deg); }"+"50% {"+o+"transform: rotateX(-10deg); }"+"90% {"+o+"transform: rotateX(15deg); }"+"100% {"+o+"transform: rotateX(0deg); }"+"}"),b.inject("@"+o+"keyframes unfold-over {"+"0% { opacity: 1.0; }"+"100% { opacity: 0.0; }"+"}"),b.inject("@"+o+"keyframes fold-over {"+"0% { opacity: 0.0; }"+"100% { opacity: 1.0; }"+"}"),b.inject(".node {position: relative;display: block;}"),b.inject(".face {pointer-events: none;position: absolute;display: block;height: 100%;width: 100%;left: 0;top: 0;}"));var i=e.extend({},e.fn.makisu.defaults,n),v=Array.prototype.slice.call(arguments,1);return this.each(function(){if(y[n])return y[n].apply(this,v);u=e(this).data(i),u.data("initialized")||(u.data("initialized",!0),l=u.children(i.selector),a=e(w.node).addClass("root"),f=a,l.each(function(t,n){h=e(n),m="fold"+(t?"":"-first"),h.css("position","relative"),h.css(b.prefix({"transform-style":"preserve-3d",transform:"translateZ(-0.1px)"})),d=e(w.back),d.css("background",h.css("background")),d.css(b.prefix({transform:"translateZ(-0.1px)"})),p=e(w.over),p.css(b.prefix({transform:"translateZ(0.1px)"})),p.css({background:i.shading,opacity:0}),c=e(w.node).append(h),c.css(b.prefix({"transform-origin":"50% 0%","transform-style":"preserve-3d",animation:m+" 1ms linear 0s 1 normal forwards"})),h.append(p),h.append(d),f.append(c),f=c}),a.css(b.prefix({"transform-origin":"50% 0%","transform-style":"preserve-3d"})),u.css(b.prefix({transform:"perspective("+i.perspective+"px)"})),u.append(a))})},e.fn.makisu.defaults={perspective:1200,shading:"rgba(0,0,0,0.12)",selector:null,overlap:.6,speed:.8,easing:"ease-in-out"},e.fn.makisu.enabled=s})(jQuery);
--------------------------------------------------------------------------------
/css/style.css:
--------------------------------------------------------------------------------
1 |
2 | html, body {
3 |
4 | -webkit-font-smoothing: antialiased;
5 | -moz-font-smoothing: antialiased;
6 |
7 | background: #ffffff;
8 | background: -moz-radial-gradient(center, ellipse cover, #ffffff 0%, #ffffff 26%, #f5f5f5 59%, #f5f5f5 77%, #cecece 100%);
9 | background: -webkit-gradient(radial, center center, 0, center center, 100%, color-stop(0%,#ffffff), color-stop(26%,#ffffff), color-stop(59%,#f5f5f5), color-stop(77%,#f5f5f5), color-stop(100%,#cecece));
10 | background: -webkit-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
11 | background: -o-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
12 | background: -ms-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
13 | background: radial-gradient(ellipse at center, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
14 | font-family: 'Days One', sans-serif;
15 | overflow: hidden;
16 | padding: 0;
17 | margin: 0;
18 | height: 100%;
19 | }
20 |
21 | body:before {
22 |
23 | background-image: url();
24 | position: absolute;
25 | content: '';
26 | opacity: 0.8;
27 | height: 100%;
28 | width: 100%;
29 | left: 0;
30 | top: 0;
31 | }
32 |
33 | a {
34 |
35 | -webkit-transition: all 250ms cubic-bezier(0.230, 1.000, 0.320, 1.000);
36 | -moz-transition: all 250ms cubic-bezier(0.230, 1.000, 0.320, 1.000);
37 | -ms-transition: all 250ms cubic-bezier(0.230, 1.000, 0.320, 1.000);
38 | -o-transition: all 250ms cubic-bezier(0.230, 1.000, 0.320, 1.000);
39 | transition: all 250ms cubic-bezier(0.230, 1.000, 0.320, 1.000);
40 |
41 | text-decoration: none;
42 | }
43 |
44 | .header {
45 |
46 | text-align: center;
47 | position: absolute;
48 | color: #333;
49 | width: 100%;
50 | top: 9%;
51 | }
52 |
53 | .header h1 {
54 |
55 | letter-spacing: -1px;
56 | text-shadow: -2px -1px 1px #fff, 1px 2px 2px rgba(0, 0, 0, 0.2);
57 | font-weight: 300;
58 | font-size: 36px;
59 | margin: 0;
60 | }
61 |
62 | .header h2 {
63 |
64 | text-transform: uppercase;
65 | text-shadow: -2px -1px 1px #fff, 1px 1px 1px rgba(0, 0, 0, 0.15);
66 | font-weight: 300;
67 | font-size: 12px;
68 | color: rgba(0,0,0,0.7);
69 | margin: 0;
70 | }
71 |
72 | .demo:after {
73 |
74 | box-shadow: 0 1px 16px rgba(0,0,0,0.15);
75 | background: #1b1b1b;
76 | position: absolute;
77 | content: '';
78 | height: 10px;
79 | width: 100%;
80 | top: 0;
81 | }
82 |
83 | /* List styles */
84 |
85 | .list {
86 |
87 | -webkit-transform-style: preserve-3d;
88 | -moz-transform-style: preserve-3d;
89 | -ms-transform-style: preserve-3d;
90 | -o-transform-style: preserve-3d;
91 | transform-style: preserve-3d;
92 |
93 | text-transform: uppercase;
94 | position: absolute;
95 | margin-left: -140px;
96 | margin-top: -280px;
97 | top: 55%;
98 | }
99 |
100 | .list a {
101 |
102 | display: block;
103 | color: #fff;
104 | }
105 |
106 | .list a:hover {
107 | text-indent: 20px;
108 | }
109 |
110 | .list dt, .list dd {
111 |
112 | text-indent: 10px;
113 | line-height: 55px;
114 | background: #E0FBAC;
115 | margin: 0;
116 | height: 55px;
117 | width: 270px;
118 | color: #fff;
119 | }
120 |
121 | .list dt {
122 |
123 | /* Since we're hiding elements behind here, we need it in 3d */
124 | -webkit-transform: translateZ(0.3px);
125 | -moz-transform: translateZ(0.3px);
126 | -ms-transform: translateZ(0.3px);
127 | -o-transform: translateZ(0.3px);
128 | transform: translateZ(0.3px);
129 |
130 | text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.2);
131 | font-size: 15px;
132 | }
133 |
134 | .list dd {
135 |
136 | border-top: 1px dashed rgba(255,255,255,0.3);
137 | line-height: 35px;
138 | font-size: 11px;
139 | height: 35px;
140 | margin: 0;
141 | }
142 |
143 | /* UI */
144 |
145 | .toggle {
146 |
147 | box-shadow: 0 1px 4px rgba(0,0,0,0.15);
148 | border-radius: 3px;
149 |
150 | text-transform: uppercase;
151 | letter-spacing: -1px;
152 | line-height: 50px;
153 | margin-left: -70px;
154 | margin-top: -20px;
155 | background: #2b2b2b;
156 | text-align: center;
157 | font-size: 12px;
158 | position: absolute;
159 | height: 50px;
160 | bottom: 10%;
161 | width: 140px;
162 | color: #fff;
163 | left: 50%;
164 | }
165 |
166 | .toggle:hover {
167 |
168 | background: #E42692;
169 | }
170 |
171 | /* No CSS 3D support warning */
172 | .warning {
173 |
174 | -webkit-transform: translateZ(2px);
175 | -moz-transform: translateZ(2px);
176 | -ms-transform: translateZ(2px);
177 | -o-transform: translateZ(2px);
178 | transform: translateZ(2px);
179 |
180 | background: rgba(255,255,255,0.6);
181 | position: fixed;
182 | display: none;
183 | z-index: 999;
184 | height: 100%;
185 | width: 100%;
186 | left: 0;
187 | top: 0;
188 | }
189 |
190 | .warning .message {
191 |
192 | box-shadow: 0 1px 8px rgba(0, 0, 0, 0.6);
193 | border-radius: 5px;
194 | text-align: center;
195 | margin-left: -150px;
196 | margin-top: -60px;
197 | line-height: 1.5;
198 | background: #222;
199 | font-size: 12px;
200 | position: absolute;
201 | padding: 10px;
202 | width: 280px;
203 | color: #fff;
204 | left: 50%;
205 | top: 50%;
206 | }
207 |
208 | .warning .message h1 {
209 |
210 | font-weight: 300;
211 | font-size: 14px;
212 | }
213 |
214 | .warning .message a {
215 |
216 | text-decoration: none;
217 | color: #73C8A9;
218 | }
219 |
220 | /* Individual styles */
221 |
222 | .sashimi dt, .sashimi dd, .sashimi a { background: #73C8A9; }
223 | .nigiri dt, .nigiri dd, .nigiri a { background: #E32551; }
224 | .maki dt, .maki dd, .maki a { background: #FFC219; }
225 |
226 | .sashimi a:hover { background: #61c19e; }
227 | .nigiri a:hover { background: #d31b46; }
228 | .maki a:hover { background: #ffbb00; }
229 |
230 | .nigiri {
231 |
232 | -webkit-transform: perspective(1200px) rotateY(40deg) !important;
233 | -moz-transform: perspective(1200px) rotateY(40deg) !important;
234 | -ms-transform: perspective(1200px) rotateY(40deg) !important;
235 | -o-transform: perspective(1200px) rotateY(40deg) !important;
236 | transform: perspective(1200px) rotateY(40deg) !important;
237 |
238 | -webkit-transform-origin: 110% 25%;
239 | -moz-transform-origin: 110% 25%;
240 | -ms-transform-origin: 110% 25%;
241 | -o-transform-origin: 110% 25%;
242 | transform-origin: 110% 25%;
243 |
244 | left: 20%;
245 | }
246 |
247 | .maki {
248 |
249 | -webkit-transform: perspective(600px) translateZ(1px) !important;
250 | -moz-transform: perspective(600px) translateZ(1px) !important;
251 | -ms-transform: perspective(600px) translateZ(1px) !important;
252 | -o-transform: perspective(600px) translateZ(1px) !important;
253 | transform: perspective(600px) translateZ(1px) !important;
254 |
255 | left: 50%;
256 | }
257 |
258 | .sashimi {
259 |
260 | -webkit-transform: perspective(1200px) rotateY(-40deg) !important;
261 | -moz-transform: perspective(1200px) rotateY(-40deg) !important;
262 | -ms-transform: perspective(1200px) rotateY(-40deg) !important;
263 | -o-transform: perspective(1200px) rotateY(-40deg) !important;
264 | transform: perspective(1200px) rotateY(-40deg) !important;
265 |
266 | -webkit-transform-origin: -10% 25%;
267 | -moz-transform-origin: -10% 25%;
268 | -ms-transform-origin: -10% 25%;
269 | -o-transform-origin: -10% 25%;
270 | transform-origin: -10% 25%;
271 |
272 | left: 80%;
273 | }
--------------------------------------------------------------------------------
/js/makisu.js:
--------------------------------------------------------------------------------
1 |
2 | /**
3 | * Copyright (C) 2012 by Justin Windle
4 | *
5 | * Permission is hereby granted, free of charge, to any person obtaining a copy
6 | * of this software and associated documentation files (the "Software"), to deal
7 | * in the Software without restriction, including without limitation the rights
8 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | * copies of the Software, and to permit persons to whom the Software is
10 | * furnished to do so, subject to the following conditions:
11 | *
12 | * The above copyright notice and this permission notice shall be included in
13 | * all copies or substantial portions of the Software.
14 | *
15 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 | * THE SOFTWARE.
22 | */
23 |
24 | (function($) {
25 |
26 | // Global initialisation flag
27 | var initialized = false;
28 |
29 | // For detecting browser prefix and capabilities
30 | var el = document.createElement( 'div' );
31 | var re = /^(Moz|(w|W)ebkit|O|ms)(?=[A-Z])/;
32 |
33 | // Establish vendor prefix and CSS 3D support
34 | var vendor = (function() { for ( var p in el.style ) if( re.test(p) ) return p.match(re)[0]; })() || '';
35 | var canRun = vendor + 'Perspective' in el.style;
36 | var prefix = '-' + vendor.toLowerCase() + '-';
37 |
38 | var $this, $root, $base, $kids, $node, $item, $over, $back;
39 | var wait, anim, last;
40 |
41 | // Public API
42 | var api = {
43 |
44 | // Toggle open / closed
45 | toggle: function() {
46 |
47 | $this = $( this );
48 | $this.makisu( $this.hasClass( 'open' ) ? 'close' : 'open' );
49 | },
50 |
51 | // Trigger the unfold animation
52 | open: function( speed, overlap, easing ) {
53 |
54 | // Cache DOM references
55 | $this = $(this);
56 | $root = $this.find( '.root' );
57 | $kids = $this.find( '.node' ).not( $root );
58 |
59 | // Establish values or fallbacks
60 | speed = utils.resolve( $this, 'speed', speed );
61 | easing = utils.resolve( $this, 'easing', easing );
62 | overlap = utils.resolve( $this, 'overlap', overlap );
63 |
64 | $kids.each( function( index, el ) {
65 |
66 | // Establish settings for this iteration
67 | anim = 'unfold' + ( !index ? '-first' : '' );
68 | last = index === $kids.length - 1;
69 | time = speed * ( 1 - overlap );
70 | wait = index * time;
71 |
72 | // Cache DOM references
73 | $item = $( el );
74 | $over = $item.find( '.over' );
75 |
76 | // Element animation
77 | $item.css(utils.prefix({
78 | 'transform': 'rotateX(180deg)',
79 | 'animation': anim + ' ' + speed + 's ' + easing + ' ' + wait + 's 1 normal forwards'
80 | }));
81 |
82 | // Shading animation happens when the next item starts
83 | if ( !last ) wait = ( index + 1 ) * time;
84 |
85 | // Shading animation
86 | $over.css(utils.prefix({
87 | 'animation': 'unfold-over ' + (speed * 0.45) + 's ' + easing + ' ' + wait + 's 1 normal forwards'
88 | }));
89 | });
90 |
91 | // Add momentum to the container
92 | $root.css(utils.prefix({
93 | 'animation': 'swing-out ' + ( $kids.length * time * 1.4 ) + 's ease-in-out 0s 1 normal forwards'
94 | }));
95 |
96 | $this.addClass( 'open' );
97 | },
98 |
99 | // Trigger the fold animation
100 | close: function( speed, overlap, easing ) {
101 |
102 | // Cache DOM references
103 | $this = $(this);
104 | $root = $this.find( '.root' );
105 | $kids = $this.find( '.node' ).not( $root );
106 |
107 | // Establish values or fallbacks
108 | speed = utils.resolve( $this, 'speed', speed ) * 0.66;
109 | easing = utils.resolve( $this, 'easing', easing );
110 | overlap = utils.resolve( $this, 'overlap', overlap );
111 |
112 | $kids.each( function( index, el ) {
113 |
114 | // Establish settings for this iteration
115 | anim = 'fold' + ( !index ? '-first' : '' );
116 | last = index === 0;
117 | time = speed * ( 1 - overlap );
118 | wait = ( $kids.length - index - 1 ) * time;
119 |
120 | // Cache DOM references
121 | $item = $( el );
122 | $over = $item.find( '.over' );
123 |
124 | // Element animation
125 | $item.css(utils.prefix({
126 | 'transform': 'rotateX(0deg)',
127 | 'animation': anim + ' ' + speed + 's ' + easing + ' ' + wait + 's 1 normal forwards'
128 | }));
129 |
130 | // Adjust delay for shading
131 | if ( !last ) wait = ( ( $kids.length - index - 2 ) * time ) + ( speed * 0.35 );
132 |
133 | // Shading animation
134 | $over.css(utils.prefix({
135 | 'animation': 'fold-over ' + (speed * 0.45) + 's ' + easing + ' ' + wait + 's 1 normal forwards'
136 | }));
137 | });
138 |
139 | // Add momentum to the container
140 | $root.css(utils.prefix({
141 | 'animation': 'swing-in ' + ( $kids.length * time * 1.0 ) + 's ease-in-out 0s 1 normal forwards'
142 | }));
143 |
144 | $this.removeClass( 'open' );
145 | }
146 | };
147 |
148 | // Utils
149 | var utils = {
150 |
151 | // Resolves argument values to defaults
152 | resolve: function( $el, key, val ) {
153 | return typeof val === 'undefined' ? $el.data( key ) : val;
154 | },
155 |
156 | // Prefixes a hash of styles with the current vendor
157 | prefix: function( style ) {
158 |
159 | for ( var key in style ) {
160 | style[ prefix + key ] = style[ key ];
161 | }
162 |
163 | return style;
164 | },
165 |
166 | // Inserts rules into the document styles
167 | inject: function( rule ) {
168 |
169 | try {
170 |
171 | var style = document.createElement( 'style' );
172 | style.innerHTML = rule;
173 | document.getElementsByTagName( 'head' )[0].appendChild( style );
174 |
175 | } catch ( error ) {}
176 | }
177 | };
178 |
179 | // Element templates
180 | var markup = {
181 | node: '',
182 | back: '',
183 | over: ''
184 | };
185 |
186 | // Plugin definition
187 | $.fn.makisu = function( options ) {
188 |
189 | // Notify if 3D isn't available
190 | if ( !canRun ) {
191 |
192 | var message = 'Failed to detect CSS 3D support';
193 |
194 | if( console && console.warn ) {
195 |
196 | // Print warning to the console
197 | console.warn( message );
198 |
199 | // Trigger errors on elements
200 | this.each( function() {
201 | $( this ).trigger( 'error', message );
202 | });
203 | }
204 |
205 | return;
206 | }
207 |
208 | // Fires only once
209 | if ( !initialized ) {
210 |
211 | initialized = true;
212 |
213 | // Unfold
214 | utils.inject( '@' + prefix + 'keyframes unfold {' +
215 |
216 | '0% {' + prefix + 'transform: rotateX(180deg); }' +
217 | '50% {' + prefix + 'transform: rotateX(-30deg); }' +
218 | '100% {' + prefix + 'transform: rotateX(0deg); }' +
219 |
220 | '}');
221 |
222 | // Unfold (first item)
223 | utils.inject( '@' + prefix + 'keyframes unfold-first {' +
224 |
225 | '0% {' + prefix + 'transform: rotateX(-90deg); }' +
226 | '50% {' + prefix + 'transform: rotateX(60deg); }' +
227 | '100% {' + prefix + 'transform: rotateX(0deg); }' +
228 |
229 | '}');
230 |
231 | // Fold
232 | utils.inject( '@' + prefix + 'keyframes fold {' +
233 |
234 | '0% {' + prefix + 'transform: rotateX(0deg); }' +
235 | '100% {' + prefix + 'transform: rotateX(180deg); }' +
236 |
237 | '}');
238 |
239 | // Fold (first item)
240 | utils.inject( '@' + prefix + 'keyframes fold-first {' +
241 |
242 | '0% {' + prefix + 'transform: rotateX(0deg); }' +
243 | '100% {' + prefix + 'transform: rotateX(-180deg); }' +
244 |
245 | '}');
246 |
247 | // Swing out
248 | utils.inject( '@' + prefix + 'keyframes swing-out {' +
249 |
250 | '0% {' + prefix + 'transform: rotateX(0deg); }' +
251 | '30% {' + prefix + 'transform: rotateX(-30deg); }' +
252 | '60% {' + prefix + 'transform: rotateX(15deg); }' +
253 | '100% {' + prefix + 'transform: rotateX(0deg); }' +
254 |
255 | '}');
256 |
257 | // Swing in
258 | utils.inject( '@' + prefix + 'keyframes swing-in {' +
259 |
260 | '0% {' + prefix + 'transform: rotateX(0deg); }' +
261 | '50% {' + prefix + 'transform: rotateX(-10deg); }' +
262 | '90% {' + prefix + 'transform: rotateX(15deg); }' +
263 | '100% {' + prefix + 'transform: rotateX(0deg); }' +
264 |
265 | '}');
266 |
267 | // Shading (unfold)
268 | utils.inject( '@' + prefix + 'keyframes unfold-over {' +
269 | '0% { opacity: 1.0; }' +
270 | '100% { opacity: 0.0; }' +
271 | '}');
272 |
273 | // Shading (fold)
274 | utils.inject( '@' + prefix + 'keyframes fold-over {' +
275 | '0% { opacity: 0.0; }' +
276 | '100% { opacity: 1.0; }' +
277 | '}');
278 |
279 | // Node styles
280 | utils.inject( '.node {' +
281 | 'position: relative;' +
282 | 'display: block;' +
283 | '}');
284 |
285 | // Face styles
286 | utils.inject( '.face {' +
287 | 'pointer-events: none;' +
288 | 'position: absolute;' +
289 | 'display: block;' +
290 | 'height: 100%;' +
291 | 'width: 100%;' +
292 | 'left: 0;' +
293 | 'top: 0;' +
294 | '}');
295 | }
296 |
297 | // Merge options & defaults
298 | var opts = $.extend( {}, $.fn.makisu.defaults, options );
299 |
300 | // Extract api method arguments
301 | var args = Array.prototype.slice.call( arguments, 1 );
302 |
303 | // Main plugin loop
304 | return this.each( function () {
305 |
306 | // If the user is calling a method...
307 | if ( api[ options ] ) {
308 | return api[ options ].apply( this, args );
309 | }
310 |
311 | // Store options in view
312 | $this = $( this ).data( opts );
313 |
314 | // Only proceed if the scene hierarchy isn't already built
315 | if ( !$this.data( 'initialized' ) ) {
316 |
317 | $this.data( 'initialized', true );
318 |
319 | // Select the first level of matching child elements
320 | $kids = $this.children( opts.selector );
321 |
322 | // Build a scene graph for elements
323 | $root = $( markup.node ).addClass( 'root' );
324 | $base = $root;
325 |
326 | // Process each element and insert into hierarchy
327 | $kids.each( function( index, el ) {
328 |
329 | $item = $( el );
330 |
331 | // Which animation should this node use?
332 | anim = 'fold' + ( !index ? '-first' : '' );
333 |
334 | // Since we're adding absolutely positioned children
335 | $item.css( 'position', 'relative' );
336 |
337 | // Give the item some depth to avoid clipping artefacts
338 | $item.css(utils.prefix({
339 | 'transform-style': 'preserve-3d',
340 | 'transform': 'translateZ(-0.1px)'
341 | }));
342 |
343 | // Create back face
344 | $back = $( markup.back );
345 | $back.css( 'background', $item.css( 'background' ) );
346 | $back.css(utils.prefix({ 'transform': 'translateZ(-0.1px)' }));
347 |
348 | // Create shading
349 | $over = $( markup.over );
350 | $over.css(utils.prefix({ 'transform': 'translateZ(0.1px)' }));
351 | $over.css({
352 | 'background': opts.shading,
353 | 'opacity': 0.0
354 | });
355 |
356 | // Begin folded
357 | $node = $( markup.node ).append( $item );
358 | $node.css(utils.prefix({
359 | 'transform-origin': '50% 0%',
360 | 'transform-style': 'preserve-3d',
361 | 'animation': anim + ' 1ms linear 0s 1 normal forwards'
362 | }));
363 |
364 | // Build display list
365 | $item.append( $over );
366 | $item.append( $back );
367 | $base.append( $node );
368 |
369 | // Use as parent in next iteration
370 | $base = $node;
371 | });
372 |
373 | // Set root transform settings
374 | $root.css(utils.prefix({
375 | 'transform-origin': '50% 0%',
376 | 'transform-style': 'preserve-3d'
377 | }));
378 |
379 | // Apply perspective
380 | $this.css(utils.prefix({
381 | 'transform': 'perspective(' + opts.perspective + 'px)'
382 | }));
383 |
384 | // Display the scene
385 | $this.append( $root );
386 | }
387 | });
388 | };
389 |
390 | // Default options
391 | $.fn.makisu.defaults = {
392 |
393 | // Perspective to apply to rotating elements
394 | perspective: 1200,
395 |
396 | // Default shading to apply (null => no shading)
397 | shading: 'rgba(0,0,0,0.12)',
398 |
399 | // Area of rotation (fraction or pixel value)
400 | selector: null,
401 |
402 | // Fraction of speed (0-1)
403 | overlap: 0.6,
404 |
405 | // Duration per element
406 | speed: 0.8,
407 |
408 | // Animation curve
409 | easing: 'ease-in-out'
410 | };
411 |
412 | $.fn.makisu.enabled = canRun;
413 |
414 | })( jQuery );
--------------------------------------------------------------------------------