| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401 | /**! * easy-pie-chart * Lightweight plugin to render simple, animated and retina optimized pie charts * * @license  * @author Robert Fleischmann <rendro87@gmail.com> (http://robert-fleischmann.de) * @version 2.1.7 **/(function (root, factory) {  if (typeof define === 'function' && define.amd) {    // AMD. Register as an anonymous module unless amdModuleId is set    define(["jquery"], function (a0) {      return (factory(a0));    });  } else if (typeof exports === 'object') {    // Node. Does not work with strict CommonJS, but    // only CommonJS-like environments that support module.exports,    // like Node.    module.exports = factory(require("jquery"));  } else {    factory(jQuery);  }}(this, function ($) {/** * Renderer to render the chart on a canvas object * @param {DOMElement} el      DOM element to host the canvas (root of the plugin) * @param {object}     options options object of the plugin */var CanvasRenderer = function(el, options) {	var cachedBackground;	var canvas = document.createElement('canvas');	el.appendChild(canvas);	if (typeof(G_vmlCanvasManager) === 'object') {		G_vmlCanvasManager.initElement(canvas);	}	var ctx = canvas.getContext('2d');	canvas.width = canvas.height = options.size;	// canvas on retina devices	var scaleBy = 1;	if (window.devicePixelRatio > 1) {		scaleBy = window.devicePixelRatio;		canvas.style.width = canvas.style.height = [options.size, 'px'].join('');		canvas.width = canvas.height = options.size * scaleBy;		ctx.scale(scaleBy, scaleBy);	}	// move 0,0 coordinates to the center	ctx.translate(options.size / 2, options.size / 2);	// rotate canvas -90deg	ctx.rotate((-1 / 2 + options.rotate / 180) * Math.PI);	var radius = (options.size - options.lineWidth) / 2;	if (options.scaleColor && options.scaleLength) {		radius -= options.scaleLength + 2; // 2 is the distance between scale and bar	}	// IE polyfill for Date	Date.now = Date.now || function() {		return +(new Date());	};	/**	 * Draw a circle around the center of the canvas	 * @param {strong} color     Valid CSS color string	 * @param {number} lineWidth Width of the line in px	 * @param {number} percent   Percentage to draw (float between -1 and 1)	 */	var drawCircle = function(color, lineWidth, percent) {		percent = Math.min(Math.max(-1, percent || 0), 1);		var isNegative = percent <= 0 ? true : false;		ctx.beginPath();		ctx.arc(0, 0, radius, 0, Math.PI * 2 * percent, isNegative);		ctx.strokeStyle = color;		ctx.lineWidth = lineWidth;		ctx.stroke();	};	/**	 * Draw the scale of the chart	 */	var drawScale = function() {		var offset;		var length;		ctx.lineWidth = 1;		ctx.fillStyle = options.scaleColor;		ctx.save();		for (var i = 24; i > 0; --i) {			if (i % 6 === 0) {				length = options.scaleLength;				offset = 0;			} else {				length = options.scaleLength * 0.6;				offset = options.scaleLength - length;			}			ctx.fillRect(-options.size/2 + offset, 0, length, 1);			ctx.rotate(Math.PI / 12);		}		ctx.restore();	};	/**	 * Request animation frame wrapper with polyfill	 * @return {function} Request animation frame method or timeout fallback	 */	var reqAnimationFrame = (function() {		return  window.requestAnimationFrame ||				window.webkitRequestAnimationFrame ||				window.mozRequestAnimationFrame ||				function(callback) {					window.setTimeout(callback, 1000 / 60);				};	}());	/**	 * Draw the background of the plugin including the scale and the track	 */	var drawBackground = function() {		if(options.scaleColor) drawScale();		if(options.trackColor) drawCircle(options.trackColor, options.trackWidth || options.lineWidth, 1);	};  /**    * Canvas accessor   */  this.getCanvas = function() {    return canvas;  };  /**    * Canvas 2D context 'ctx' accessor   */  this.getCtx = function() {    return ctx;  };	/**	 * Clear the complete canvas	 */	this.clear = function() {		ctx.clearRect(options.size / -2, options.size / -2, options.size, options.size);	};	/**	 * Draw the complete chart	 * @param {number} percent Percent shown by the chart between -100 and 100	 */	this.draw = function(percent) {		// do we need to render a background		if (!!options.scaleColor || !!options.trackColor) {			// getImageData and putImageData are supported			if (ctx.getImageData && ctx.putImageData) {				if (!cachedBackground) {					drawBackground();					cachedBackground = ctx.getImageData(0, 0, options.size * scaleBy, options.size * scaleBy);				} else {					ctx.putImageData(cachedBackground, 0, 0);				}			} else {				this.clear();				drawBackground();			}		} else {			this.clear();		}		ctx.lineCap = options.lineCap;		// if barcolor is a function execute it and pass the percent as a value		var color;		if (typeof(options.barColor) === 'function') {			color = options.barColor(percent);		} else {			color = options.barColor;		}		// draw bar		drawCircle(color, options.lineWidth, percent / 100);	}.bind(this);	/**	 * Animate from some percent to some other percentage	 * @param {number} from Starting percentage	 * @param {number} to   Final percentage	 */	this.animate = function(from, to) {		var startTime = Date.now();		options.onStart(from, to);		var animation = function() {			var process = Math.min(Date.now() - startTime, options.animate.duration);			var currentValue = options.easing(this, process, from, to - from, options.animate.duration);			this.draw(currentValue);			options.onStep(from, to, currentValue);			if (process >= options.animate.duration) {				options.onStop(from, to);			} else {				reqAnimationFrame(animation);			}		}.bind(this);		reqAnimationFrame(animation);	}.bind(this);};var EasyPieChart = function(el, opts) {	var defaultOptions = {		barColor: '#ef1e25',		trackColor: '#f9f9f9',		scaleColor: '#dfe0e0',		scaleLength: 5,		lineCap: 'round',		lineWidth: 3,		trackWidth: undefined,		size: 110,		rotate: 0,		animate: {			duration: 1000,			enabled: true		},		easing: function (x, t, b, c, d) { // more can be found here: http://gsgd.co.uk/sandbox/jquery/easing/			t = t / (d/2);			if (t < 1) {				return c / 2 * t * t + b;			}			return -c/2 * ((--t)*(t-2) - 1) + b;		},		onStart: function(from, to) {			return;		},		onStep: function(from, to, currentValue) {			return;		},		onStop: function(from, to) {			return;		}	};	// detect present renderer	if (typeof(CanvasRenderer) !== 'undefined') {		defaultOptions.renderer = CanvasRenderer;	} else if (typeof(SVGRenderer) !== 'undefined') {		defaultOptions.renderer = SVGRenderer;	} else {		throw new Error('Please load either the SVG- or the CanvasRenderer');	}	var options = {};	var currentValue = 0;	/**	 * Initialize the plugin by creating the options object and initialize rendering	 */	var init = function() {		this.el = el;		this.options = options;		// merge user options into default options		for (var i in defaultOptions) {			if (defaultOptions.hasOwnProperty(i)) {				options[i] = opts && typeof(opts[i]) !== 'undefined' ? opts[i] : defaultOptions[i];				if (typeof(options[i]) === 'function') {					options[i] = options[i].bind(this);				}			}		}		// check for jQuery easing		if (typeof(options.easing) === 'string' && typeof(jQuery) !== 'undefined' && jQuery.isFunction(jQuery.easing[options.easing])) {			options.easing = jQuery.easing[options.easing];		} else {			options.easing = defaultOptions.easing;		}		// process earlier animate option to avoid bc breaks		if (typeof(options.animate) === 'number') {			options.animate = {				duration: options.animate,				enabled: true			};		}		if (typeof(options.animate) === 'boolean' && !options.animate) {			options.animate = {				duration: 1000,				enabled: options.animate			};		}		// create renderer		this.renderer = new options.renderer(el, options);		// initial draw		this.renderer.draw(currentValue);		// initial update		if (el.dataset && el.dataset.percent) {			this.update(parseFloat(el.dataset.percent));		} else if (el.getAttribute && el.getAttribute('data-percent')) {			this.update(parseFloat(el.getAttribute('data-percent')));		}	}.bind(this);	/**	 * Update the value of the chart	 * @param  {number} newValue Number between 0 and 100	 * @return {object}          Instance of the plugin for method chaining	 */	this.update = function(newValue) {		newValue = parseFloat(newValue);		if (options.animate.enabled) {			this.renderer.animate(currentValue, newValue);		} else {			this.renderer.draw(newValue);		}		currentValue = newValue;		return this;	}.bind(this);	/**	 * Disable animation	 * @return {object} Instance of the plugin for method chaining	 */	this.disableAnimation = function() {		options.animate.enabled = false;		return this;	};	/**	 * Enable animation	 * @return {object} Instance of the plugin for method chaining	 */	this.enableAnimation = function() {		options.animate.enabled = true;		return this;	};	init();};$.fn.easyPieChart = function(options) {	return this.each(function() {		var instanceOptions;		if (!$.data(this, 'easyPieChart')) {			instanceOptions = $.extend({}, options, $(this).data());			$.data(this, 'easyPieChart', new EasyPieChart(this, instanceOptions));		}	});};}));document.addEventListener('DOMContentLoaded', function () {	/* 	Easy pie chart Snippet		DOC: make sure to include this snippet in your project to be able to use the easy 		configurations without any jquery implementations	 */	$('.js-easy-pie-chart').each(function() {		var $this = $(this),			barcolor = $this.css('color') || color.primary._700,			trackcolor = $this.data('trackcolor') || 'rgba(0,0,0,0.04)',			size = parseInt($this.data('piesize')) || 50,			scalecolor =   $this.data('scalecolor') || $this.css('color'),			scalelength = parseInt($this.data('scalelength')) || 0,			linewidth = parseInt($this.data('linewidth')) ||  parseInt(size / 8.5),			linecap = $this.data('linecap') || 'butt'; //butt, round and square.					$this.easyPieChart({			size : size,			barColor : barcolor,			trackColor : trackcolor,			scaleColor: scalecolor,			scaleLength: scalelength, //Length of the scale lines (reduces the radius of the chart).			lineCap : linecap, //Defines how the ending of the bar line looks like. Possible values are: butt, round and square.			lineWidth : linewidth,			animate: {				duration: 1500,				enabled: true			},			onStep: function(from, to, percent) {				$(this.el).find('.js-percent').text(Math.round(percent));			}		});		$this = null;	});});
 |