AutorÃa | Ultima modificación | Ver Log |
/**
* o-------------------------------------------------------------------------------o
* | This file is part of the RGraph package. RGraph is Free software, licensed |
* | under the MIT license - so it's free to use for all purposes. Extended |
* | support is available if required and donations are always welcome! You can |
* | read more here: |
* | http://www.rgraph.net/support |
* o-------------------------------------------------------------------------------o
*/
/**
* Initialise the various objects
*/
if (typeof(RGraph) == 'undefined') RGraph = {isRGraph:true,type:'common'};
RGraph.Highlight = {};
RGraph.Registry = {};
RGraph.Registry.store = [];
RGraph.Registry.store['chart.event.handlers'] = [];
RGraph.Registry.store['__rgraph_event_listeners__'] = []; // Used in the new system for tooltips
RGraph.background = {};
RGraph.objects = [];
RGraph.Resizing = {};
RGraph.events = [];
RGraph.cursor = [];
RGraph.HTML = RGraph.HTML || {};
RGraph.ObjectRegistry = {};
RGraph.ObjectRegistry.objects = {};
RGraph.ObjectRegistry.objects.byUID = [];
RGraph.ObjectRegistry.objects.byCanvasID = [];
/**
* Some "constants"
*/
PI = Math.PI;
HALFPI = PI / 2;
TWOPI = PI * 2;
ISFF = navigator.userAgent.indexOf('Firefox') != -1;
ISOPERA = navigator.userAgent.indexOf('Opera') != -1;
ISCHROME = navigator.userAgent.indexOf('Chrome') != -1;
ISSAFARI = navigator.userAgent.indexOf('Safari') != -1 && !ISCHROME;
ISWEBKIT = navigator.userAgent.indexOf('WebKit') != -1;
//ISIE is defined below
//ISIE6 is defined below
//ISIE7 is defined below
//ISIE8 is defined below
//ISIE9 is defined below
//ISIE9 is defined below
//ISIE9UP is defined below
//ISIE10 is defined below
//ISIE10UP is defined below
//ISIE11 is defined below
//ISIE11UP is defined below
//ISOLD is defined below
/**
* Returns five values which are used as a nice scale
*
* @param max int The maximum value of the graph
* @param obj object The graph object
* @return array An appropriate scale
*/
RGraph.getScale = function (max, obj)
{
/**
* Special case for 0
*/
if (max == 0) {
return ['0.2', '0.4', '0.6', '0.8', '1.0'];
}
var original_max = max;
/**
* Manually do decimals
*/
if (max <= 1) {
if (max > 0.5) {
return [0.2,0.4,0.6,0.8, Number(1).toFixed(1)];
} else if (max >= 0.1) {
return obj.Get('chart.scale.round') ? [0.2,0.4,0.6,0.8,1] : [0.1,0.2,0.3,0.4,0.5];
} else {
var tmp = max;
var exp = 0;
while (tmp < 1.01) {
exp += 1;
tmp *= 10;
}
var ret = ['2e-' + exp, '4e-' + exp, '6e-' + exp, '8e-' + exp, '10e-' + exp];
if (max <= ('5e-' + exp)) {
ret = ['1e-' + exp, '2e-' + exp, '3e-' + exp, '4e-' + exp, '5e-' + exp];
}
return ret;
}
}
// Take off any decimals
if (String(max).indexOf('.') > 0) {
max = String(max).replace(/\.\d+$/, '');
}
var interval = Math.pow(10, Number(String(Number(max)).length - 1));
var topValue = interval;
while (topValue < max) {
topValue += (interval / 2);
}
// Handles cases where the max is (for example) 50.5
if (Number(original_max) > Number(topValue)) {
topValue += (interval / 2);
}
// Custom if the max is greater than 5 and less than 10
if (max < 10) {
topValue = (Number(original_max) <= 5 ? 5 : 10);
}
/**
* Added 02/11/2010 to create "nicer" scales
*/
if (obj && typeof(obj.Get('chart.scale.round')) == 'boolean' && obj.Get('chart.scale.round')) {
topValue = 10 * interval;
}
return [topValue * 0.2, topValue * 0.4, topValue * 0.6, topValue * 0.8, topValue];
}
/**
* Returns an appropriate scale. The return value is actualy anm object consiosting of:
* scale.max
* scale.min
* scale.scale
*
* @param obj object The graph object
* @param prop object An object consisting of configuration properties
* @return object An object containg scale information
*/
RGraph.getScale2 = function (obj, opt)
{
var RG = RGraph;
var ca = obj.canvas;
var co = obj.context;
var prop = obj.properties;
var numlabels = typeof(opt['ylabels.count']) == 'number' ? opt['ylabels.count'] : 5;
var units_pre = typeof(opt['units.pre']) == 'string' ? opt['units.pre'] : '';
var units_post = typeof(opt['units.post']) == 'string' ? opt['units.post'] : '';
var max = Number(opt['max']);
var min = typeof(opt['min']) == 'number' ? opt['min'] : 0;
var strict = opt['strict'];
var decimals = Number(opt['scale.decimals']); // Sometimes the default is null
var point = opt['scale.point']; // Default is a string in all chart libraries so no need to cast it
var thousand = opt['scale.thousand']; // Default is a string in all chart libraries so no need to cast it
var original_max = max;
var round = opt['scale.round'];
var scale = {'max':1,'labels':[]};
/**
* Special case for 0
*
* ** Must be first **
*/
if (!max) {
var max = 1;
var scale = {max:1,min:0,labels:[]};
for (var i=0; i<numlabels; ++i) {
var label = ((((max - min) / numlabels) + min) * (i + 1)).toFixed(decimals);
scale.labels.push(units_pre + label + units_post);
}
/**
* Manually do decimals
*/
} else if (max <= 1 && !strict) {
if (max > 0.5) {
max = 1;
min = min;
scale.min = min;
for (var i=0; i<numlabels; ++i) {
var label = ((((max - min) / numlabels) * (i + 1)) + min).toFixed(decimals);
scale.labels.push(units_pre + label + units_post);
}
} else if (max >= 0.1) {
max = 0.5;
min = min;
scale = {'max': 0.5, 'min':min,'labels':[]}
for (var i=0; i<numlabels; ++i) {
var label = ((((max - min) / numlabels) + min) * (i + 1)).toFixed(decimals);
scale.labels.push(units_pre + label + units_post);
}
} else {
scale = {'min':min,'labels':[]}
var max_str = String(max);
if (max_str.indexOf('e') > 0) {
var numdecimals = Math.abs(max_str.substring(max_str.indexOf('e') + 1));
} else {
var numdecimals = String(max).length - 2;
}
var max = 1 / Math.pow(10,numdecimals - 1);
for (var i=0; i<numlabels; ++i) {
var label = ((((max - min) / numlabels) + min) * (i + 1));
label = label.toExponential();
label = label.split(/e/);
label[0] = Math.round(label[0]);
label = label.join('e');
scale.labels.push(label);
}
//This makes the top scale value of the format 10e-2 instead of 1e-1
tmp = scale.labels[scale.labels.length - 1].split(/e/);
tmp[0] += 0;
tmp[1] = Number(tmp[1]) - 1;
tmp = tmp[0] + 'e' + tmp[1];
scale.labels[scale.labels.length - 1] = tmp;
// Add the units
for (var i=0; i<scale.labels.length ; ++i) {
scale.labels[i] = units_pre + scale.labels[i] + units_post;
}
scale.max = Number(max);
}
} else if (!strict) {
/**
* Now comes the scale handling for integer values
*/
// This accomodates decimals by rounding the max up to the next integer
max = Math.ceil(max);
var interval = Math.pow(10, Math.max(1, Number(String(Number(max) - Number(min)).length - 1)) );
var topValue = interval;
while (topValue < max) {
topValue += (interval / 2);
}
// Handles cases where the max is (for example) 50.5
if (Number(original_max) > Number(topValue)) {
topValue += (interval / 2);
}
// Custom if the max is greater than 5 and less than 10
if (max <= 10) {
topValue = (Number(original_max) <= 5 ? 5 : 10);
}
// Added 02/11/2010 to create "nicer" scales
if (obj && typeof(round) == 'boolean' && round) {
topValue = 10 * interval;
}
scale.max = topValue;
// Now generate the scale. Temporarily set the objects chart.scale.decimal and chart.scale.point to those
//that we've been given as the number_format functuion looks at those instead of using argumrnts.
var tmp_point = prop['chart.scale.point'];
var tmp_thousand = prop['chart.scale.thousand'];
obj.Set('chart.scale.thousand', thousand);
obj.Set('chart.scale.point', point);
for (var i=0; i<numlabels; ++i) {
scale.labels.push( RG.number_format(obj, ((((i+1) / numlabels) * (topValue - min)) + min).toFixed(decimals), units_pre, units_post) );
}
obj.Set('chart.scale.thousand', tmp_thousand);
obj.Set('chart.scale.point', tmp_point);
} else if (typeof(max) == 'number' && strict) {
/**
* ymax is set and also strict
*/
for (var i=0; i<numlabels; ++i) {
scale.labels.push( RG.number_format(obj, ((((i+1) / numlabels) * (max - min)) + min).toFixed(decimals), units_pre, units_post) );
}
// ???
scale.max = max;
}
scale.units_pre = units_pre;
scale.units_post = units_post;
scale.point = point;
scale.decimals = decimals;
scale.thousand = thousand;
scale.numlabels = numlabels;
scale.round = Boolean(round);
scale.min = min;
return scale;
}
/**
* Returns the maximum numeric value which is in an array
*
* @param array arr The array (can also be a number, in which case it's returned as-is)
* @param int Whether to ignore signs (ie negative/positive)
* @return int The maximum value in the array
*/
RGraph.array_max = function (arr)
{
var max = null;
var MathLocal = Math;
if (typeof(arr) == 'number') {
return arr;
}
if (RGraph.is_null(arr)) {
return 0;
}
for (var i=0,len=arr.length; i<len; ++i) {
if (typeof(arr[i]) == 'number') {
var val = arguments[1] ? MathLocal.abs(arr[i]) : arr[i];
if (typeof max == 'number') {
max = MathLocal.max(max, val);
} else {
max = val;
}
}
}
return max;
}
/**
* Returns the maximum value which is in an array
*
* @param array arr The array
* @param int len The length to pad the array to
* @param mixed The value to use to pad the array (optional)
*/
RGraph.array_pad = function (arr, len)
{
if (arr.length < len) {
var val = arguments[2] ? arguments[2] : null;
for (var i=arr.length; i<len; i+=1) {
arr[i] = val;
}
}
return arr;
}
/**
* An array sum function
*
* @param array arr The array to calculate the total of
* @return int The summed total of the arrays elements
*/
RGraph.array_sum = function (arr)
{
// Allow integers
if (typeof(arr) == 'number') {
return arr;
}
// Account for null
if (RGraph.is_null(arr)) {
return 0;
}
var i, sum;
var len = arr.length;
for(i=0,sum=0;i<len;sum+=arr[i++]);
return sum;
}
/**
* Takes any number of arguments and adds them to one big linear array
* which is then returned
*
* @param ... mixed The data to linearise. You can strings, booleans, numbers or arrays
*/
RGraph.array_linearize = function ()
{
var arr = [];
var args = arguments;
var RG = RGraph;
for (var i=0,len=args.length; i<len; ++i) {
if (typeof(args[i]) == 'object' && args[i]) {
for (var j=0; j<args[i].length; ++j) {
var sub = RG.array_linearize(args[i][j]);
for (var k=0; k<sub.length; ++k) {
arr.push(sub[k]);
}
}
} else {
arr.push(args[i]);
}
}
return arr;
}
/**
* This is a useful function which is basically a shortcut for drawing left, right, top and bottom alligned text.
*
* @param object context The context
* @param string font The font
* @param int size The size of the text
* @param int x The X coordinate
* @param int y The Y coordinate
* @param string text The text to draw
* @parm string The vertical alignment. Can be null. "center" gives center aligned text, "top" gives top aligned text.
* Anything else produces bottom aligned text. Default is bottom.
* @param string The horizontal alignment. Can be null. "center" gives center aligned text, "right" gives right aligned text.
* Anything else produces left aligned text. Default is left.
* @param bool Whether to show a bounding box around the text. Defaults not to
* @param int The angle that the text should be rotate at (IN DEGREES)
* @param string Background color for the text
* @param bool Whether the text is bold or not
*/
RGraph.Text = function (context, font, size, x, y, text)
{
// "Cache" the args as a local variable
var args = arguments;
// Handle undefined - change it to an empty string
if ((typeof(text) != 'string' && typeof(text) != 'number') || text == 'undefined') {
return;
}
/**
* This accommodates multi-line text
*/
if (typeof(text) == 'string' && text.match(/\r\n/)) {
var dimensions = RGraph.MeasureText('M', args[11], font, size);
/**
* Measure the text (width and height)
*/
var arr = text.split('\r\n');
/**
* Adjust the Y position
*/
// This adjusts the initial y position
if (args[6] && args[6] == 'center') y = (y - (dimensions[1] * ((arr.length - 1) / 2)));
for (var i=1; i<arr.length; ++i) {
RGraph.Text(context,
font,
size,
args[9] == -90 ? (x + (size * 1.5)) : x,
y + (dimensions[1] * i),
arr[i],
args[6] ? args[6] : null,
args[7],
args[8],
args[9],
args[10],
args[11],
args[12]);
}
// Update text to just be the first line
text = arr[0];
}
// Accommodate MSIE
if (document.all && ISOLD) {
y += 2;
}
context.font = (args[11] ? 'Bold ': '') + size + 'pt ' + font;
var i;
var origX = x;
var origY = y;
var originalFillStyle = context.fillStyle;
var originalLineWidth = context.lineWidth;
// Need these now the angle can be specified, ie defaults for the former two args
if (typeof(args[6]) == 'undefined') args[6] = 'bottom'; // Vertical alignment. Default to bottom/baseline
if (typeof(args[7]) == 'undefined') args[7] = 'left'; // Horizontal alignment. Default to left
if (typeof(args[8]) == 'undefined') args[8] = null; // Show a bounding box. Useful for positioning during development. Defaults to false
if (typeof(args[9]) == 'undefined') args[9] = 0; // Angle (IN DEGREES) that the text should be drawn at. 0 is middle right, and it goes clockwise
// The alignment is recorded here for purposes of Opera compatibility
if (navigator.userAgent.indexOf('Opera') != -1) {
context.canvas.__rgraph_valign__ = args[6];
context.canvas.__rgraph_halign__ = args[7];
}
// First, translate to x/y coords
context.save();
context.canvas.__rgraph_originalx__ = x;
context.canvas.__rgraph_originaly__ = y;
context.translate(x, y);
x = 0;
y = 0;
// Rotate the canvas if need be
if (args[9]) {
context.rotate(args[9] / (180 / PI));
}
// Vertical alignment - defaults to bottom
if (args[6]) {
var vAlign = args[6];
if (vAlign == 'center') {
context.textBaseline = 'middle';
} else if (vAlign == 'top') {
context.textBaseline = 'top';
}
}
// Hoeizontal alignment - defaults to left
if (args[7]) {
var hAlign = args[7];
var width = context.measureText(text).width;
if (hAlign) {
if (hAlign == 'center') {
context.textAlign = 'center';
} else if (hAlign == 'right') {
context.textAlign = 'right';
}
}
}
context.fillStyle = originalFillStyle;
/**
* Draw a bounding box if requested
*/
context.save();
context.fillText(text,0,0);
context.lineWidth = 1;
var width = context.measureText(text).width;
var width_offset = (hAlign == 'center' ? (width / 2) : (hAlign == 'right' ? width : 0));
var height = size * 1.5; // !!!
var height_offset = (vAlign == 'center' ? (height / 2) : (vAlign == 'top' ? height : 0));
var ieOffset = ISOLD ? 2 : 0;
if (args[8]) {
context.strokeRect(-3 - width_offset,
0 - 3 - height - ieOffset + height_offset,
width + 6,
height + 6);
/**
* If requested, draw a background for the text
*/
if (args[10]) {
context.fillStyle = args[10];
context.fillRect(-3 - width_offset,
0 - 3 - height - ieOffset + height_offset,
width + 6,
height + 6);
}
context.fillStyle = originalFillStyle;
/**
* Do the actual drawing of the text
*/
context.fillText(text,0,0);
}
context.restore();
// Reset the lineWidth
context.lineWidth = originalLineWidth;
context.restore();
}
/**
* Clears the canvas by setting the width. You can specify a colour if you wish.
*
* @param object canvas The canvas to clear
*/
RGraph.Clear = function (ca)
{
var RG = RGraph;
var co = ca.getContext('2d');
var color = arguments[1];
if (!ca) {
return;
}
RG.FireCustomEvent(ca.__object__, 'onbeforeclear');
if (ISIE8 && !color) {
color = 'white';
}
/**
* Can now clear the canvas back to fully transparent
*/
if (!color || (color && color == 'rgba(0,0,0,0)' || color == 'transparent')) {
co.clearRect(0,0,ca.width, ca.height);
// Reset the globalCompositeOperation
co.globalCompositeOperation = 'source-over';
} else {
co.fillStyle = color;
co.beginPath();
if (ISIE8) {
co.fillRect(0,0,ca.width,ca.height);
} else {
co.fillRect(-10,-10,ca.width + 20,ca.height + 20);
}
co.fill();
}
//if (RG.ClearAnnotations) {
//RG.ClearAnnotations(ca.id);
//}
/**
* This removes any background image that may be present
*/
if (RG.Registry.Get('chart.background.image.' + ca.id)) {
var img = RG.Registry.Get('chart.background.image.' + ca.id);
img.style.position = 'absolute';
img.style.left = '-10000px';
img.style.top = '-10000px';
}
/**
* This hides the tooltip that is showing IF it has the same canvas ID as
* that which is being cleared
*/
if (RG.Registry.Get('chart.tooltip')) {
RG.HideTooltip(ca);
//RG.Redraw();
}
/**
* Set the cursor to default
*/
ca.style.cursor = 'default';
RG.FireCustomEvent(ca.__object__, 'onclear');
}
/**
* Draws the title of the graph
*
* @param object canvas The canvas object
* @param string text The title to write
* @param integer gutter The size of the gutter
* @param integer The center X point (optional - if not given it will be generated from the canvas width)
* @param integer Size of the text. If not given it will be 14
*/
RGraph.DrawTitle = function (obj, text, gutterTop)
{
var RG = RGraph;
var ca = canvas = obj.canvas;
var co = context = obj.context;
var prop = obj.properties;
var gutterLeft = prop['chart.gutter.left'];
var gutterRight = prop['chart.gutter.right'];
var gutterTop = gutterTop;
var gutterBottom = prop['chart.gutter.bottom'];
var size = arguments[4] ? arguments[4] : 12;
var bold = prop['chart.title.bold'];
var centerx = (arguments[3] ? arguments[3] : ((ca.width - gutterLeft - gutterRight) / 2) + gutterLeft);
var keypos = prop['chart.key.position'];
var vpos = prop['chart.title.vpos'];
var hpos = prop['chart.title.hpos'];
var bgcolor = prop['chart.title.background'];
var x = prop['chart.title.x'];
var y = prop['chart.title.y'];
var halign = 'center';
var valign = 'center';
// Account for 3D effect by faking the key position
if (obj.type == 'bar' && prop['chart.variant'] == '3d') {
keypos = 'gutter';
}
co.beginPath();
co.fillStyle = prop['chart.text.color'] ? prop['chart.text.color'] : 'black';
/**
* Vertically center the text if the key is not present
*/
if (keypos && keypos != 'gutter') {
var valign = 'center';
} else if (!keypos) {
var valign = 'center';
} else {
var valign = 'bottom';
}
// if chart.title.vpos is a number, use that
if (typeof(prop['chart.title.vpos']) == 'number') {
vpos = prop['chart.title.vpos'] * gutterTop;
if (prop['chart.xaxispos'] == 'top') {
vpos = prop['chart.title.vpos'] * gutterBottom + gutterTop + (ca.height - gutterTop - gutterBottom);
}
} else {
vpos = gutterTop - size - 5;
if (prop['chart.xaxispos'] == 'top') {
vpos = ca.height - gutterBottom + size + 5;
}
}
// if chart.title.hpos is a number, use that. It's multiplied with the (entire) canvas width
if (typeof(hpos) == 'number') {
centerx = hpos * ca.width;
}
/**
* Now the chart.title.x and chart.title.y settings override (is set) the above
*/
if (typeof(x) == 'number') centerx = x;
if (typeof(y) == 'number') vpos = y;
/**
* Horizontal alignment can now (Jan 2013) be specified
*/
if (typeof(prop['chart.title.halign']) == 'string') {
halign = prop['chart.title.halign'];
}
/**
* Vertical alignment can now (Jan 2013) be specified
*/
if (typeof(prop['chart.title.valign']) == 'string') {
valign = prop['chart.title.valign'];
}
// Set the colour
if (typeof(prop['chart.title.color'] != null)) {
var oldColor = co.fillStyle
var newColor = prop['chart.title.color']
co.fillStyle = newColor ? newColor : 'black';
}
/**
* Default font is Arial
*/
var font = prop['chart.text.font'];
/**
* Override the default font with chart.title.font
*/
if (typeof(prop['chart.title.font']) == 'string') {
font = prop['chart.title.font'];
}
/**
* Draw the title
*/
RG.Text2(obj,{'font':font,
'size':size,
'x':centerx,
'y':vpos,
'text':text,
'valign':valign,
'halign':halign,
'bounding':bgcolor != null,
'bounding.fill':bgcolor,
'bold':bold,
'tag':'title'
});
// Reset the fill colour
co.fillStyle = oldColor;
}
/**
* This function returns the mouse position in relation to the canvas
*
* @param object e The event object.
*
RGraph.getMouseXY = function (e)
{
var el = (ISOLD ? event.srcElement : e.target);
var x;
var y;
// ???
var paddingLeft = el.style.paddingLeft ? parseInt(el.style.paddingLeft) : 0;
var paddingTop = el.style.paddingTop ? parseInt(el.style.paddingTop) : 0;
var borderLeft = el.style.borderLeftWidth ? parseInt(el.style.borderLeftWidth) : 0;
var borderTop = el.style.borderTopWidth ? parseInt(el.style.borderTopWidth) : 0;
if (ISIE8) e = event;
// Browser with offsetX and offsetY
if (typeof(e.offsetX) == 'number' && typeof(e.offsetY) == 'number') {
x = e.offsetX;
y = e.offsetY;
// FF and other
} else {
x = 0;
y = 0;
while (el != document.body && el) {
x += el.offsetLeft;
y += el.offsetTop;
el = el.offsetParent;
}
x = e.pageX - x;
y = e.pageY - y;
}
return [x, y];
}*/
RGraph.getMouseXY = function(e)
{
var el = e.target;
var ca = el;
var caStyle = ca.style;
var offsetX = 0;
var offsetY = 0;
var x;
var y;
var ISFIXED = (ca.style.position == 'fixed');
var borderLeft = parseInt(caStyle.borderLeftWidth) || 0;
var borderTop = parseInt(caStyle.borderTopWidth) || 0;
var paddingLeft = parseInt(caStyle.paddingLeft) || 0
var paddingTop = parseInt(caStyle.paddingTop) || 0
var additionalX = borderLeft + paddingLeft;
var additionalY = borderTop + paddingTop;
if (typeof(e.offsetX) == 'number' && typeof(e.offsetY) == 'number') {
if (ISFIXED) {
if (ISOPERA) {
x = e.offsetX;
y = e.offsetY;
} else if (ISWEBKIT) {
x = e.offsetX - paddingLeft - borderLeft;
y = e.offsetY - paddingTop - borderTop;
} else if (ISIE) {
x = e.offsetX - paddingLeft;
y = e.offsetY - paddingTop;
} else {
x = e.offsetX;
y = e.offsetY;
}
} else {
if (!ISIE && !ISOPERA) {
x = e.offsetX - borderLeft - paddingLeft;
y = e.offsetY - borderTop - paddingTop;
} else if (ISIE) {
x = e.offsetX - paddingLeft;
y = e.offsetY - paddingTop;
} else {
x = e.offsetX;
y = e.offsetY;
}
}
} else {
if (typeof(el.offsetParent) != 'undefined') {
do {
offsetX += el.offsetLeft;
offsetY += el.offsetTop;
} while ((el = el.offsetParent));
}
x = e.pageX - offsetX - additionalX;
y = e.pageY - offsetY - additionalY;
x -= (2 * (parseInt(document.body.style.borderLeftWidth) || 0));
y -= (2 * (parseInt(document.body.style.borderTopWidth) || 0));
//x += (parseInt(caStyle.borderLeftWidth) || 0);
//y += (parseInt(caStyle.borderTopWidth) || 0);
}
// We return a javascript array with x and y defined
return [x, y];
}
/**
* This function returns a two element array of the canvas x/y position in
* relation to the page
*
* @param object canvas
*/
RGraph.getCanvasXY = function (canvas)
{
var x = 0;
var y = 0;
var el = canvas; // !!!
do {
x += el.offsetLeft;
y += el.offsetTop;
// ACCOUNT FOR TABLES IN wEBkIT
if (el.tagName.toLowerCase() == 'table' && (ISCHROME || ISSAFARI)) {
x += parseInt(el.border) || 0;
y += parseInt(el.border) || 0;
}
el = el.offsetParent;
} while (el && el.tagName.toLowerCase() != 'body');
var paddingLeft = canvas.style.paddingLeft ? parseInt(canvas.style.paddingLeft) : 0;
var paddingTop = canvas.style.paddingTop ? parseInt(canvas.style.paddingTop) : 0;
var borderLeft = canvas.style.borderLeftWidth ? parseInt(canvas.style.borderLeftWidth) : 0;
var borderTop = canvas.style.borderTopWidth ? parseInt(canvas.style.borderTopWidth) : 0;
if (navigator.userAgent.indexOf('Firefox') > 0) {
x += parseInt(document.body.style.borderLeftWidth) || 0;
y += parseInt(document.body.style.borderTopWidth) || 0;
}
return [x + paddingLeft + borderLeft, y + paddingTop + borderTop];
}
/**
* This function determines whther a canvas is fixed (CSS positioning) or not. If not it returns
* false. If it is then the element that is fixed is returned (it may be a parent of the canvas).
*
* @return Either false or the fixed positioned element
*/
RGraph.isFixed = function (canvas)
{
var obj = canvas;
var i = 0;
while (obj && obj.tagName.toLowerCase() != 'body' && i < 99) {
if (obj.style.position == 'fixed') {
return obj;
}
obj = obj.offsetParent;
}
return false;
}
/**
* Registers a graph object (used when the canvas is redrawn)
*
* @param object obj The object to be registered
*/
RGraph.Register = function (obj)
{
// Checking this property ensures the object is only registered once
if (!obj.Get('chart.noregister')) {
// As of 21st/1/2012 the object registry is now used
RGraph.ObjectRegistry.Add(obj);
obj.Set('chart.noregister', true);
}
}
/**
* Causes all registered objects to be redrawn
*
* @param string An optional color to use to clear the canvas
*/
RGraph.Redraw = function ()
{
var objectRegistry = RGraph.ObjectRegistry.objects.byCanvasID;
// Get all of the canvas tags on the page
var tags = document.getElementsByTagName('canvas');
for (var i=0,len=tags.length; i<len; ++i) {
if (tags[i].__object__ && tags[i].__object__.isRGraph) {
// Only clear the canvas if it's not Trace'ing - this applies to the Line/Scatter Trace effects
if (!tags[i].noclear) {
RGraph.Clear(tags[i], arguments[0] ? arguments[0] : null);
}
}
}
// Go through the object registry and redraw *all* of the canvas'es that have been registered
for (var i=0,len=objectRegistry.length; i<len; ++i) {
if (objectRegistry[i]) {
var id = objectRegistry[i][0];
objectRegistry[i][1].Draw();
}
}
}
/**
* Causes all registered objects ON THE GIVEN CANVAS to be redrawn
*
* @param canvas object The canvas object to redraw
* @param bool Optional boolean which defaults to true and determines whether to clear the canvas
*/
RGraph.RedrawCanvas = function (canvas)
{
var objects = RGraph.ObjectRegistry.getObjectsByCanvasID(canvas.id);
/**
* First clear the canvas
*/
if (!arguments[1] || (typeof(arguments[1]) == 'boolean' && !arguments[1] == false) ) {
// TODO This function should really support passing a color as the second optional argument - which is then used in the below
// call
RGraph.Clear(canvas);
}
/**
* Now redraw all the charts associated with that canvas
*/
for (var i=0,len=objects.length; i<len; ++i) {
if (objects[i]) {
if (objects[i] && objects[i].isRGraph) { // Is it an RGraph object ??
objects[i].Draw();
}
}
}
}
/**
* This function draws the background for the bar chart, line chart and scatter chart.
*
* @param object obj The graph object
*/
RGraph.background.Draw = function (obj)
{
var RG = RGraph;
var ca = canvas = obj.canvas;
var co = context = obj.context;
var prop = obj.properties;
var height = 0;
var gutterLeft = obj.gutterLeft;
var gutterRight = obj.gutterRight;
var gutterTop = obj.gutterTop;
var gutterBottom = obj.gutterBottom;
var variant = prop['chart.variant'];
co.fillStyle = prop['chart.text.color'];
// If it's a bar and 3D variant, translate
if (variant == '3d') {
co.save();
co.translate(10, -5);
}
// X axis title
if (typeof(prop['chart.title.xaxis']) == 'string' && prop['chart.title.xaxis'].length) {
var size = prop['chart.text.size'] + 2;
var font = prop['chart.text.font'];
var bold = prop['chart.title.xaxis.bold'];
if (typeof(prop['chart.title.xaxis.size']) == 'number') {
size = prop['chart.title.xaxis.size'];
}
if (typeof(prop['chart.title.xaxis.font']) == 'string') {
font = prop['chart.title.xaxis.font'];
}
var hpos = ((ca.width - gutterLeft - gutterRight) / 2) + gutterLeft;
var vpos = ca.height - gutterBottom + 25;
if (typeof(prop['chart.title.xaxis.pos']) == 'number') {
vpos = ca.height - (gutterBottom * prop['chart.title.xaxis.pos']);
}
// Specifically specified X/Y positions
if (typeof prop['chart.title.xaxis.x'] == 'number') {
hpos = prop['chart.title.xaxis.x'];
}
if (typeof prop['chart.title.xaxis.y'] == 'number') {
vpos = prop['chart.title.xaxis.y'];
}
RG.Text2(obj, {'font':font,
'size':size,
'x':hpos,
'y':vpos,
'text':prop['chart.title.xaxis'],
'halign':'center',
'valign':'center',
'bold':bold,
'tag': 'title xaxis'
});
}
// Y axis title
if (typeof(prop['chart.title.yaxis']) == 'string' && prop['chart.title.yaxis'].length) {
var size = prop['chart.text.size'] + 2;
var font = prop['chart.text.font'];
var angle = 270;
var bold = prop['chart.title.yaxis.bold'];
var color = prop['chart.title.yaxis.color'];
if (typeof(prop['chart.title.yaxis.pos']) == 'number') {
var yaxis_title_pos = prop['chart.title.yaxis.pos'] * gutterLeft;
} else {
var yaxis_title_pos = ((gutterLeft - 25) / gutterLeft) * gutterLeft;
}
if (typeof(prop['chart.title.yaxis.size']) == 'number') {
size = prop['chart.title.yaxis.size'];
}
if (typeof(prop['chart.title.yaxis.font']) == 'string') {
font = prop['chart.title.yaxis.font'];
}
if (prop['chart.title.yaxis.align'] == 'right' || prop['chart.title.yaxis.position'] == 'right') {
angle = 90;
yaxis_title_pos = prop['chart.title.yaxis.pos'] ? (ca.width - gutterRight) + (prop['chart.title.yaxis.pos'] * gutterRight) :
ca.width - gutterRight + prop['chart.text.size'] + 5;
} else {
yaxis_title_pos = yaxis_title_pos;
}
var y = ((ca.height - gutterTop - gutterBottom) / 2) + gutterTop;
// Specifically specified X/Y positions
if (typeof prop['chart.title.yaxis.x'] == 'number') {
yaxis_title_pos = prop['chart.title.yaxis.x'];
}
if (typeof prop['chart.title.yaxis.y'] == 'number') {
y = prop['chart.title.yaxis.y'];
}
co.fillStyle = color;
RG.Text2(obj, {'font':font,
'size':size,
'x':yaxis_title_pos,
'y':y,
'valign':'center',
'halign':'center',
'angle':angle,
'bold':bold,
'text':prop['chart.title.yaxis'],
'tag':'title yaxis'
});
}
/**
* If the background color is spec ified - draw that. It's a rectangle that fills the
* entire are within the gutters
*/
var bgcolor = prop['chart.background.color'];
if (bgcolor) {
co.fillStyle = bgcolor;
co.fillRect(gutterLeft, gutterTop, ca.width - gutterLeft - gutterRight, ca.height - gutterTop - gutterBottom);
}
/**
* Draw horizontal background bars
*/
co.beginPath(); // Necessary?
co.fillStyle = prop['chart.background.barcolor1'];
co.strokeStyle = co.fillStyle;
height = (ca.height - gutterBottom);
for (var i=gutterTop; i<height ; i+=80) {
co.fillRect(gutterLeft, i, ca.width - gutterLeft - gutterRight, Math.min(40, ca.height - gutterBottom - i) );
}
co.fillStyle = prop['chart.background.barcolor2'];
co.strokeStyle = co.fillStyle;
height = (ca.height - gutterBottom);
for (var i= (40 + gutterTop); i<height; i+=80) {
co.fillRect(gutterLeft, i, ca.width - gutterLeft - gutterRight, i + 40 > (ca.height - gutterBottom) ? ca.height - (gutterBottom + i) : 40);
}
//context.stroke();
co.beginPath();
// Draw the background grid
if (prop['chart.background.grid']) {
// If autofit is specified, use the .numhlines and .numvlines along with the width to work
// out the hsize and vsize
if (prop['chart.background.grid.autofit']) {
/**
* Align the grid to the tickmarks
*/
if (prop['chart.background.grid.autofit.align']) {
// Align the horizontal lines
obj.Set('chart.background.grid.autofit.numhlines', prop['chart.ylabels.count']);
// Align the vertical lines for the line
if (obj.type == 'line') {
if (prop['chart.labels'] && prop['chart.labels'].length) {
obj.Set('chart.background.grid.autofit.numvlines', prop['chart.labels'].length - 1);
} else {
obj.Set('chart.background.grid.autofit.numvlines', obj.data[0].length - 1);
}
// Align the vertical lines for the bar
} else if (obj.type == 'bar' && prop['chart.labels'] && prop['chart.labels'].length) {
obj.Set('chart.background.grid.autofit.numvlines', prop['chart.labels'].length);
}
}
var vsize = ((ca.width - gutterLeft - gutterRight)) / prop['chart.background.grid.autofit.numvlines'];
var hsize = (ca.height - gutterTop - gutterBottom) / prop['chart.background.grid.autofit.numhlines'];
obj.Set('chart.background.grid.vsize', vsize);
obj.Set('chart.background.grid.hsize', hsize);
}
co.beginPath();
co.lineWidth = prop['chart.background.grid.width'] ? prop['chart.background.grid.width'] : 1;
co.strokeStyle = prop['chart.background.grid.color'];
// Dashed background grid
if (prop['chart.background.grid.dashed'] && typeof co.setLineDash == 'function') {
co.setLineDash([3,2]);
}
// Dotted background grid
if (prop['chart.background.grid.dotted'] && typeof co.setLineDash == 'function') {
co.setLineDash([1,2]);
}
// Draw the horizontal lines
if (prop['chart.background.grid.hlines']) {
height = (ca.height - gutterBottom)
for (y=gutterTop; y<height; y+=prop['chart.background.grid.hsize']) {
context.moveTo(gutterLeft, Math.round(y));
context.lineTo(ca.width - gutterRight, Math.round(y));
}
}
if (prop['chart.background.grid.vlines']) {
// Draw the vertical lines
var width = (ca.width - gutterRight)
for (x=gutterLeft; x<=width; x+=prop['chart.background.grid.vsize']) {
co.moveTo(Math.round(x), gutterTop);
co.lineTo(Math.round(x), ca.height - gutterBottom);
}
}
if (prop['chart.background.grid.border']) {
// Make sure a rectangle, the same colour as the grid goes around the graph
co.strokeStyle = prop['chart.background.grid.color'];
co.strokeRect(Math.round(gutterLeft), Math.round(gutterTop), ca.width - gutterLeft - gutterRight, ca.height - gutterTop - gutterBottom);
}
}
context.stroke();
// Reset the line dash
if (typeof co.setLineDash == 'function') {
co.setLineDash([1,0]);
}
// If it's a bar and 3D variant, translate
if (variant == '3d') {
co.restore();
}
// Draw the title if one is set
if ( typeof(prop['chart.title']) == 'string') {
if (obj.type == 'gantt') {
gutterTop -= 10;
}
RG.DrawTitle(obj,
prop['chart.title'],
gutterTop,
null,
prop['chart.title.size'] ? prop['chart.title.size'] : prop['chart.text.size'] + 2);
}
co.stroke();
}
/**
* Makes a clone of an object
*
* @param obj val The object to clone
*/
RGraph.array_clone = function (obj)
{
var RG = RGraph;
if(obj == null || typeof(obj) != 'object') {
return obj;
}
var temp = [];
for (var i=0,len=obj.length;i<len; ++i) {
if (typeof(obj[i]) == 'number') {
temp[i] = (function (arg) {return Number(arg);})(obj[i]);
} else if (typeof(obj[i]) == 'string') {
temp[i] = (function (arg) {return String(arg);})(obj[i]);
} else if (typeof(obj[i]) == 'function') {
temp[i] = obj[i];
} else {
temp[i] = RG.array_clone(obj[i]);
}
}
return temp;
}
/**
* Formats a number with thousand seperators so it's easier to read
*
* @param integer obj The chart object
* @param integer num The number to format
* @param string The (optional) string to prepend to the string
* @param string The (optional) string to append to the string
* @return string The formatted number
*/
RGraph.number_format = function (obj, num)
{
var RG = RGraph;
var ca = obj.canvas;
var co = obj.context;
var prop = obj.properties;
var i;
var prepend = arguments[2] ? String(arguments[2]) : '';
var append = arguments[3] ? String(arguments[3]) : '';
var output = '';
var decimal = '';
var decimal_seperator = typeof(prop['chart.scale.point']) == 'string' ? prop['chart.scale.point'] : '.';
var thousand_seperator = typeof(prop['chart.scale.thousand']) == 'string' ? prop['chart.scale.thousand'] : ',';
RegExp.$1 = '';
var i,j;
if (typeof(prop['chart.scale.formatter']) == 'function') {
return prop['chart.scale.formatter'](obj, num);
}
// Ignore the preformatted version of "1e-2"
if (String(num).indexOf('e') > 0) {
return String(prepend + String(num) + append);
}
// We need then number as a string
num = String(num);
// Take off the decimal part - we re-append it later
if (num.indexOf('.') > 0) {
var tmp = num;
num = num.replace(/\.(.*)/, ''); // The front part of the number
decimal = tmp.replace(/(.*)\.(.*)/, '$2'); // The decimal part of the number
}
// Thousand seperator
//var seperator = arguments[1] ? String(arguments[1]) : ',';
var seperator = thousand_seperator;
/**
* Work backwards adding the thousand seperators
*/
var foundPoint;
for (i=(num.length - 1),j=0; i>=0; j++,i--) {
var character = num.charAt(i);
if ( j % 3 == 0 && j != 0) {
output += seperator;
}
/**
* Build the output
*/
output += character;
}
/**
* Now need to reverse the string
*/
var rev = output;
output = '';
for (i=(rev.length - 1); i>=0; i--) {
output += rev.charAt(i);
}
// Tidy up
//output = output.replace(/^-,/, '-');
if (output.indexOf('-' + prop['chart.scale.thousand']) == 0) {
output = '-' + output.substr(('-' + prop['chart.scale.thousand']).length);
}
// Reappend the decimal
if (decimal.length) {
output = output + decimal_seperator + decimal;
decimal = '';
RegExp.$1 = '';
}
// Minor bugette
if (output.charAt(0) == '-') {
output = output.replace(/-/, '');
prepend = '-' + prepend;
}
return prepend + output + append;
}
/**
* Draws horizontal coloured bars on something like the bar, line or scatter
*/
RGraph.DrawBars = function (obj)
{
var prop = obj.properties;
var co = obj.context;
var ca = obj.canvas;
var RG = RGraph;
var hbars = prop['chart.background.hbars'];
if (hbars === null) {
return;
}
/**
* Draws a horizontal bar
*/
co.beginPath();
for (i=0,len=hbars.length; i<len; ++i) {
var start = hbars[i][0];
var length = hbars[i][1];
var color = hbars[i][2];
// Perform some bounds checking
if(RG.is_null(start))start = obj.scale2.max
if (start > obj.scale2.max) start = obj.scale2.max;
if (RG.is_null(length)) length = obj.scale2.max - start;
if (start + length > obj.scale2.max) length = obj.scale2.max - start;
if (start + length < (-1 * obj.scale2.max) ) length = (-1 * obj.scale2.max) - start;
if (prop['chart.xaxispos'] == 'center' && start == obj.scale2.max && length < (obj.scale2.max * -2)) {
length = obj.scale2.max * -2;
}
/**
* Draw the bar
*/
var x = prop['chart.gutter.left'];
var y = obj.getYCoord(start);
var w = ca.width - prop['chart.gutter.left'] - prop['chart.gutter.right'];
var h = obj.getYCoord(start + length) - y;
// Accommodate Opera :-/
if (ISOPERA != -1 && prop['chart.xaxispos'] == 'center' && h < 0) {
h *= -1;
y = y - h;
}
/**
* Account for X axis at the top
*/
if (prop['chart.xaxispos'] == 'top') {
y = ca.height - y;
h *= -1;
}
co.fillStyle = color;
co.fillRect(x, y, w, h);
}
/*
// If the X axis is at the bottom, and a negative max is given, warn the user
if (obj.Get('chart.xaxispos') == 'bottom' && (hbars[i][0] < 0 || (hbars[i][1] + hbars[i][1] < 0)) ) {
alert('[' + obj.type.toUpperCase() + ' (ID: ' + obj.id + ') BACKGROUND HBARS] You have a negative value in one of your background hbars values, whilst the X axis is in the center');
}
var ystart = (obj.grapharea - (((hbars[i][0] - obj.scale2.min) / (obj.scale2.max - obj.scale2.min)) * obj.grapharea));
//var height = (Math.min(hbars[i][1], obj.max - hbars[i][0]) / (obj.scale2.max - obj.scale2.min)) * obj.grapharea;
var height = obj.getYCoord(hbars[i][0]) - obj.getYCoord(hbars[i][1]);
// Account for the X axis being in the center
if (obj.Get('chart.xaxispos') == 'center') {
ystart /= 2;
//height /= 2;
}
ystart += obj.Get('chart.gutter.top')
var x = obj.Get('chart.gutter.left');
var y = ystart - height;
var w = obj.canvas.width - obj.Get('chart.gutter.left') - obj.Get('chart.gutter.right');
var h = height;
// Accommodate Opera :-/
if (navigator.userAgent.indexOf('Opera') != -1 && obj.Get('chart.xaxispos') == 'center' && h < 0) {
h *= -1;
y = y - h;
}
/**
* Account for X axis at the top
*/
//if (obj.Get('chart.xaxispos') == 'top') {
// y = obj.canvas.height - y;
// h *= -1;
//}
//obj.context.fillStyle = hbars[i][2];
//obj.context.fillRect(x, y, w, h);
//}
}
/**
* Draws in-graph labels.
*
* @param object obj The graph object
*/
RGraph.DrawInGraphLabels = function (obj)
{
var RG = RGraph;
var ca = obj.canvas;
var co = obj.context;
var prop = obj.properties;
var labels = prop['chart.labels.ingraph'];
var labels_processed = [];
// Defaults
var fgcolor = 'black';
var bgcolor = 'white';
var direction = 1;
if (!labels) {
return;
}
/**
* Preprocess the labels array. Numbers are expanded
*/
for (var i=0,len=labels.length; i<len; i+=1) {
if (typeof(labels[i]) == 'number') {
for (var j=0; j<labels[i]; ++j) {
labels_processed.push(null);
}
} else if (typeof(labels[i]) == 'string' || typeof(labels[i]) == 'object') {
labels_processed.push(labels[i]);
} else {
labels_processed.push('');
}
}
/**
* Turn off any shadow
*/
RG.NoShadow(obj);
if (labels_processed && labels_processed.length > 0) {
for (var i=0,len=labels_processed.length; i<len; ++i) {
if (labels_processed[i]) {
var coords = obj.coords[i];
if (coords && coords.length > 0) {
var x = (obj.type == 'bar' ? coords[0] + (coords[2] / 2) : coords[0]);
var y = (obj.type == 'bar' ? coords[1] + (coords[3] / 2) : coords[1]);
var length = typeof(labels_processed[i][4]) == 'number' ? labels_processed[i][4] : 25;
co.beginPath();
co.fillStyle = 'black';
co.strokeStyle = 'black';
if (obj.type == 'bar') {
/**
* X axis at the top
*/
if (obj.Get('chart.xaxispos') == 'top') {
length *= -1;
}
if (prop['chart.variant'] == 'dot') {
co.moveTo(Math.round(x), obj.coords[i][1] - 5);
co.lineTo(Math.round(x), obj.coords[i][1] - 5 - length);
var text_x = Math.round(x);
var text_y = obj.coords[i][1] - 5 - length;
} else if (prop['chart.variant'] == 'arrow') {
co.moveTo(Math.round(x), obj.coords[i][1] - 5);
co.lineTo(Math.round(x), obj.coords[i][1] - 5 - length);
var text_x = Math.round(x);
var text_y = obj.coords[i][1] - 5 - length;
} else {
co.arc(Math.round(x), y, 2.5, 0, 6.28, 0);
co.moveTo(Math.round(x), y);
co.lineTo(Math.round(x), y - length);
var text_x = Math.round(x);
var text_y = y - length;
}
co.stroke();
co.fill();
} else if (obj.type == 'line') {
if (
typeof(labels_processed[i]) == 'object' &&
typeof(labels_processed[i][3]) == 'number' &&
labels_processed[i][3] == -1
) {
co.moveTo(Math.round(x), y + 5);
co.lineTo(Math.round(x), y + 5 + length);
co.stroke();
co.beginPath();
// This draws the arrow
co.moveTo(Math.round(x), y + 5);
co.lineTo(Math.round(x) - 3, y + 10);
co.lineTo(Math.round(x) + 3, y + 10);
co.closePath();
var text_x = x;
var text_y = y + 5 + length;
} else {
var text_x = x;
var text_y = y - 5 - length;
co.moveTo(Math.round(x), y - 5);
co.lineTo(Math.round(x), y - 5 - length);
co.stroke();
co.beginPath();
// This draws the arrow
co.moveTo(Math.round(x), y - 5);
co.lineTo(Math.round(x) - 3, y - 10);
co.lineTo(Math.round(x) + 3, y - 10);
co.closePath();
}
co.fill();
}
// Taken out on the 10th Nov 2010 - unnecessary
//var width = context.measureText(labels[i]).width;
co.beginPath();
// Fore ground color
co.fillStyle = (typeof(labels_processed[i]) == 'object' && typeof(labels_processed[i][1]) == 'string') ? labels_processed[i][1] : 'black';
RG.Text2(obj,{'font':prop['chart.text.font'],
'size':prop['chart.text.size'],
'x':text_x,
'y':text_y,
'text': (typeof(labels_processed[i]) == 'object' && typeof(labels_processed[i][0]) == 'string') ? labels_processed[i][0] : labels_processed[i],
'valign': 'bottom',
'halign':'center',
'bounding':true,
'bounding.fill': (typeof(labels_processed[i]) == 'object' && typeof(labels_processed[i][2]) == 'string') ? labels_processed[i][2] : 'white',
'tag':'labels ingraph'
});
co.fill();
}
}
}
}
}
/**
* This function "fills in" key missing properties that various implementations lack
*
* @param object e The event object
*/
RGraph.FixEventObject = function (e)
{
if (ISOLD) {
var e = event;
e.pageX = (event.clientX + document.body.scrollLeft);
e.pageY = (event.clientY + document.body.scrollTop);
e.target = event.srcElement;
if (!document.body.scrollTop && document.documentElement.scrollTop) {
e.pageX += parseInt(document.documentElement.scrollLeft);
e.pageY += parseInt(document.documentElement.scrollTop);
}
}
// Any browser that doesn't implement stopPropagation() (MSIE)
if (!e.stopPropagation) {
e.stopPropagation = function () {window.event.cancelBubble = true;}
}
return e;
}
/**
* Thisz function hides the crosshairs coordinates
*/
RGraph.HideCrosshairCoords = function ()
{
var RG = RGraph;
var div = RG.Registry.Get('chart.coordinates.coords.div');
if ( div
&& div.style.opacity == 1
&& div.__object__.Get('chart.crosshairs.coords.fadeout')
) {
var style = RG.Registry.Get('chart.coordinates.coords.div').style;
setTimeout(function() {style.opacity = 0.9;}, 25);
setTimeout(function() {style.opacity = 0.8;}, 50);
setTimeout(function() {style.opacity = 0.7;}, 75);
setTimeout(function() {style.opacity = 0.6;}, 100);
setTimeout(function() {style.opacity = 0.5;}, 125);
setTimeout(function() {style.opacity = 0.4;}, 150);
setTimeout(function() {style.opacity = 0.3;}, 175);
setTimeout(function() {style.opacity = 0.2;}, 200);
setTimeout(function() {style.opacity = 0.1;}, 225);
setTimeout(function() {style.opacity = 0;}, 250);
setTimeout(function() {style.display = 'none';}, 275);
}
}
/**
* Draws the3D axes/background
*/
RGraph.Draw3DAxes = function (obj)
{
var prop = obj.properties;
var co = obj.context;
var ca = obj.canvas;
var gutterLeft = prop['chart.gutter.left'];
var gutterRight = prop['chart.gutter.right'];
var gutterTop = prop['chart.gutter.top'];
var gutterBottom = prop['chart.gutter.bottom'];
co.strokeStyle = '#aaa';
co.fillStyle = '#ddd';
// Draw the vertical left side
co.beginPath();
co.moveTo(gutterLeft, gutterTop);
co.lineTo(gutterLeft + 10, gutterTop - 5);
co.lineTo(gutterLeft + 10, ca.height - gutterBottom - 5);
co.lineTo(gutterLeft, ca.height - gutterBottom);
co.closePath();
co.stroke();
co.fill();
// Draw the bottom floor
co.beginPath();
co.moveTo(gutterLeft, ca.height - gutterBottom);
co.lineTo(gutterLeft + 10, ca.height - gutterBottom - 5);
co.lineTo(ca.width - gutterRight + 10, ca.height - gutterBottom - 5);
co.lineTo(ca.width - gutterRight, ca.height - gutterBottom);
co.closePath();
co.stroke();
co.fill();
}
/**
* This function attempts to "fill in" missing functions from the canvas
* context object. Only two at the moment - measureText() nd fillText().
*
* @param object context The canvas 2D context
*/
RGraph.OldBrowserCompat = function (co)
{
if (!co) {
return;
}
if (!co.measureText) {
// This emulates the measureText() function
co.measureText = function (text)
{
var textObj = document.createElement('DIV');
textObj.innerHTML = text;
textObj.style.position = 'absolute';
textObj.style.top = '-100px';
textObj.style.left = 0;
document.body.appendChild(textObj);
var width = {width: textObj.offsetWidth};
textObj.style.display = 'none';
return width;
}
}
if (!co.fillText) {
// This emulates the fillText() method
co.fillText = function (text, targetX, targetY)
{
return false;
}
}
// If IE8, add addEventListener()
if (!co.canvas.addEventListener) {
window.addEventListener = function (ev, func, bubble)
{
return this.attachEvent('on' + ev, func);
}
co.canvas.addEventListener = function (ev, func, bubble)
{
return this.attachEvent('on' + ev, func);
}
}
}
/**
* Draws a rectangle with curvy corners
*
* @param co object The context
* @param x number The X coordinate (top left of the square)
* @param y number The Y coordinate (top left of the square)
* @param w number The width of the rectangle
* @param h number The height of the rectangle
* @param number The radius of the curved corners
* @param boolean Whether the top left corner is curvy
* @param boolean Whether the top right corner is curvy
* @param boolean Whether the bottom right corner is curvy
* @param boolean Whether the bottom left corner is curvy
*/
RGraph.strokedCurvyRect = function (co, x, y, w, h)
{
// The corner radius
var r = arguments[5] ? arguments[5] : 3;
// The corners
var corner_tl = (arguments[6] || arguments[6] == null) ? true : false;
var corner_tr = (arguments[7] || arguments[7] == null) ? true : false;
var corner_br = (arguments[8] || arguments[8] == null) ? true : false;
var corner_bl = (arguments[9] || arguments[9] == null) ? true : false;
co.beginPath();
// Top left side
co.moveTo(x + (corner_tl ? r : 0), y);
co.lineTo(x + w - (corner_tr ? r : 0), y);
// Top right corner
if (corner_tr) {
co.arc(x + w - r, y + r, r, PI + HALFPI, TWOPI, false);
}
// Top right side
co.lineTo(x + w, y + h - (corner_br ? r : 0) );
// Bottom right corner
if (corner_br) {
co.arc(x + w - r, y - r + h, r, TWOPI, HALFPI, false);
}
// Bottom right side
co.lineTo(x + (corner_bl ? r : 0), y + h);
// Bottom left corner
if (corner_bl) {
co.arc(x + r, y - r + h, r, HALFPI, PI, false);
}
// Bottom left side
co.lineTo(x, y + (corner_tl ? r : 0) );
// Top left corner
if (corner_tl) {
co.arc(x + r, y + r, r, PI, PI + HALFPI, false);
}
co.stroke();
}
/**
* Draws a filled rectangle with curvy corners
*
* @param context object The context
* @param x number The X coordinate (top left of the square)
* @param y number The Y coordinate (top left of the square)
* @param w number The width of the rectangle
* @param h number The height of the rectangle
* @param number The radius of the curved corners
* @param boolean Whether the top left corner is curvy
* @param boolean Whether the top right corner is curvy
* @param boolean Whether the bottom right corner is curvy
* @param boolean Whether the bottom left corner is curvy
*/
RGraph.filledCurvyRect = function (co, x, y, w, h)
{
// The corner radius
var r = arguments[5] ? arguments[5] : 3;
// The corners
var corner_tl = (arguments[6] || arguments[6] == null) ? true : false;
var corner_tr = (arguments[7] || arguments[7] == null) ? true : false;
var corner_br = (arguments[8] || arguments[8] == null) ? true : false;
var corner_bl = (arguments[9] || arguments[9] == null) ? true : false;
co.beginPath();
// First draw the corners
// Top left corner
if (corner_tl) {
co.moveTo(x + r, y + r);
co.arc(x + r, y + r, r, PI, PI + HALFPI, false);
} else {
co.fillRect(x, y, r, r);
}
// Top right corner
if (corner_tr) {
co.moveTo(x + w - r, y + r);
co.arc(x + w - r, y + r, r, PI + HALFPI, 0, false);
} else {
co.moveTo(x + w - r, y);
co.fillRect(x + w - r, y, r, r);
}
// Bottom right corner
if (corner_br) {
co.moveTo(x + w - r, y + h - r);
co.arc(x + w - r, y - r + h, r, 0, HALFPI, false);
} else {
co.moveTo(x + w - r, y + h - r);
co.fillRect(x + w - r, y + h - r, r, r);
}
// Bottom left corner
if (corner_bl) {
co.moveTo(x + r, y + h - r);
co.arc(x + r, y - r + h, r, HALFPI, PI, false);
} else {
co.moveTo(x, y + h - r);
co.fillRect(x, y + h - r, r, r);
}
// Now fill it in
co.fillRect(x + r, y, w - r - r, h);
co.fillRect(x, y + r, r + 1, h - r - r);
co.fillRect(x + w - r - 1, y + r, r + 1, h - r - r);
co.fill();
}
/**
* Hides the zoomed canvas
*/
RGraph.HideZoomedCanvas = function ()
{
var interval = 15;
var frames = 10;
if (typeof(__zoomedimage__) == 'object') {
var obj = __zoomedimage__.obj;
var prop = obj.properties;
} else {
return;
}
if (prop['chart.zoom.fade.out']) {
for (var i=frames,j=1; i>=0; --i, ++j) {
if (typeof(__zoomedimage__) == 'object') {
setTimeout("__zoomedimage__.style.opacity = " + String(i / 10), j * interval);
}
}
if (typeof(__zoomedbackground__) == 'object') {
setTimeout("__zoomedbackground__.style.opacity = " + String(i / frames), j * interval);
}
}
if (typeof(__zoomedimage__) == 'object') {
setTimeout("__zoomedimage__.style.display = 'none'", prop['chart.zoom.fade.out'] ? (frames * interval) + 10 : 0);
}
if (typeof(__zoomedbackground__) == 'object') {
setTimeout("__zoomedbackground__.style.display = 'none'", prop['chart.zoom.fade.out'] ? (frames * interval) + 10 : 0);
}
}
/**
* Adds an event handler
*
* @param object obj The graph object
* @param string event The name of the event, eg ontooltip
* @param object func The callback function
*/
RGraph.AddCustomEventListener = function (obj, name, func)
{
var RG = RGraph;
if (typeof(RG.events[obj.uid]) == 'undefined') {
RG.events[obj.uid] = [];
}
RG.events[obj.uid].push([obj, name, func]);
return RG.events[obj.uid].length - 1;
}
/**
* Used to fire one of the RGraph custom events
*
* @param object obj The graph object that fires the event
* @param string event The name of the event to fire
*/
RGraph.FireCustomEvent = function (obj, name)
{
var RG = RGraph;
if (obj && obj.isRGraph) {
// New style of adding custom events
if (obj[name]) {
(obj[name])(obj);
}
var uid = obj.uid;
if ( typeof(uid) == 'string'
&& typeof(RG.events) == 'object'
&& typeof(RG.events[uid]) == 'object'
&& RG.events[uid].length > 0) {
for(var j=0; j<RG.events[uid].length; ++j) {
if (RG.events[uid][j] && RG.events[uid][j][1] == name) {
RG.events[uid][j][2](obj);
}
}
}
}
}
/**
* If you prefer, you can use the SetConfig() method to set the configuration information
* for your chart. You may find that setting the configuration this way eases reuse.
*
* @param object obj The graph object
* @param object config The graph configuration information
*/
RGraph.SetConfig = function (obj, config)
{
for (i in config) {
if (typeof(i) == 'string') {
obj.Set(i, config[i]);
}
}
return obj;
}
/**
* Clears all the custom event listeners that have been registered
*
* @param string Limits the clearing to this object ID
*/
RGraph.RemoveAllCustomEventListeners = function ()
{
var RG = RGraph;
var id = arguments[0];
if (id && RG.events[id]) {
RG.events[id] = [];
} else {
RG.events = [];
}
}
/**
* Clears a particular custom event listener
*
* @param object obj The graph object
* @param number i This is the index that is return by .AddCustomEventListener()
*/
RGraph.RemoveCustomEventListener = function (obj, i)
{
var RG = RGraph;
if ( typeof(RG.events) == 'object'
&& typeof(RG.events[obj.id]) == 'object'
&& typeof(RG.events[obj.id][i]) == 'object') {
RG.events[obj.id][i] = null;
}
}
/**
* This draws the background
*
* @param object obj The graph object
*/
RGraph.DrawBackgroundImage = function (obj)
{
var prop = obj.properties;
var ca = obj.canvas;
var co = obj.context;
var RG = RGraph;
if (typeof(prop['chart.background.image']) == 'string') {
if (typeof(ca.__rgraph_background_image__) == 'undefined') {
var img = new Image();
img.__object__ = obj;
img.__canvas__ = ca;
img.__context__ = co;
img.src = obj.Get('chart.background.image');
ca.__rgraph_background_image__ = img;
} else {
img = ca.__rgraph_background_image__;
}
// When the image has loaded - redraw the canvas
img.onload = function ()
{
obj.__rgraph_background_image_loaded__ = true;
RG.Clear(ca);
RG.RedrawCanvas(ca);
}
var gutterLeft = obj.gutterLeft;
var gutterRight = obj.gutterRight;
var gutterTop = obj.gutterTop;
var gutterBottom = obj.gutterBottom;
var stretch = prop['chart.background.image.stretch'];
var align = prop['chart.background.image.align'];
// Handle chart.background.image.align
if (typeof(align) == 'string') {
if (align.indexOf('right') != -1) {
var x = ca.width - img.width - gutterRight;
} else {
var x = gutterLeft;
}
if (align.indexOf('bottom') != -1) {
var y = ca.height - img.height - gutterBottom;
} else {
var y = gutterTop;
}
} else {
var x = gutterLeft || 25;
var y = gutterTop || 25;
}
// X/Y coords take precedence over the align
var x = typeof(prop['chart.background.image.x']) == 'number' ? prop['chart.background.image.x'] : x;
var y = typeof(prop['chart.background.image.y']) == 'number' ? prop['chart.background.image.y'] : y;
var w = stretch ? ca.width - gutterLeft - gutterRight : img.width;
var h = stretch ? ca.height - gutterTop - gutterBottom : img.height;
/**
* You can now specify the width and height of the image
*/
if (typeof(prop['chart.background.image.w']) == 'number') w = prop['chart.background.image.w'];
if (typeof(prop['chart.background.image.h']) == 'number') h = prop['chart.background.image.h'];
co.drawImage(img,x,y,w, h);
}
}
/**
* This function determines wshether an object has tooltips or not
*
* @param object obj The chart object
*/
RGraph.hasTooltips = function (obj)
{
var prop = obj.properties;
if (typeof(prop['chart.tooltips']) == 'object' && prop['chart.tooltips']) {
for (var i=0,len=prop['chart.tooltips'].length; i<len; ++i) {
if (!RGraph.is_null(obj.Get('chart.tooltips')[i])) {
return true;
}
}
} else if (typeof(prop['chart.tooltips']) == 'function') {
return true;
}
return false;
}
/**
* This function creates a (G)UID which can be used to identify objects.
*
* @return string (g)uid The (G)UID
*/
RGraph.CreateUID = function ()
{
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c)
{
var r = Math.random()*16|0, v = c == 'x' ? r : (r&0x3|0x8);
return v.toString(16);
});
}
/**
* This is the new object registry, used to facilitate multiple objects per canvas.
*
* @param object obj The object to register
*/
RGraph.ObjectRegistry.Add = function (obj)
{
var uid = obj.uid;
var id = obj.canvas.id;
var RG = RGraph;
/**
* Index the objects by UID
*/
RG.ObjectRegistry.objects.byUID.push([uid, obj]);
/**
* Index the objects by the canvas that they're drawn on
*/
RG.ObjectRegistry.objects.byCanvasID.push([id, obj]);
}
/**
* Remove an object from the object registry
*
* @param object obj The object to remove.
*/
RGraph.ObjectRegistry.Remove = function (obj)
{
var id = obj.id;
var uid = obj.uid;
var RG = RGraph;
for (var i=0; i<RG.ObjectRegistry.objects.byUID.length; ++i) {
if (RG.ObjectRegistry.objects.byUID[i] && RG.ObjectRegistry.objects.byUID[i][1].uid == uid) {
RG.ObjectRegistry.objects.byUID[i] = null;
}
}
for (var i=0; i<RG.ObjectRegistry.objects.byCanvasID.length; ++i) {
if ( RG.ObjectRegistry.objects.byCanvasID[i]
&& RG.ObjectRegistry.objects.byCanvasID[i][1]
&& RG.ObjectRegistry.objects.byCanvasID[i][1].uid == uid) {
RG.ObjectRegistry.objects.byCanvasID[i] = null;
}
}
}
/**
* Removes all objects from the ObjectRegistry. If either the ID of a canvas is supplied,
* or the canvas itself, then only objects pertaining to that canvas are cleared.
*
* @param mixed Either a canvas object (as returned by document.getElementById()
* or the ID of a canvas (ie a string)
*/
RGraph.ObjectRegistry.Clear = function ()
{
var RG = RGraph;
// If an ID is supplied restrict the learing to that
if (arguments[0]) {
var id = (typeof(arguments[0]) == 'object' ? arguments[0].id : arguments[0]);
var objects = RG.ObjectRegistry.getObjectsByCanvasID(id);
for (var i=0; i<objects.length; ++i) {
RG.ObjectRegistry.Remove(objects[i]);
}
} else {
RG.ObjectRegistry.objects = {};
RG.ObjectRegistry.objects.byUID = [];
RG.ObjectRegistry.objects.byCanvasID = [];
}
}
/**
* Lists all objects in the ObjectRegistry
*
* @param boolean ret Whether to return the list or alert() it
*/
RGraph.ObjectRegistry.List = function ()
{
var list = [];
var RG = RGraph;
for (var i=0,len=RG.ObjectRegistry.objects.byUID.length; i<len; ++i) {
if (RG.ObjectRegistry.objects.byUID[i]) {
list.push(RG.ObjectRegistry.objects.byUID[i][1].type);
}
}
if (arguments[0]) {
return list;
} else {
p(list);
}
}
/**
* Clears the ObjectRegistry of objects that are of a certain given type
*
* @param type string The type to clear
*/
RGraph.ObjectRegistry.ClearByType = function (type)
{
var RG = RGraph;
var objects = RG.ObjectRegistry.objects.byUID;
for (var i=0; i<objects.length; ++i) {
if (objects[i]) {
var uid = objects[i][0];
var obj = objects[i][1];
if (obj && obj.type == type) {
RG.ObjectRegistry.Remove(obj);
}
}
}
}
/**
* This function provides an easy way to go through all of the objects that are held in the
* Registry
*
* @param func function This function is run for every object. Its passed the object as an argument
* @param string type Optionally, you can pass a type of object to look for
*/
RGraph.ObjectRegistry.Iterate = function (func)
{
var objects = RGraph.ObjectRegistry.objects.byUID;
for (var i=0; i<objects.length; ++i) {
if (typeof arguments[1] == 'string') {
var types = arguments[1].split(/,/);
for (var j=0; j<types.length; ++j) {
if (types[j] == objects[i][1].type) {
func(objects[i][1]);
}
}
} else {
func(objects[i][1]);
}
}
}
/**
* Retrieves all objects for a given canvas id
*
* @patarm id string The canvas ID to get objects for.
*/
RGraph.ObjectRegistry.getObjectsByCanvasID = function (id)
{
var store = RGraph.ObjectRegistry.objects.byCanvasID;
var ret = [];
// Loop through all of the objects and return the appropriate ones
for (var i=0; i<store.length; ++i) {
if (store[i] && store[i][0] == id ) {
ret.push(store[i][1]);
}
}
return ret;
}
/**
* Retrieves the relevant object based on the X/Y position.
*
* @param object e The event object
* @return object The applicable (if any) object
*/
RGraph.ObjectRegistry.getFirstObjectByXY =
RGraph.ObjectRegistry.getObjectByXY = function (e)
{
var canvas = e.target;
var ret = null;
var objects = RGraph.ObjectRegistry.getObjectsByCanvasID(canvas.id);
for (var i=(objects.length - 1); i>=0; --i) {
var obj = objects[i].getObjectByXY(e);
if (obj) {
return obj;
}
}
}
/**
* Retrieves the relevant objects based on the X/Y position.
* NOTE This function returns an array of objects
*
* @param object e The event object
* @return An array of pertinent objects. Note the there may be only one object
*/
RGraph.ObjectRegistry.getObjectsByXY = function (e)
{
var canvas = e.target;
var ret = [];
var objects = RGraph.ObjectRegistry.getObjectsByCanvasID(canvas.id);
// Retrieve objects "front to back"
for (var i=(objects.length - 1); i>=0; --i) {
var obj = objects[i].getObjectByXY(e);
if (obj) {
ret.push(obj);
}
}
return ret;
}
/**
* Retrieves the object with the corresponding UID
*
* @param string uid The UID to get the relevant object for
*/
RGraph.ObjectRegistry.getObjectByUID = function (uid)
{
var objects = RGraph.ObjectRegistry.objects.byUID;
for (var i=0; i<objects.length; ++i) {
if (objects[i] && objects[i][1].uid == uid) {
return objects[i][1];
}
}
}
/**
* Brings a chart to the front of the ObjectRegistry by
* removing it and then readding it at the end and then
* redrawing the canvas
*
* @param object obj The object to bring to the front
* @param boolean redraw Whether to redraw the canvas after the
* object has been moved
*/
RGraph.ObjectRegistry.bringToFront = function (obj)
{
var redraw = typeof arguments[1] == 'undefined' ? true : arguments[1];
RGraph.ObjectRegistry.Remove(obj);
RGraph.ObjectRegistry.Add(obj);
if (redraw) {
RGraph.RedrawCanvas(obj.canvas);
}
}
/**
* Retrieves the objects that are the given type
*
* @param mixed canvas The canvas to check. It can either be the canvas object itself or just the ID
* @param string type The type to look for
* @return array An array of one or more objects
*/
RGraph.ObjectRegistry.getObjectsByType = function (type)
{
var objects = RGraph.ObjectRegistry.objects.byUID;
var ret = [];
for (var i=0; i<objects.length; ++i) {
if (objects[i] && objects[i][1] && objects[i][1].type && objects[i][1].type && objects[i][1].type == type) {
ret.push(objects[i][1]);
}
}
return ret;
}
/**
* Retrieves the FIRST object that matches the given type
*
* @param string type The type of object to look for
* @return object The FIRST object that matches the given type
*/
RGraph.ObjectRegistry.getFirstObjectByType = function (type)
{
var objects = RGraph.ObjectRegistry.objects.byUID;
for (var i=0; i<objects.length; ++i) {
if (objects[i] && objects[i][1] && objects[i][1].type == type) {
return objects[i][1];
}
}
return null;
}
/**
* This takes centerx, centery, x and y coordinates and returns the
* appropriate angle relative to the canvas angle system. Remember
* that the canvas angle system starts at the EAST axis
*
* @param number cx The centerx coordinate
* @param number cy The centery coordinate
* @param number x The X coordinate (eg the mouseX if coming from a click)
* @param number y The Y coordinate (eg the mouseY if coming from a click)
* @return number The relevant angle (measured in in RADIANS)
*/
RGraph.getAngleByXY = function (cx, cy, x, y)
{
var angle = Math.atan((y - cy) / (x - cx));
angle = Math.abs(angle)
if (x >= cx && y >= cy) {
angle += TWOPI;
} else if (x >= cx && y < cy) {
angle = (HALFPI - angle) + (PI + HALFPI);
} else if (x < cx && y < cy) {
angle += PI;
} else {
angle = PI - angle;
}
/**
* Upper and lower limit checking
*/
if (angle > TWOPI) {
angle -= TWOPI;
}
return angle;
}
/**
* This function returns the distance between two points. In effect the
* radius of an imaginary circle that is centered on x1 and y1. The name
* of this function is derived from the word "Hypoteneuse", which in
* trigonmetry is the longest side of a triangle
*
* @param number x1 The original X coordinate
* @param number y1 The original Y coordinate
* @param number x2 The target X coordinate
* @param number y2 The target Y coordinate
*/
RGraph.getHypLength = function (x1, y1, x2, y2)
{
var ret = Math.sqrt(((x2 - x1) * (x2 - x1)) + ((y2 - y1) * (y2 - y1)));
return ret;
}
/**
* This function gets the end point (X/Y coordinates) of a given radius.
* You pass it the center X/Y and the radius and this function will return
* the endpoint X/Y coordinates.
*
* @param number cx The center X coord
* @param number cy The center Y coord
* @param number r The lrngth of the radius
*/
RGraph.getRadiusEndPoint = function (cx, cy, angle, radius)
{
var x = cx + (Math.cos(angle) * radius);
var y = cy + (Math.sin(angle) * radius);
return [x, y];
}
/**
* This installs all of the event listeners
*
* @param object obj The chart object
*/
RGraph.InstallEventListeners = function (obj)
{
var RG = RGraph;
var prop = obj.properties;
/**
* Don't attempt to install event listeners for older versions of MSIE
*/
if (ISOLD) {
return;
}
/**
* If this function exists, then the dynamic file has been included.
*/
if (RG.InstallCanvasClickListener) {
RG.InstallWindowMousedownListener(obj);
RG.InstallWindowMouseupListener(obj);
RG.InstallCanvasMousemoveListener(obj);
RG.InstallCanvasMouseupListener(obj);
RG.InstallCanvasMousedownListener(obj);
RG.InstallCanvasClickListener(obj);
} else if ( RG.hasTooltips(obj)
|| prop['chart.adjustable']
|| prop['chart.annotatable']
|| prop['chart.contextmenu']
|| prop['chart.resizable']
|| prop['chart.key.interactive']
|| prop['chart.events.click']
|| prop['chart.events.mousemove']
|| typeof obj.onclick == 'function'
|| typeof obj.onmousemove == 'function'
) {
alert('[RGRAPH] You appear to have used dynamic features but not included the file: RGraph.common.dynamic.js');
}
}
/**
* Loosly mimicks the PHP function print_r();
*/
RGraph.pr = function (obj)
{
var indent = (arguments[2] ? arguments[2] : ' ');
var str = '';
var counter = typeof arguments[3] == 'number' ? arguments[3] : 0;
if (counter >= 5) {
return '';
}
switch (typeof obj) {
case 'string': str += obj + ' (' + (typeof obj) + ', ' + obj.length + ')'; break;
case 'number': str += obj + ' (' + (typeof obj) + ')'; break;
case 'boolean': str += obj + ' (' + (typeof obj) + ')'; break;
case 'function': str += 'function () {}'; break;
case 'undefined': str += 'undefined'; break;
case 'null': str += 'null'; break;
case 'object':
// In case of null
if (RGraph.is_null(obj)) {
str += indent + 'null\n';
} else {
str += indent + 'Object {' + '\n'
for (j in obj) {
str += indent + ' ' + j + ' => ' + RGraph.pr(obj[j], true, indent + ' ', counter + 1) + '\n';
}
str += indent + '}';
}
break;
default:
str += 'Unknown type: ' + typeof obj + '';
break;
}
/**
* Finished, now either return if we're in a recursed call, or alert()
* if we're not.
*/
if (!arguments[1]) {
alert(str);
}
return str;
}
/**
* Produces a dashed line
*
* @param object co The 2D context
* @param number x1 The start X coordinate
* @param number y1 The start Y coordinate
* @param number x2 The end X coordinate
* @param number y2 The end Y coordinate
*/
RGraph.DashedLine = function(co, x1, y1, x2, y2)
{
/**
* This is the size of the dashes
*/
var size = 5;
/**
* The optional fifth argument can be the size of the dashes
*/
if (typeof(arguments[5]) == 'number') {
size = arguments[5];
}
var dx = x2 - x1;
var dy = y2 - y1;
var num = Math.floor(Math.sqrt((dx * dx) + (dy * dy)) / size);
var xLen = dx / num;
var yLen = dy / num;
var count = 0;
do {
(count % 2 == 0 && count > 0) ? co.lineTo(x1, y1) : co.moveTo(x1, y1);
x1 += xLen;
y1 += yLen;
} while(count++ <= num);
}
/**
* Makes an AJAX call. It calls the given callback (a function) when ready
*
* @param string url The URL to retrieve
* @param function callback A function that is called when the response is ready, there's an example below
* called "myCallback".
*/
RGraph.AJAX = function (url, callback)
{
// Mozilla, Safari, ...
if (window.XMLHttpRequest) {
var httpRequest = new XMLHttpRequest();
// MSIE
} else if (window.ActiveXObject) {
var httpRequest = new ActiveXObject("Microsoft.XMLHTTP");
}
httpRequest.onreadystatechange = function ()
{
if (this.readyState == 4 && this.status == 200) {
this.__user_callback__ = callback;
this.__user_callback__(this.responseText);
}
}
httpRequest.open('GET', url, true);
httpRequest.send();
}
/**
* Makes an AJAX POST request. It calls the given callback (a function) when ready
*
* @param string url The URL to retrieve
* @param object data The POST data
* @param function callback A function that is called when the response is ready, there's an example below
* called "myCallback".
*/
RGraph.AJAX.POST = function (url, data, callback)
{
// Used when building the POST string
var crumbs = [];
// Mozilla, Safari, ...
if (window.XMLHttpRequest) {
var httpRequest = new XMLHttpRequest();
// MSIE
} else if (window.ActiveXObject) {
var httpRequest = new ActiveXObject("Microsoft.XMLHTTP");
}
httpRequest.onreadystatechange = function ()
{
if (this.readyState == 4 && this.status == 200) {
this.__user_callback__ = callback;
this.__user_callback__(this.responseText);
}
}
httpRequest.open('POST', url, true);
httpRequest.setRequestHeader("Content-type","application/x-www-form-urlencoded");
for (i in data) {
if (typeof i == 'string') {
crumbs.push(i + '=' + encodeURIComponent(data[i]));
}
}
httpRequest.send(crumbs.join('&'));
}
/**
* Uses the above function but calls the call back passing a number as its argument
*
* @param url string The URL to fetch
* @param callback function Your callback function (which is passed the number as an argument)
*/
RGraph.AJAX.getNumber = function (url, callback)
{
RGraph.AJAX(url, function ()
{
var num = parseFloat(this.responseText);
callback(num);
});
}
/**
* Uses the above function but calls the call back passing a string as its argument
*
* @param url string The URL to fetch
* @param callback function Your callback function (which is passed the string as an argument)
*/
RGraph.AJAX.getString = function (url, callback)
{
RGraph.AJAX(url, function ()
{
var str = String(this.responseText);
callback(str);
});
}
/**
* Uses the above function but calls the call back passing JSON (ie a JavaScript object ) as its argument
*
* @param url string The URL to fetch
* @param callback function Your callback function (which is passed the JSON object as an argument)
*/
RGraph.AJAX.getJSON = function (url, callback)
{
RGraph.AJAX(url, function ()
{
var json = eval('(' + this.responseText + ')');
callback(json);
});
}
/**
* Uses the above RGraph.AJAX function but calls the call back passing an array as its argument.
* Useful if you're retrieving CSV data
*
* @param url string The URL to fetch
* @param callback function Your callback function (which is passed the CSV/array as an argument)
*/
RGraph.AJAX.getCSV = function (url, callback)
{
var seperator = arguments[2] ? arguments[2] : ',';
RGraph.AJAX(url, function ()
{
var regexp = new RegExp(seperator);
var arr = this.responseText.split(regexp);
// Convert the strings to numbers
for (var i=0,len=arr.length;i<len;++i) {
arr[i] = parseFloat(arr[i]);
}
callback(arr);
});
}
/**
* Rotates the canvas
*
* @param object canvas The canvas to rotate
* @param int x The X coordinate about which to rotate the canvas
* @param int y The Y coordinate about which to rotate the canvas
* @param int angle The angle(in RADIANS) to rotate the canvas by
*/
RGraph.RotateCanvas = function (ca, x, y, angle)
{
var co = ca.getContext('2d');
co.translate(x, y);
co.rotate(angle);
co.translate(0 - x, 0 - y);
}
/**
* Measures text by creating a DIV in the document and adding the relevant text to it.
* Then checking the .offsetWidth and .offsetHeight.
*
* @param string text The text to measure
* @param bool bold Whether the text is bold or not
* @param string font The font to use
* @param size number The size of the text (in pts)
* @return array A two element array of the width and height of the text
*/
RGraph.MeasureText = function (text, bold, font, size)
{
// Add the sizes to the cache as adding DOM elements is costly and causes slow downs
if (typeof(__rgraph_measuretext_cache__) == 'undefined') {
__rgraph_measuretext_cache__ = [];
}
var str = text + ':' + bold + ':' + font + ':' + size;
if (typeof(__rgraph_measuretext_cache__) == 'object' && __rgraph_measuretext_cache__[str]) {
return __rgraph_measuretext_cache__[str];
}
if (!__rgraph_measuretext_cache__['text-div']) {
var div = document.createElement('DIV');
div.style.position = 'absolute';
div.style.top = '-100px';
div.style.left = '-100px';
document.body.appendChild(div);
// Now store the newly created DIV
__rgraph_measuretext_cache__['text-div'] = div;
} else if (__rgraph_measuretext_cache__['text-div']) {
var div = __rgraph_measuretext_cache__['text-div'];
}
div.innerHTML = text.replace(/\r\n/g, '<br />');
div.style.fontFamily = font;
div.style.fontWeight = bold ? 'bold' : 'normal';
div.style.fontSize = (size || 12) + 'pt';
var size = [div.offsetWidth, div.offsetHeight];
//document.body.removeChild(div);
__rgraph_measuretext_cache__[str] = size;
return size;
}
/* New text function. Accepts two arguments:
* o obj - The chart object
* o opt - An object/hash/map of properties. This can consist of:
* x The X coordinate (REQUIRED)
* y The Y coordinate (REQUIRED)
* text The text to show (REQUIRED)
* font The font to use
* size The size of the text (in pt)
* bold Whether the text shouldd be bold or not
* marker Whether to show a marker that indicates the X/Y coordinates
* valign The vertical alignment
* halign The horizontal alignment
* bounding Whether to draw a bounding box for the text
* boundingStroke The strokeStyle of the bounding box
* boundingFill The fillStyle of the bounding box
*/
RGraph.Text2 = function (obj, opt)
{
/**
* An RGraph object can be given, or a string or the 2D rendering context
* The coords are placed on the obj.coordsText variable ONLY if it's an RGraph object. The function
* still returns the cooords though in all cases.
*/
if (obj && obj.isRGraph) {
var co = obj.context;
var ca = obj.canvas;
} else if (typeof obj == 'string') {
var ca = document.getElementById(obj);
var co = ca.getContext('2d');
} else if (typeof obj.getContext == 'function') {
var ca = obj;
var co = ca.getContext('2d');
} else if (obj.toString().indexOf('CanvasRenderingContext2D') != -1) {
var co = obj;
var ca = obj.context;
}
var x = opt.x;
var y = opt.y;
var originalX = x;
var originalY = y;
var text = opt.text;
var text_multiline = text.split(/\r?\n/g);
var numlines = text_multiline.length;
var font = opt.font ? opt.font : 'Arial';
var size = opt.size ? opt.size : 10;
var size_pixels = size * 1.5;
var bold = opt.bold;
var halign = opt.halign ? opt.halign : 'left';
var valign = opt.valign ? opt.valign : 'bottom';
var tag = typeof opt.tag == 'string' && opt.tag.length > 0 ? opt.tag : '';
var marker = opt.marker;
var angle = opt.angle || 0;
/**
* Changed the name of boundingFill/boundingStroke - this allows you to still use those names
*/
if (typeof opt.boundingFill == 'string') opt['bounding.fill'] = opt.boundingFill;
if (typeof opt.boundingStroke == 'string') opt['bounding.stroke'] = opt.boundingStroke;
var bounding = opt.bounding;
var bounding_stroke = opt['bounding.stroke'] ? opt['bounding.stroke'] : 'black';
var bounding_fill = opt['bounding.fill'] ? opt['bounding.fill'] : 'rgba(255,255,255,0.7)';
var bounding_shadow = opt['bounding.shadow'];
var bounding_shadow_color = opt['bounding.shadow.color'] || '#ccc';
var bounding_shadow_blur = opt['bounding.shadow.blur'] || 3;
var bounding_shadow_offsetx = opt['bounding.shadow.offsetx'] || 3;
var bounding_shadow_offsety = opt['bounding.shadow.offsety'] || 3;
var bounding_linewidth = opt['bounding.linewidth'] || 1;
/**
* Initialize the return value to an empty object
*/
var ret = {};
/**
* The text arg must be a string or a number
*/
if (typeof text == 'number') {
text = String(text);
}
if (typeof text != 'string') {
alert('[RGRAPH TEXT] The text given must a string or a number');
return;
}
/**
* This facilitates vertical text
*/
if (angle != 0) {
co.save();
co.translate(x, y);
co.rotate((Math.PI / 180) * angle)
x = 0;
y = 0;
}
/**
* Set the font
*/
co.font = (opt.bold ? 'bold ' : '') + size + 'pt ' + font;
/**
* Measure the width/height. This must be done AFTER the font has been set
*/
var width=0;
for (var i=0; i<numlines; ++i) {
width = Math.max(width, co.measureText(text_multiline[i]).width);
}
var height = size_pixels * numlines;
/**
* Accommodate old MSIE 7/8
*/
//if (document.all && ISOLD) {
//y += 2;
//}
/**
* If marker is specified draw a marker at the X/Y coordinates
*/
if (opt.marker) {
var marker_size = 10;
var strokestyle = co.strokeStyle;
co.beginPath();
co.strokeStyle = 'red';
co.moveTo(x, y - marker_size);
co.lineTo(x, y + marker_size);
co.moveTo(x - marker_size, y);
co.lineTo(x + marker_size, y);
co.stroke();
co.strokeStyle = strokestyle;
}
/**
* Set the horizontal alignment
*/
if (halign == 'center') {
co.textAlign = 'center';
var boundingX = x - 2 - (width / 2);
} else if (halign == 'right') {
co.textAlign = 'right';
var boundingX = x - 2 - width;
} else {
co.textAlign = 'left';
var boundingX = x - 2;
}
/**
* Set the vertical alignment
*/
if (valign == 'center') {
co.textBaseline = 'middle';
// Move the text slightly
y -= 1;
y -= ((numlines - 1) / 2) * size_pixels;
var boundingY = y - (size_pixels / 2) - 2;
} else if (valign == 'top') {
co.textBaseline = 'top';
var boundingY = y - 2;
} else {
co.textBaseline = 'bottom';
// Move the Y coord if multiline text
if (numlines > 1) {
y -= ((numlines - 1) * size_pixels);
}
var boundingY = y - size_pixels - 2;
}
var boundingW = width + 4;
var boundingH = height + 4;
/**
* Draw a bounding box if required
*/
if (bounding) {
var pre_bounding_linewidth = co.lineWidth;
var pre_bounding_strokestyle = co.strokeStyle;
var pre_bounding_fillstyle = co.fillStyle;
var pre_bounding_shadowcolor = co.shadowColor;
var pre_bounding_shadowblur = co.shadowBlur;
var pre_bounding_shadowoffsetx = co.shadowOffsetX;
var pre_bounding_shadowoffsety = co.shadowOffsetY;
co.lineWidth = bounding_linewidth;
co.strokeStyle = bounding_stroke;
co.fillStyle = bounding_fill;
if (bounding_shadow) {
co.shadowColor = bounding_shadow_color;
co.shadowBlur = bounding_shadow_blur;
co.shadowOffsetX = bounding_shadow_offsetx;
co.shadowOffsetY = bounding_shadow_offsety;
}
//obj.context.strokeRect(boundingX, boundingY, width + 6, (size_pixels * numlines) + 4);
//obj.context.fillRect(boundingX, boundingY, width + 6, (size_pixels * numlines) + 4);
co.strokeRect(boundingX, boundingY, boundingW, boundingH);
co.fillRect(boundingX, boundingY, boundingW, boundingH);
// Reset the linewidth,colors and shadow to it's original setting
co.lineWidth = pre_bounding_linewidth;
co.strokeStyle = pre_bounding_strokestyle;
co.fillStyle = pre_bounding_fillstyle;
co.shadowColor = pre_bounding_shadowcolor
co.shadowBlur = pre_bounding_shadowblur
co.shadowOffsetX = pre_bounding_shadowoffsetx
co.shadowOffsetY = pre_bounding_shadowoffsety
}
/**
* Draw the text
*/
if (numlines > 1) {
for (var i=0; i<numlines; ++i) {
co.fillText(text_multiline[i], x, y + (size_pixels * i));
}
} else {
co.fillText(text, x, y);
}
/**
* If the text is at 90 degrees restore() the canvas - getting rid of the rotation
* and the translate that we did
*/
if (angle != 0) {
if (angle == 90) {
if (halign == 'left') {
if (valign == 'bottom') {boundingX = originalX - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height / 2) - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - height - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
} else if (halign == 'center') {
if (valign == 'bottom') {boundingX = originalX - 2; boundingY = originalY - (width / 2) - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height / 2) - 2; boundingY = originalY - (width / 2) - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - height - 2; boundingY = originalY - (width / 2) - 2; boundingW = height + 4; boundingH = width + 4;}
} else if (halign == 'right') {
if (valign == 'bottom') {boundingX = originalX - 2; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height / 2) - 2; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - height - 2; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
}
} else if (angle == 180) {
if (halign == 'left') {
if (valign == 'bottom') {boundingX = originalX - width - 2; boundingY = originalY - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'center') {boundingX = originalX - width - 2; boundingY = originalY - (height / 2) - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'top') {boundingX = originalX - width - 2; boundingY = originalY - height - 2; boundingW = width + 4; boundingH = height + 4;}
} else if (halign == 'center') {
if (valign == 'bottom') {boundingX = originalX - (width / 2) - 2; boundingY = originalY - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'center') {boundingX = originalX - (width / 2) - 2; boundingY = originalY - (height / 2) - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'top') {boundingX = originalX - (width / 2) - 2; boundingY = originalY - height - 2; boundingW = width + 4; boundingH = height + 4;}
} else if (halign == 'right') {
if (valign == 'bottom') {boundingX = originalX - 2; boundingY = originalY - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'center') {boundingX = originalX - 2; boundingY = originalY - (height / 2) - 2; boundingW = width + 4; boundingH = height + 4;}
if (valign == 'top') {boundingX = originalX - 2; boundingY = originalY - height - 2; boundingW = width + 4; boundingH = height + 4;}
}
} else if (angle == 270) {
if (halign == 'left') {
if (valign == 'bottom') {boundingX = originalX - height - 2; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height / 2) - 4; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - 2; boundingY = originalY - width - 2; boundingW = height + 4; boundingH = width + 4;}
} else if (halign == 'center') {
if (valign == 'bottom') {boundingX = originalX - height - 2; boundingY = originalY - (width/2) - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height/2) - 4; boundingY = originalY - (width/2) - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - 2; boundingY = originalY - (width/2) - 2; boundingW = height + 4; boundingH = width + 4;}
} else if (halign == 'right') {
if (valign == 'bottom') {boundingX = originalX - height - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'center') {boundingX = originalX - (height/2) - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
if (valign == 'top') {boundingX = originalX - 2; boundingY = originalY - 2; boundingW = height + 4; boundingH = width + 4;}
}
}
co.restore();
}
/**
* Reset the text alignment so that text rendered
*/
co.textBaseline = 'alphabetic';
co.textAlign = 'left';
/**
* Fill the ret variable with details of the text
*/
ret.x = boundingX;
ret.y = boundingY;
ret.width = boundingW;
ret.height = boundingH
ret.object = obj;
ret.text = text;
ret.tag = tag;
/**
* Save and then return the details of the text (but oly
* if it's an RGraph object that was given)
*/
if (obj && obj.isRGraph && obj.coordsText) {
obj.coordsText.push(ret);
}
return ret;
}
/**
* Takes a sequential index abd returns the group/index variation of it. Eg if you have a
* sequential index from a grouped bar chart this function can be used to convert that into
* an appropriate group/index combination
*
* @param nindex number The sequential index
* @param data array The original data (which is grouped)
* @return The group/index information
*/
RGraph.sequentialIndexToGrouped = function (index, data)
{
var group = 0;
var grouped_index = 0;
while (--index >= 0) {
if (RGraph.is_null(data[group])) {
group++;
grouped_index = 0;
continue;
}
// Allow for numbers as well as arrays in the dataset
if (typeof data[group] == 'number') {
group++
grouped_index = 0;
continue;
}
grouped_index++;
if (grouped_index >= data[group].length) {
group++;
grouped_index = 0;
}
}
return [group, grouped_index];
}
/**
* Similar to the jQuery each() function - this lets you iterate easily over an array. The 'this' variable is set]
* to the array in the callback function.
*
* @param array arr The array
* @param function func The function to call
* @param object Optionally you can specify the object that the "this" variable is set to
*/
RGraph.each = function (arr, func)
{
for(var i=0, len=arr.length; i<len; i+=1) {
if (typeof arguments[2] !== 'undefined') {
var ret = func.call(arguments[2], i, arr[i]);
} else {
var ret = func.call(arr, i, arr[i]);
}
if (ret === false) {
return;
}
}
}
/**
* Checks whether strings or numbers are empty or not. It also
* handles null or variables set to undefined. If a variable really
* is undefined - ie it hasn't been declared at all - you need to use
* "typeof variable" and check the return value - which will be undefined.
*
* @param mixed value The variable to check
*/
function empty (value)
{
if (!value || value.length <= 0) {
return true;
}
return false;
}
/**
* This function highlights a rectangle
*
* @param object obj The chart object
* @param number shape The coordinates of the rect to highlight
*/
RGraph.Highlight.Rect = function (obj, shape)
{
var ca = obj.canvas;
var co = obj.context;
var prop = obj.properties;
if (prop['chart.tooltips.highlight']) {
// Safari seems to need this
co.lineWidth = 1;
/**
* Draw a rectangle on the canvas to highlight the appropriate area
*/
co.beginPath();
co.strokeStyle = prop['chart.highlight.stroke'];
co.fillStyle = prop['chart.highlight.fill'];
co.strokeRect(shape['x'],shape['y'],shape['width'],shape['height']);
co.fillRect(shape['x'],shape['y'],shape['width'],shape['height']);
co.stroke;
co.fill();
}
}
/**
* This function highlights a point
*
* @param object obj The chart object
* @param number shape The coordinates of the rect to highlight
*/
RGraph.Highlight.Point = function (obj, shape)
{
var prop = obj.properties;
var ca = obj.canvas;
var co = obj.context;
if (prop['chart.tooltips.highlight']) {
/**
* Draw a rectangle on the canvas to highlight the appropriate area
*/
co.beginPath();
co.strokeStyle = prop['chart.highlight.stroke'];
co.fillStyle = prop['chart.highlight.fill'];
var radius = prop['chart.highlight.point.radius'] || 2;
co.arc(shape['x'],shape['y'],radius, 0, TWOPI, 0);
co.stroke();
co.fill();
}
}
/**
* Creates an HTML tag
*
* @param string type
* @param obj parent
* @param obj
* @param obj
*/
RGraph.HTML.create = function (type, parent)
{
var obj = document.createElement(type);
// Add the attributes
if (arguments[2]) {
this.attr(obj, arguments[2]);
}
// Add the styles
if (arguments[3]) {
this.css(obj, arguments[3]);
}
/**
* Add the tag to the object that has been provided (usually the document)
*/
parent.appendChild(obj);
return obj;
}
/**
* Sets attributes on a HTML object
*
* @param object obj
* @param object attr
*/
RGraph.HTML.attr = function (obj, attr)
{
for (i in attr) {
if (typeof i == 'string') {
obj[i] = attr[i];
}
}
}
/**
* Sets CSS on a HTML object
*
* @param object obj
* @param object css
*/
RGraph.HTML.css = function (obj, styles)
{
var style = obj.style;
for (i in styles) {
if (typeof i == 'string') {
style[i] = styles[i];
}
}
}
/**
* This is the same as Date.parse - though a little more flexible.
*
* @param string str The date string to parse
* @return Returns the same thing as Date.parse
*/
RGraph.parseDate = function (str)
{
str.trim();
// Allow for: now (just the word "now")
if (str === 'now') {
str = (new Date()).toString();
}
// Allow for: 2013-11-22 12:12:12 or 2013/11/22 12:12:12
if (str.match(/^(\d\d\d\d)(-|\/)(\d\d)(-|\/)(\d\d)( |T)(\d\d):(\d\d):(\d\d)$/)) {
str = RegExp.$1 + '-' + RegExp.$3 + '-' + RegExp.$5 + 'T' + RegExp.$7 + ':' + RegExp.$8 + ':' + RegExp.$9;
}
// Allow for: 2013-11-22
if (str.match(/^\d\d\d\d-\d\d-\d\d$/)) {
str = str.replace(/-/, '/');
}
// Allow for: 12:09:44 (time only using todays date)
if (str.match(/^\d\d:\d\d:\d\d$/)) {
var dateObj = new Date();
var date = dateObj.getDate();
var month = dateObj.getMonth() + 1;
var year = dateObj.getFullYear();
str = (year + '-' + month + '-' + date) + ' ' + str;
}
return Date.parse(str);
}
// Reset all of the color values to their original values
RGraph.resetColorsToOriginalValues = function (obj)
{
if (obj.original_colors) {
// Reset the colors to their original values
for (var j in obj.original_colors) {
if (typeof j === 'string') {
obj.properties[j] = RGraph.array_clone(obj.original_colors[j]);
}
}
}
// Reset the colorsParsed flag so that they're parsed for gradients again
obj.colorsParsed = false;
}
/**
* This function is a short-cut for the canvas path syntax (which can be rather verbose)
*
* @param mixed obj This can either be the 2D context or an RGraph object
* @param array path The path details
*/
RGraph.Path = function (obj, path)
{
/**
* Allow either the RGraph object or the context to be used as the first argument
*/
if (obj.isRGraph && typeof obj.type === 'string') {
var co = obj.context;
} else if (obj.toString().indexOf('CanvasRenderingContext2D') > 0) {
var co = obj;
}
/**
* If the Path information has been passed as a string - split it up
*/
if (typeof path == 'string') {
path = path.split(/ +/);
}
/**
* Go through the path information
*/
for (var i=0,len=path.length; i<len; i+=1) {
var op = path[i];
// 100,100,50,0,Math.PI * 1.5, false
switch (op) {
case 'b':co.beginPath();break;
case 'c':co.closePath();break;
case 'm':co.moveTo(parseFloat(path[i+1]),parseFloat(path[i+2]));i+=2;break;
case 'l':co.lineTo(parseFloat(path[i+1]),parseFloat(path[i+2]));i+=2;break;
case 's':co.strokeStyle=path[i+1];co.stroke();i+=1;break;
case 'f':co.fillStyle=path[i+1];co.fill();i+=1;break;
case 'qc':co.quadraticCurveTo(parseFloat(path[i+1]),parseFloat(path[i+2]),parseFloat(path[i+3]),parseFloat(path[i+4]));i+=4;break;
case 'bc':co.bezierCurveTo(parseFloat(path[i+1]),parseFloat(path[i+2]),parseFloat(path[i+3]),parseFloat(path[i+4]),parseFloat(path[i+5]),parseFloat(path[i+6]));i+=6;break;
case 'r':co.rect(parseFloat(path[i+1]),parseFloat(path[i+2]),parseFloat(path[i+3]),parseFloat(path[i+4]));i+=4;break;
case 'a':co.arc(parseFloat(path[i+1]),parseFloat(path[i+2]),parseFloat(path[i+3]),parseFloat(path[i+4]),parseFloat(path[i+5]),path[i+6]==='true'||path[i+6]===true?true:false);i+=6;break;
case 'at':co.arcTo(parseFloat(path[i+1]),parseFloat(path[i+2]),parseFloat(path[i+3]),parseFloat(path[i+4]),parseFloat(path[i+5]));i+=5;break;
case 'lw':co.lineWidth=parseFloat(path[i+1]);i+=1;break;
case 'lj':co.lineJoin=path[i+1];i+=1;break;
case 'lc':co.lineCap=path[i+1];i+=1;break;
case 'sc':co.shadowColor=path[i+1];i+=1;break;
case 'sb':co.shadowBlur=parseFloat(path[i+1]);i+=1;break;
case 'sx':co.shadowOffsetX=parseFloat(path[i+1]);i+=1;break;
case 'sy':co.shadowOffsetY=parseFloat(path[i+1]);i+=1;break;
case 'fu':(path[i+1])(obj);i+=1;break;
}
}
}
// Some other functions. Because they're rarely changed - they're hand minified
RGraph.LinearGradient=function(obj,x1,y1,x2,y2,color1,color2){var gradient=obj.context.createLinearGradient(x1,y1,x2,y2);var numColors=arguments.length-5;for (var i=5;i<arguments.length;++i){var color=arguments[i];var stop=(i-5)/(numColors-1);gradient.addColorStop(stop,color);}return gradient;}
RGraph.RadialGradient=function(obj,x1,y1,r1,x2,y2,r2,color1,color2){var gradient=obj.context.createRadialGradient(x1,y1,r1,x2,y2,r2);var numColors=arguments.length-7;for(var i=7;i<arguments.length; ++i){var color=arguments[i];var stop=(i-7)/(numColors-1);gradient.addColorStop(stop,color);}return gradient;}
RGraph.array_shift=function(arr){var ret=[];for(var i=1;i<arr.length;++i){ret.push(arr[i]);}return ret;}
RGraph.AddEventListener=function(id,e,func){var type=arguments[3]?arguments[3]:'unknown';RGraph.Registry.Get('chart.event.handlers').push([id,e,func,type]);}
RGraph.ClearEventListeners=function(id){if(id&&id=='window'){window.removeEventListener('mousedown',window.__rgraph_mousedown_event_listener_installed__,false);window.removeEventListener('mouseup',window.__rgraph_mouseup_event_listener_installed__,false);}else{var canvas = document.getElementById(id);canvas.removeEventListener('mouseup',canvas.__rgraph_mouseup_event_listener_installed__,false);canvas.removeEventListener('mousemove',canvas.__rgraph_mousemove_event_listener_installed__,false);canvas.removeEventListener('mousedown',canvas.__rgraph_mousedown_event_listener_installed__,false);canvas.removeEventListener('click',canvas.__rgraph_click_event_listener_installed__,false);}}
RGraph.HidePalette=function(){var div=RGraph.Registry.Get('palette');if(typeof(div)=='object'&&div){div.style.visibility='hidden';div.style.display='none';RGraph.Registry.Set('palette',null);}}
RGraph.random=function(min,max){var dp=arguments[2]?arguments[2]:0;var r=Math.random();return Number((((max - min) * r) + min).toFixed(dp));}
RGraph.random.array=function(num,min,max){var arr = [];for(var i=0;i<num;i++)arr.push(RGraph.random(min,max));return arr;}
RGraph.NoShadow=function(obj){obj.context.shadowColor='rgba(0,0,0,0)';obj.context.shadowBlur=0;obj.context.shadowOffsetX=0;obj.context.shadowOffsetY=0;}
RGraph.SetShadow=function(obj,color,offsetx,offsety,blur){obj.context.shadowColor=color;obj.context.shadowOffsetX=offsetx;obj.context.shadowOffsetY=offsety;obj.context.shadowBlur=blur;}
RGraph.array_reverse=function(arr){var newarr=[];for(var i=arr.length-1;i>=0;i--){newarr.push(arr[i]);}return newarr;}
RGraph.Registry.Set=function(name,value){RGraph.Registry.store[name]=value;return value;}
RGraph.Registry.Get=function(name){return RGraph.Registry.store[name];}
RGraph.degrees2Radians=function(degrees){return degrees*(PI/180);}
RGraph.log=(function(n,base){var log=Math.log;return function(n,base){return log(n)/(base?log(base):1);};})();
RGraph.is_array=function(obj){return obj!=null&&obj.constructor.toString().indexOf('Array')!=-1;}
RGraph.trim=function(str){return RGraph.ltrim(RGraph.rtrim(str));}
RGraph.ltrim=function(str){return str.replace(/^(\s|\0)+/, '');}
RGraph.rtrim=function(str){return str.replace(/(\s|\0)+$/, '');}
RGraph.GetHeight=function(obj){return obj.canvas.height;}
RGraph.GetWidth=function(obj){return obj.canvas.width;}
RGraph.is_null=function(arg){if(arg==null||(typeof(arg))=='object'&&!arg){return true;}return false;}
RGraph.Timer=function(label){if(typeof(RGraph.TIMER_LAST_CHECKPOINT)=='undefined'){RGraph.TIMER_LAST_CHECKPOINT=Date.now();}var now=Date.now();console.log(label+': '+(now-RGraph.TIMER_LAST_CHECKPOINT).toString());RGraph.TIMER_LAST_CHECKPOINT=now;}
RGraph.Async=function(func){return setTimeout(func,arguments[1]?arguments[1]:1);}
RGraph.isIE=function(){return navigator.userAgent.indexOf('Trident')>0||navigator.userAgent.indexOf('MSIE')>0;};ISIE=RGraph.isIE();
RGraph.isIE6=function(){return navigator.userAgent.indexOf('MSIE 6')>0;};ISIE6=RGraph.isIE6();
RGraph.isIE7=function(){return navigator.userAgent.indexOf('MSIE 7')>0;};ISIE7=RGraph.isIE7();
RGraph.isIE8=function(){return navigator.userAgent.indexOf('MSIE 8')>0;};ISIE8=RGraph.isIE8();
RGraph.isIE9=function(){return navigator.userAgent.indexOf('MSIE 9')>0;};ISIE9=RGraph.isIE9();
RGraph.isIE10=function(){return navigator.userAgent.indexOf('MSIE 10')>0;};ISIE10=RGraph.isIE10();
RGraph.isIE11=function(){return navigator.userAgent.indexOf('MSIE')==-1&&navigator.userAgent.indexOf('Trident')>0;};ISIE11=RGraph.isIE11();
RGraph.isIE9up=function(){return ISIE9||ISIE10||ISIE11;};ISIE9UP=RGraph.isIE9up();
RGraph.isIE10up=function(){return ISIE10||ISIE11};ISIE10UP=RGraph.isIE10up();
RGraph.isIE11up=function(){return ISIE11};ISIE11UP=RGraph.isIE11up();
RGraph.isOld=function(){return ISIE6||ISIE7||ISIE8;};ISOLD=RGraph.isOld();
RGraph.Reset=function(canvas){canvas.width=canvas.width;RGraph.ObjectRegistry.Clear(canvas);canvas.__rgraph_aa_translated__=false;}
function pd(variable){RGraph.pr(variable);}
function p(variable){RGraph.pr(arguments[0],arguments[1],arguments[3]);}
function a(variable){alert(variable);}
function cl(variable){return console.log(variable);}