Skip to content

Instantly share code, notes, and snippets.

@grvcoelho
Created December 28, 2014 17:42
Show Gist options
  • Save grvcoelho/60d5926b3f908607052e to your computer and use it in GitHub Desktop.
Save grvcoelho/60d5926b3f908607052e to your computer and use it in GitHub Desktop.
MYjqLq
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Material Design Ripple Effect</title>
</head>
<body>
<div class="btn"></div>
</body>
</html>
(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();
})
.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