| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102 |
- // version: 2014-06-26
- /**
- * 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. If you want to |
- * | donate to help keep the project going then you can do so here: |
- * | |
- * | http://www.rgraph.net/donate |
- * o--------------------------------------------------------------------------------o
- */
- RGraph = window.RGraph || {isRGraph: true};
- /**
- * The chart constuctor
- *
- * @param object canvas
- * @param array data
- */
- RGraph.RScatter =
- RGraph.Rscatter = function (id)
- {
- var tmp = RGraph.getCanvasTag(id);
- // Get the canvas and context objects
- this.id = tmp[0];
- this.canvas = tmp[1];
- this.context = this.canvas.getContext ? this.canvas.getContext("2d", {alpha: (typeof id === 'object' && id.alpha === false) ? false : true}) : null;
- this.canvas.__object__ = this;
- this.type = 'rscatter';
- this.hasTooltips = false;
- this.isRGraph = true;
- this.uid = RGraph.CreateUID();
- this.canvas.uid = this.canvas.uid ? this.canvas.uid : RGraph.CreateUID();
- this.colorsParsed = false;
- this.coordsText = [];
- this.original_colors = [];
- this.firstDraw = true; // After the first draw this will be false
- /**
- * Compatibility with older browsers
- */
- //RGraph.OldBrowserCompat(this.context);
- this.centerx = 0;
- this.centery = 0;
- this.radius = 0;
- this.max = 0;
-
- this.properties =
- {
- 'chart.radius': null,
- 'chart.colors': [], // This is used internally for the key
- 'chart.colors.default': 'black',
- 'chart.gutter.left': 25,
- 'chart.gutter.right': 25,
- 'chart.gutter.top': 25,
- 'chart.gutter.bottom': 25,
- 'chart.title': '',
- 'chart.title.background': null,
- 'chart.title.hpos': null,
- 'chart.title.vpos': null,
- 'chart.title.bold': true,
- 'chart.title.font': null,
- 'chart.title.x': null,
- 'chart.title.y': null,
- 'chart.title.halign': null,
- 'chart.title.valign': null,
- 'chart.labels': null,
- 'chart.labels.position': 'center',
- 'chart.labels.axes': 'nsew',
- 'chart.text.color': 'black',
- 'chart.text.font': 'Arial',
- 'chart.text.size': 10,
- 'chart.key': null,
- 'chart.key.background': 'white',
- 'chart.key.position': 'graph',
- 'chart.key.halign': 'right',
- 'chart.key.shadow': false,
- 'chart.key.shadow.color': '#666',
- 'chart.key.shadow.blur': 3,
- 'chart.key.shadow.offsetx': 2,
- 'chart.key.shadow.offsety': 2,
- 'chart.key.position.gutter.boxed':false,
- 'chart.key.position.x': null,
- 'chart.key.position.y': null,
- 'chart.key.color.shape': 'square',
- 'chart.key.rounded': true,
- 'chart.key.linewidth': 1,
- 'chart.key.colors': null,
- 'chart.key.interactive': false,
- 'chart.key.interactive.highlight.chart.fill':'rgba(255,0,0,0.9)',
- 'chart.key.interactive.highlight.label':'rgba(255,0,0,0.2)',
- 'chart.key.text.color': 'black',
- 'chart.contextmenu': null,
- 'chart.tooltips': null,
- 'chart.tooltips.event': 'onmousemove',
- 'chart.tooltips.effect': 'fade',
- 'chart.tooltips.css.class': 'RGraph_tooltip',
- 'chart.tooltips.highlight': true,
- 'chart.tooltips.hotspot': 3,
- 'chart.tooltips.coords.page': false,
- 'chart.annotatable': false,
- 'chart.annotate.color': 'black',
- 'chart.zoom.factor': 1.5,
- 'chart.zoom.fade.in': true,
- 'chart.zoom.fade.out': true,
- 'chart.zoom.hdir': 'right',
- 'chart.zoom.vdir': 'down',
- 'chart.zoom.frames': 25,
- 'chart.zoom.delay': 16.666,
- 'chart.zoom.shadow': true,
- 'chart.zoom.background': true,
- 'chart.zoom.action': 'zoom',
- 'chart.resizable': false,
- 'chart.resize.handle.background': null,
- 'chart.ymax': null,
- 'chart.ymin': 0,
- 'chart.tickmarks': 'cross',
- 'chart.ticksize': 3,
- 'chart.scale.decimals': null,
- 'chart.scale.point': '.',
- 'chart.scale.thousand': ',',
- 'chart.scale.round': false,
- 'chart.units.pre': '',
- 'chart.units.post': '',
- 'chart.events.mousemove': null,
- 'chart.events.click': null,
- 'chart.highlight.stroke': 'transparent',
- 'chart.highlight.fill': 'rgba(255,255,255,0.7)',
- 'chart.highlight.point.radius': 3,
- 'chart.labels.count': 5
- }
-
- this.data = [];
- // Handle multiple datasets being given as one argument
- if (arguments[1][0] && arguments[1][0][0] && typeof arguments[1][0][0] == 'object') {
- // Store the data set(s)
- for (var i=0; i<arguments[1].length; ++i) {
- this.data[i] = arguments[1][i];
- }
- // Handle multiple data sets being supplied as seperate arguments
- } else {
- // Store the data set(s)
- for (var i=1; i<arguments.length; ++i) {
- this.data[i - 1] = RGraph.array_clone(arguments[i]);
- }
- }
- /**
- * Create the $ objects so that functions can be added to them
- */
- for (var i=0,idx=0; i<this.data.length; ++i) {
- for (var j=0,len=this.data[i].length; j<len; j+=1,idx+=1) {
- this['$' + idx] = {}
- }
- }
- /**
- * Translate half a pixel for antialiasing purposes - but only if it hasn't beeen
- * done already
- */
- if (!this.canvas.__rgraph_aa_translated__) {
- this.context.translate(0.5,0.5);
-
- this.canvas.__rgraph_aa_translated__ = true;
- }
- // Short variable names
- var RG = RGraph;
- var ca = this.canvas;
- var co = ca.getContext('2d');
- var prop = this.properties;
- var jq = jQuery;
- var pa = RG.Path;
- var win = window;
- var doc = document;
- var ma = Math;
-
-
-
- /**
- * "Decorate" the object with the generic effects if the effects library has been included
- */
- if (RG.Effects && typeof RG.Effects.decorate === 'function') {
- RG.Effects.decorate(this);
- }
- /**
- * A simple setter
- *
- * @param string name The name of the property to set
- * @param string value The value of the property
- */
- this.set =
- this.Set = function (name, value)
- {
- /**
- * This should be done first - prepend the property name with "chart." if necessary
- */
- if (name.substr(0,6) != 'chart.') {
- name = 'chart.' + name;
- }
- prop[name.toLowerCase()] = value;
-
- return this;
- };
- /**
- * A simple getter
- *
- * @param string name The name of the property to get
- */
- this.get =
- this.Get = function (name)
- {
- /**
- * This should be done first - prepend the property name with "chart." if necessary
- */
- if (name.substr(0,6) != 'chart.') {
- name = 'chart.' + name;
- }
-
- return prop[name.toLowerCase()];
- };
- /**
- * This method draws the rose chart
- */
- this.draw =
- this.Draw = function ()
- {
- /**
- * Fire the onbeforedraw event
- */
- RG.FireCustomEvent(this, 'onbeforedraw');
-
-
- /**
- * This doesn't affect the chart, but is used for compatibility
- */
- this.gutterLeft = prop['chart.gutter.left'];
- this.gutterRight = prop['chart.gutter.right'];
- this.gutterTop = prop['chart.gutter.top'];
- this.gutterBottom = prop['chart.gutter.bottom'];
-
- // Calculate the radius
- this.radius = (Math.min(ca.width - this.gutterLeft - this.gutterRight, ca.height - this.gutterTop - this.gutterBottom) / 2);
- this.centerx = ((ca.width - this.gutterLeft - this.gutterRight) / 2) + this.gutterLeft;
- this.centery = ((ca.height - this.gutterTop - this.gutterBottom) / 2) + this.gutterTop;
- this.coords = [];
- this.coords2 = [];
- /**
- * Stop this growing uncontrollably
- */
- this.coordsText = [];
- /**
- * If there's a user specified radius/centerx/centery, use them
- */
- if (typeof(prop['chart.centerx']) == 'number') this.centerx = prop['chart.centerx'];
- if (typeof(prop['chart.centery']) == 'number') this.centery = prop['chart.centery'];
- if (typeof(prop['chart.radius']) == 'number') this.radius = prop['chart.radius'];
-
-
-
- /**
- * Parse the colors for gradients. Its down here so that the center X/Y can be used
- */
- if (!this.colorsParsed) {
-
- this.parseColors();
-
- // Don't want to do this again
- this.colorsParsed = true;
- }
-
-
- /**
- * Work out the scale
- */
- var max = prop['chart.ymax'];
- var min = prop['chart.ymin'];
-
- if (typeof(max) == 'number') {
- this.max = max;
- this.scale2 = RG.getScale2(this, {'max':max,
- 'min':min,
- 'strict':true,
- 'scale.decimals':Number(prop['chart.scale.decimals']),
- 'scale.point':prop['chart.scale.point'],
- 'scale.thousand':prop['chart.scale.thousand'],
- 'scale.round':prop['chart.scale.round'],
- 'units.pre':prop['chart.units.pre'],
- 'units.post':prop['chart.units.post'],
- 'ylabels.count':prop['chart.labels.count']
- });
- } else {
-
- for (var i=0; i<this.data.length; i+=1) {
- for (var j=0,len=this.data[i].length; j<len; j+=1) {
- this.max = Math.max(this.max, this.data[i][j][1]);
- }
- }
- this.min = prop['chart.ymin'];
-
- this.scale2 = RG.getScale2(this, {'max':this.max,
- 'min':min,
- 'scale.decimals':Number(prop['chart.scale.decimals']),
- 'scale.point':prop['chart.scale.point'],
- 'scale.thousand':prop['chart.scale.thousand'],
- 'scale.round':prop['chart.scale.round'],
- 'units.pre':prop['chart.units.pre'],
- 'units.post':prop['chart.units.post'],
- 'ylabels.count':prop['chart.labels.count']
- });
- this.max = this.scale2.max;
- }
-
- /**
- * Change the centerx marginally if the key is defined
- */
- if (prop['chart.key'] && prop['chart.key'].length > 0 && prop['chart.key'].length >= 3) {
- this.centerx = this.centerx - prop['chart.gutter.right'] + 5;
- }
-
- /**
- * Populate the colors array for the purposes of generating the key
- */
- if (typeof(prop['chart.key']) == 'object' && RG.is_array(prop['chart.key']) && prop['chart.key'][0]) {
- // Reset the colors array
- prop['chart.colors'] = [];
- for (var i=0; i<this.data.length; i+=1) {
- for (var j=0,len=this.data[i].length; j<len; j+=1) {
- if (typeof this.data[i][j][2] == 'string') {
- prop['chart.colors'].push(this.data[i][j][2]);
- }
- }
- }
- }
-
-
-
- /**
- * Populate the chart.tooltips array
- */
- this.Set('chart.tooltips', []);
- for (var i=0; i<this.data.length; i+=1) {
- for (var j=0,len=this.data[i].length; j<len; j+=1) {
- if (typeof this.data[i][j][3] == 'string') {
- prop['chart.tooltips'].push(this.data[i][j][3]);
- }
- }
- }
-
-
-
- // This resets the chart drawing state
- co.beginPath();
-
- this.DrawBackground();
- this.DrawRscatter();
- this.DrawLabels();
-
- /**
- * Setup the context menu if required
- */
- if (prop['chart.contextmenu']) {
- RG.ShowContext(this);
- }
-
-
-
- // Draw the title if any has been set
- if (prop['chart.title']) {
- RG.DrawTitle(this,
- prop['chart.title'],
- this.centery - this.radius - 10,
- this.centerx,
- prop['chart.title.size'] ? prop['chart.title.size'] : prop['chart.text.size'] + 2);
- }
-
-
- /**
- * This function enables resizing
- */
- if (prop['chart.resizable']) {
- RG.AllowResizing(this);
- }
-
-
- /**
- * This installs the event listeners
- */
- RG.InstallEventListeners(this);
- /**
- * Fire the onfirstdraw event
- */
- if (this.firstDraw) {
- RG.fireCustomEvent(this, 'onfirstdraw');
- this.firstDrawFunc();
- this.firstDraw = false;
- }
- /**
- * Fire the RGraph ondraw event
- */
- RG.FireCustomEvent(this, 'ondraw');
- return this;
- };
- /**
- * This method draws the rscatter charts background
- */
- this.drawBackground =
- this.DrawBackground = function ()
- {
- co.lineWidth = 1;
-
-
-
- // Draw the background grey circles
- co.strokeStyle = '#ccc'; // TODO Use a property here - eg chart.background.circles.color
- // Radius must be greater than 0 for Opera to work
- var r = this.radius / 10;
- for (var i=0,len=this.radius; i<=len; i+=r) {
- //co.moveTo(this.centerx + i, this.centery);
-
- // Radius must be greater than 0 for Opera to work
- co.arc(this.centerx, this.centery, i, 0, RG.TWOPI, 0);
- }
- co.stroke();
-
-
-
-
-
-
-
- // Draw the background lines that go from the center outwards
- co.beginPath();
- for (var i=15; i<360; i+=15) {
-
- // Radius must be greater than 0 for Opera to work
- co.arc(this.centerx, this.centery, this.radius, i / (180 / RG.PI), (i + 0.01) / (180 / RG.PI), 0);
-
- co.lineTo(this.centerx, this.centery);
- }
- co.stroke();
-
-
-
-
-
-
-
-
-
-
-
-
-
- co.beginPath();
- co.strokeStyle = 'black';
-
- // Draw the X axis
- co.moveTo(this.centerx - this.radius, Math.round(this.centery));
- co.lineTo(this.centerx + this.radius, Math.round(this.centery));
-
- // Draw the X ends
- co.moveTo(Math.round(this.centerx - this.radius), this.centery - 5);
- co.lineTo(Math.round(this.centerx - this.radius), this.centery + 5);
- co.moveTo(Math.round(this.centerx + this.radius), this.centery - 5);
- co.lineTo(Math.round(this.centerx + this.radius), this.centery + 5);
-
- // Draw the X check marks
- for (var i=(this.centerx - this.radius); i<(this.centerx + this.radius); i+=(this.radius / 10)) {
- co.moveTo(Math.round(i), this.centery - 3);
- co.lineTo(Math.round(i), this.centery + 3);
- }
-
- // Draw the Y check marks
- for (var i=(this.centery - this.radius); i<(this.centery + this.radius); i+=(this.radius / 10)) {
- co.moveTo(this.centerx - 3, Math.round(i));
- co.lineTo(this.centerx + 3, Math.round(i));
- }
-
- // Draw the Y axis
- co.moveTo(Math.round(this.centerx), this.centery - this.radius);
- co.lineTo(Math.round(this.centerx), this.centery + this.radius);
-
- // Draw the Y ends
- co.moveTo(this.centerx - 5, Math.round(this.centery - this.radius));
- co.lineTo(this.centerx + 5, Math.round(this.centery - this.radius));
-
- co.moveTo(this.centerx - 5, Math.round(this.centery + this.radius));
- co.lineTo(this.centerx + 5, Math.round(this.centery + this.radius));
-
- // Stroke it
- co.closePath();
- co.stroke();
- };
- /**
- * This method draws a set of data on the graph
- */
- this.drawRscatter =
- this.DrawRscatter = function ()
- {
- for (var dataset=0; dataset<this.data.length; dataset+=1) {
- var data = this.data[dataset];
- this.coords2[dataset] = [];
- for (var i=0; i<data.length; ++i) {
-
- var d1 = data[i][0];
- var d2 = data[i][1];
- var a = d1 / (180 / RG.PI); // RADIANS
- var r = ( (d2 - prop['chart.ymin']) / (this.scale2.max - this.scale2.min) ) * this.radius;
- var x = Math.sin(a) * r;
- var y = Math.cos(a) * r;
- var color = data[i][2] ? data[i][2] : prop['chart.colors.default'];
- var tooltip = data[i][3] ? data[i][3] : null;
-
- if (tooltip && String(tooltip).length) {
- this.hasTooltips = true;
- }
-
- /**
- * Account for the correct quadrant
- */
- x = x + this.centerx;
- y = this.centery - y;
-
-
- this.DrawTick(x, y, color);
-
- // Populate the coords array with the coordinates and the tooltip
- this.coords.push([x, y, color, tooltip]);
- this.coords2[dataset].push([x, y, color, tooltip]);
- }
- }
- };
- /**
- * Unsuprisingly, draws the labels
- */
- this.drawLabels =
- this.DrawLabels = function ()
- {
- co.lineWidth = 1;
-
- // Default the color to black
- co.fillStyle = 'black';
- co.strokeStyle = 'black';
-
- var key = prop['chart.key'];
- var r = this.radius;
- var color = prop['chart.text.color'];
- var font = prop['chart.text.font'];
- var size = prop['chart.text.size'];
- var axes = prop['chart.labels.axes'].toLowerCase();
- var units_pre = prop['chart.units.pre'];
- var units_post = prop['chart.units.post'];
- var decimals = prop['chart.scale.decimals'];
- var centerx = this.centerx;
- var centery = this.centery;
-
- co.fillStyle = prop['chart.text.color'];
-
- // Draw any labels
- if (typeof prop['chart.labels'] == 'object' && prop['chart.labels']) {
- this.DrawCircularLabels(co, prop['chart.labels'], font , size, r);
- }
-
-
- var color = 'rgba(255,255,255,0.8)';
- // Draw the axis labels
- for (var i=0,len=this.scale2.labels.length; i<len; ++i) {
- if (axes.indexOf('n') > -1) RG.Text2(this, {'tag': 'scale','font':font,'size':size,'x':centerx,'y':centery - (r * ((i+1) / len)),'text':this.scale2.labels[i],'valign':'center','halign':'center','bounding':true,'boundingFill':color});
- if (axes.indexOf('s') > -1) RG.Text2(this, {'tag': 'scale','font':font,'size':size,'x':centerx,'y':centery + (r * ((i+1) / len)),'text':this.scale2.labels[i],'valign':'center','halign':'center','bounding':true,'boundingFill':color});
- if (axes.indexOf('e') > -1) RG.Text2(this, {'tag': 'scale','font':font,'size':size,'x':centerx + (r * ((i+1) / len)),'y':centery,'text':this.scale2.labels[i],'valign':'center','halign':'center','bounding':true,'boundingFill':color});
- if (axes.indexOf('w') > -1) RG.Text2(this, {'tag': 'scale','font':font,'size':size,'x':centerx - (r * ((i+1) / len)),'y':centery,'text':this.scale2.labels[i],'valign':'center','halign':'center','bounding':true,'boundingFill':color});
- }
-
- // Draw the center minimum value (but only if there's at least one axes labels stipulated)
- if (prop['chart.labels.axes'].length > 0) {
- RG.Text2(this, {'font':font,
- 'size':size,
- 'x':centerx,
- 'y':centery,
- 'text':RG.number_format(this, Number(this.scale2.min).toFixed(this.scale2.decimals), this.scale2.units_pre, this.scale2.units_post),
- 'valign':'center',
- 'halign':'center',
- 'bounding':true,
- 'boundingFill':color,
- 'tag': 'scale'
- });
- }
-
- /**
- * Draw the key
- */
- if (key && key.length) {
- RG.DrawKey(this, key, prop['chart.colors']);
- }
- };
- /**
- * Draws the circular labels that go around the charts
- *
- * @param labels array The labels that go around the chart
- */
- this.drawCircularLabels =
- this.DrawCircularLabels = function (context, labels, font_face, font_size, r)
- {
- var position = prop['chart.labels.position'];
- var r = r + 10;
-
- for (var i=0; i<labels.length; ++i) {
-
-
- var a = (360 / labels.length) * (i + 1) - (360 / (labels.length * 2));
- var a = a - 90 + (prop['chart.labels.position'] == 'edge' ? ((360 / labels.length) / 2) : 0);
-
- var x = Math.cos(a / (180/RG.PI) ) * (r + 10);
- var y = Math.sin(a / (180/RG.PI)) * (r + 10);
-
- RG.Text2(this, {'font':font_face,
- 'size':font_size,
- 'x':this.centerx + x,
- 'y':this.centery + y,
- 'text':String(labels[i]),
- 'valign':'center',
- 'halign':'center',
- 'tag': 'labels'
- });
- }
- };
- /**
- * Draws a single tickmark
- */
- this.drawTick =
- this.DrawTick = function (x, y, color)
- {
- var tickmarks = prop['chart.tickmarks'];
- var ticksize = prop['chart.ticksize'];
-
- co.strokeStyle = color;
- co.fillStyle = color;
-
- // Cross
- if (tickmarks == 'cross') {
-
- co.beginPath();
- co.moveTo(x + ticksize, y + ticksize);
- co.lineTo(x - ticksize, y - ticksize);
- co.stroke();
-
- co.beginPath();
- co.moveTo(x - ticksize, y + ticksize);
- co.lineTo(x + ticksize, y - ticksize);
- co.stroke();
-
- // Circle
- } else if (tickmarks == 'circle') {
-
- co.beginPath();
- co.arc(x, y, ticksize, 0, 6.2830, false);
- co.fill();
-
- // Square
- } else if (tickmarks == 'square') {
-
- co.beginPath();
- co.fillRect(x - ticksize, y - ticksize, 2 * ticksize, 2 * ticksize);
- co.fill();
-
- // Diamond shape tickmarks
- } else if (tickmarks == 'diamond') {
-
- co.beginPath();
- co.moveTo(x, y - ticksize);
- co.lineTo(x + ticksize, y);
- co.lineTo(x, y + ticksize);
- co.lineTo(x - ticksize, y);
- co.closePath();
- co.fill();
-
- // Plus style tickmarks
- } else if (tickmarks == 'plus') {
-
- co.lineWidth = 1;
-
- co.beginPath();
- co.moveTo(x, y - ticksize);
- co.lineTo(x, y + ticksize);
- co.moveTo(x - ticksize, y);
- co.lineTo(x + ticksize, y);
- co.stroke();
- }
- };
- /**
- * This function makes it much easier to get the (if any) point that is currently being hovered over.
- *
- * @param object e The event object
- */
- this.getShape =
- this.getPoint = function (e)
- {
- var mouseXY = RG.getMouseXY(e);
- var mouseX = mouseXY[0];
- var mouseY = mouseXY[1];
- var overHotspot = false;
- var offset = prop['chart.tooltips.hotspot']; // This is how far the hotspot extends
-
- for (var i=0,len=this.coords.length; i<len; ++i) {
-
- var x = this.coords[i][0];
- var y = this.coords[i][1];
- var tooltip = this.coords[i][3];
-
- if (
- mouseX < (x + offset) &&
- mouseX > (x - offset) &&
- mouseY < (y + offset) &&
- mouseY > (y - offset)
- ) {
-
- var tooltip = RG.parseTooltipText(prop['chart.tooltips'], i);
-
- return {0:this,1:x,2:y,3:i,'object':this, 'x':x, 'y':y, 'index':i, 'tooltip': tooltip};
- }
- }
- };
- /**
- * This function facilitates the installation of tooltip event listeners if
- * tooltips are defined.
- */
- this.allowTooltips =
- this.AllowTooltips = function ()
- {
- // Preload any tooltip images that are used in the tooltips
- RG.PreLoadTooltipImages(this);
-
-
- /**
- * This installs the window mousedown event listener that lears any
- * highlight that may be visible.
- */
- RG.InstallWindowMousedownTooltipListener(this);
-
-
- /**
- * This installs the canvas mousemove event listener. This function
- * controls the pointer shape.
- */
- RG.InstallCanvasMousemoveTooltipListener(this);
-
-
- /**
- * This installs the canvas mouseup event listener. This is the
- * function that actually shows the appropriate tooltip (if any).
- */
- RG.InstallCanvasMouseupTooltipListener(this);
- };
- /**
- * Each object type has its own Highlight() function which highlights the appropriate shape
- *
- * @param object shape The shape to highlight
- */
- this.highlight =
- this.Highlight = function (shape)
- {
- // Add the new highlight
- RG.Highlight.Point(this, shape);
- };
- /**
- * The getObjectByXY() worker method. Don't call this call:
- *
- * RGraph.ObjectRegistry.getObjectByXY(e)
- *
- * @param object e The event object
- */
- this.getObjectByXY = function (e)
- {
- var mouseXY = RG.getMouseXY(e);
- var mouseX = mouseXY[0];
- var mouseY = mouseXY[1];
- var centerx = this.centerx;
- var centery = this.centery;
- var radius = this.radius;
-
- if (
- mouseX > (centerx - radius)
- && mouseX < (centerx + radius)
- && mouseY > (centery - radius)
- && mouseY < (centery + radius)
- ) {
-
- return this;
- }
- };
- /**
- * This function positions a tooltip when it is displayed
- *
- * @param obj object The chart object
- * @param int x The X coordinate specified for the tooltip
- * @param int y The Y coordinate specified for the tooltip
- * @param objec tooltip The tooltips DIV element
- */
- this.positionTooltip = function (obj, x, y, tooltip, idx)
- {
- var coordX = obj.coords[tooltip.__index__][0];
- var coordY = obj.coords[tooltip.__index__][1];
- var canvasXY = RG.getCanvasXY(obj.canvas);
- var gutterLeft = obj.gutterLeft;
- var gutterTop = obj.gutterTop;
- var width = tooltip.offsetWidth;
-
- // Set the top position
- tooltip.style.left = 0;
- tooltip.style.top = parseInt(tooltip.style.top) - 7 + 'px';
-
- // By default any overflow is hidden
- tooltip.style.overflow = '';
-
- // The arrow
- var img = new Image();
- img.src = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABEAAAAFCAYAAACjKgd3AAAARUlEQVQYV2NkQAN79+797+RkhC4M5+/bd47B2dmZEVkBCgcmgcsgbAaA9GA1BCSBbhAuA/AagmwQPgMIGgIzCD0M0AMMAEFVIAa6UQgcAAAAAElFTkSuQmCC';
- img.style.position = 'absolute';
- img.id = '__rgraph_tooltip_pointer__';
- img.style.top = (tooltip.offsetHeight - 2) + 'px';
- tooltip.appendChild(img);
-
- // Reposition the tooltip if at the edges:
-
- // LEFT edge
- if ((canvasXY[0] + coordX - (width / 2)) < 10) {
- tooltip.style.left = (canvasXY[0] + coordX - (width * 0.1)) + 'px';
- img.style.left = ((width * 0.1) - 8.5) + 'px';
-
- // RIGHT edge
- } else if ((canvasXY[0] + coordX + (width / 2)) > doc.body.offsetWidth) {
- tooltip.style.left = canvasXY[0] + coordX - (width * 0.9) + 'px';
- img.style.left = ((width * 0.9) - 8.5) + 'px';
-
- // Default positioning - CENTERED
- } else {
- tooltip.style.left = (canvasXY[0] + coordX - (width * 0.5)) + 'px';
- img.style.left = ((width * 0.5) - 8.5) + 'px';
- }
- };
- /**
- * This function returns the radius (ie the distance from the center) for a particular
- * value.
- *
- * @param number value The value you want the radius for
- */
- this.getRadius = function (value)
- {
- var max = this.max;
- if (value < 0 || value > max) {
- return null;
- }
-
- var r = (value / max) * this.radius;
-
- return r;
- };
- /**
- * This allows for easy specification of gradients
- */
- this.parseColors = function ()
- {
- // Save the original colors so that they can be restored when the canvas is reset
- if (this.original_colors.length === 0) {
- this.original_colors['data'] = RG.array_clone(this.data);
- this.original_colors['chart.highlight.stroke'] = RG.array_clone(prop['chart.highlight.stroke']);
- this.original_colors['chart.highlight.fill'] = RG.array_clone(prop['chart.highlight.fill']);
- this.original_colors['chart.colors.default'] = RG.array_clone(prop['chart.colors.default']);
- }
- // Go through the data
- for (var i=0; i<this.data.length; i+=1) {
- for (var j=0,len=this.data[i].length; j<len; j+=1) {
- this.data[i][j][2] = this.parseSingleColorForGradient(this.data[i][j][2]);
- }
- }
-
- prop['chart.highlight.stroke'] = this.parseSingleColorForGradient(prop['chart.highlight.stroke']);
- prop['chart.highlight.fill'] = this.parseSingleColorForGradient(prop['chart.highlight.fill']);
- prop['chart.colors.default'] = this.parseSingleColorForGradient(prop['chart.colors.default']);
- };
- /**
- * This parses a single color value
- */
- this.parseSingleColorForGradient = function (color)
- {
- if (!color || typeof color != 'string') {
- return color;
- }
- if (color.match(/^gradient\((.*)\)$/i)) {
-
- var parts = RegExp.$1.split(':');
-
- // Create the gradient
- var grad = co.createRadialGradient(this.centerx, this.centery, 0, this.centerx, this.centery, this.radius);
-
- var diff = 1 / (parts.length - 1);
-
- grad.addColorStop(0, RG.trim(parts[0]));
-
- for (var j=1; j<parts.length; ++j) {
- grad.addColorStop(j * diff, RG.trim(parts[j]));
- }
- }
-
- return grad ? grad : color;
- };
- /**
- * This function handles highlighting an entire data-series for the interactive
- * key
- *
- * @param int index The index of the data series to be highlighted
- */
- this.interactiveKeyHighlight = function (index)
- {
- if (this.coords2 && this.coords2[index] && this.coords2[index].length) {
- this.coords2[index].forEach(function (value, idx, arr)
- {
- co.beginPath();
- co.fillStyle = prop['chart.key.interactive.highlight.chart.fill'];
- co.arc(value[0], value[1], prop['chart.ticksize'] + 2, 0, RG.TWOPI, false);
- co.fill();
- });
- }
- };
- /**
- * Using a function to add events makes it easier to facilitate method chaining
- *
- * @param string type The type of even to add
- * @param function func
- */
- this.on = function (type, func)
- {
- if (type.substr(0,2) !== 'on') {
- type = 'on' + type;
- }
-
- this[type] = func;
-
- return this;
- };
- /**
- * This helps the Gantt reset colors when the reset function is called.
- * It handles going through the data and resetting the colors.
- */
- this.resetColorsToOriginalValues = function ()
- {
- /**
- * Copy the original colors over for single-event-per-line data
- */
- for (var i=0,len=this.original_colors['data'].length; i<len; ++i) {
- for (var j=0,len2=this.original_colors['data'][i].length; j<len2;++j) {
- this.data[i][j][2] = RG.array_clone(this.original_colors['data'][i][j][2]);
- }
- }
- };
- /**
- * This function runs once only
- * (put at the end of the file (before any effects))
- */
- this.firstDrawFunc = function ()
- {
- };
- /**
- * Register the object
- */
- RG.Register(this);
- };
|