// o---------------------------------------------------------------------------------o // | This file is part of the RGraph package - you can learn more at: | // | | // | https://www.rgraph.net/license.html | // | | // | RGraph is dual-licensed under the Open Source GPL license. That means that it's | // | free to use and there are no restrictions on what you can use RGraph for! | // | If the GPL license does not suit you however, then there's an inexpensive | // | commercial license option available. See the URL above for more details. | // o---------------------------------------------------------------------------------o RGraph = window.RGraph || {isrgraph:true,isRGraph:true,rgraph:true}; // // The Activity meter constructor // RGraph.Activity = function (conf) { var id = conf.id, canvas = document.getElementById(id), min = conf.min, max = conf.max, value = conf.value; // id, min, max, value // Get the canvas and context objects this.id = id; this.canvas = canvas; this.context = this.canvas.getContext ? this.canvas.getContext("2d", {alpha: (typeof id === 'object' && id.alpha === false) ? false : true}) : null; this.canvas.__object__ = this; this.type = 'activity'; this.min = RGraph.stringsToNumbers(min); this.max = RGraph.stringsToNumbers(max); this.value = RGraph.stringsToNumbers(value); this.centerx = null; this.centery = null; this.radius = null; this.isRGraph = true; this.isrgraph = true; this.rgraph = true; this.currentValue = null; this.uid = RGraph.createUID(); this.canvas.uid = this.canvas.uid ? this.canvas.uid : RGraph.createUID(); this.colorsParsed = false; this.coordsText = []; this.angles = this.coords = []; this.original_colors = []; this.images = []; this.firstDraw = true; // After the first draw this will be false this.stopAnimationRequested = false;// Used to control the animations // // If the value is zero set it to very // slightly more than zero so the meter // is drawn correctly. // // Likewise with the maximum value // if (typeof this.value === 'number') { this.value = [this.value]; } for (var i=0; i<this.value.length; ++i) { if (this.value[i] <= 0.0000001) { this.value[i] = 0.0000001; } } // Various config type stuff this.properties = { radius: null, centerx: null, centery: null, width: null, ends: 'round', marginLeft: 15, marginRight: 15, marginTop: 15, marginBottom: 15, marginInner: 1, backgroundGrid: false, backgroundGridColor: '#666', backgroundGridCircles: true, backgroundGridCirclesCount: null, backgroundGridRadials: true, backgroundGridRadialsCount: 8, backgroundRings: true, backgroundRingsColors: null, backgroundRingsAlpha: 0.5, backgroundColor: 'black', colors: ['#F45B5B','#90EE7E','#2B908F','red','green','blue','yellow','pink'], icons: [], iconsWidth: null, iconsHeight: null, iconsOffsetx: 0, iconsOffsety: 0, textFont: 'Arial, Verdana, sans-serif', textSize: 12, textColor: '#aaa', textBold: false, textItalic: false, textAccessible: false, textAccessibleOverflow: 'visible', textAccessiblePointerevents: false, text: null, labelsCenter: false, labelsCenterIndex: 0, labelsCenterFont: null, labelsCenterSize: 50, labelsCenterColor: null, labelsCenterBold: null, labelsCenterItalic: null, labelsCenterUnitsPre: '', labelsCenterUnitsPost: '', labelsCenterDecimals: 0, labelsCenterPoint: '.', labelsCenterThousand: ',', labelsCenterSpecific: '', labelsCenterOffsetx: 0, labelsCenterOffsety: 0, labels: [], labelsColor: null, labelsFont: null, labelsSize: null, labelsBold: null, labelsItalic: null, labelsBackgroundFill: 'transparent', labelsBackgroundStroke: 'transparent', labelsOffsetx: 0, labelsOffsety: 0, labelsFormattedDecimals: 0, labelsFormattedPoint: '.', labelsFormattedThousand: ',', labelsFormattedUnitsPre: '', labelsFormattedUnitsPost: '', contextmenu: null, annotatable: false, annotatableColor: 'black', annotatableLinewidth: 1, adjustable: false, tooltips: null, tooltipsEffect: 'slide', tooltipsCssClass: 'RGraph_tooltip', tooltipsCss: null, tooltipsEvent: 'onclick', tooltipsHighlight: true, tooltipsPersistent: false, tooltipsHotspotXonly: false, tooltipsHotspotIgnore: null, tooltipsFormattedThousand: ',', tooltipsFormattedPoint: '.', tooltipsFormattedDecimals: 0, tooltipsFormattedUnitsPre: '', tooltipsFormattedUnitsPost: '', tooltipsFormattedKeyColors: null, tooltipsFormattedKeyColorsShape: 'square', tooltipsFormattedKeyLabels: [], tooltipsFormattedListType: 'ul', tooltipsFormattedListItems: null, tooltipsFormattedTableHeaders: null, tooltipsFormattedTableData: null, tooltipsPointer: true, tooltipsPointerOffsetx: 0, tooltipsPointerOffsety: 0, tooltipsPositionStatic: true, highlightStyle: null, highlightStroke: 'rgba(0,0,0,0)', highlightFill: 'rgba(255,255,255,0.5)', clearto: 'rgba(0,0,0,0)' } // // Add the reverse look-up table for property names // so that property names can be specified in any case. // this.properties_lowercase_map = []; for (var i in this.properties) { if (typeof i === 'string') { this.properties_lowercase_map[i.toLowerCase()] = i; } } // Check for support if (!this.canvas) { alert('[ACTIVITY] No canvas support'); return; } // Easy access to properties and the path function var properties = this.properties; this.path = RGraph.pathObjectFunction; // // "Decorate" the object with the generic effects if the effects library has been included // if (RGraph.Effects && typeof RGraph.Effects.decorate === 'function') { RGraph.Effects.decorate(this); } // Add the responsive method. This method resides in the common file. this.responsive = RGraph.responsive; // // A setter // this.set = function (name) { var value = typeof arguments[1] === 'undefined' ? null : arguments[1]; // Go through all of the properties and make sure // that they're using the correct capitalisation if (typeof name === 'string') { name = this.properties_lowercase_map[name.toLowerCase()] || name; } // Set the colorsParsed flag to false if the colors // property is being set if ( name === 'colors' || name === 'backgroundColor' ) { this.colorsParsed = false; } // the number of arguments is only one and it's an // object - parse it for configuration data and return. if (arguments.length === 1 && typeof arguments[0] === 'object') { for (i in arguments[0]) { if (typeof i === 'string') { this.set(i, arguments[0][i]); } } return this; } properties[name] = value; return this; }; // // A getter // // @param name string The name of the property to get // this.get = function (name) { // Go through all of the properties and make sure // that they're using the correct capitalisation name = this.properties_lowercase_map[name.toLowerCase()] || name; return properties[name]; }; // // The function you call to draw the bar chart // this.draw = function () { // // Fire the onbeforedraw event // RGraph.fireCustomEvent(this, 'onbeforedraw'); // Translate half a pixel for antialiasing purposes - // but only if it hasn't been done already // // MUST be the first thing done to the canvas! // if (!this.canvas.__rgraph_aa_translated__) { this.context.translate(0.5,0.5); this.canvas.__rgraph_aa_translated__ = true; } // Ensure that the obj.value property is an array if (typeof this.value === 'number') { this.value = [this.value]; } // // Constrain the value to be within the min and max // for (var i=0; i<this.value.length; ++i) { if (this.value[i] > this.max) this.value[i] = this.max; if (this.value[i] < this.min) this.value[i] = this.min; } // // Set the current value // this.currentValue = RGraph.arrayClone(this.value); // // Make the margins easy to access // this.marginLeft = properties.marginLeft; this.marginRight = properties.marginRight; this.marginTop = properties.marginTop; this.marginBottom = properties.marginBottom; this.centerx = ((this.canvas.width - this.marginLeft - this.marginRight) / 2) + this.marginLeft; this.centery = ((this.canvas.height - this.marginBottom - this.marginTop) / 2) + this.marginTop; this.radius = Math.min( (this.canvas.width - this.marginLeft - this.marginRight) / 2, (this.canvas.height - this.marginTop - this.marginBottom) / 2 ); // // Stop this growing uncontrollably // this.coordsText = []; // // Custom centerx, centery and radius // if (typeof properties.centerx === 'number') this.centerx = properties.centerx; if (typeof properties.centery === 'number') this.centery = properties.centery; if (typeof properties.radius === 'number') this.radius = properties.radius; // // Allow the centerx/centery/radius to be a plus/minus // if (typeof this.properties.radius === 'string' && this.properties.radius.match(/^\+|-\d+$/) ) this.radius += parseFloat(this.properties.radius); if (typeof this.properties.centerx === 'string' && this.properties.centerx.match(/^\+|-\d+$/) ) this.centerx += parseFloat(this.properties.centerx); if (typeof this.properties.centery === 'string' && this.properties.centery.match(/^\+|-\d+$/) ) this.centery += parseFloat(this.properties.centery); // The width variable is the width of an individual ring this.width = typeof properties.width === 'number' ? properties.width : ((this.radius * 0.75) / this.value.length); this.width = this.width - properties.marginInner - properties.marginInner; // Allow for +/-xx style width if (typeof properties.width === 'string') { this.width += Number(properties.width); } // // Parse the colors for gradients. Its down here so that the center X/Y can be used // if (!this.colorsParsed) { this.parseColors(); // Don't want to do this again this.colorsParsed = true; } // // Install clipping // if (!RGraph.isNull(this.properties.clip)) { RGraph.clipTo.start(this, this.properties.clip); } // // Draw the meter and its label // this.drawBackground(); this.drawMeter(); this.drawLabels(); this.drawIcons(); this.drawEnds(); // // Setup the context menu if required // if (properties.contextmenu) { RGraph.showContext(this); } // // Add custom text thats specified // RGraph.addCustomText(this); // // This installs the event listeners // RGraph.installEventListeners(this); // // End clipping // if (!RGraph.isNull(this.properties.clip)) { RGraph.clipTo.end(); } // // Fire the onfirstdraw event // if (this.firstDraw) { this.firstDraw = false; RGraph.fireCustomEvent(this, 'onfirstdraw'); this.firstDrawFunc(); } // // Fire the RGraph draw event // RGraph.fireCustomEvent(this, 'ondraw'); // // Install any inline responsive configuration. This // should be last in the draw function - even after // the draw events. // RGraph.installInlineResponsive(this); return this; }; // // Used in chaining. Runs a function there and then - not waiting for // the events to fire (eg the onbeforedraw event) // // @param function func The function to execute // this.exec = function (func) { func(this); return this; }; // // Draw the background"grid" // this.drawBackground = function () { // First thing to do is clear the canvas to the backgroundColor if (properties.backgroundColor) { this.path( 'fs % fr -5 -5 % %', properties.backgroundColor, this.canvas.width + 10, this.canvas.height + 10 ); } // Draw the grid? if (properties.backgroundGrid) { // Draw the background grey circles if (properties.backgroundGridCircles) { // How many background circles? if (typeof properties.backgroundGridCirclesCount === 'number') { var count = properties.backgroundGridCirclesCount; } else { var count = this.value.length + 1; } for (var i=0; i<count; i++) { var radius = this.radius - (i * (this.width + (2 * properties.marginInner))); this.path( 'b a % % % 0 6.29 false s %', this.centerx, this.centery, radius, properties.backgroundGridColor ); } // // Draw the background lines that go from the center outwards // if (properties.backgroundGridRadials) { var angle = (RGraph.TWOPI / properties.backgroundGridRadialsCount); for (var i=0; i<=properties.backgroundGridRadialsCount; ++i) { // Radius must be greater than 0 for Opera to work this.path( 'b a % % % % % % a % % % % % % s %', this.centerx, this.centery, radius, (i * angle) - RGraph.HALFPI, (i * angle) - RGraph.HALFPI, false, this.centerx, this.centery, this.radius, (i * angle) - RGraph.HALFPI, (i * angle) - RGraph.HALFPI, false, properties.backgroundGridColor ); } } } } }; // // Draws the meter // this.drawMeter = function () { var color; // This is the maximum radius of the outer ring var r = (this.radius - properties.marginInner); // Draw the background rings if enabled if (properties.backgroundRings) { // // Loop through the values and draw the background rings // this.context.globalAlpha = properties.backgroundRingsAlpha; for (var i=0; i<this.value.length; ++i) { // Determine the color if (RGraph.isArray(properties.backgroundRingsColors) && typeof properties.backgroundRingsColors[i] === 'string') { color = properties.backgroundRingsColors[i]; } else { color = properties.colors[i]; } this.path( 'b a % % % % % false a % % % % % true f %', this.centerx, this.centery, r - (i * (properties.marginInner + properties.marginInner + this.width)), 0 - RGraph.HALFPI, RGraph.TWOPI - RGraph.HALFPI, this.centerx, this.centery, r - this.width - (i * (properties.marginInner + properties.marginInner + this.width)), RGraph.TWOPI - RGraph.HALFPI, 0 - RGraph.HALFPI, color ); } // Reset the alpha value this.context.globalAlpha = 1; } // // Now loop through the values and draw each circle // for (var i=0; i<this.value.length; ++i) { var radians = ((this.value[i] - this.min) / (this.max - this.min)) * RGraph.TWOPI; this.path( 'b a % % % % % false a % % % % % true f %', this.centerx, this.centery, r - (i * (properties.marginInner + properties.marginInner + this.width)), 0 - RGraph.HALFPI, radians - RGraph.HALFPI, this.centerx, this.centery, r - this.width - (i * (properties.marginInner + properties.marginInner + this.width)), radians - RGraph.HALFPI, 0 - RGraph.HALFPI, properties.colors[i] ); // Store the coordinates of the ring this.angles[i] = this.coords[i] = { x: this.centerx, y: this.centery, angleStart: 0 - RGraph.HALFPI, angleEnd: radians - RGraph.HALFPI, radiusInner: r - this.width - (i * (properties.marginInner + properties.marginInner + this.width)), radiusOuter: r - (i * (properties.marginInner + properties.marginInner + this.width)), color: properties.colors[i], }; } // // By calling the RGraph.text() function with a blank // string the text accessible wrapper DIV will be // created and the responsive function will work // as expected (ie the css: and parentCss: options). // RGraph.text({ object: this, x: -10000, y: -10000, color: 'transparent', text: '' }); }; // // Draws the center label and the vertical labels if enabled // this.drawLabels = function () { // // Draw the label if specified // if (properties.labelsCenter) { var textConf = RGraph.getTextConf({ object: this, prefix: 'labelsCenter' }); RGraph.text({ object: this, text: String(properties.labelsCenterSpecific ? properties.labelsCenterSpecific : RGraph.numberFormat({ object: this, number: this.value[properties.labelsCenterIndex].toFixed(properties.labelsCenterDecimals), unitspre: properties.labelsCenterUnitsPre, unitspost: properties.labelsCenterUnitsPost, point: properties.labelsCenterPoint, thousand: properties.labelsCenterThousand })), x: this.centerx + properties.labelsCenterOffsetx, y: this.centery + properties.labelsCenterOffsety, halign: 'center', valign: 'center', color: textConf.color, size: textConf.size, font: textConf.font, bold: textConf.bold, italic: textConf.italic }); } var textConf = RGraph.getTextConf({ object: this, prefix: 'labels' }); if (typeof properties.labels === 'string') { properties.labels = RGraph.arrayPad({ array: [], length: this.coords.length, value: properties.labels }); } // Loop thru the labels for (var i=0; i<properties.labels.length; ++i) { if ( typeof properties.labels === 'object' && properties.labels.length && typeof properties.labels[i] === 'string' ) { var text = RGraph.labelSubstitution({ object: this, text: properties.labels[i], index: i, value: this.value[i], decimals: properties.labelsFormattedDecimals, point: properties.labelsFormattedPoint, thousand: properties.labelsFormattedThousand, unitsPre: properties.labelsFormattedUnitsPre, unitsPost: properties.labelsFormattedUnitsPost, }); } RGraph.text({ object: this, text: ( RGraph.isString(properties.labels[i]) || RGraph.isNumber(properties.labels[i]) ) ? text : '', x: this.coords[i].x - 5 + properties.labelsOffsetx, y: this.coords[i].y - this.coords[i].radiusOuter + ((this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2) + properties.labelsOffsety, valign: 'center', halign: 'right', bounding: true, boundingFill: properties.labelsBackgroundFill, boundingStroke: properties.labelsBackgroundStroke, textConfPrefix: 'labels', }); } }; // // Draws the icons at the start of the circles. // this.drawIcons = function () { for (var i=0; i<this.value.length; ++i) { if (typeof properties.icons[i] === 'string' && properties.icons[i].length) { this.images[i] = new Image(); this.images[i].src = properties.icons[i]; this.images[i].index = i; var obj = this; this.images[i].onload = function () { // Aids readability var img = this; // IMPORTANT: // // In this callback the 'this' variable is the image object. The // 'obj' variable is the chart object. // var width = img.width; var height = width; // Reduce the size of the icon if necessary if (width >= obj.width) { width -= 10; height -= 10; } // If a width has been stipulated then use it if (typeof properties.iconsWidth === 'number') { width = properties.iconsWidth; } // If a height has been specified then use it if (typeof properties.iconsHeight === 'number') { height = properties.iconsHeight; } var x = obj.centerx - (width / 3) + ((properties.ends === 'straight' || properties.ends === 'square') ? 15 : 0); var y = (obj.centery - obj.coords[this.index].radiusOuter + ((obj.coords[this.index].radiusOuter - obj.coords[this.index].radiusInner) / 2) - (height / 2) ); x += properties.iconsOffsetx; y += properties.iconsOffsety; obj.context.drawImage( this, x, y, width, height ); }; } } }; // // Draws the rounded ends to the bars // this.drawEnds = function () { if (properties.ends === 'straight' || properties.ends === 'square') { return; } for (var i=0; i<this.coords.length; ++i) { var x = this.coords[i].x; var y = this.coords[i].y - ((this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2) - this.coords[i].radiusInner; this.path( 'b a % % % % % false f %', x, y, this.width / 2, 0, 6.29, properties.colors[i] ); // cx,cy,angle,radius var xy = RGraph.getRadiusEndPoint({ cx: this.centerx, cy: this.centery, angle: this.coords[i].angleEnd, radius: ((this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2) + this.coords[i].radiusInner }); this.path( 'b a % % % % % false f %', xy[0], xy[1], (this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2, 0, 6.29, properties.colors[i] ); } }; // // A placeholder function // // @param object The event object // this.getShape = function (e) { var mouseXY = RGraph.getMouseXY(e); var mouseX = mouseXY[0]; var mouseY = mouseXY[1]; // Go through the obj.coords array and find the correct index/shape for (var i=0; i<this.coords.length; ++i) { if (RGraph.tooltipsHotspotIgnore(this, i)) { continue; } // Recreate the shape but don't fill/stroke it this.path( 'b a % % % % % false a % % % % % true', this.coords[i].x, this.coords[i].y, this.coords[i].radiusOuter, this.coords[i].angleStart, this.coords[i].angleEnd, this.coords[i].x, this.coords[i].y, this.coords[i].radiusInner, this.coords[i].angleEnd, this.coords[i].angleStart ); // // Add the ends to the path if the bars are capped // if (properties.ends) { this.path( 'a % % % % % false', this.coords[i].x, this.coords[i].y - this.coords[i].radiusInner - (this.width / 2), this.width / 2, RGraph.HALFPI, RGraph.PI + RGraph.HALFPI ); // cx,cy,angle,radius of the circle at the end of the bar var xy = RGraph.getRadiusEndPoint({ cx: this.centerx, cy: this.centery, angle: this.coords[i].angleEnd, radius: ((this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2) + this.coords[i].radiusInner }); this.path( 'm % % a % % % % % false', xy[0], xy[1], xy[0], xy[1], (this.coords[i].radiusOuter - this.coords[i].radiusInner) / 2, 0, 6.29 ); } if ( this.context.isPointInPath(mouseX, mouseY) && (this.properties.clip ? RGraph.clipTo.test(this, mouseX, mouseY) : true) ) { if (RGraph.parseTooltipText) { var tooltip = RGraph.parseTooltipText(properties.tooltips, i); } return { object: this, x: this.coords[i].x, y: this.coords[i].y, radiusInner: this.coords[i].radiusInner, radiusOuter: this.coords[i].radiusOuter, angleStart: this.coords[i].angleStart, angleEnd: this.coords[i].angleEnd, tooltip: typeof tooltip === 'string' ? tooltip : null, label: properties.labels && typeof properties.labels[i] === 'string' ? properties.labels[i] : null, dataset: 0, index: i, sequentialIndex: i }; } } }; // // This function returns the pertinent value for a particular click (or other mouse event) // // @param obj e The event object // this.getValue = function (e) { var mouseXY = RGraph.getMouseXY(e); var angle = RGraph.getAngleByXY( this.centerx, this.centery, mouseXY[0], mouseXY[1] ); // Adjust the angle because canvas angles // start at the east axis angle += RGraph.HALFPI; if (angle > RGraph.TWOPI) { angle -= RGraph.TWOPI; } // Calculate the value based on the angle and min/max values var value = ((angle / RGraph.TWOPI) * (this.max - this.min)) + this.min; // Ensure that the value is in range value = Math.max(value, this.min); value = Math.min(value, this.max); return value; }; // // The getObjectByXY() worker method. Don't call this call: // // RGraph.ObjectRegistry.getObjectByXY(e) // // @param object e The event object // this.getObjectByXY = function (e) { var mouseXY = RGraph.getMouseXY(e), width = this.width; // Work out the radius var radius = RGraph.getHypLength( this.centerx, this.centery, mouseXY[0], mouseXY[1] ); if (radius > this.radius || radius < (this.radius * 0.25) ) { return null; } return this; }; // // This method handles the adjusting calculation for when the mouse is moved // // @param object e The event object // this.adjusting_mousemove = function (e) { // // Handle adjusting for the Bar // if (properties.adjustable && RGraph.Registry.get('adjusting') && RGraph.Registry.get('adjusting').uid == this.uid) { var mouseXY = RGraph.getMouseXY(e); var radius = RGraph.getHypLength( this.centerx, this.centery, mouseXY[0], mouseXY[1] ); var index = this.getIndexByRadius(radius); // Store the index in the radius if (typeof RGraph.Registry.get('adjusting.index') !== 'number') { RGraph.Registry.set('adjusting.index', index); RGraph.addCustomEventListener(this, 'onadjustend', function () { RGraph.Registry.set('adjusting.index', null); }); } else { index = RGraph.Registry.get('adjusting.index') } if (typeof index === 'number') { this.value[index] = this.getValue(e); RGraph.clear(this.canvas); RGraph.redrawCanvas(this.canvas); RGraph.fireCustomEvent(this, 'onadjust'); } } }; // // Returns the relevant index of the relevant datapoint. // But only if it's in range. // // @param number radius The radius to check // this.getIndexByRadius = function (radius) { // Loop through the shapes to determine the correct shape for (var i=0; i<this.coords.length; ++i) { if (radius < this.coords[i].radiusOuter && radius > this.coords[i].radiusInner) { return i; } } return null; }; // // This method returns the appropriate angle for a value // // @param number value The value // this.getAngle = function (value) { // Higher than max if (value > this.max || value < this.min) { return null; } var angle = (((value - this.min) / (this.max - this.min)) * RGraph.TWOPI) - RGraph.HALFPI; if (value === this.max) angle -= 0.00001; if (value === this.min) angle += 0.00001; return angle; }; // // Each object type has its own Highlight() function which highlights the appropriate shape // // @param object shape The shape to highlight // this.highlight = function (shape) { if (typeof properties.highlightStyle === 'function') { if (properties.tooltipsHighlight) { (properties.highlightStyle)(shape); } // Highlight all of the rings except this one - essentially an inverted highlight } else if (typeof properties.highlightStyle === 'string' && properties.highlightStyle === 'invert') { if (properties.tooltipsHighlight) { this.context.beginPath(); for (var i=0; i<this.coords.length; ++i) { if (shape.index !== i) { this.pathBar(i); } } this.context.fillStyle = properties.highlightFill; this.context.strokeStyle = properties.highlightStroke; this.context.stroke(); this.context.fill(); } // Regular highlighting } else { if (properties.tooltipsHighlight) { this.context.beginPath(); this.pathBar(shape.index); this.context.fillStyle = properties.highlightFill; this.context.strokeStyle = properties.highlightStroke; this.context.stroke(); this.context.fill(); } } }; // // Paths a bar given an index but does not: // o stroke // o fill // o issue a beginPath() call // o issue a closePath() call // // Do NOT use when drawing the bars initially - the .coords array // must already be populated. // this.pathBar = function (index) { // Path the outer edge this.path( 'a % % % % % false', this.coords[index].x, this.coords[index].y, this.coords[index].radiusOuter, 0 - RGraph.HALFPI, this.coords[index].angleEnd ); // Get the center x/y coords of the end of the bar var xy = RGraph.getRadiusEndPoint({ cx: this.centerx, cy: this.centery, angle: this.coords[index].angleEnd, radius: this.coords[index].radiusInner + ((this.coords[index].radiusOuter - this.coords[index].radiusInner) / 2) }); // Path the ending curved end if the ends are round if (properties.ends === 'round') { this.path( 'a % % % % % false', xy[0], xy[1], this.width / 2, this.coords[index].angleEnd, this.coords[index].angleEnd + RGraph.PI ); } // Path the inner edge this.path( 'a % % % % % true', this.coords[index].x, this.coords[index].y, this.coords[index].radiusInner, this.coords[index].angleEnd, this.coords[index].angleStart ); // Path the starting curved end if the ends are round if (properties.ends === 'round') { this.path( 'a % % % % % false', this.centerx, this.centery - this.coords[index].radiusInner - ((this.coords[index].radiusOuter - this.coords[index].radiusInner) / 2), this.width / 2, RGraph.HALFPI, RGraph.PI + RGraph.HALFPI ); } }; // // A worker function that handles Activity chart specific tooltip substitutions // this.tooltipSubstitutions = function (opt) { return { index: opt.index, dataset: 0, sequentialIndex: opt.index, value: this.value[opt.index], values: this.value }; }; // // A worker function that returns the correct color/label/value // // @param object specific The indexes that are applicable // @param number index The appropriate index // this.tooltipsFormattedCustom = function (specific, index) { return { label: properties.tooltipsFormattedKeyLabels[index] }; }; // // This allows for static tooltip positioning // this.positionTooltipStatic = function (args) { var obj = args.object, e = args.event, tooltip = args.tooltip, index = args.index, canvasXY = RGraph.getCanvasXY(obj.canvas), coords = this.coords[args.index]; //cx,cy,angle,radius var point = RGraph.getRadiusEndPoint({ cx: this.centerx, cy: this.centery, radius: ((coords.radiusOuter - coords.radiusInner) / 2) + coords.radiusInner, angle: ((coords.angleEnd - coords.angleStart) / 2) + (0 - RGraph.HALFPI) }); // Position the tooltip in the X direction //args.tooltip.style.left = point[0] - (tooltip.offsetWidth / 2) + 'px'; args.tooltip.style.left = canvasXY[0] + point[0] - (tooltip.offsetWidth / 2) + obj.properties.tooltipsOffsetx + 'px'; args.tooltip.style.top = canvasXY[1] + point[1] - tooltip.offsetHeight - 10 + obj.properties.tooltipsOffsety + 'px'; // If the top of the tooltip is off the top of the page // then move the tooltip down if(parseFloat(args.tooltip.style.top) < 0) { args.tooltip.style.top = 5 + 'px'; } }; // // This allows for easy specification of gradients // this.parseColors = function () { // Save the original colors so that they can be restored when the canvas is reset if (this.original_colors.length === 0) { this.original_colors.backgroundColor = RGraph.arrayClone(properties.backgroundColor); this.original_colors.colors = RGraph.arrayClone(properties.colors); } // Parse the background color properties.backgroundColor = this.parseSingleColorForGradient(properties.backgroundColor); // Parse colors var colors = properties.colors; if (colors && colors.length) { for (var i=0; i<colors.length; ++i) { colors[i] = this.parseSingleColorForGradient(colors[i]); } } }; // // Use this function to reset the object to the post-constructor state. Eg reset colors if // need be etc // this.reset = function () { }; // // This parses a single color value // this.parseSingleColorForGradient = function (color) { if (!color || typeof color != 'string') { return color; } if (color.match(/^gradient\((.*)\)$/i)) { // Allow for JSON gradients if (color.match(/^gradient\(({.*})\)$/i)) { return RGraph.parseJSONGradient({ object: this, def: RegExp.$1, radial: true }); } var parts = RegExp.$1.split(':'); // Create the gradient var grad = this.context.createLinearGradient( properties.marginLeft, 0, this.canvas.width - properties.marginLeft - properties.marginRight, 0 ); var diff = 1 / (parts.length - 1); grad.addColorStop(0, RGraph.trim(parts[0])); for (var j=1,len=parts.length; j<len; ++j) { grad.addColorStop(j * diff, RGraph.trim(parts[j])); } } return grad ? grad : color; }; // // Using a function to add events makes it easier to facilitate method chaining // // @param string type The type of even to add // @param function func // this.on = function (type, func) { if (type.substr(0,2) !== 'on') { type = 'on' + type; } if (typeof this[type] !== 'function') { this[type] = func; } else { RGraph.addCustomEventListener(this, type, func); } return this; }; // // This function runs once only // (put at the end of the file (before any effects)) // this.firstDrawFunc = function () { }; // // Activity chart Grow effect // // This effect gradually increases the represented value // // @param An object of options - eg: {frames: 60} // @param function An optional callback function // this.grow = function () { // Cancel any stop request if one is pending this.cancelStopAnimation(); var obj = this; // // Convert the value to an array if its a number // if (typeof this.value === 'number') { this.value = [this.value]; } //this.currentValue = this.currentValue || obj.min; if (RGraph.isNull(this.currentValue)) { this.currentValue=[]; for (var i=0; i<this.value.length; ++i) { this.currentValue[i] = this.min; } } var opt = arguments[0] || {}, frames = opt.frames || 30, frame = 0, diff = [], step = [], callback = arguments[1] || function () {}, initial = []; // Set a few properties too by looping through the data for (var i=0; i<this.value.length; ++i) { diff[i] = this.value[i] - this.currentValue[i]; step[i] = diff[i] / frames; initial[i] = this.currentValue[i]; } function iterator () { if (obj.stopAnimationRequested) { // Reset the flag obj.stopAnimationRequested = false; return; } for (var i=0; i<obj.value.length; ++i) { obj.value[i] = initial[i] + (frame * step[i]); } // Increment the frame frame++; RGraph.clear(obj.canvas); RGraph.redrawCanvas(obj.canvas); if (frame <= frames) { RGraph.Effects.updateCanvas(iterator); } else { callback(obj); } } iterator(); return this; }; // // Couple of functions that allow you to control the // animation effect // this.stopAnimation = function () { this.stopAnimationRequested = true; }; this.cancelStopAnimation = function () { this.stopAnimationRequested = false; }; // // This function handles clipping to scale values. Because // each chart handles scales differently, a worker function // is needed instead of it all being done centrally in the // RGraph.clipTo.start() function. // // @param string clip The clip string as supplied by the // user in the chart configuration // this.clipToScaleWorker = function (clip) { // The Regular expression is actually done by the // calling RGraph.clipTo.start() function in the core // library if (RegExp.$1 === 'min') from = this.min; else from = Number(RegExp.$1); if (RegExp.$2 === 'max') to = this.max; else to = Number(RegExp.$2); var a1 = this.getAngle(from), a2 = this.getAngle(to); // Change the radius if the number is "min" if (RegExp.$1 === 'min') { a1 = this.getAngle(this.min); } // Change the radius if the number is "max" if (RegExp.$2 === 'max') { a2 = this.getAngle(this.max); } this.path( 'sa b m % % a % % % % % false c cl', this.centerx, this.centery, this.centerx, this.centery, Math.max(this.canvas.width, this.canvas.height), a1, a2 ); }; // // This function handles TESTING clipping to scale values. // Because each chart handles scales differently, a worker // function is needed instead of it all being done // centrally in the RGraph.clipTo.start() function. // // @param string clip The clip string as supplied by the // user in the chart configuration // this.clipToScaleTestWorker = function (clip) { // The Regular expression is actually done by the // calling RGraph.clipTo.start() function in the core // library if (RegExp.$1 === 'min') from = this.min; else from = Number(RegExp.$1); if (RegExp.$2 === 'max') to = this.max; else to = Number(RegExp.$2); var a1 = this.getAngle(from), a2 = this.getAngle(to); // Change the radius if the number is "min" if (RegExp.$1 === 'min') { a1 = this.getAngle(this.min); } // Change the radius if the number is "max" if (RegExp.$2 === 'max') { a2 = this.getAngle(this.max); } this.path( 'b m % % a % % % % % false c', this.centerx, this.centery, this.centerx, this.centery, Math.max(this.canvas.width, this.canvas.height), a1, a2 ); }; // // Register the object // RGraph.register(this); // // This is the 'end' of the constructor so if the first argument // contains configuration data - handle that. // RGraph.parseObjectStyleConfig(this, conf.options); };