| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- //Anonymous sely-executing function
- (function (root, factory) {
- factory(root.jQuery);
- }(this, function ($) {
- var CanvasRenderer = function (element, options) {
- var cachedBackground;
- var canvas = document.createElement('canvas');
- element.appendChild(canvas);
- var ctx = canvas.getContext('2d');
- canvas.width = canvas.height = options.size;
- // 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;
- Date.now = Date.now || function () {
- //convert to milliseconds
- return +(new Date());
- };
- 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();
- };
- /**
- * 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 track
- */
- var drawBackground = function () {
- if (options.trackColor) drawCircle(options.trackColor, options.lineWidth, 1);
- };
- /**
- * Clear the complete canvas
- */
- this.clear = function () {
- ctx.clearRect(options.size / -2, options.size / -2, options.size, options.size);
- };
- /**
- * Draw the complete chart
- * param percent Percent shown by the chart between -100 and 100
- */
- this.draw = function (percent) {
- if (!!options.trackColor) {
- // getImageData and putImageData are supported
- if (ctx.getImageData && ctx.putImageData) {
- if (!cachedBackground) {
- drawBackground();
- cachedBackground = ctx.getImageData(0, 0, options.size, options.size);
- } else {
- ctx.putImageData(cachedBackground, 0, 0);
- }
- } else {
- this.clear();
- drawBackground();
- }
- } else {
- this.clear();
- }
- ctx.lineCap = options.lineCap;
- // draw bar
- drawCircle(options.barColor, options.lineWidth, percent / 100);
- }.bind(this);
- this.animate = function (from, to) {
- var startTime = Date.now();
- 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);
- //Show the number at the center of the circle
- options.onStep(from, to, currentValue);
- reqAnimationFrame(animation);
- }.bind(this);
- reqAnimationFrame(animation);
- }.bind(this);
- };
- var pieChart = function (element, userOptions) {
- var defaultOptions = {
- barColor: '#ef1e25',
- trackColor: '#f9f9f9',
- lineCap: 'round',
- lineWidth: 3,
- size: 100,
- rotate: 0,
- animate: {
- duration: 1000,
- enabled: true
- },
- easing: function (x, t, b, c, d) {//copy from jQuery easing animate
- t = t / (d / 2);
- if (t < 1) {
- return c / 2 * t * t + b;
- }
- return -c / 2 * ((--t) * (t - 2) - 1) + b;
- },
- onStep: function (from, to, currentValue) {
- return;
- },
- renderer: CanvasRenderer//Maybe SVGRenderer more later
- };
- var options = {};
- var currentValue = 0;
- var init = function () {
- this.element = element;
- this.options = options;
- // merge user options into default options
- for (var i in defaultOptions) {
- if (defaultOptions.hasOwnProperty(i)) {
- options[i] = userOptions && typeof(userOptions[i]) !== 'undefined' ? userOptions[i] : defaultOptions[i];
- if (typeof(options[i]) === 'function') {
- options[i] = options[i].bind(this);
- }
- }
- }
- // check for jQuery easing, use jQuery easing first
- 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;
- }
- // create renderer
- this.renderer = new options.renderer(element, options);
- // initial draw
- this.renderer.draw(currentValue);
- if (element.getAttribute && element.getAttribute('data-percent')) {
- var newValue = parseFloat(element.getAttribute('data-percent'));
- if (options.animate.enabled) {
- this.renderer.animate(currentValue, newValue);
- } else {
- this.renderer.draw(newValue);
- }
- currentValue = newValue;
- }
- }.bind(this)();
- };
- $.fn.pieChart = function (options) {
- //Iterate all the dom to draw the pie-charts
- return this.each(function () {
- if (!$.data(this, 'pieChart')) {
- var userOptions = $.extend({}, options, $(this).data());
- $.data(this, 'pieChart', new pieChart(this, userOptions));
- }
- });
- };
- }));
|