A Pen by Guilherme Rv Coelho on CodePen.
Created
December 28, 2014 17:42
-
-
Save grvcoelho/60d5926b3f908607052e to your computer and use it in GitHub Desktop.
MYjqLq
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<html lang="en"> | |
<head> | |
<meta charset="UTF-8"> | |
<title>Material Design Ripple Effect</title> | |
</head> | |
<body> | |
<div class="btn"></div> | |
</body> | |
</html> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(function($, window, document, undefined) { | |
'use strict'; | |
/** | |
* Define the name of the plugin | |
*/ | |
var ripple = 'ripple'; | |
/** | |
* Get an instance of the plugin | |
*/ | |
var self = null; | |
/** | |
* Define the defaults of the plugin | |
*/ | |
var defaults = {}; | |
/** | |
* Create the main plugin function | |
*/ | |
function Ripples(element, options) { | |
self = this; | |
this.element = $(element); | |
this.options = $.extend({}, defaults, options); | |
this._defaults = defaults; | |
this._name = ripple; | |
this.init(); | |
} | |
/** | |
* Initialize the plugin | |
*/ | |
Ripples.prototype.init = function() { | |
var $element = this.element; | |
$element.on('mousedown touchstart', function(event) { | |
/** | |
* Verify if the user is just touching on a device and return if so | |
*/ | |
if(self.isTouch() && event.type === 'mousedown') { | |
return false; | |
} | |
/** | |
* Verify if the current element already has a ripple wrapper element and | |
* creates if it doesn't | |
*/ | |
if(!($element.find('.ripple-wrapper').length)) { | |
$element.append('<div class="ripple-wrapper"></div>'); | |
} | |
/** | |
* Find the ripple wrapper | |
*/ | |
var $wrapper = $element.find('.ripple-wrapper'); | |
/** | |
* Get relY and relX positions | |
*/ | |
var relY = self.getRelY(event); | |
var relX = self.getRelX(event); | |
/** | |
* If relY and/or relX are false, return the event | |
*/ | |
if(!relY && !relX) { | |
return; | |
} | |
/** | |
* Get the ripple color | |
*/ | |
var rippleColor = self.getRippleColor(); | |
/** | |
* Create the ripple element | |
*/ | |
var $ripple = $('<div></div>'); | |
$ripple | |
.addClass('ripple') | |
.css({ | |
'left': relX, | |
'top': relY, | |
'background-color': rippleColor | |
}); | |
/** | |
* Append the ripple to the wrapper | |
*/ | |
$wrapper.append($ripple); | |
/** | |
* Make sure the ripple has the styles applied (ugly hack but it works) | |
*/ | |
(function() { return window.getComputedStyle($ripple[0]).opacity; })(); | |
/** | |
* Turn on the ripple animation | |
*/ | |
self.rippleOn($ripple); | |
/** | |
* Call the rippleEnd function when the transition 'on' ends | |
*/ | |
setTimeout(function() { | |
self.rippleEnd($ripple); | |
}, 500); | |
/** | |
* Detect when the user leaves the element | |
*/ | |
$element.on('mouseup mouseleave touchend', function() { | |
$ripple.data('mousedown', 'off'); | |
if($ripple.data('animating') === 'off') { | |
self.rippleOut($ripple); | |
} | |
}); | |
}); | |
}; | |
/** | |
* Get the new size based on the element height/width and the ripple width | |
*/ | |
Ripples.prototype.getNewSize = function($ripple) { | |
var $element = this.element; | |
return (Math.max($element.outerWidth(), $element.outerHeight()) / $ripple.outerWidth()) * 2.5; | |
}; | |
/** | |
* Get the relX | |
*/ | |
Ripples.prototype.getRelX = function(event) { | |
var $element = this.element; | |
var wrapperOffset = $element.find('.ripple-wrapper').offset(); | |
if(!self.isTouch()) { | |
/** | |
* Get the mouse position relative to the ripple wrapper | |
*/ | |
return event.pageX - wrapperOffset.left; | |
} else { | |
/** | |
* Make sure the user is using only one finger and then get the touch | |
* position relative to the ripple wrapper | |
*/ | |
event = event.originalEvent; | |
if(event.touches.length !== 1) { | |
return event.touches[0].pageX - wrapperOffset.left; | |
} | |
return false; | |
} | |
}; | |
/** | |
* Get the relY | |
*/ | |
Ripples.prototype.getRelY = function(event) { | |
var $element = this.element; | |
var wrapperOffset = $element.find('.ripple-wrapper').offset(); | |
if(!self.isTouch()) { | |
/** | |
* Get the mouse position relative to the ripple wrapper | |
*/ | |
return event.pageY - wrapperOffset.top; | |
} else { | |
/** | |
* Make sure the user is using only one finger and then get the touch | |
* position relative to the ripple wrapper | |
*/ | |
event = event.originalEvent; | |
if(event.touches.length !== 1) { | |
return event.touches[0].pageY - wrapperOffset.top; | |
} | |
return false; | |
} | |
}; | |
/** | |
* Get the ripple color | |
*/ | |
Ripples.prototype.getRippleColor = function() { | |
var $element = this.element; | |
var color = $element.data('ripple-color') ? $element.data('ripple-color') : window.getComputedStyle($element[0]).color; | |
return color; | |
}; | |
/** | |
* Verify if the client browser has transistion support | |
*/ | |
Ripples.prototype.hasTransitionSupport = function() { | |
var thisBody = document.body || document.documentElement; | |
var thisStyle = thisBody.style; | |
var support = ( | |
thisStyle.transition !== undefined || | |
thisStyle.WebkitTransition !== undefined || | |
thisStyle.MozTransition !== undefined || | |
thisStyle.MsTransition !== undefined || | |
thisStyle.OTransition !== undefined | |
); | |
return support; | |
}; | |
/** | |
* Verify if the client is using a mobile device | |
*/ | |
Ripples.prototype.isTouch = function() { | |
return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent); | |
}; | |
/** | |
* End the animation of the ripple | |
*/ | |
Ripples.prototype.rippleEnd = function($ripple) { | |
$ripple.data('animating', 'off'); | |
if($ripple.data('mousedown') === 'off') { | |
self.rippleOut($ripple); | |
} | |
}; | |
/** | |
* Turn off the ripple effect | |
*/ | |
Ripples.prototype.rippleOut = function($ripple) { | |
$ripple.off(); | |
if(self.hasTransitionSupport()) { | |
$ripple.addClass('ripple-out'); | |
} else { | |
$ripple.animate({'opacity': 0}, 100, function() { | |
$ripple.trigger('transitionend'); | |
}); | |
} | |
$ripple.on('transitionend webkitTransitionEnd oTransitionEnd MSTransitionEnd', function() { | |
$ripple.remove(); | |
}); | |
}; | |
/** | |
* Turn on the ripple effect | |
*/ | |
Ripples.prototype.rippleOn = function($ripple) { | |
var size = self.getNewSize($ripple); | |
var $element = this.element; | |
if(self.hasTransitionSupport()) { | |
$ripple | |
.css({ | |
'-ms-transform': 'scale(' + size + ')', | |
'-moz-transform': 'scale(' + size + ')', | |
'-webkit-transform': 'scale(' + size + ')', | |
'transform': 'scale(' + size + ')' | |
}) | |
.addClass('ripple-on') | |
.data('animating', 'on') | |
.data('mousedown', 'on'); | |
} else { | |
$ripple.animate({ | |
'width': Math.max($element.outerWidth(), $element.outerHeight()) * 2, | |
'height': Math.max($element.outerWidth(), $element.outerHeight()) * 2, | |
'margin-left': Math.max($element.outerWidth(), $element.outerHeight()) * (-1), | |
'margin-top': Math.max($element.outerWidth(), $element.outerHeight()) * (-1), | |
'opacity': 0.2 | |
}, 500, function() { | |
$ripple.trigger('transitionend'); | |
}); | |
} | |
}; | |
/** | |
* Create the jquery plugin function | |
*/ | |
$.fn.ripple = function(options) { | |
return this.each(function() { | |
if(!$.data(this, 'plugin_' + ripple)) { | |
$.data(this, 'plugin_' + ripple, new Ripples(this, options)); | |
} | |
}); | |
}; | |
})(jQuery, window, document); | |
$(document).ready(function() { | |
$('.btn').ripple(); | |
}) |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
.btn { | |
position: relative; | |
background: #283593; | |
display: inline-block; | |
margin: 20px; | |
width: 200px; height: 200px; | |
color: #fff; | |
border-radius: 50% | |
} | |
.ripple-wrapper { | |
position: absolute; | |
top: 0; | |
left: 0; | |
z-index: 1; | |
width: 100%; | |
height: 100%; | |
overflow: hidden; | |
border-radius: inherit; | |
pointer-events: none; | |
} | |
.ripple { | |
position: absolute; | |
width: 20px; | |
height: 20px; | |
margin-left: -10px; | |
margin-top: -10px; | |
border-radius: 100%; | |
background-color: rgba(0,0,0,0.05); | |
-webkit-transform: scale(1); | |
-ms-transform: scale(1); | |
transform: scale(1); | |
-webkit-transform-origin: 50%; | |
-ms-transform-origin: 50%; | |
transform-origin: 50%; | |
-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=0)"; | |
filter: alpha(opacity=0); | |
opacity: 0; | |
pointer-events: none; | |
} | |
.ripple.ripple-on { | |
-webkit-transition: opacity 0.15s ease-in 0s, transform 0.5s cubic-bezier(0.4, 0, 0.2, 1) 0.1s; | |
transition: opacity 0.15s ease-in 0s, transform 0.5s cubic-bezier(0.4, 0, 0.2, 1) 0.1s; | |
-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=15)"; | |
filter: alpha(opacity=15); | |
opacity: 0.15; | |
} | |
.ripple.ripple-out { | |
-webkit-transition: opacity 0.1s linear 0s !important; | |
transition: opacity 0.1s linear 0s !important; | |
-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=0)"; | |
filter: alpha(opacity=0); | |
opacity: 0; | |
} | |
/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/styl/ripple.styl","../node_modules/kouto-swiss/lib/kouto-swiss/utilities/overflow.styl","../node_modules/kouto-swiss/lib/kouto-swiss/mixins/vendors.styl"],"names":[],"mappings":"AAEA;EACE,UAAS,SAAT;EACA,KAAI,EAAJ;EACA,MAAK,EAAL;EACA,SAAQ,EAAR;EACA,OAAM,KAAN;EACA,QAAO,KAAP;ECFM,UAAS,OAAT;EDIN,eAAc,QAAd;EACA,gBAAe,KAAf;;AAEF;EACE,UAAS,SAAT;EACA,OAAM,KAAN;EACA,QAAO,KAAP;EACA,aAAY,MAAZ;EACA,YAAW,MAAX;EACA,eAAc,KAAd;EACA,kBAAmC,iBAAnC;EEb+B,mBAAG,SAAH;EAAA,eAAG,SAAH;EAShB,WAAG,SAAH;EATgB,0BAAG,IAAH;EAAA,sBAAG,IAAH;EAShB,kBAAG,IAAH;EAiID,YAAsE,qDAAtE;EACA,QAA8B,iBAA9B;EACR,SAAQ,EAAR;EF3HN,gBAAe,KAAf;;AAEF;EEnBiC,oBAAG,2EAAH;EAShB,YAAG,2EAAH;EAiID,YAAsE,sDAAtE;EACA,QAA8B,kBAA9B;EACR,SAAQ,KAAR;;AFrHR;EEvBiC,oBAAG,kCAAH;EAShB,YAAG,kCAAH;EAiID,YAAsE,qDAAtE;EACA,QAA8B,iBAA9B;EACR,SAAQ,EAAR","file":"ripple.css","sourceRoot":"..","sourcesContent":["@import 'kouto-swiss'\r\n\r\n.ripple-wrapper\r\n  position absolute\r\n  top 0\r\n  left 0\r\n  z-index 1\r\n  width 100%\r\n  height 100%\r\n  overflow hidden\r\n  border-radius inherit\r\n  pointer-events none\r\n\r\n.ripple\r\n  position absolute\r\n  width 20px\r\n  height 20px\r\n  margin-left -10px\r\n  margin-top -10px\r\n  border-radius 100%\r\n  background-color rgba(0, 0, 0, 0.05)\r\n  transform scale(1)\r\n  transform-origin 50%\r\n  opacity 0\r\n  pointer-events none\r\n\r\n.ripple.ripple-on\r\n  transition opacity 0.15s ease-in 0s, transform 0.5s cubic-bezier(0.4, 0, 0.2, 1) 0.1s\r\n  opacity 0.15\r\n\r\n.ripple.ripple-out\r\n  transition opacity 0.1s linear 0s !important\r\n  opacity 0","ks-overflow( value, args... )\n    if value == ellipsis\n        white-space nowrap\n        text-overflow ellipsis\n        overflow hidden\n    else\n        overflow value args\n\noverflow = ks-overflow unless ks-no-conflict\n","ks-vendors-prefixes = recommended\n\nks-vendor( property, value, feature = null, prefixes = null, official = true )\n    unless \"vendor\" in called-from\n        if feature isnt null and !( feature in ks-support-ignore-features )\n            for prefix in caniuse-prefixes( feature )\n                prefixedproperty = \"-\" + prefix + \"-\" + property\n                {prefixedproperty}: value\n        else if ks-vendors-prefixes isnt false\n                if ks-vendors-prefixes isnt recommended\n                    prefixes = ks-vendors-prefixes\n                if prefixes\n                    for prefix in prefixes\n                        prefixedproperty = \"-\" + prefix + \"-\" + property\n                        {prefixedproperty}: value\n    if official\n        {property}: value\n\nvendor = ks-vendor unless ks-no-conflict\n\nunless ks-no-conflict\n    animation()\n        ks-vendor( \"animation\", arguments, feature: \"css-animation\" )\n    animation-delay()\n        ks-vendor( \"animation-delay\", arguments, feature: \"css-animation\" )\n    animation-direction()\n        ks-vendor( \"animation-direction\", arguments, feature: \"css-animation\" )\n    animation-duration()\n        ks-vendor( \"animation-duration\", arguments, feature: \"css-animation\" )\n    animation-fill-mode()\n        ks-vendor( \"animation-fill-mode\", arguments, feature: \"css-animation\" )\n    animation-iteration-count()\n        ks-vendor( \"animation-iteration-count\", arguments, feature: \"css-animation\" )\n    animation-name()\n        ks-vendor( \"animation-name\", arguments, feature: \"css-animation\" )\n    animation-play-state()\n        ks-vendor( \"animation-play-state\", arguments, feature: \"css-animation\" )\n    animation-timing-function()\n        ks-vendor( \"animation-timing-function\", arguments, feature: \"css-animation\" )\n\n    appearance()\n        ks-vendor( \"appearance\", arguments, prefixes: webkit moz )\n\n    backface-visibility()\n        ks-vendor( \"backface-visibility\", arguments, prefixes: webkit ms )\n\n    background-clip()\n        ks-vendor( \"background-clip\", arguments, feature: \"background-img-opts\" )\n    background-origin()\n        ks-vendor( \"background-origin\", arguments, feature: \"background-img-opts\" )\n    background-size()\n        ks-vendor( \"background-size\", arguments, feature: \"background-img-opts\" )\n\n    box-sizing()\n        ks-vendor( \"box-sizing\", arguments, feature: \"css3-boxsizing\" )\n\n    border-image-source()\n        ks-vendor( \"border-image-source\", arguments, feature: \"border-image\" )\n    border-image-slice()\n        ks-vendor( \"border-image-slice\", arguments, feature: \"border-image\" )\n    border-image-width()\n        ks-vendor( \"border-image-width\", arguments, feature: \"border-image\" )\n    border-image-outset()\n        ks-vendor( \"border-image-outset\", arguments, feature: \"border-image\" )\n    border-image-repeat()\n        ks-vendor( \"border-image-repeat\", arguments, feature: \"border-image\" )\n    border-image()\n        ks-vendor( \"border-image\", arguments, feature: \"border-image\" )\n\n    clip-path()\n        ks-vendor( \"clip-path\", arguments, prefixes: webkit )\n\n    column-count()\n        ks-vendor( \"column-count\", arguments, feature: \"multicolumn\" )\n    column-fill()\n        ks-vendor( \"column-fill\", arguments, feature: \"multicolumn\" )\n    column-gap()\n        ks-vendor( \"column-gap\", arguments, feature: \"multicolumn\" )\n    column-rule()\n        ks-vendor( \"column-rule\", arguments, feature: \"multicolumn\" )\n    column-rule-color()\n        ks-vendor( \"column-rule-color\", arguments, feature: \"multicolumn\" )\n    column-rule-style()\n        ks-vendor( \"column-rule-style\", arguments, feature: \"multicolumn\" )\n    column-rule-width()\n        ks-vendor( \"column-rule-width\", arguments, feature: \"multicolumn\" )\n    column-span()\n        ks-vendor( \"column-span\", arguments, feature: \"multicolumn\")\n    column-width()\n        ks-vendor( \"column-width\", arguments, feature: \"multicolumn\" )\n    columns()\n        ks-vendor( \"columns\", arguments, feature: \"multicolumn\" )\n\n    grid()\n        ks-vendor( \"grid\", arguments, feature: \"css-grid\" )\n    grid-area()\n        ks-vendor( \"grid-area\", arguments, feature: \"css-grid\" )\n    grid-auto-columns()\n        ks-vendor( \"grid-auto-columns\", arguments, feature: \"css-grid\" )\n    grid-auto-flow()\n        ks-vendor( \"grid-auto-flow\", arguments, feature: \"css-grid\" )\n    grid-auto-position()\n        ks-vendor( \"grid-auto-position\", arguments, feature: \"css-grid\" )\n    grid-auto-rows()\n        ks-vendor( \"grid-auto-rows\", arguments, feature: \"css-grid\" )\n    grid-column()\n        ks-vendor( \"grid-column\", arguments, feature: \"css-grid\" )\n    grid-column-end()\n        ks-vendor( \"grid-column-end\", arguments, feature: \"css-grid\" )\n    grid-column-start()\n        ks-vendor( \"grid-column-start\", arguments, feature: \"css-grid\" )\n    grid-row()\n        ks-vendor( \"grid-row\", arguments, feature: \"css-grid\" )\n    grid-row-end()\n        ks-vendor( \"grid-row-end\", arguments, feature: \"css-grid\" )\n    grid-row-start()\n        ks-vendor( \"grid-row-start\", arguments, feature: \"css-grid\" )\n    grid-template()\n        ks-vendor( \"grid-template\", arguments, feature: \"css-grid\" )\n    grid-template-areas()\n        ks-vendor( \"grid-template-areas\", arguments, feature: \"css-grid\" )\n    grid-template-columns()\n        ks-vendor( \"grid-template-columns\", arguments, feature: \"css-grid\" )\n    grid-template-rows()\n        ks-vendor( \"grid-template-rows\", arguments, feature: \"css-grid\" )\n\n    hyphens()\n        ks-vendor( \"hyphens\", arguments, feature: \"css-hyphens\" )\n\n    justify-content()\n        ks-vendor( \"justify-content\", arguments, feature: \"flexbox\" )\n\n    mask()\n        ks-vendor( \"mask\", arguments, feature: \"css-masks\" )\n\n    opacity( value )\n        if value == inherit or value == initial\n            -ms-filter value\n            filter value\n        else\n            val = round( value * 100 )\n            if val == 100\n                -ms-filter: none\n                filter: none\n            else\n                -ms-filter: '\"progid:DXImageTransform.Microsoft.Alpha(Opacity=%s)\"' % val\n                filter: 'alpha(opacity=%s)' % val\n        opacity value\n\n    order()\n        ks-vendor( \"order\", arguments, feature: \"flexbox\" )\n\n    perspective()\n        if current-property[ 0 ] is \"perspective\"\n            ks-vendor( \"perspective\", arguments, feature: \"transforms3d\" )\n        else\n            unquote( \"perspective(\" + arguments + \")\" )\n\n    perspective-origin()\n        ks-vendor( \"perspective-origin\", arguments, feature: \"transforms3d\" )\n\n    transform()\n        ks-vendor( \"transform\", arguments, feature: \"transforms2d\" )\n    transform-origin()\n        ks-vendor( \"transform-origin\", arguments, feature: \"transforms2d\" )\n    transform-style()\n        ks-vendor( \"transform-style\", arguments, feature: \"transforms3d\" )\n\n    transition()\n        ks-vendor( \"transition\", arguments, feature: \"css-transitions\" )\n    transition-delay()\n        ks-vendor( \"transition-delay\", arguments, feature: \"css-transitions\" )\n    transition-duration()\n        ks-vendor( \"transition-duration\", arguments, feature: \"css-transitions\" )\n    transition-property()\n        ks-vendor( \"transition-property\", arguments, feature: \"css-transitions\" )\n    transition-timing-function()\n        ks-vendor( \"transition-timing-function\", arguments, feature: \"css-transitions\" )\n"]} */ |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment