All Downloads are FREE. Search and download functionalities are using the official Maven repository.

META-INF.adf.jsLibsDebug.ApacheChart.js Maven / Gradle / Ivy

/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 * 
 *  http://www.apache.org/licenses/LICENSE-2.0
 * 
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 */

/////////////////////////////////////////////
// Some utility functions
/////////////////////////////////////////////

/**
* Base Class that provides inheritence for charting subsystem
* @constructor 
*/
function ApacheChartObj()
{
  this.Init();
}

ApacheChartObj.prototype = new Object();
ApacheChartObj.prototype.constructor = ApacheChartObj;
ApacheChartObj._tempConstructor = function(){}

ApacheChartObj.Inherit = function(baseClass, extendingClass)
{
  var tempConstructor = ApacheChartObj._tempConstructor;

  tempConstructor.prototype = baseClass.prototype;
  extendingClass.prototype = new tempConstructor();

  extendingClass.prototype.constructor = extendingClass;
  extendingClass.superclass = baseClass.prototype;
}

/**
* Asserts arg is true; else throws error with msg.
*/
ApacheChartObj.Assert = function(arg, msg)
{
  if (!arg)
  {
    throw new Error(msg);
  }
}


/**
* StringBuffer utility used by the charting
*/
function ApacheChartBuffer(size)
{
  this.maxStreamLength = document.all?5000:100000;  
  this.data = new Array(size?size:100);
  this.iStr = 0;
}

ApacheChartBuffer.prototype.append = function(obj)
{
  this.data[this.iStr++] = obj;
  if (this.data.length > this.maxStreamLength)
  {
    this.data = [this.data.join("")];
    this.data.length = 100;
    this.iStr = 1;
  }
  return this;  
}

ApacheChartBuffer.prototype.toString = function()
{
  return this.data.join("");
}

////////////////////////////////////////////////////////////////////
// Abstract Data Structure representing the model for drawing a chart control
////////////////////////////////////////////////////////////////////
function ApacheChartModel(seriesLabels, groupLabels, yValues, xValues, seriesColors)
{
  // An array representing series labels
  this._seriesLabels = seriesLabels;
  // An array representing Group labels
  this._groupLabels = groupLabels;
  // A 2D array representing values for Y axis
  this._yValues = yValues;
  // A 2D array representing values for X axis.
  // The x axis values are used only for scatter plots/XYline 
  this._xValues = xValues;
  // The array of strings with colors. Used for display of the series
  this._seriesColors = seriesColors;
  
  var labelCount = seriesLabels.length;
  var colorCount = seriesColors.length;

  if(colorCount < labelCount)
  {
    var toHexFunc = ApacheChart._to_hex;
    for(i = colorCount; i < labelCount; i++)
    {
      // generate random colors
      var rVal = Math.floor(Math.random()*1000)%255;
      var gVal = Math.floor(Math.random()*1000)%255;
      var bVal = Math.floor(Math.random()*1000)%255;
      seriesColors[i] = "#"+toHexFunc(rVal)+toHexFunc(gVal)+toHexFunc(bVal);
    }
  }
  // the maximum value used to display the y-axis. 
  // Default is 120% of maximum of the yValues
  //this._maxYValue = undefined;
  
  // the minimum value used to display the y-axis. Default is 80% of minimum of values
  //this._minYValue = undefined;

  // the maximum value used to display the X-axis. 
  // Default is 120% of maximum of the xValues
  //this._maxXValue = undefined;
  
  // the minimum value used to display the y-axis. Default is 80% of minimum of values 
  //this._minXValue = undefined;
    
  // The title for the graph
  //this._title = undefined;
  
  // The sub-title for the graph
  //this._subTitle = undefined;
  
  // The foot node for the graph
  //this._footNote = undefined;
}

ApacheChartModel.prototype.getSeriesLabels = function()
{
  return this._seriesLabels; 
}

ApacheChartModel.prototype.getGroupLabels = function()
{
  return this._groupLabels;
}

ApacheChartModel.prototype.getSeriesColors = function()
{
  return this._seriesColors; 
}

ApacheChartModel.prototype.getXValues = function()
{
  return this._xValues; 
}

ApacheChartModel.prototype.getYValues = function()
{
  return this._yValues; 
}

ApacheChartModel.prototype.setMaxYValue = function(maxYValue)
{
  this._maxYValue = maxYValue; 
}

ApacheChartModel.prototype.getMaxYValue = function()
{
  return this._maxYValue; 
}

ApacheChartModel.prototype.setMinYValue = function(minYValue)
{
  this._minYValue = minYValue; 
}

ApacheChartModel.prototype.getMinYValue = function()
{
  return this._minYValue; 
}

ApacheChartModel.prototype.setMaxXValue = function(maxXValue)
{
  this._maxXValue = maxXValue; 
}

ApacheChartModel.prototype.getMaxXValue = function()
{
  return this._maxXValue; 
}

ApacheChartModel.prototype.setMinXValue = function(minXValue)
{
  this._minXValue = minXValue; 
}

ApacheChartModel.prototype.getMinXValue = function()
{
  return this._minXValue; 
}

ApacheChartModel.prototype.setTitle = function(title)
{
  this._title = title; 
}

ApacheChartModel.prototype.getTitle = function()
{
  return this._title; 
}

ApacheChartModel.prototype.setSubTitle = function(subTitle)
{
  this._subTitle = subTitle; 
}

ApacheChartModel.prototype.getSubTitle = function()
{
  return this._subTitle; 
}

ApacheChartModel.prototype.setFootNote = function(footNote)
{
  this._footNote = footNote; 
}

ApacheChartModel.prototype.getFootNote = function()
{
  return this._footNote; 
}

////////////////////////////////////////////////////////////////////
// A Chart Event that is triggered in response to a user click
// This event can be marshalled to the server if integration is done
// with a J2EE platform platform
////////////////////////////////////////////////////////////////////
function ApacheChartEvent(seriesIndices, yValueIndices, yValues, xValues)
{
  this._seriesIndices = seriesIndices;
  this._yValueIndices = yValueIndices;
  this._yValues = yValues;
  this._xValues = xValues;
}

ApacheChartEvent.prototype.getSeriesIndices = function()
{
  return this._seriesIndices;
}

ApacheChartEvent.prototype.getYValueIndices = function()
{
  return this._yValueIndices;
}

ApacheChartEvent.prototype.getYValues = function()
{
  return this._yValues;
}

ApacheChartEvent.prototype.getXValues = function()
{
  return this._xValues;
}

ApacheChartEvent.prototype.toString = function()
{
  var sb = new ApacheChartBuffer();
  if(this._seriesIndices)
    sb.append("seriesIndices = "+ this._seriesIndices.join(","));
  if(this._yValueIndices)
    sb.append("\yValueIndices = "+ this._yValueIndices.join(","));
  sb.append("\nyValues = "+ this._yValues.join(","));
  if(this._xValues)
    sb.append("\nxValues = "+ this._xValues.join(","));
  
  return sb.toString();
}

ApacheChartEvent.prototype.marshall = function()
{
  var value = new Array();
  if(this._seriesIndices)
    value.push("seriesIndices\t"+this._seriesIndices.join("\t"));
  if(this._yValueIndices)
    value.push("yValueIndices\t"+this._yValueIndices.join("\t"));
  value.push("yValues\t"+this._yValues.join("\t"));
  if(this._xValues)
    value.push("xValues\t"+this._xValues.join("\t"));
  
  return value.join("$adf$");
} 
////////////////////////////////////////////////////////////////////
// Abstract Base Class for rendering a Chart control
////////////////////////////////////////////////////////////////////
function ApacheChart(type, model, svgEmbedId, isPerspective, legendPosition)
{
  this.Init(type, model, svgEmbedId, isPerspective, legendPosition);
}

// Hack for IE. The SVG has to be loaded externally otherwise,
// it needs to be activated by clicking
ApacheChart.createSVG = function(containerId, 
  svgEmbedId, sourceUrl, inlineStyle, styleClass)
{
  var svgContainer = document.getElementById(containerId);
  var embed = document.createElement("embed");
  var agent = window._agent;

  if(agent && agent.isIE)
  {
    var semiColIndex = sourceUrl.indexOf(";");
    if(semiColIndex!=-1)
      sourceUrl = sourceUrl.substr(0,semiColIndex);
  }
  embed.setAttribute("src",sourceUrl);
  embed.setAttribute("id",svgEmbedId);
  embed.setAttribute("wmode","transparent");
  embed.setAttribute("type","image/svg+xml"); 
  
  if(agent && agent.isOpera)
  {
    // opera does not like 100% width and 100% height.
    var style = document.defaultView.getComputedStyle(svgContainer, null);
    var embedStyle = embed.style;
    embedStyle.width = style.width;
    embedStyle.height = style.height;
  }
  else
  {
    embed.style.cssText = inlineStyle;
  }
  if(styleClass)
  {
    embed.className = styleClass;
  }
  svgContainer.appendChild(embed);
}

/**
 * Method to detect if Adobe ActiveX controll is initialized on the machine
 */
ApacheChart.isASVInstalled = function()
{
  try{
      var asv = new ActiveXObject("Adobe.SVGCtl");
      return true;
  }
  catch(e){
  }
  return false;
}
ApacheChartObj.Inherit(ApacheChartObj, ApacheChart);

ApacheChart.prototype.Init = function(type, model, svgEmbedId, 
  isPerspective, legendPosition)
{  
  this._type = type;
  this._model = model;
  this._svgEmbedId = svgEmbedId;
  this._margins = {left:2,right:2,top:2,bottom:2};
  // By default the graphs are drawn with a perspective(2.5D)
  this._isPerspective = isPerspective;
  this._legendPosition = legendPosition;
  //this._rootElement = undefined;
  this._toolTip = null;
  this._toolTipVisible = false;
  //this._svgDoc = undefined;
  //this._width = undefined;
  //this._height = undefined;
  //this._vLabelContainer = undefined;
  //this._hLabelContainer = undefined;
  
  // By default the graph animates for 1.5 s
  this._animDuration = 1500; 
  this._dataElems = [];
  this._labelElems = [];
  this._gridElems = [];

  // The group labels
  this._groupLabelElems = [];
  
  // The number of Major Line sections to draw on y axis
  this._yMajorGridCount = 8;
  // The number of Minor Line section to draw on y axis
  this._yMinorGridCount = -1;

  // The number of Major Line sections to draw on x axis
  this._xMajorGridCount = -1;
  
  // Controls if the tooltips are displayed or not
  this._tooltipsVisible = true;
        
  // By default we will use gradients
  this._gradientsUsed = true;

  // The maximum precision of the numbers displayed on yaxis/tooltips
  this._maxPrecision = 0;
  
  // The decimal separator
  this._decimalSep = null;

  // The form from which we will submit
  this._formName = null;
  this._partialSubmit = true;
      
  this._svgCheckTotal = 0;
  this._errorTextNode = null;
  this._isIE = false;
  if(window._agent) // use trinidad agent
    this._isIE = _agent.isIE;
    
  if(this._isIE)
    this._errorHtml = "

Unable to load SVG plugin. Please install the plugin from Adobe

"; else this._errorHtml = "

This component needs an SVG enabled browser like Internet Explorer, Firefox 1.5+ or Opera 9.0+

"; this._statusHtml = "

Please Wait. Attempting to load SVG document...

"; this.ComputeMinMaxValues(); } /** * Error text to be displayed in the container of the embed, * in case the svg document fails to load. * NOTE: the error text can be customized by browser for e.g. in IE it can contain * the link to SVG plugin download */ ApacheChart.prototype.setErrorHtml = function(errorHtml) { this._errorHtml = errorHtml; } /** * Status text that is displayed when the SVG document is taking a while to load */ ApacheChart.prototype.setStatusHtml = function(statusHtml) { this._statusHtml = statusHtml; } ApacheChart.prototype.setYMajorGridLineCount = function(count) { // number of sections is 1 greater than the lines this._yMajorGridCount = count>0?count+1:count; } ApacheChart.prototype.setYMinorGridLineCount = function(count) { // number of sections is 1 greater than the lines this._yMinorGridCount = count>0?count+1:count; } ApacheChart.prototype.setXMajorGridLineCount = function(count) { // number of sections is 1 greater than the lines this._xMajorGridCount = count>0?count+1:count; } ApacheChart.prototype.setAnimationDuration = function(dur) { this._animDuration = dur; } ApacheChart.prototype.setGradientsUsed = function(gradient) { this._gradientsUsed = gradient; } ApacheChart.prototype.setMaxPrecision = function(precision) { this._maxPrecision = precision; } ApacheChart.prototype.setFormName = function(formName) { this._formName = formName; } ApacheChart.prototype.setPartialSubmit = function(partial) { this._partialSubmit = partial; } ApacheChart.prototype.setTooltipsVisible = function(visible) { this._tooltipsVisible = visible; } ApacheChart.prototype.getToolTip = function() { return this._toolTip; } ApacheChart.prototype.setToolTip = function(tt) { this._toolTip = tt; } ApacheChart.prototype.ComputeMinMaxValues = function() { var model = this._model, yValues = model.getYValues(), xValues = model.getXValues(), maxYValue = model.getMaxYValue(), maxXValue = model.getMaxXValue(), minYValue = model.getMinYValue(), minXValue = model.getMinXValue(), seriesLabels = model.getSeriesLabels(); if(yValues != null && (maxYValue == null || minYValue == null)) { var minMax = this._computeAxisMinMaxValues(yValues, seriesLabels.length); if(maxYValue == null) model.setMaxYValue(minMax.max); if(minYValue == null) model.setMinYValue(minMax.min); } if(xValues != null && (maxXValue == null || minXValue == null)) { var minMax = this._computeAxisMinMaxValues(xValues, seriesLabels.length); if(maxXValue == null) model.setMaxXValue(minMax.max); if(minXValue == null) model.setMinXValue(minMax.min); } } ApacheChart.prototype._computeAxisMinMaxValues = function(values, seriesSize) { var stackedTotal, value, maxValue = Number.NEGATIVE_INFINITY, minValue = Number.POSITIVE_INFINITY, type = this._type, isStacked = false, groupsCount = values.length; if(type == ApacheChart.TYPE_VBAR_STACKED || type == ApacheChart.TYPE_HBAR_STACKED || type == ApacheChart.TYPE_AREA_STACKED) { isStacked = true; } for (var i = 0; i < groupsCount; ++i) { stackedTotal = 0; for (var j = 0; j < seriesSize; ++j) { value = values[i][j]; if (isStacked) stackedTotal += value; else { maxValue = Math.max(maxValue, value); minValue = Math.min(minValue, value); } } if (isStacked) { maxValue = Math.max(maxValue, stackedTotal); minValue = Math.min(minValue, stackedTotal); } } var maxMult = maxValue>0?ApacheChart._MAX_MULTIPLIER:ApacheChart._MIN_MULTIPLIER, minMult = minValue>0?ApacheChart._MIN_MULTIPLIER:ApacheChart._MAX_MULTIPLIER; return {max: maxValue*maxMult, min: minValue*minMult}; } ApacheChart.TYPE_VBAR = 1; ApacheChart.TYPE_HBAR = 2; ApacheChart.TYPE_VBAR_STACKED = 3; ApacheChart.TYPE_HBAR_STACKED = 4; ApacheChart.TYPE_PIE = 5; ApacheChart.TYPE_AREA = 6; ApacheChart.TYPE_AREA_STACKED = 7; ApacheChart.TYPE_LINE = 8; ApacheChart.TYPE_BAR_LINE_COMBO = 9; ApacheChart.TYPE_XYLINE = 10; ApacheChart.TYPE_SCATTER_PLOT = 11; ApacheChart.TYPE_RADAR = 12; ApacheChart.TYPE_RADAR_AREA = 13; ApacheChart.TYPE_FUNNEL = 14; ApacheChart.CIRCULAR_GAUGE = 15; ApacheChart.SEMI_CIRCULAR_GAUGE = 16; ApacheChart.LEGEND_LOCATION_NONE = "none"; ApacheChart.LEGEND_LOCATION_TOP = "top"; ApacheChart.LEGEND_LOCATION_END = "end"; ApacheChart.LEGEND_LOCATION_BOTTOM = "bottom"; ApacheChart.LEGEND_LOCATION_START = "start"; ApacheChart._MAX_MULTIPLIER = 1.2; ApacheChart._MIN_MULTIPLIER = .8; ApacheChart._XOFFSET_PERSPECTIVE = 10; ApacheChart._YOFFSET_PERSPECTIVE = 5; // margin generally used around text ApacheChart._TEXT_MARGIN = 4; ApacheChart._DEFAULT_STOP_OPACITY = .9; ApacheChart._BORDER_SIZE = 6; // Animate at 15 fps ApacheChart._ANIMATE_INTERVAL = 66; ApacheChart._SVGCHECK_INTERVAL = 100; ApacheChart._SVGCHECK_STATUS_LIMIT = 5000; ApacheChart._SVGCHECK_MAX_LIMIT = 20000; ApacheChart.createChart = function( type, model, svgEmbedId, isPerspective, legendPosition) { var chart = null; if(type == this.TYPE_VBAR || type == this.TYPE_VBAR_STACKED || type == this.TYPE_BAR_LINE_COMBO) { chart = new ApacheBarChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_HBAR || type == this.TYPE_HBAR_STACKED) { chart = new ApacheHBarChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_PIE) { chart = new ApachePieChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_AREA || type == this.TYPE_AREA_STACKED) { chart = new ApacheAreaChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_LINE) { chart = new ApacheLineChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_SCATTER_PLOT) { chart = new ApacheScatterPlotChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_XYLINE) { chart = new ApacheXYLineChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_RADAR || type == this.TYPE_RADAR_AREA) { chart = new ApacheRadarChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.TYPE_FUNNEL) { chart = new ApacheFunnelChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.SEMI_CIRCULAR_GAUGE) { chart = new ApacheSemiGaugeChart(type, model, svgEmbedId, isPerspective, legendPosition); } else if(type == this.CIRCULAR_GAUGE) { chart = new ApacheGaugeChart(type, model, svgEmbedId, isPerspective, legendPosition); } return chart; } ApacheChart.prototype.setPerspective = function(isPerpective) { this._isPerspective = isPerpective; } ApacheChart.prototype.clear = function() { var rootElem = this._rootElement; var childNode = rootElem.firstChild; while (childNode) { rootElem.removeChild(childNode); childNode = rootElem.firstChild; } } ApacheChart.prototype.draw = function() { if(!this._initDocument()) return; // Initialize our gradients if necessary if (this._gradientsUsed && !this._gradientsInitialized) { this.InitializeGradients(); this._gradientsInitialized = true; } if(this._tooltipsVisible) { this.ShowToolTipCallback = TrUIUtils.createCallback(this, this.ShowToolTip); this.HideToolTipCallback = TrUIUtils.createCallback(this, this.HideToolTip); } this.ClickCallback = TrUIUtils.createCallback(this, this.Click); // Note the ordering is important. The grid takes the space after the title etc. this.DrawBorder(); this.DrawTitles(); // First just draw the label elements so that we can estimate the space requirements this.DrawGroupLabels(); this.DrawYValueLabels(); // Now adjust margins based on the labels this.AdjustMarginsForGroupLabels(); this.AdjustMarginsForYLabels(); // Now start drawing the graph so that it gobbles the left over space this.DrawLegend(); this.LayoutGroupLabels(); this.LayoutYValueLabels(); this.DrawGrid(); this.DrawChartData(); this.Animate(); } ApacheChart.prototype._initDocument = function() { // Get hold of the svgDocument var svgEmbed = document.getElementById(this._svgEmbedId); var isIE = this._isIE; if(isIE && !ApacheChart.isASVInstalled()) { this._displayErrorHtml(svgEmbed); return false; } try { var svgDoc = svgEmbed.getSVGDocument(); this._rootElement = svgDoc.getElementById("chartRoot"); if(!this._rootElement) // make sure that the document is loaded throw "not yet loaded"; this._svgDoc = svgDoc; this._width = svgEmbed.clientWidth; this._height = svgEmbed.clientHeight; if(this._errorTextNode != null) { svgEmbed.parentNode.removeChild(this._errorTextNode); svgEmbed.style.display = ""; } } catch(e) { this._svgCheckTotal += ApacheChart._SVGCHECK_INTERVAL; if(this._svgCheckTotal > ApacheChart._SVGCHECK_MAX_LIMIT) { // We are out of our chances this._displayErrorHtml(svgEmbed); return false; } else if(null == this._errorTextNode && this._svgCheckTotal > ApacheChart._SVGCHECK_STATUS_LIMIT) { // display a status message this._displayStatusHtml(svgEmbed); } if(!this._drawCallback) this._drawCallback = TrUIUtils.createCallback(this, this.draw); // Lets try again window.setTimeout(this._drawCallback, ApacheChart._SVGCHECK_INTERVAL); return false; } return true; } ApacheChart.prototype._displayStatusHtml = function(svgEmbed) { var errorTextNode = this._errorTextNode = document.createElement("span"); errorTextNode.innerHTML = this._statusHtml; svgEmbed.parentNode.insertBefore(errorTextNode, svgEmbed); svgEmbed.style.display = "none"; } ApacheChart.prototype._displayErrorHtml = function(svgEmbed) { if(this._errorTextNode) { this._errorTextNode.innerHTML = this._errorHtml; return; } else { var errorTextNode = this._errorTextNode = document.createElement("span"); errorTextNode.innerHTML = this._errorHtml; svgEmbed.parentNode.insertBefore(errorTextNode, svgEmbed); } svgEmbed.style.display = "none"; } ApacheChart.prototype.DrawChartData = function() { // no default implementation. Subclasses have to override this } ApacheChart.prototype.Animate = function() { var animateDuration = this._animDuration; if(animateDuration > 0) { if(this._animCallback == null) this._animCallback = TrUIUtils.createCallback(this, this.DoAnimation); this._startTime = (new Date()).getTime(); this._intervalId = window.setInterval(this._animCallback, ApacheChart._ANIMATE_INTERVAL); } } ApacheChart.prototype.DoAnimation = function() { var animateDuration = this._animDuration; var diffTime = (new Date()).getTime() - this._startTime; if(diffTime >= animateDuration) { window.clearInterval(this._intervalId); this.SetDataAnimStep(1); this.SetLabelsAnimStep(1); this.SetGridAnimStep(1); // we do not need the elements any more. delete this._dataElems; delete this._labelElems; delete this._gridElems; } else { var ratio = (diffTime)/animateDuration; this.SetDataAnimStep(ratio); this.SetLabelsAnimStep(ratio); this.SetGridAnimStep(ratio); } } ApacheChart.prototype.SetDataAnimStep = function(ratio) { var animElems = this._dataElems, animCount = animElems.length; var margins = this._margins, animHorizontal = this.AnimAlongXAxis(); // Default implementation is to make the elements appear from x axis or y axis if(animHorizontal) { var marginLeft = margins.left; for(var i = 0; i < animCount; ++i) { var tx = (1-ratio)*marginLeft; animElems[i].setAttribute("transform", "translate("+tx+",0) scale("+ratio+",1)"); } } else { var marginBottom = margins.bottom, cy = (this._height - marginBottom); for(var i = 0; i < animCount; ++i) { var ty = (1-ratio)*cy; animElems[i].setAttribute("transform", "translate(0,"+ty+") scale(1,"+ratio+")"); } } } ApacheChart.prototype.AnimAlongXAxis = function(ratio) { return false; } ApacheChart.prototype.SetLabelsAnimStep = function(ratio) { var animElems = this._labelElems, animCount = animElems.length; // Default implementation is to make the elements fade in for(var i = 0; i < animCount; ++i) { animElems[i].setAttribute("fill-opacity", ratio); } } ApacheChart.prototype.SetGridAnimStep = function(ratio) { var animElems = this._gridElems, animCount = animElems.length; var margins = this._margins, animHorizontal = this.AnimAlongXAxis(); // Default implementation is to make the grid appear along the x axis or y axis if(animHorizontal) { var marginBottom = margins.bottom, cy = (this._height - marginBottom); // reverse the animation for horizontal chart for(var i = 0; i < animCount; ++i) { var ty = (1-ratio)*cy; animElems[i].setAttribute("transform", "translate(0,"+ty+") scale(1,"+ratio+")"); } } else { var marginLeft = margins.left; for(var i = 0; i < animCount; ++i) { var tx = (1-ratio)*marginLeft; animElems[i].setAttribute("transform", "translate("+tx+",0) scale("+ratio+",1)"); } } } ApacheChart.prototype.InitializeGradients = function() { var svgDoc = this._svgDoc, model = this._model, seriesColors = model.getSeriesColors(), seriesCount = model.getSeriesLabels().length; var gradients = svgDoc.getElementById("gradients"); ApacheChartObj.Assert(gradients, "No Gradients element in the SVG document"); var gradientElements = gradients.childNodes; ApacheChartObj.Assert(gradients.childNodes.length>1, "No Gradient Template in the SVG document"); var gradientElement, gradientTemplate = null; for (var i = 0; i< seriesCount; ++i) { gradientElement = svgDoc.getElementById("gradient"+i); if(gradientElement == null) { if(gradientTemplate == null) { gradientTemplate = gradients.firstChild; while(gradientTemplate.nodeType == 3 && gradientTemplate != null) gradientTemplate = gradientTemplate.nextSibling; } gradientElement = gradientTemplate.cloneNode(true); gradientElement.id = "gradient"+i; gradients.appendChild(gradientElement); } var childNode = gradientElement.firstChild; var stopIndex = 0; while (childNode) { if (childNode.nodeName == "stop") { var color = seriesColors[i]; color = (stopIndex == 0)?color:this._getLighterColor(color); childNode.setAttribute("stop-color",color); this.SetStopOpacity(childNode); if(stopIndex>=1) break; stopIndex++; } childNode = childNode.nextSibling; } } } ApacheChart.prototype.SetStopOpacity = function(stopNode) { // no default implementation stopNode.setAttribute("stop-opacity", ApacheChart._DEFAULT_STOP_OPACITY); } ApacheChart.prototype._getLighterColor = function(color) { if(color.indexOf("#") >=0 ) { color = color.substr(1); var rVal = color.substr(0,2), gVal = color.substr(2,2), bVal = color.substr(4); color = "#"+this._getLighterNumberStr(rVal)+this._getLighterNumberStr(gVal)+ this._getLighterNumberStr(bVal); } else { color = color.toLowerCase().replace(" ", ""); color = color.substring(4, color.length-1); var arr = color.split(","); color = "#"+this._getLighterNumberStr(arr[0])+this._getLighterNumberStr(arr[1])+ this._getLighterNumberStr(arr[2]); } return color; } ApacheChart.prototype._getLighterNumberStr = function(valStr) { var val = Math.round(parseInt(valStr, 16)*1.7); if(val>255) val = 255; return ApacheChart._to_hex(val); } ApacheChart._to_hex = function(n) { var digit_array = ApacheChart._digit_array; if(digit_array == null) { digit_array = ApacheChart._digit_array = ['0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f']; } var hex_result='' var the_start=true; for(var i=32;i>0;) { i-=4; var one_digit=(n>>i)&0xf; if(!the_start||one_digit!=0) { the_start=false; hex_result+=digit_array[one_digit]; } } return ''+(hex_result==''?'0':hex_result); } ApacheChart.prototype.DrawBorder = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement; var rectElem = svgDoc.getElementById("borderPrototype").cloneNode(false); var borderSize = ApacheChart._BORDER_SIZE, stroke = borderSize/2; rectElem.setAttribute("x", 0); rectElem.setAttribute("y", 0); rectElem.setAttribute("rx", stroke); rectElem.setAttribute("ry", stroke); rectElem.setAttribute("width", this._width-stroke); rectElem.setAttribute("height", this._height-stroke); rectElem.setAttribute("stroke-width", stroke); rootElem.appendChild(rectElem); var margins = this._margins; margins.left += borderSize; margins.right += borderSize; margins.top += borderSize; margins.bottom += borderSize; } ApacheChart.prototype.DrawTitles = function() { var model = this._model, title = model.getTitle(), subTitle = model.getSubTitle(), footNote = model.getFootNote(); if(title) this._drawTitleElem("titleTextPrototype", title, false); if(subTitle) this._drawTitleElem("subTitleTextPrototype", subTitle, false); if(footNote) this._drawTitleElem("footNoteTextPrototype", footNote, true); } ApacheChart.prototype._drawTitleElem = function(template, title, isFooter) { var svgDoc = this._svgDoc, rootElem = this._rootElement; var margins = this._margins, gridWidth = (this._width - margins.left - margins.right); var labelElems = this._labelElems, animate = (this._animDuration>0); var textElem = svgDoc.getElementById(template).cloneNode(true); if(animate) { labelElems.push(textElem); textElem.setAttribute("fill-opacity","0"); } textElem.firstChild.data = title; rootElem.appendChild(textElem); var textBBox = textElem.getBBox(), textWidth = textBBox.width, dx=margins.left; if(isFooter && this._width > textWidth + margins.right) dx = (this._width-textWidth)-margins.right; if(!isFooter && gridWidth > textWidth) dx = (gridWidth-textWidth)/2+margins.left; textElem.setAttribute("x",dx); if(isFooter) { textElem.setAttribute("y",this._height-margins.bottom); margins.bottom += textBBox.height+ApacheChart._TEXT_MARGIN; } else { margins.top += textBBox.height; textElem.setAttribute("y",margins.top); margins.top += ApacheChart._TEXT_MARGIN; } } ApacheChart.prototype.DrawGroupLabels = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var container = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); this._hLabelContainer = container; var groupLabels = model.getGroupLabels(), vLineCount = groupLabels.length; var labelElem, labelElems = this._labelElems, animate = (this._animDuration>0); var labelText, gLabelElems = this._groupLabelElems; for (var i = 0; i< vLineCount; ++i) { // draw the horizontal label if(i==0) { labelElem = svgDoc.getElementById("groupLabelPrototype"); } labelText = groupLabels[i]; if(!labelText) continue; labelElem = labelElem.cloneNode(true); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } labelElem.firstChild.data = labelText; container.appendChild(labelElem); gLabelElems[i] = labelElem; } rootElem.appendChild(container); } ApacheChart.prototype.LayoutGroupLabels = function() { var model = this._model, margins = this._margins, marginLeft = margins.left; var container = this._hLabelContainer, childNodes = container.childNodes; if(childNodes.length == 0) return; if(this._isPerspective) marginLeft += ApacheChart._XOFFSET_PERSPECTIVE; var gridWidth = (this._width - marginLeft - margins.right); var isCenterAligned = this.IsGroupLabelCentered(); var groupLabels = model.getGroupLabels(), vLineCount = groupLabels.length; var labelElem, groupWidth = gridWidth/(isCenterAligned?vLineCount:vLineCount-1); var dx = 0, dy = this._height - margins.bottom+container.getBBox().height+ApacheChart._TEXT_MARGIN; var gLabelElems = this._groupLabelElems; for (var i = 0; i< vLineCount; ++i) { labelElem = gLabelElems[i]; if(!labelElem) continue; labelElem.setAttribute("y", dy); var textWidth = labelElem.getBBox().width; if(isCenterAligned) { if(groupWidth > textWidth) dx = (groupWidth-textWidth)/2; else dx = 2; } else { dx = (-textWidth)/2; if(this._isPerspective) dx -= ApacheChart._XOFFSET_PERSPECTIVE; } labelElem.setAttribute("x", marginLeft+dx+i*groupWidth); } } /** * Indicates if the group label should be center aligned or edge aligned * @return true(String) indicates center aligned, false indicates it is edge aligned */ ApacheChart.prototype.IsGroupLabelCentered = function() { return true; } ApacheChart.prototype.AdjustMarginsForGroupLabels = function() { var container = this._hLabelContainer; if(container && container.childNodes.length > 0) { this._margins.bottom += container.getBBox().height+ApacheChart._TEXT_MARGIN; var isCentered = this.IsGroupLabelCentered(); if(!isCentered) { var textWidth = container.lastChild.getBBox().width; if(textWidth/2> this._margins.right) this._margins.right = textWidth/2; } } } ApacheChart.prototype.DrawLegend = function() { var legendPosition = this._legendPosition; if(legendPosition == ApacheChart.LEGEND_LOCATION_NONE) { return; } var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var gradientsUsed = this._gradientsUsed; var seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length, seriesColors = model.getSeriesColors(); var labelElem, rectElem, legendRectHeight, legendGroup = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var labelElems = this._labelElems, animate = (this._animDuration>0); rootElem.appendChild(legendGroup); if(this._isPerspective) { marginLeft += ApacheChart._XOFFSET_PERSPECTIVE; } var gridWidth = (this._width - marginLeft - margins.right), gridHeight = (this._height - marginTop - margins.bottom); if(animate) { labelElems.push(legendGroup); legendGroup.setAttribute("fill-opacity","0"); } var dx = 0, dy = 0, tx = marginLeft, ty = this._height - margins.bottom; var drawSideWays = (legendPosition == ApacheChart.LEGEND_LOCATION_START || legendPosition == ApacheChart.LEGEND_LOCATION_END) for (var i = 0; i < seriesCount; ++i) { if(i == 0) { labelElem = svgDoc.getElementById("legendTextPrototype"); rectElem = svgDoc.getElementById("legendRectPrototype"); legendRectHeight = parseInt(rectElem.getAttribute("height")); } if(drawSideWays) dx = 0; rectElem = rectElem.cloneNode(false); rectElem.setAttribute("x", dx); rectElem.setAttribute("y", dy-legendRectHeight); if(gradientsUsed) rectElem.setAttribute("fill", "url(#gradient"+i+")"); else rectElem.setAttribute("fill", seriesColors[i]); rectElem.setAttribute("stroke", "#000000"); // TODO: Legend elements should fire on click event //rectElem.setAttribute("onclick", "parent."+onclickStrings[i]); legendGroup.appendChild(rectElem); dx += 1.5*legendRectHeight; labelElem = labelElem.cloneNode(true); labelElem.setAttribute("x", dx); labelElem.setAttribute("y", dy); labelElem.firstChild.data = seriesLabels[i]; legendGroup.appendChild(labelElem); // TODO: Legend elements should fire on click event //labelElem.setAttribute("onclick", "parent."+onclickStrings[i]); if(!drawSideWays) dx += labelElem.getBBox().width+legendRectHeight; else dy += 1.5*legendRectHeight; if(i == 0 && !drawSideWays) { var rect = labelElem.getBBox(); if(legendPosition == ApacheChart.LEGEND_LOCATION_TOP) { ty = this.SetLegendTopAdjustment(margins.top+rect.height); margins.top += rect.height+ApacheChart._TEXT_MARGIN; } else { ty = this.SetLegendBottomAdjustment(ty); margins.bottom += rect.height+ApacheChart._TEXT_MARGIN; } } } if(!drawSideWays && gridWidth > dx) tx = (gridWidth-dx)/2+marginLeft; if(drawSideWays) { var lBBox = legendGroup.getBBox(); if(legendPosition == ApacheChart.LEGEND_LOCATION_START) { tx = this.SetLegendLeftAdjustment(margins.left); margins.left += lBBox.width+ApacheChart._TEXT_MARGIN; } else { margins.right += lBBox.width+ApacheChart._TEXT_MARGIN; tx = this._width - margins.right + ApacheChart._TEXT_MARGIN; tx = this.SetLegendRightAdjustment(tx); } if(gridHeight > dy) ty = (gridHeight-lBBox.height)/2+marginTop; else ty = gridHeight+marginTop-lBBox.height; } legendGroup.setAttribute("transform", "translate("+tx+","+ty+")"); } /** * Adjusts the legend location when it is at the top * @param ty(int) the original y location of the legend */ ApacheChart.prototype.SetLegendTopAdjustment = function(ty) { // By default we need not adjust anything return ty; } /** * Adjusts the legend location when it is at the bottom * @param ty(int) the original y location of the legend */ ApacheChart.prototype.SetLegendBottomAdjustment = function(ty) { var container = this._hLabelContainer; if(container && container.childNodes.length > 0) { ty += container.getBBox().height+ApacheChart._TEXT_MARGIN; } return ty; } /** * Adjusts the legend location when it is at the Left * @param tx(int) the original x location of the legend */ ApacheChart.prototype.SetLegendLeftAdjustment = function(tx) { var container = this._vLabelContainer; if(container) { tx -= container.getBBox().width+ApacheChart._TEXT_MARGIN; } return tx; } /** * Adjusts the legend location when it is at the Right * @param tx{int} the original x location of the legend */ ApacheChart.prototype.SetLegendRightAdjustment = function(tx) { // By default we need not adjust anything return tx; } ApacheChart.prototype.DrawGrid = function() { if(this._isPerspective) this.DrawPerspectiveGrid(); else this.Draw2DGrid(); } ApacheChart.prototype.Draw2DGrid = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var gridElems = this._gridElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var gradientsUsed = this._gradientsUsed; var rectElem = svgDoc.getElementById("gridRectPrototype").cloneNode(false); rectElem.setAttribute("x", margins.left); rectElem.setAttribute("y", (marginTop)); rectElem.setAttribute("width", gridWidth); rectElem.setAttribute("height", gridHeight); if(gradientsUsed) rectElem.setAttribute("fill", "url(#gridGradient)"); this._rootElement.appendChild(rectElem); var pathElem = svgDoc.getElementById("gridPathPrototype").cloneNode(false); if(animate) { gridElems.push(pathElem); pathElem.setAttribute("transform", "scale(0.00001,1)"); } var sb = new ApacheChartBuffer(), vLineCount = this.GetVLineCount(), hLineCount = this.GetHLineCount(); // horizontal lines for (var i = 0; i< hLineCount-1; ++i) { sb.append("M").append(marginLeft).append(",").append((i+1)*gridHeight/hLineCount+marginTop); sb.append("h").append(gridWidth); } // vertical lines for (var i = 0; i< vLineCount-1; ++i) { sb.append("M").append(marginLeft+((i+1)*gridWidth/vLineCount)).append(",").append(marginTop); sb.append("v").append(gridHeight); } pathElem.setAttribute("d", sb.toString()); pathElem.removeAttribute("id"); this._rootElement.appendChild(pathElem); } ApacheChart.prototype.GetVLineCount = function() { var xMajorCount = this._xMajorGridCount; if(xMajorCount >= 0) return xMajorCount; else return this._model.getGroupLabels().length; } ApacheChart.prototype.GetHLineCount = function() { return this._yMajorGridCount; } ApacheChart.prototype.DrawPerspectiveGrid = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var gridElems = this._gridElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var rectElem = svgDoc.getElementById("gridRectPrototype").cloneNode(false); var gradientsUsed = this._gradientsUsed; rectElem.setAttribute("x", marginLeft+ApacheChart._XOFFSET_PERSPECTIVE); rectElem.setAttribute("y", marginTop); rectElem.setAttribute("width", (gridWidth)); rectElem.setAttribute("height", (gridHeight)); if(gradientsUsed) rectElem.setAttribute("fill", "url(#gridGradient)"); rectElem.removeAttribute("id"); this._rootElement.appendChild(rectElem); var sb = new ApacheChartBuffer(); var pathElem = svgDoc.getElementById("gridPath3dRectPrototype").cloneNode(false); sb.append("M").append(marginLeft+xOffset).append(",").append(marginTop); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("v").append(gridHeight); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("m").append(gridWidth).append(",").append(0); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("h").append(-gridWidth); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gridGradient)"); pathElem.setAttribute("d", sb.toString()); pathElem.removeAttribute("id"); this._rootElement.appendChild(pathElem); pathElem = svgDoc.getElementById("gridPathPrototype").cloneNode(false); if(animate) { pathElem.setAttribute("transform", "scale(0.00001,1)"); gridElems.push(pathElem); } var vLineCount = this.GetVLineCount(), hLineCount = this.GetHLineCount(); sb = new ApacheChartBuffer(); // horizontal lines for (var i = 0; i< hLineCount-1; ++i) { sb.append("M").append(marginLeft).append(",").append((i+1)*gridHeight/hLineCount+marginTop+yOffset); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("h").append(gridWidth); } // vertical lines for (var i = 0; i< vLineCount-1; ++i) { sb.append("M").append(marginLeft+xOffset+((i+1)*gridWidth/vLineCount)).append(",").append(marginTop); sb.append("v").append(gridHeight); sb.append("l").append(-xOffset).append(",").append(yOffset); } pathElem.setAttribute("d", sb.toString()); this._rootElement.appendChild(pathElem); } ApacheChart.prototype.DrawYValueLabels = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var container = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); this._vLabelContainer = container; var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var labelElems = this._labelElems, animate = (this._animDuration>0); var labelElem = svgDoc.getElementById("yLabelPrototype").cloneNode(true); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } labelElem.firstChild.data = this._formatValue(minValue); container.appendChild(labelElem); labelElem = labelElem.cloneNode(true); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } labelElem.firstChild.data = this._formatValue(maxValue); container.appendChild(labelElem); var hLineCount = this._yMajorGridCount; // horizontal lines for (var i = 0; i< hLineCount-1; ++i) { var value = ((maxValue-minValue)*(i+1)/hLineCount) + minValue; labelElem = labelElem.cloneNode(true); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } labelElem.firstChild.data = this._formatValue(value); container.appendChild(labelElem); } rootElem.appendChild(container); } ApacheChart.prototype._formatValue = function(value) { // Initialize the decimal separtor var decimalSep = this._decimalSep; if(decimalSep == null) { var symbols = window.getLocaleSymbols?getLocaleSymbols():null; if (symbols) { this._decimalSep = symbols.getDecimalSeparator(); } else this._decimalSep = "."; decimalSep = this._decimalSep; } value = value.toFixed(this._maxPrecision); value = value.toString(); if(value.indexOf(decimalSep) == -1) { value = value.replace(".", decimalSep); } return value; } ApacheChart.prototype.AdjustMarginsForYLabels = function() { var container = this._vLabelContainer; if(container && container.childNodes.length > 0) this._margins.left += container.getBBox().width+ApacheChart._TEXT_MARGIN; } ApacheChart.prototype.LayoutYValueLabels = function() { var model = this._model, margins = this._margins; var marginLeft = margins.left, marginTop = margins.top; var container = this._vLabelContainer, childNodes = container.childNodes; var gridHeight = (this._height - marginTop - margins.bottom); if(this._isPerspective) gridHeight -= ApacheChart._YOFFSET_PERSPECTIVE; var bBox = container.getBBox(), textHeight = bBox.height; this.SetVerticalLabelAt(childNodes.item(0), gridHeight+marginTop, marginLeft, textHeight); this.SetVerticalLabelAt(childNodes.item(1), marginTop, marginLeft, textHeight); var hLineCount = this._yMajorGridCount; // horizontal lines for (var i = 0; i< hLineCount-1; ++i) { this.SetVerticalLabelAt(childNodes.item(i+2), (hLineCount -i -1)*gridHeight/hLineCount+marginTop, marginLeft, textHeight); } } ApacheChart.prototype.SetVerticalLabelAt = function( labelElem, y, marginLeft, textHeight) { if(this._isPerspective) y += ApacheChart._YOFFSET_PERSPECTIVE; // readjust to right align var labelMargin = ApacheChart._TEXT_MARGIN, textLength = labelElem.getBBox().width, dx = labelMargin; if(marginLeft>textLength+labelMargin) dx = marginLeft-textLength-labelMargin; labelElem.setAttribute("x", dx); labelElem.setAttribute("y", y+textHeight/2); } ApacheChart.prototype.DrawGroupLabelTitle = function( label, container, labelElem, dx, dy, quadWidth, quadHeight) { if(!label) return quadHeight; var labelElems = this._labelElems, animate = (this._animDuration>0); labelElem.setAttribute("y", dy+quadHeight); labelElem.firstChild.data = label; container.appendChild(labelElem); var rect = labelElem.getBBox(); var textWidth = rect.width; if(quadWidth > textWidth) dx += (quadWidth-textWidth)/2; else dx += 2; labelElem.setAttribute("x", dx); if(animate) labelElems.push(labelElem); quadHeight -= rect.height+ApacheChart._TEXT_MARGIN; return quadHeight; } ApacheChart.prototype.ShowToolTip = function(e) { if (this._toolTipVisible) return; var model = this._model, seriesColors = model.getSeriesColors(); var toolTip = this.getToolTip(); if (toolTip == null) { toolTip = this._svgDoc.getElementById("toolTip").cloneNode(true); this.setToolTip(toolTip); this._rootElement.appendChild(toolTip); } toolTip.style.setProperty("visibility","visible",""); var circleElem = toolTip.firstChild.nextSibling; var boundingRectElem = circleElem.nextSibling.nextSibling; this.FillToolTipData(boundingRectElem, circleElem, e); var ttBBox = toolTip.getBBox(); var pt = this.GetToolTipLocation(e, ttBBox); var dx = pt.x, dy = pt.y; if(dx + ttBBox.width > this._width) { dx -= ttBBox.width; circleElem.setAttribute("cx",boundingRectElem.getBBox().width); } else { circleElem.setAttribute("cx",0); } if(dy - ttBBox.height < 0) { dy += ttBBox.height; circleElem.setAttribute("cy",0); } else { circleElem.setAttribute("cy",boundingRectElem.getBBox().height); } if(this._isPerspective && this._type != ApacheChart.TYPE_PIE) dy += ApacheChart._YOFFSET_PERSPECTIVE/2 toolTip.setAttribute("transform","translate("+dx+","+dy+")"); this._toolTipVisible = true; } ApacheChart.prototype.GetToolTipLocation = function(e, ttBBox) { var targetBBox = e.target.getBBox(); return {x:(targetBBox.x+targetBBox.width/2), y:(targetBBox.y - ttBBox.height)}; } ApacheChart.prototype.GetChartEvent = function(e) { var evtTarget = e.target; var i = parseInt(evtTarget.getAttribute("yValueIndex")), j = parseInt(evtTarget.getAttribute("seriesIndex")); var model = this._model, yValues = model.getYValues(); return new ApacheChartEvent([j],[i], [yValues[i][j]],null); } ApacheChart.prototype.FillToolTipData = function(boundingRectElem, circleElem, e) { var chartEvent = this.GetChartEvent(e); var j = chartEvent.getSeriesIndices()[0]; var model = this._model, groupLabels = model.getGroupLabels(), seriesLabels = model.getSeriesLabels(), yValues = chartEvent.getYValues(); //top label var textElem = boundingRectElem.nextSibling.nextSibling; textElem.firstChild.data = seriesLabels[j]; var labelWidth = textElem.getBBox().width; //actual value textElem = textElem.nextSibling.nextSibling; textElem.firstChild.data = this._formatValue(yValues[0]); var dataWidth = textElem.getBBox().width; // leave a clearance on either end of the text var xMargin = ApacheChart._TEXT_MARGIN, dx = xMargin; if (labelWidth > dataWidth) dx = (labelWidth-dataWidth)/2+xMargin; textElem.setAttribute("x",dx); var rectWidth = Math.max(labelWidth,dataWidth)+2*xMargin; boundingRectElem.setAttribute("width",rectWidth); boundingRectElem.setAttribute("stroke", seriesColors[j]); circleElem.setAttribute("stroke",seriesColors[j]); } ApacheChart.prototype.HideToolTip = function(e) { var toolTip = this.getToolTip(); if(toolTip) toolTip.style.setProperty("visibility","hidden",""); this._toolTipVisible = false; } ApacheChart.prototype.Click = function(e) { var chartEvent = this.GetChartEvent(e); var formName = this._formName; if(formName !=null) { var svgEmbed = document.getElementById(this._svgEmbedId); var sourceId = svgEmbed.parentNode.id; var chartValue ={ 'event': 'chartDrillDown', 'source':sourceId, 'value':chartEvent.marshall()}; if(this._partialSubmit) { _submitPartialChange(formName,'0',chartValue); } else { submitForm(formName,'0',chartValue); } } else alert(chartEvent); } //////////////////////////////////////////////////////////////////// // Bar Chart subclass //////////////////////////////////////////////////////////////////// function ApacheBarChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheBarChart); ApacheBarChart.prototype.DrawChartData = function() { var isCombo = this._type == ApacheChart.TYPE_BAR_LINE_COMBO; var isPerspective = this._isPerspective; if(isPerspective) this._drawPerspectiveBars(isCombo); else this._drawBars(isCombo); // delegate to the line chart for combos if(isCombo) { if(isPerspective) this.__drawPerspectiveLines = ApacheLineChart.prototype.__drawPerspectiveLines; else this.__drawLines = ApacheLineChart.prototype.__drawLines; ApacheLineChart.prototype.DrawChartData.call(this, isCombo); } } ApacheBarChart.prototype._drawBars = function(isCombo) { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var rectElem = svgDoc.getElementById("barRectPrototype"); var barItemPadding = ApacheBarChart._BARITEM_PADDING; var isStacked = (this._type == ApacheChart.TYPE_VBAR_STACKED); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var barDivider = isStacked?1:(isCombo?Math.ceil(seriesCount/2): seriesCount); var yValueCount = yValues.length; var barWidth = (gridWidth/Math.max(yValueCount,groupCount)-2*barItemPadding)/barDivider; var dx = marginLeft, dy, barHeight, stackBase = minValue; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(1,0.00001)"; for (var i = 0; i< yValueCount; ++i) { dx += barItemPadding; dy = gridHeight + marginTop; for (var j = 0; j < seriesCount; ++j) { // for combo charts we draw every alternate(even) bar. if(isCombo && j%2>0) continue; // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. if(isStacked) stackBase = (j==0?minValue:0); rectElem = rectElem.cloneNode(false); if(animate) { dataElems.push(rectElem); rectElem.setAttribute("transform",defaultTransform); } rectElem.setAttribute("x", dx); barHeight = gridHeight*(yValues[i][j]-stackBase)/(maxValue-minValue); if(isStacked) dy -= barHeight; else dy = gridHeight + marginTop - barHeight; rectElem.setAttribute("y", dy); rectElem.setAttribute("width", barWidth); rectElem.setAttribute("height", barHeight); if(gradientsUsed) rectElem.setAttribute("fill", "url(#gradient"+j+")"); else rectElem.setAttribute("fill", seriesColors[j]); rectElem.setAttribute("stroke", seriesColors[j]); rectElem.setAttribute("stroke-width", 1); rectElem.setAttribute("yValueIndex", i); rectElem.setAttribute("seriesIndex", j); if(this._tooltipsVisible) { rectElem.addEventListener("mouseover",this.ShowToolTipCallback,false); rectElem.addEventListener("mouseout",this.HideToolTipCallback,false); } rectElem.addEventListener("click",this.ClickCallback,false); rootElem.appendChild(rectElem); if(!isStacked) dx += barWidth; } if(isStacked) dx += barWidth; dx += barItemPadding; } } ApacheBarChart.prototype._drawPerspectiveBars = function(isCombo) { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var pathElem = svgDoc.getElementById("barPathPrototype"); var barItemPadding = ApacheBarChart._BARITEM_PADDING; var isStacked = (this._type == ApacheChart.TYPE_VBAR_STACKED); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var seriesBars = isCombo?Math.ceil(seriesCount/2): seriesCount, barWidth; var yValueCount = yValues.length; if(isStacked) barWidth = gridWidth/Math.max(yValueCount,groupCount)-2*barItemPadding; else barWidth = (gridWidth/Math.max(yValueCount,groupCount) -2*barItemPadding - (seriesBars)*barItemPadding)/seriesBars; var dx = marginLeft, dy, barHeight, stackBase = minValue; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(1, 0.00001)"; for (var i = 0; i< yValueCount; ++i) { dx += barItemPadding; dy = gridHeight + marginTop+yOffset; for (var j = 0; j < seriesCount; ++j) { // for combo charts we draw every alternate(even) bar. if(isCombo && j%2>0) continue; // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. if(isStacked) stackBase = (j==0?minValue:0); barHeight = gridHeight*(yValues[i][j]-stackBase)/(maxValue-minValue); if(isStacked) dy -= barHeight; else dy = gridHeight + yOffset + marginTop - barHeight; pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } var sb = new ApacheChartBuffer(); sb.append("M").append(dx).append(",").append(dy); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("h").append(barWidth); sb.append("v").append(barHeight); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("v").append(-barHeight); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("h").append(-barWidth); sb.append("v").append(barHeight); sb.append("h").append(barWidth); sb.append("v").append(-barHeight); pathElem.setAttribute("stroke", seriesColors[j]); pathElem.setAttribute("stroke-width", 1); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+j+")"); else pathElem.setAttribute("fill", seriesColors[j]); pathElem.setAttribute("d", sb.toString()); pathElem.setAttribute("yValueIndex", i); pathElem.setAttribute("seriesIndex", j); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); rootElem.appendChild(pathElem); if(!isStacked) { dx += barWidth; dx += barItemPadding; } } if(isStacked) dx += barWidth; dx += barItemPadding; } } ApacheBarChart.prototype.ShowToolTip = function(e) { if(this._type == ApacheChart.TYPE_BAR_LINE_COMBO) { var i = parseInt(e.target.getAttribute("seriesIndex")); if(i%2>0) { try { // Maybe we need a generic framework for combos so that we can delegate. // Till that time... this.GetToolTipLocation = ApacheLineChart.prototype.GetToolTipLocation; this.FillToolTipData = ApacheLineChart.prototype.FillToolTipData; this.GetChartEvent = ApacheLineChart.prototype.GetChartEvent; ApacheLineChart.prototype.ShowToolTip.call(this, e); } finally { // restore this.GetToolTipLocation = ApacheBarChart.prototype.GetToolTipLocation; this.FillToolTipData = ApacheBarChart.prototype.FillToolTipData; this.GetChartEvent = ApacheBarChart.prototype.GetChartEvent; } return; } } ApacheBarChart.superclass.ShowToolTip.call(this, e); } // number of pixels on either side of the bar item ApacheBarChart._BARITEM_PADDING = 2; //////////////////////////////////////////////////////////////////// // Bar chart subclass //////////////////////////////////////////////////////////////////// function ApacheHBarChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheHBarChart); ApacheHBarChart.prototype.DrawChartData = function() { if(this._isPerspective) this._drawPerspectiveBars(); else this._drawBars(); } ApacheHBarChart.prototype.AnimAlongXAxis = function() { // horizontal bar animates around x axis return true; } ApacheHBarChart.prototype.DrawYValueLabels = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var container = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); // Since the horizontal bar chart is flipped Y labels are horizontal this._vLabelContainer = container; var labelElem = svgDoc.getElementById("groupLabelPrototype"); var labelElems = this._labelElems, animate = (this._animDuration>0); var groupLabels = model.getGroupLabels(), hLineCount = groupLabels.length; var labelText, gLabelElems = this._groupLabelElems; // horizontal lines for (var i = 0; i< hLineCount; ++i) { labelText = groupLabels[i]; if(!labelText) continue; labelElem = labelElem.cloneNode(true); labelElem.firstChild.data = labelText; container.appendChild(labelElem); gLabelElems[i] = labelElem; if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } } rootElem.appendChild(container); } ApacheHBarChart.prototype.LayoutYValueLabels = function() { var model = this._model, margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var gridHeight = (this._height - marginTop - margins.bottom); if(this._isPerspective) gridHeight -= ApacheChart._YOFFSET_PERSPECTIVE; var container = this._vLabelContainer, childNodes = container.childNodes; if(childNodes.length == 0) return; var labelElem, bBox = container.getBBox(), textHeight = bBox.height; var groupLabels = model.getGroupLabels(), hLineCount = groupLabels.length; var gLabelElems = this._groupLabelElems; // horizontal lines for (var i = 0; i< hLineCount; ++i) { labelElem = gLabelElems[i]; if(!labelElem) continue; this.SetVerticalLabelAt(labelElem, (hLineCount -i)*gridHeight/hLineCount+marginTop-(gridHeight/(2*hLineCount)), marginLeft, textHeight); } } ApacheHBarChart.prototype.IsGroupLabelCentered = function() { return false; } ApacheHBarChart.prototype.DrawGroupLabels = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var vLineCount = this._yMajorGridCount; var container = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); // Since the horizontal bar chart is flipped group labels are vertical this._hLabelContainer = container; var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var labelElem = svgDoc.getElementById("yLabelPrototype"); var value, labelElems = this._labelElems, animate = (this._animDuration>0); for (var i = 0; i< vLineCount+1; ++i) { // draw the horizontal label labelElem = labelElem.cloneNode(true); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } if(i==0) value = minValue; else if(i==vLineCount) value = maxValue; else value = (((maxValue-minValue)*(i)/vLineCount) + minValue); labelElem.firstChild.data = this._formatValue(value); container.appendChild(labelElem); } rootElem.appendChild(container); } ApacheHBarChart.prototype.LayoutGroupLabels = function() { var model = this._model, margins = this._margins, marginLeft = margins.left; var gridWidth = (this._width - marginLeft - margins.right); var container = this._hLabelContainer, childNodes = container.childNodes; if(this._isPerspective) gridWidth -= ApacheChart._XOFFSET_PERSPECTIVE; var vLineCount = this._yMajorGridCount; var labelElem, yValWidth = gridWidth/vLineCount; var bBox = container.getBBox(); var dx = 0, dy = this._height - margins.bottom + bBox.height+ApacheChart._TEXT_MARGIN; var labelElems = this._labelElems, animate = (this._animDuration>0); for (var i = 0; i< vLineCount+1; ++i) { // draw the horizontal label labelElem = childNodes.item(i); labelElem.setAttribute("y", dy); var textWidth = labelElem.getBBox().width; labelElem.setAttribute("x", marginLeft-textWidth/2+i*yValWidth); } } ApacheHBarChart.prototype.GetVLineCount = function() { return this._yMajorGridCount; } ApacheHBarChart.prototype.GetHLineCount = function() { var xMajorCount = this._xMajorGridCount; if(xMajorCount >= 0) return xMajorCount; else return this._model.getGroupLabels().length; } ApacheHBarChart.prototype._drawBars = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var rectElem = svgDoc.getElementById("barRectPrototype"); var barItemPadding = ApacheBarChart._BARITEM_PADDING; var isStacked = (this._type == ApacheChart.TYPE_HBAR_STACKED); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var barDivider = isStacked?1:seriesCount, stackBase = minValue; var yValueCount = yValues.length; var barHeight = (gridHeight/Math.max(yValueCount,groupCount)-2*barItemPadding)/barDivider; var dx = marginLeft, dy=gridHeight+marginTop, barWidth; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(0.00001,1)"; for (var i = 0; i< yValueCount; ++i) { dy -= barItemPadding; dx = marginLeft; for (var j = 0; j < seriesCount; ++j) { // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. if(isStacked) stackBase = (j==0?minValue:0); rectElem = rectElem.cloneNode(false); if(animate) { dataElems.push(rectElem); rectElem.setAttribute("transform",defaultTransform); } rectElem.setAttribute("x", dx); barWidth = gridWidth*(yValues[i][j]-stackBase)/(maxValue-minValue); if(isStacked) dx += barWidth; rectElem.setAttribute("y", dy-barHeight); rectElem.setAttribute("width", barWidth); rectElem.setAttribute("height", barHeight); if(gradientsUsed) rectElem.setAttribute("fill", "url(#gradient"+j+")"); else rectElem.setAttribute("fill", seriesColors[j]); rectElem.setAttribute("stroke", seriesColors[j]); rectElem.setAttribute("stroke-width", 1); rectElem.setAttribute("yValueIndex", i); rectElem.setAttribute("seriesIndex", j); if(this._tooltipsVisible) { rectElem.addEventListener("mouseover",this.ShowToolTipCallback,false); rectElem.addEventListener("mouseout",this.HideToolTipCallback,false); } rectElem.addEventListener("click",this.ClickCallback,false); rootElem.appendChild(rectElem); if(!isStacked) dy -= barHeight; } if(isStacked) dy -= barHeight; dy -= barItemPadding; } } ApacheHBarChart.prototype._drawPerspectiveBars = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var pathElem = svgDoc.getElementById("barPathPrototype"); var barItemPadding = ApacheBarChart._BARITEM_PADDING; var isStacked = (this._type == ApacheChart.TYPE_HBAR_STACKED); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var yValueCount = yValues.length; var barHeight, stackBase = minValue; if(isStacked) barHeight = gridHeight/Math.max(yValueCount,groupCount)-2*barItemPadding; else barHeight = (gridHeight/Math.max(yValueCount,groupCount)-2*barItemPadding - (seriesCount)*barItemPadding)/seriesCount; var dx = marginLeft, dy=gridHeight+marginTop+yOffset, barWidth; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(0.00001,1)"; for (var i = 0; i< yValueCount; ++i) { dy -= barItemPadding; dx = marginLeft; for (var j = 0; j < seriesCount; ++j) { // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. if(isStacked) stackBase = (j==0?minValue:0); barWidth = gridWidth*(yValues[i][j]-stackBase)/(maxValue-minValue); pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } var sb = new ApacheChartBuffer(); sb.append("M").append(dx).append(",").append(dy); sb.append("h").append(barWidth); sb.append("v").append(-barHeight); sb.append("h").append(-barWidth); sb.append("v").append(barHeight); sb.append("M").append(dx).append(",").append(dy-barHeight); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("h").append(barWidth); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("z"); sb.append("M").append(dx+barWidth).append(",").append(dy); sb.append("v").append(-barHeight); sb.append("l").append(xOffset).append(",").append(-yOffset); sb.append("v").append(barHeight); sb.append("z"); pathElem.setAttribute("stroke", seriesColors[j]); pathElem.setAttribute("stroke-width", 1); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+j+")"); else pathElem.setAttribute("fill", seriesColors[j]); pathElem.setAttribute("d", sb.toString()); pathElem.setAttribute("yValueIndex", i); pathElem.setAttribute("seriesIndex", j); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); rootElem.appendChild(pathElem); if(isStacked) dx += barWidth; else { dy -= barHeight; dy -= barItemPadding; } } if(isStacked) dy -= barHeight; dy -= barItemPadding; } } //////////////////////////////////////////////////////////////////// // Pie chart subclass //////////////////////////////////////////////////////////////////// function ApachePieChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApachePieChart); ApachePieChart.prototype.Init = function( type, model, svgEmbedId, isPerspective, legendPosition) { ApachePieChart.superclass.Init.call(this, type, model, svgEmbedId, isPerspective, legendPosition); //this._pieAnimAngles = undefined; //this._pieAnimRadii = undefined; } ApachePieChart.prototype.DrawChartData = function() { var rootElem = this._rootElement; // calculate the number of rows and columns var model = this._model, yValues = model.getYValues(), yValueCount = yValues.length; var groupLabels = model.getGroupLabels(), groupCount = groupLabels?groupLabels.length:1; var nCols = Math.ceil(Math.sqrt(yValueCount)), nRows = Math.round(Math.sqrt(yValueCount)); var labelElem = this._svgDoc.getElementById("groupLabelPrototype"); var margins = this._margins, dx=margins.left, dy=margins.top; var quadWidth = (this._width - margins.left - margins.right)/nCols; var animate = (this._animDuration>0), isPerspective = this._isPerspective; var pieAnimRadii, vGap = 2*ApacheChart._TEXT_MARGIN; var quadHeight = (this._height - margins.top - margins.bottom - (nRows-1)*vGap)/nRows; if(animate) { this._pieAnimAngles = []; pieAnimRadii = this._pieAnimRadii = []; } for(var i = 0; i= yValueCount) break; var groupLabel = (iGroup == -1)?null:groupLabels[iGroup]; var pieContainer = rootElem.cloneNode(false); rootElem.appendChild(pieContainer); var newHeight = this.DrawGroupLabelTitle(groupLabel, rootElem, labelElem.cloneNode(true), dx, dy, quadWidth, quadHeight); var newWidth = quadWidth - 2*ApacheChart._TEXT_MARGIN; var cx= dx+quadWidth/2+ApacheChart._TEXT_MARGIN, cy = dy+newHeight/2; if(animate) { pieAnimRadii.push(Math.max(cx, cy)); } if(isPerspective) { this._draw3DPies(pieContainer, newWidth, newHeight, iGroup); // The chart is draw with the center at 0 so we need to compensate for it. pieContainer.setAttribute("transform", "translate("+cx+","+cy+") scale(1.0,0.707)"); } else { this._drawPies(pieContainer, newWidth, newHeight, iGroup); pieContainer.setAttribute("transform", "translate("+cx+","+cy+")"); } dx +=quadWidth; } dx=margins.left; dy +=quadHeight+vGap; } } ApachePieChart.prototype.ComputeMinMaxValues = function() { } ApachePieChart.prototype.DrawGroupLabels = function() { } ApachePieChart.prototype.LayoutGroupLabels = function() { } ApachePieChart.prototype.DrawGrid = function() { } ApachePieChart.prototype.DrawYValueLabels = function() { } ApachePieChart.prototype.LayoutYValueLabels = function() { } ApachePieChart.prototype.SetDataAnimStep = function(ratio) { var pieAnimRadii = this._pieAnimRadii, pieAnimAngles = this._pieAnimAngles, isPerspective = this._isPerspective, agleIndex = 0, elemIndex = 0; var animElems = this._dataElems, chartCount = pieAnimRadii.length; var model = this._model, yValues = model.getYValues(); // We are animating parependiculat to the tangent to the middle of the pie for(var i = 0; i < chartCount; ++i) { var nPies = yValues[i].length; var radius = pieAnimRadii[i]*(1-ratio); for (var j = 0; j0); var gradientsUsed = this._gradientsUsed; var defaultTransform = "translate(-10000, -10000)", pieAnimAngles = this._pieAnimAngles; for (var i = 0; i= .5) // major arc { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 0 0 "); } else { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 1 0 "); } sb.append(x2); sb.append(",").append(y2); sb.append("z"); // set the centroids as expandos if(this._tooltipsVisible) { pathElem.setAttribute("_apcGx", Math.round((x1+x2)/3)); pathElem.setAttribute("_apcGy", Math.round((y1+y2)/3)); } pathElem.setAttribute("d", sb.toString()); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("yValueIndex", iGroup); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); pieStart += valueRatio; pieElems[i] = pathElem; } for (var i = 0; i< nPies; ++i) { // calculate the pie gradient: pieContainer.appendChild(pieElems[i]); } } ApachePieChart.prototype._draw3DPies = function( pieContainer, quadWidth, quadHeight, iGroup) { var svgDoc = this._svgDoc, model = this._model, yValues = model.getYValues(); var groupLabels = model.getGroupLabels(), seriesColors = model.getSeriesColors(); var pieSize = Math.min(quadWidth/2, quadHeight/2); var pieTotal = 0; if(iGroup == -1) iGroup = 0; var nPies = yValues[iGroup].length; for (var i = 0; i < nPies; ++i) { pieTotal += yValues[iGroup][i]; } var perspectiveHeight = pieSize/4, pieElems = new Array(nPies), ringElems = new Array(nPies), edgeElems = new Array(nPies); var dataElems = this._dataElems, animate = (this._animDuration>0); if( perspectiveHeight> ApachePieChart._MAX_PERSPECTIVE_HEIGHT ) perspectiveHeight = ApachePieChart._MAX_PERSPECTIVE_HEIGHT; var pathElem = svgDoc.getElementById("piePathPrototype"), pieStart = 0; var gradientsUsed = this._gradientsUsed; var defaultTransform = "translate(-10000, -10000)", pieAnimAngles = this._pieAnimAngles; for (var i = 0; i < nPies; ++i) { pathElem = pathElem.cloneNode(false); var valueRatio = 1 - (yValues[iGroup][i])/(pieTotal); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); pieAnimAngles.push(pieStart+valueRatio/2); } var arcBeginX, arcBeginY, arcEndX, arcEndY; arcBeginX = pieSize*Math.cos(pieStart*Math.PI*2); arcBeginY = pieSize*Math.sin(pieStart*Math.PI*2); var sb = new ApacheChartBuffer(); sb.append("M0,0L").append(arcBeginX).append(",").append(arcBeginY); arcEndX = pieSize*Math.cos((pieStart+valueRatio)*Math.PI*2); arcEndY = pieSize*Math.sin((pieStart+valueRatio)*Math.PI*2); if (valueRatio >= .5) { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 0 0 "); } else { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 1 0 "); } sb.append(arcEndX).append(",").append(arcEndY); sb.append("z"); // set the centroid as expandos if(this._tooltipsVisible) { pathElem.setAttribute("_apcGx", Math.round((arcBeginX+arcEndX)/3)); pathElem.setAttribute("_apcGy", Math.round((arcBeginY+arcEndY)/3)); } if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("yValueIndex", iGroup); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); var pathRingElem = pathElem.cloneNode(false); var pathEdgeElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathRingElem); pathRingElem.setAttribute("transform",defaultTransform); dataElems.push(pathEdgeElem); pathEdgeElem.setAttribute("transform",defaultTransform); } pathElem.setAttribute("d", sb.toString()); sb = new ApacheChartBuffer(); sb.append("M").append(arcBeginX).append(",").append(arcBeginY); if (valueRatio >= .5) // major arc { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 0 0 "); } else { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 1 0 "); } sb.append(arcEndX).append(",").append(arcEndY); sb.append("v").append(perspectiveHeight); if (valueRatio >= .5) // major arc { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 0 1 "); } else { sb.append("A").append(pieSize).append(" ").append(pieSize).append(" 1 1 1 "); } sb.append(arcBeginX).append(",").append(arcBeginY+perspectiveHeight); sb.append("z"); pathRingElem.setAttribute("d", sb.toString()); sb = new ApacheChartBuffer(); sb.append("M0,0L"); sb.append(arcBeginX).append(",").append(arcBeginY); sb.append("v").append(perspectiveHeight); sb.append("L").append(0).append(",").append(perspectiveHeight); sb.append("z"); sb.append("M0,0L"); sb.append(arcEndX).append(",").append(arcEndY); sb.append("v").append(perspectiveHeight); sb.append("L").append(0).append(",").append(perspectiveHeight); sb.append("z"); pathEdgeElem.setAttribute("d", sb.toString()); pieStart += valueRatio; pieElems[i] = pathElem; ringElems[i] = pathRingElem; edgeElems[i] = pathEdgeElem; } // For the top half, edges have preference over rings var totalRatio = 0; for (var i = 0; i< nPies; ++i) { if(totalRatio <= .5) pieContainer.appendChild(ringElems[i]); totalRatio += (yValues[iGroup][i])/(pieTotal); } totalRatio = 0; for (var i = 0; i< nPies; ++i) { if(totalRatio <= .5) pieContainer.appendChild(edgeElems[i]); totalRatio += (yValues[iGroup][i])/(pieTotal); } // For the bottom half, rings have preference over edges totalRatio = 0; for (var i = 0; i< nPies; ++i) { if(totalRatio > .5) pieContainer.appendChild(edgeElems[i]); totalRatio += (yValues[iGroup][i])/(pieTotal); } totalRatio = 0; for (var i = 0; i< nPies; ++i) { if(totalRatio > .5) pieContainer.appendChild(ringElems[i]); totalRatio += (yValues[iGroup][i])/(pieTotal); } for (var i = 0; i< nPies; ++i) { pieContainer.appendChild(pieElems[i]); } } ApachePieChart.prototype.GetToolTipLocation = function(e, ttBBox) { var evtTarget = e.target; var ctm = evtTarget.parentNode.getCTM(); // display the tooltip at the centroid return {x:(ctm.e + parseInt(evtTarget.getAttribute("_apcGx"))), y:(ctm.f + ctm.d*parseInt(evtTarget.getAttribute("_apcGy")) - ttBBox.height)}; } ApachePieChart._MAX_PERSPECTIVE_HEIGHT = 30; //////////////////////////////////////////////////////////////////// // Area chart subclass //////////////////////////////////////////////////////////////////// function ApacheAreaChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheAreaChart); ApacheAreaChart.prototype.Init = function( type, model, svgEmbedId, isPerspective, legendPosition) { ApacheAreaChart.superclass.Init.call(this, type, model, svgEmbedId, isPerspective, legendPosition); this._toolTips = []; } ApacheAreaChart.prototype.SetStopOpacity = function(stopNode) { // In gecko opacity does not mix with stop-opacity, so use a lower value stopNode.setAttribute("stop-opacity", ApacheChart._DEFAULT_STOP_OPACITY/2); } ApacheAreaChart.prototype.DrawChartData = function() { var rootElem = this._rootElement; if(this._tooltipsVisible) { rootElem.addEventListener("mousemove",this.ShowToolTipCallback,false); rootElem.addEventListener("mouseout",this.HideToolTipCallback,false); } if(this._isPerspective) this._drawPerspectiveAreas(); else this._drawAreas(); } ApacheAreaChart.prototype.SetDataAnimStep = function(ratio) { var model = this._model, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var yValues = model.getYValues(), yValueCount = yValues.length; var animElems = this._dataElems, animPathCount = (this._isPerspective)? (yValueCount-1):1; var margins = this._margins, marginBottom = margins.bottom; var cy = (this._height - marginBottom); var newRatio = ratio*seriesCount, animSeriesIndex = 0; if(newRatio > 1) { animSeriesIndex = Math.floor(newRatio); if(animSeriesIndex >= seriesCount) animSeriesIndex = seriesCount - 1; newRatio = newRatio - Math.floor(newRatio); } // We will make each series appear separately var i = animSeriesIndex; for (var j = 0; j < animPathCount; ++j) { var ty = (1-newRatio)*cy; animElems[i*animPathCount+j].setAttribute("transform", "translate(0,"+ty+") scale(1,"+newRatio+")"); if(i>0) { animElems[(i-1)*animPathCount+j].setAttribute("transform", "scale(1,1)"); } } // make sure that everything is scaled properly at the end if(ratio == 1) { for(var i = 0; i < seriesCount; ++i) { for (var j = 0; j < animPathCount; ++j) { animElems[i*animPathCount+j].setAttribute("transform", "scale(1,1)"); } } } } /** * Overridden to indicate that the group label is edge aligned instead of center aligned */ ApacheAreaChart.prototype.IsGroupLabelCentered = function() { return false; } ApacheAreaChart.prototype.GetVLineCount = function() { var xMajorCount = this._xMajorGridCount; if(xMajorCount >= 0) return xMajorCount; else { // Area Chart has one vertical line less since the // first line represents a value return this._model.getGroupLabels().length-1; } } ApacheAreaChart.prototype._drawAreas = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var pathElem = svgDoc.getElementById("areaPathPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var isStacked = (this._type == ApacheChart.TYPE_AREA_STACKED); var yValueCount = yValues.length; var barWidth = (gridWidth/(Math.max(yValueCount,groupCount)-1)); var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(1,0.00001)"; var dx, dy, cumYs = [], stackBase; for (var i = 0; i< seriesCount; ++i) { dx = marginLeft; dy = marginTop + gridHeight; var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); if(i == 0 || !isStacked) sb.append("M").append(dx).append(",").append(dy); else if(isStacked) sb.append("M").append(dx).append(",").append(cumYs[0]); // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. stackBase = (i==0?minValue:0); for (var j = 0; j < yValueCount; ++j) { if(isStacked) { if(null == cumYs[j]) cumYs[j] = gridHeight + marginTop; dy = (cumYs[j] -= gridHeight*(yValues[j][i]-stackBase)/(maxValue-minValue)); } else dy = gridHeight + marginTop - gridHeight*(yValues[j][i]- minValue)/(maxValue-minValue); sb.append("L").append(dx).append(",").append(dy); if(j != yValueCount - 1) dx += barWidth; } if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("seriesIndex", i); pathElem.addEventListener("click",this.ClickCallback,false); if(i == 0 || !isStacked) { sb.append("L").append(dx).append(",").append(gridHeight + marginTop); sb.append("Z"); } else { for (var j = yValueCount-1; j>=0; --j) { var prevY = cumYs[j]+gridHeight*(yValues[j][i]-stackBase)/(maxValue-minValue); sb.append("L").append(dx).append(",").append(prevY); dx -= barWidth; } } pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); } } ApacheAreaChart.prototype._drawPerspectiveAreas = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var pathElem = svgDoc.getElementById("areaPathPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var isStacked = (this._type == ApacheChart.TYPE_AREA_STACKED); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var yValueCount = yValues.length; var barWidth = (gridWidth/(Math.max(yValueCount,groupCount)-1)), stackBase; var gridBottom = gridHeight + marginTop + yOffset, dx, dy, cumYs = []; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(1,0.00001)"; for (var i = 0; i< seriesCount; ++i) { dx = marginLeft; var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); // If we use non zero min and it is a stacked graph, we need to remove the min for only // the first series. stackBase = (i==0?minValue:0); for (var j = 0; j < yValueCount; ++j) { if(isStacked) { if(null == cumYs[j]) cumYs[j] = gridBottom; dy = (cumYs[j] -= gridHeight*(yValues[j][i]-stackBase)/(maxValue-minValue)); } else dy = gridBottom - gridHeight*(yValues[j][i]-minValue)/(maxValue-minValue); if(j != yValueCount - 1) { pathElem = pathElem.cloneNode(false); sb.append("M").append(dx).append(",").append(dy); sb.append("l").append(xOffset).append(",").append(-yOffset); var nextdy, nextdx = dx+barWidth; if(isStacked) { if(null == cumYs[j+1]) cumYs[j+1] = gridBottom; nextdy = (cumYs[j+1] - gridHeight*(yValues[j+1][i]-stackBase)/(maxValue-minValue)); } else nextdy = gridBottom - gridHeight*(yValues[j+1][i]-minValue)/(maxValue-minValue) sb.append("L").append(nextdx+xOffset).append(",").append(nextdy-yOffset); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("L").append(dx).append(",").append(dy); sb.append("M").append(nextdx).append(",").append(nextdy); sb.append("l").append(xOffset).append(",").append(-yOffset); var prevSeriesY, prevSeriesY2; if(i == 0 || !isStacked) { sb.append("L").append(nextdx+xOffset).append(",").append(gridHeight + marginTop); } else { sb.append("L").append(nextdx+xOffset).append(",").append(cumYs[j+1]-yOffset); } sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("L").append(nextdx).append(",").append(nextdy); sb.append("M").append(dx).append(",").append(dy); sb.append("L").append(nextdx).append(",").append(nextdy); if(i == 0 || !isStacked) { sb.append("L").append(nextdx).append(",").append(gridBottom); sb.append("L").append(dx).append(",").append(gridBottom); } else { sb.append("L").append(nextdx).append(",").append(cumYs[j+1]); sb.append("L").append(dx).append(",").append( cumYs[j]+gridHeight*(yValues[j][i]-stackBase)/(maxValue-minValue)); } sb.append("L").append(dx).append(",").append(dy); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("yValueIndex", j); pathElem.setAttribute("seriesIndex", i); pathElem.addEventListener("click",this.ClickCallback,false); dx += barWidth; pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } } } } } ApacheAreaChart.prototype.GetChartEvent = function(e, seriesYs) { var clientX = e.clientX, clientY = e.clientY, evtTarget = e.target; var isStacked = (this._type == ApacheChart.TYPE_AREA_STACKED); var isPerspective = this._isPerspective; var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var model = this._model, yValues = model.getYValues(); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length; var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var yValueCount = yValues.length; var barWidth = (gridWidth/(Math.max(yValueCount,groupCount)-1)); if(isPerspective) { gridWidth -= xOffset; gridHeight -= yOffset; } if(clientX < marginLeft || clientX>(marginLeft + gridWidth + (isPerspective?xOffset:0)) || clientY < marginTop || clientY>(marginTop + gridHeight + (isPerspective?yOffset:0))) { return null; } var dx, dy, dy1, cumYs = [], seriesIndices = [], seriesValues = []; var gridBottom = gridHeight + marginTop +(isPerspective?yOffset:0); var seriesCount = model.getSeriesLabels().length, stackBase, insideStacked = false; if(!seriesYs) seriesYs = []; for (var i = 0; i< seriesCount && !insideStacked; ++i) { dx = marginLeft; stackBase = (i==0?minValue:0); for (var j = 0; j < yValueCount; ++j) { if(isStacked) { if(null == cumYs[j]) cumYs[j] = gridBottom; if(null == cumYs[j+1] && (j != yValueCount -1)) cumYs[j+1] = gridBottom; cumYs[j] -= gridHeight*(yValues[j][i]-stackBase)/(maxValue-minValue); } if(j == yValueCount - 1) continue; if(clientX > dx && clientX < (dx+barWidth)) { if(isStacked) { dy1 = cumYs[j]; dy2 = (cumYs[j+1] - gridHeight*(yValues[j+1][i]-stackBase)/(maxValue-minValue)); dy = dy1 - (dy1-dy2)*(clientX - dx)/barWidth; if(clientY >= dy) { value = yValues[j][i] + (yValues[j+1][i]-yValues[j][i])*(clientX-dx)/barWidth; seriesValues.push(value); seriesIndices.push(i); seriesYs.push(dy); insideStacked = true; break; } } else { dy1 = gridBottom - gridHeight*(yValues[j][i]-minValue)/(maxValue-minValue); dy = dy1 - (gridHeight*(yValues[j+1][i]-yValues[j][i])/(maxValue-minValue))*(clientX-dx)/barWidth; // find all the series that the y point matches if(dy<=clientY) { value = yValues[j][i] + (yValues[j+1][i]-yValues[j][i])*(clientX - dx)/barWidth; seriesValues.push(value); seriesIndices.push(i); seriesYs.push(dy); } break; } } dx += barWidth; } } return new ApacheChartEvent(seriesIndices,null, seriesValues,null); } ApacheAreaChart.prototype.ShowToolTip = function(e) { // Hide any existing tooltips this.HideToolTip(); var seriesYs = []; var chartEvent = this.GetChartEvent(e, seriesYs); if(chartEvent == null || chartEvent.getYValues().length == 0) { return; } this._displayToolTips(chartEvent.getYValues(), chartEvent.getSeriesIndices(), seriesYs, e); } ApacheAreaChart.prototype._displayToolTips = function( seriesValues, seriesIndices, seriesYs, e) { var svgDoc = this._svgDoc, rootElem = this._rootElement; var model = this._model, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length, seriesColors = model.getSeriesColors(); var tooltipCount = seriesIndices.length, toolTips = this._toolTips; var clientX = e.clientX; var dx, dy; for(var i = 0; i this._width) { dx -= ttBBox.width; circleElem.setAttribute("cx",boundingRectElem.getBBox().width); } else { circleElem.setAttribute("cx",0); } if(dy - ttBBox.height < 0) { dy += ttBBox.height; circleElem.setAttribute("cy",0); } else { circleElem.setAttribute("cy",boundingRectElem.getBBox().height); } boundingRectElem.setAttribute("stroke", seriesColors[seriesIndex]); circleElem.setAttribute("stroke",seriesColors[seriesIndex]); toolTip.setAttribute("transform","translate("+dx+","+dy+")"); } } ApacheAreaChart.prototype.HideToolTip = function(e) { var tooltips = this._toolTips, tooltipCount = tooltips.length; for(var i = 0; i0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var lineDot = svgDoc.getElementById("lineDotPrototype"), pathElem = svgDoc.getElementById("linePathPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var yValueCount = yValues.length; var barWidth = gridWidth/Math.max(yValueCount,groupCount); var dx, dy; var gradientsUsed = this._gradientsUsed; // Adobe plugin does not like 0 for scale and gecko does not like a low number for circles var defaultTransform = this._isIE?"scale(0.00001,1)":"scale(0,1)"; for (var i = 0; i< seriesCount; ++i) { // For combo charts we will draw every alternate(odd) series if(isCombo && i%2 == 0) continue; dx = marginLeft+barWidth/2; var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); for (var j = 0; j < yValueCount; ++j) { dy = gridHeight + marginTop - gridHeight*(yValues[j][i]-minValue)/(maxValue-minValue); if(j == 0) sb.append("M").append(dx).append(",").append(dy); else sb.append("L").append(dx).append(",").append(dy); lineDot = lineDot.cloneNode(false); lineDot.setAttribute("cx", dx); lineDot.setAttribute("cy", dy); if(gradientsUsed) lineDot.setAttribute("fill", "url(#gradient"+i+")"); else lineDot.setAttribute("fill", seriesColors[i]); lineDot.setAttribute("stroke", seriesColors[i]); if(animate) { lineDot.setAttribute("transform",defaultTransform); dataElems.push(lineDot); } rootElem.appendChild(lineDot); // There is no fill for lines pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); dx += barWidth; } pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } } } ApacheLineChart.prototype.__drawPerspectiveLines = function(isCombo) { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var pathElem = svgDoc.getElementById("linePath3dPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var yValueCount = yValues.length; var barWidth = (gridWidth/Math.max(yValueCount,groupCount)); var gridBottom = gridHeight + marginTop + yOffset, dx, dy; var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(0.00001,1)"; for (var i = 0; i< seriesCount; ++i) { // For combo charts we will draw every alternate(odd) series if(isCombo && i%2 == 0) continue; dx = marginLeft+barWidth/2; var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); for (var j = 0; j < yValueCount; ++j) { dy = gridBottom - gridHeight*(yValues[j][i]-minValue)/(maxValue-minValue); if(j != yValueCount - 1) { var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); sb.append("M").append(dx).append(",").append(dy); sb.append("l").append(xOffset).append(",").append(-yOffset); var nextdy = gridBottom - gridHeight*(yValues[j+1][i]-minValue)/(maxValue-minValue); var nextdx = dx+barWidth; sb.append("L").append(nextdx+xOffset).append(",").append(nextdy-yOffset); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("L").append(dx).append(",").append(dy); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mousemove",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); dx += barWidth; pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } } } } } ApacheLineChart.prototype.ShowToolTip = function(e) { // first hide any existing tooltip this.HideToolTip(); ApacheLineChart.superclass.ShowToolTip.call(this, e); } ApacheLineChart.prototype.GetToolTipLocation = function(e, ttBBox) { return {x:(e.clientX+20), y:(e.clientY+20)}; } ApacheLineChart.prototype.GetChartEvent = function(e) { var evtTarget = e.target; var i = parseInt(evtTarget.getAttribute("seriesIndex")); var clientX = e.clientX; var isPerspective = this._isPerspective; var yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var model = this._model, yValues = model.getYValues(); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length; var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var yValueCount = yValues.length; var barWidth = (gridWidth/Math.max(yValueCount,groupCount)); var gridBottom = gridHeight + marginTop +(isPerspective?yOffset:0); var dx = marginLeft+barWidth/2, value = 0.0; for (var j = 0; j < yValueCount; ++j) { if(j == yValueCount - 1) continue; if(clientX > dx && clientX < (dx+barWidth)) { value = yValues[j][i] + (yValues[j+1][i]-yValues[j][i])*(clientX - dx)/barWidth; break; } dx += barWidth; } return new ApacheChartEvent([i],null, [value],null); } ApacheLineChart.prototype.FillToolTipData = function(boundingRectElem, circleElem, e) { var chartEvent = this.GetChartEvent(e); var i = chartEvent.getSeriesIndices()[0], value = chartEvent.getYValues()[0]; var seriesLabels = this._model.getSeriesLabels(); var textElem = boundingRectElem.nextSibling.nextSibling; textElem.firstChild.data = seriesLabels[i]+ ": "+this._formatValue(value); var labelWidth = textElem.getBBox().width; //We do not need the next label textElem = textElem.nextSibling.nextSibling; textElem.firstChild.data = ""; var rectWidth = labelWidth+2*ApacheChart._TEXT_MARGIN; boundingRectElem.setAttribute("width",rectWidth); boundingRectElem.setAttribute("stroke", seriesColors[i]); circleElem.setAttribute("r",0); } //////////////////////////////////////////////////////////////////// // ScatterPlot chart subclass //////////////////////////////////////////////////////////////////// function ApacheScatterPlotChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheScatterPlotChart); ApacheScatterPlotChart.prototype.Init = function( type, model, svgEmbedId, isPerspective, legendPosition) { ApacheScatterPlotChart.superclass.Init.call(this, type, model, svgEmbedId, isPerspective, legendPosition); //this._cxs = undefined; //this._cys = undefined; } ApacheScatterPlotChart.prototype.DrawChartData = function() { if(this._isPerspective) this._drawPerspectivePoints(); else this._drawPoints(); } ApacheScatterPlotChart.prototype.SetDataAnimStep = function(ratio) { var isPerspective = this._isPerspective; var cxs = this._cxs, cys = this._cys, gridCx, gridCy; var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var animElems = this._dataElems, animCount = cxs.length, elemIndex = 0; if(isPerspective) marginLeft += ApacheChart._XOFFSET_PERSPECTIVE; gridCx = gridWidth/2 + marginLeft; gridCy = gridHeight/2 + marginTop; // we are going to animate by starting the dot at the middle and work towards its destination for(var i = 0; i < animCount; ++i) { var cx = gridCx - (gridCx - cxs[i])*ratio; var cy = gridCy - (gridCy - cys[i])*ratio; var elem = animElems[elemIndex++]; elem.setAttribute("cx", cx); elem.setAttribute("cy", cy); if(isPerspective) { elem = animElems[elemIndex++]; elem.setAttribute("cx", cx); elem.setAttribute("cy", cy); } } } ApacheScatterPlotChart.prototype.SetGridAnimStep = function(ratio) { var animElems = this._gridElems, animCount = animElems.length; for(var i = 0; i < animCount; ++i) { animElems[i].setAttribute("fill-opacity", ratio); animElems[i].setAttribute("transform", "scale(1,1)"); } } /** * Overridden to indicate that the group label is edge aligned instead of center aligned */ ApacheScatterPlotChart.prototype.IsGroupLabelCentered = function() { return false; } ApacheScatterPlotChart.prototype.GetVLineCount = function() { var xMajorCount = this._xMajorGridCount; if(xMajorCount >= 0) return xMajorCount; else { // Area Chart has one vertical line less since the // first line represents a value return this._model.getGroupLabels().length-1; } } ApacheScatterPlotChart.prototype._drawPoints = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var dotElem = svgDoc.getElementById("scatterDotPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), xValues = model.getXValues(), yValues = model.getYValues(), nValues = yValues.length; var defaultTransform = "translate(0,100000)"; var minYValue = model.getMinYValue(), maxYValue = model.getMaxYValue(); var minXValue = model.getMinXValue(), maxXValue = model.getMaxXValue(); var barWidth = (gridWidth/(groupCount-1)); var cxs, cys, dx, dy, gridCx, gridCY; var gradientsUsed = this._gradientsUsed; if(animate) { cxs = this._cxs = []; cys = this._cys = []; gridCx = gridWidth/2 + marginLeft; gridCy = gridHeight/2 + marginTop; } for (var i = 0; i< seriesCount; ++i) { for (var j = 0; j < nValues; ++j) { dy = gridHeight + marginTop - gridHeight*(yValues[j][i]-minYValue)/(maxYValue-minYValue); dx = marginLeft + gridWidth*(xValues[j][i]-minXValue)/(maxXValue-minXValue); dotElem = dotElem.cloneNode(false); if(gradientsUsed) dotElem.setAttribute("fill", "url(#gradient"+i+")"); else dotElem.setAttribute("fill", seriesColors[i]); dotElem.setAttribute("stroke", seriesColors[i]); dotElem.setAttribute("yValueIndex", j); dotElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { dotElem.addEventListener("mouseover",this.ShowToolTipCallback,false); dotElem.addEventListener("mouseout",this.HideToolTipCallback,false); } dotElem.addEventListener("click",this.ClickCallback,false); if(animate) { dataElems.push(dotElem); dotElem.setAttribute("cx", gridCx); dotElem.setAttribute("cy", gridCy); // we will use it during animation cxs.push(dx); cys.push(dy); } else { dotElem.setAttribute("cx", dx); dotElem.setAttribute("cy", dy); } rootElem.appendChild(dotElem); } } } ApacheScatterPlotChart.prototype._drawPerspectivePoints = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var dotElem = svgDoc.getElementById("scatter3dDotPrototype"); var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), xValues = model.getXValues(), yValues = model.getYValues(), nValues = yValues.length; var minYValue = model.getMinYValue(), maxYValue = model.getMaxYValue(); var minXValue = model.getMinXValue(), maxXValue = model.getMaxXValue(); var barWidth = (gridWidth/(groupCount-1)); var gridBottom = gridHeight + marginTop + yOffset, cxs, cys, dx, dy, gridCx, gridCY; var gradientsUsed = this._gradientsUsed; if(animate) { cxs = this._cxs = []; cys = this._cys = []; gridCx = gridWidth/2 + marginLeft +xOffset; gridCy = gridHeight/2 + marginTop; } for (var i = 0; i< seriesCount; ++i) { for (var j = 0; j < nValues; ++j) { dy = gridBottom - gridHeight*(yValues[j][i]-minYValue)/(maxYValue-minYValue); dx = marginLeft + gridWidth*(xValues[j][i]-minXValue)/(maxXValue-minXValue); dotElem = dotElem.cloneNode(false); if(animate) { dataElems.push(dotElem); dotElem.setAttribute("cx", gridCx); dotElem.setAttribute("cy", gridCy); // we will use it during animation cxs.push(dx); cys.push(dy); } else { dotElem.setAttribute("cx", dx); dotElem.setAttribute("cy", dy); } if(gradientsUsed) dotElem.setAttribute("fill", "url(#gradient"+i+")"); else dotElem.setAttribute("fill", seriesColors[i]); dotElem.setAttribute("stroke", seriesColors[i]); dotElem.setAttribute("yValueIndex", j); dotElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { dotElem.addEventListener("mouseover",this.ShowToolTipCallback,false); dotElem.addEventListener("mouseout",this.HideToolTipCallback,false); } dotElem.addEventListener("click",this.ClickCallback,false); var shadowElem = dotElem.cloneNode(false); if(animate) { dataElems.push(shadowElem); } shadowElem.setAttribute("fill","#333333"); shadowElem.setAttribute("opacity","0.5"); shadowElem.setAttribute("stroke","none"); shadowElem.setAttribute("transform", "translate(3,3)"); rootElem.appendChild(shadowElem); rootElem.appendChild(dotElem); } } } ApacheScatterPlotChart.prototype.GetChartEvent = function(e) { var evtTarget = e.target; var i = parseInt(evtTarget.getAttribute("seriesIndex")), j = parseInt(evtTarget.getAttribute("yValueIndex")); var model = this._model, xValues = model.getXValues(), yValues = model.getYValues(); return new ApacheChartEvent([i],[j], [yValues[j][i]],[xValues[j][i]]); } ApacheScatterPlotChart.prototype.FillToolTipData = function(boundingRectElem, circleElem, e) { var chartEvent = this.GetChartEvent(e); var i = chartEvent.getSeriesIndices()[0], yValue = chartEvent.getYValues()[0] xValue = chartEvent.getXValues()[0]; var model = this._model, seriesLabels = model.getSeriesLabels(); var textElem = boundingRectElem.nextSibling.nextSibling; textElem.firstChild.data = seriesLabels[i]+ ": ("+ this._formatValue(xValue) + ") (" + this._formatValue(yValue) +")"; var labelWidth = textElem.getBBox().width; //We do not need the next label textElem = textElem.nextSibling.nextSibling; textElem.firstChild.data = ""; var rectWidth = labelWidth+2*ApacheChart._TEXT_MARGIN; boundingRectElem.setAttribute("width",rectWidth); boundingRectElem.setAttribute("stroke", seriesColors[i]); circleElem.setAttribute("r",0); } //////////////////////////////////////////////////////////////////// // XYLine Chart subclass // Note: chart x values should be in the ascending order other wise // the chart does not make sense //////////////////////////////////////////////////////////////////// function ApacheXYLineChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheScatterPlotChart, ApacheXYLineChart); ApacheXYLineChart.prototype.SetDataAnimStep = function(ratio) { // bypass the parent and go directly into Chart ApacheChart.prototype.SetDataAnimStep.call(this, ratio); } ApacheXYLineChart.prototype.DrawChartData = function() { if(this._isPerspective) this._drawPerspectiveXYValues(); else this._drawXYValues(); } ApacheXYLineChart.prototype.AnimAlongXAxis = function() { // always around x axis return true; } ApacheXYLineChart.prototype._drawXYValues = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var dotElem, pathElem; var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), xValues = model.getXValues(), yValues = model.getYValues(), nValues = yValues.length; var defaultTransform = "scale(0.00001,1)"; var minYValue = model.getMinYValue(), maxYValue = model.getMaxYValue(); var minXValue = model.getMinXValue(), maxXValue = model.getMaxXValue(); var barWidth = (gridWidth/(groupCount-1)); var dx, dy; pathElem = svgDoc.getElementById("linePathPrototype"); for (var i = 0; i< seriesCount; ++i) { var sb = new ApacheChartBuffer(); dx = marginLeft; dy = gridHeight + marginTop; pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } for (var j = 0; j < nValues; ++j) { dy = gridHeight + marginTop - gridHeight*(yValues[j][i]-minYValue)/(maxYValue-minYValue); dx = marginLeft + gridWidth*(xValues[j][i]-minXValue)/(maxXValue-minXValue); if(j==0) sb.append("M").append(dx).append(",").append(dy); else sb.append("L").append(dx).append(",").append(dy); } pathElem.setAttribute("seriesIndex", i); pathElem.setAttribute("stroke", seriesColors[i]); if(this._tooltipsVisible) { pathElem.addEventListener("mousemove",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); } } ApacheXYLineChart.prototype._drawPerspectiveXYValues = function() { var svgDoc = this._svgDoc, model = this._model, margins = this._margins; var rootElem = this._rootElement, dataElems = this._dataElems, animate = (this._animDuration>0); var xOffset = ApacheChart._XOFFSET_PERSPECTIVE, yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right - xOffset); var gridHeight = (this._height - marginTop - margins.bottom - yOffset); var dotElem, pathElem; var groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), xValues = model.getXValues(), yValues = model.getYValues(), nValues = yValues.length; var defaultTransform = "scale(0.00001,1)"; var minYValue = model.getMinYValue(), maxYValue = model.getMaxYValue(); var minXValue = model.getMinXValue(), maxXValue = model.getMaxXValue(); var gridBottom = gridHeight + marginTop+yOffset, dx, dy; var gradientsUsed = this._gradientsUsed; pathElem = svgDoc.getElementById("linePath3dPrototype"); for (var i = 0; i< seriesCount; ++i) { var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } for (var j = 0; j < nValues; ++j) { dy = gridBottom - gridHeight*(yValues[j][i]-minYValue)/(maxYValue-minYValue); dx = marginLeft + gridWidth*(xValues[j][i]-minXValue)/(maxXValue-minXValue); if(j != nValues - 1) { sb.append("M").append(dx).append(",").append(dy); sb.append("l").append(xOffset).append(",").append(-yOffset); var nextdy, nextdx; nextdx = marginLeft + gridWidth*(xValues[j+1][i]-minXValue)/(maxXValue-minXValue); nextdy = gridBottom - gridHeight*(yValues[j+1][i]-minYValue)/(maxYValue-minYValue); sb.append("L").append(nextdx+xOffset).append(",").append(nextdy-yOffset); sb.append("l").append(-xOffset).append(",").append(yOffset); sb.append("L").append(dx).append(",").append(dy); } } if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mousemove",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); } } ApacheXYLineChart.prototype.ShowToolTip = function(e) { // first hide any existing tooltip this.HideToolTip(); ApacheXYLineChart.superclass.ShowToolTip.call(this, e); } ApacheXYLineChart.prototype.GetToolTipLocation = function(e, ttBBox) { return {x:(e.clientX+20), y:(e.clientY+20)}; } ApacheXYLineChart.prototype.GetChartEvent = function(e) { var evtTarget = e.target; var i = parseInt(evtTarget.getAttribute("seriesIndex")); var clientX = e.clientX, clientY = e.clientY, evtTarget = e.target; var isPerspective = this._isPerspective; var yOffset = ApacheChart._YOFFSET_PERSPECTIVE; var model = this._model, xValues = model.getXValues(), yValues = model.getYValues(), nValues = yValues.length; var minYValue = model.getMinYValue(), maxYValue = model.getMaxYValue(); var minXValue = model.getMinXValue(), maxXValue = model.getMaxXValue(); var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var gridBottom = gridHeight + marginTop +(isPerspective?yOffset:0); var dx, dy, xValue = 0.0, yValue = 0.0; var nextdy, nextdx; for (var j = 0; j < nValues; ++j) { if(j != nValues - 1) { dx = marginLeft + gridWidth*(xValues[j][i]-minXValue)/(maxXValue-minXValue); nextdx = marginLeft + gridWidth*(xValues[j+1][i]-minXValue)/(maxXValue-minXValue); if(clientX > dx && clientX < (dx+nextdx)) { dy = gridBottom - gridHeight*(yValues[j][i]-minYValue)/(maxYValue-minYValue); nextdy = gridBottom - gridHeight*(yValues[j+1][i]-minYValue)/(maxYValue-minYValue); yValue = yValues[j][i] + (yValues[j+1][i]-yValues[j][i])*(clientY - dy)/(nextdy-dy); xValue = xValues[j][i] + (xValues[j+1][i]-xValues[j][i])*(clientX - dx)/(nextdx-dx); break; } } } return new ApacheChartEvent([i],null, [yValue],[xValue]); } ApacheXYLineChart.prototype.FillToolTipData = function(boundingRectElem, circleElem, e) { var chartEvent = this.GetChartEvent(e); var i = chartEvent.getSeriesIndices()[0], yValue = chartEvent.getYValues()[0] xValue = chartEvent.getXValues()[0]; var model = this._model, seriesLabels = model.getSeriesLabels(); var textElem = boundingRectElem.nextSibling.nextSibling; textElem.firstChild.data = seriesLabels[i]+ ": ("+ this._formatValue(xValue) + ") (" + this._formatValue(yValue) +")"; var labelWidth = textElem.getBBox().width; //We do not need the next label textElem = textElem.nextSibling.nextSibling; textElem.firstChild.data = ""; var rectWidth = labelWidth+2*ApacheChart._TEXT_MARGIN; boundingRectElem.setAttribute("width",rectWidth); boundingRectElem.setAttribute("stroke", seriesColors[i]); circleElem.setAttribute("r",0); } //////////////////////////////////////////////////////////////////// // Radar chart subclass //////////////////////////////////////////////////////////////////// function ApacheRadarChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheRadarChart); ApacheRadarChart.prototype.Init = function( type, model, svgEmbedId, isPerspective, legendPosition) { ApacheRadarChart.superclass.Init.call(this, type, model, svgEmbedId, isPerspective, legendPosition); this._toolTips = []; } ApacheRadarChart.prototype.draw = function() { this._yLabels = []; ApacheRadarChart.superclass.draw.call(this); delete this._yLabels; } ApacheRadarChart.prototype.SetGridAnimStep = function(ratio) { var animElems = this._gridElems, animCount = animElems.length; for(var i = 0; i < animCount; ++i) { animElems[i].setAttribute("fill-opacity",ratio); } } ApacheRadarChart.prototype.SetDataAnimStep = function(ratio) { var animElems = this._dataElems, animCount = animElems.length; var margins = this._margins, marginLeft = margins.left, marginTop = margins.top; var isRadarArea =(this._type == ApacheChart.TYPE_RADAR_AREA); var model = this._model, groupLabels = model.getGroupLabels(), groupCount = groupLabels.length, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var cx = marginLeft+gridWidth/2, cy = marginTop+gridHeight/2; var newRatio = ratio*seriesCount, animSeriesIndex = 0; if(newRatio > 1) { animSeriesIndex = Math.floor(newRatio); if(animSeriesIndex >= seriesCount) animSeriesIndex = seriesCount - 1; newRatio = newRatio - Math.floor(newRatio); } var tx = (1-newRatio)*cx, ty = (1-newRatio)*cy; var transform = "translate("+tx+","+ ty+") scale("+newRatio+","+newRatio+")"; // We will make each series appear separately var i = animSeriesIndex; this._setRadarSeriesAnimStep(i, animElems, isRadarArea, transform); if(i>0) { this._setRadarSeriesAnimStep(i-1, animElems, isRadarArea, "scale(1,1)"); } // make sure that everything is scaled properly at the end if(ratio == 1) { for(var i = 0; i < seriesCount; ++i) { this._setRadarSeriesAnimStep(i, animElems, isRadarArea, "scale(1,1)"); } } } ApacheRadarChart.prototype._setRadarSeriesAnimStep = function(i, animElems, isRadarArea, transform) { animElems[i].setAttribute("transform", transform); if(!isRadarArea) { var dots = animElems["dots"+i]; for(var j = dots.length-1; j>=0; --j) { dots[j].setAttribute("transform", transform); } } } ApacheRadarChart.prototype.DrawChartData = function() { this._drawRadar(); // Move the y-value labels to the top since currently there is not concept of z-index in svg var yLabels = this._yLabels, rootElem = this._rootElement; for(var i = yLabels.length-1; i >=0; i--) { var label = yLabels[i]; if(label) { rootElem.removeChild(label); rootElem.appendChild(label); } } } ApacheRadarChart.prototype.SetStopOpacity = function(stopNode) { // In gecko opacity does not mix with stop-opacity, so use a lower value stopNode.setAttribute("stop-opacity", ApacheChart._DEFAULT_STOP_OPACITY/2); } /** * Adjusts the legend location when it is at the top * @param ty(int) the original y location of the legend */ ApacheRadarChart.prototype.SetLegendTopAdjustment = function(ty) { var container = this._hLabelContainer; ty -= container.getBBox().height+ApacheChart._TEXT_MARGIN; return ty; } /** * Adjusts the legend location when it is at the bottom * @param ty(int) the original y location of the legend */ ApacheRadarChart.prototype.SetLegendBottomAdjustment = function(ty) { var container = this._hLabelContainer; if(container.childNodes.length > 0) { ty += container.getBBox().height+ApacheChart._TEXT_MARGIN; } return ty; } /** * Adjusts the legend location when it is at the Left * @param tx(int) the original x location of the legend */ ApacheRadarChart.prototype.SetLegendLeftAdjustment = function(tx) { var container = this._hLabelContainer; if(container.childNodes.length > 0) { tx -= container.getBBox().width+ApacheChart._TEXT_MARGIN; } return tx; } /** * Adjusts the legend location when it is at the Right * @param tx(int) the original x location of the legend */ ApacheRadarChart.prototype.SetLegendRightAdjustment = function(tx) { var container = this._hLabelContainer; if(container.childNodes.length > 0) tx += container.getBBox().width+ApacheChart._TEXT_MARGIN; return tx; } ApacheRadarChart.prototype.DrawGroupLabels = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model; var container = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); this._hLabelContainer = container; var labelElems = this._labelElems, animate = (this._animDuration>0); var groupLabels = model.getGroupLabels(), vLineCount = groupLabels.length; var labelElem = svgDoc.getElementById("groupLabelPrototype"); var labelText, gLabelElems = this._groupLabelElems; for(var i = 0; i 0) { var bBox = container.getBBox(); var dxVertical = bBox.width+ApacheChart._TEXT_MARGIN, dyVertical = bBox.height+ApacheChart._TEXT_MARGIN; this._margins.top += dyVertical; this._margins.bottom += dyVertical; this._margins.left += dxVertical; this._margins.right += dxVertical; } } ApacheRadarChart.prototype.LayoutGroupLabels = function() { var model = this._model, margins = this._margins; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight, cy; var cx = marginLeft+gridWidth/2, radius; var groupLabels = model.getGroupLabels(), vLineCount = groupLabels.length; var labelElem, groupWidth = gridWidth/vLineCount; var container = this._hLabelContainer, childNodes = container.childNodes; var firstLabel = false, gLabelElems = this._groupLabelElems; if(childNodes.length == 0) return; for(var i = 0; i Math.PI) dx -= textWidth; labelElem.setAttribute("x", dx); } } ApacheRadarChart.prototype.DrawGrid = function() { // No perspective support for radar this.Draw2DGrid(); } ApacheRadarChart.prototype.Draw2DGrid = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model, margins = this._margins; var gridElems = this._gridElems, animate = (this._animDuration>0); var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var cx = marginLeft+gridWidth/2, cy = marginTop+gridHeight/2; var radius = Math.min(gridWidth, gridHeight)/2; var gradientsUsed = this._gradientsUsed; var circle = svgDoc.getElementById("radarCirclePrototype").cloneNode(false); circle.setAttribute("cx", cx); circle.setAttribute("cy", cy); circle.setAttribute("r", radius); if(gradientsUsed) circle.setAttribute("fill", "url(#gridGradient)"); if(animate) { gridElems.push(circle); circle.setAttribute("fill-opacity","0"); } rootElem.appendChild(circle); var vLineCount = this.GetVLineCount(), circleCount = this.GetHLineCount(); // inner circles circle = svgDoc.getElementById("radarInnerCirclePrototype"); circle.setAttribute("cx", cx); circle.setAttribute("cy", cy); for (var i = 0; i< circleCount-1; ++i) { circle = circle.cloneNode(false); if(animate) { gridElems.push(circle); circle.setAttribute("fill-opacity","0"); } var newRadius = radius - (i+1)*radius/circleCount; circle.setAttribute("r", newRadius); rootElem.appendChild(circle); } var sb = new ApacheChartBuffer(); var pathElem = svgDoc.getElementById("radarGridPathPrototype").cloneNode(false); sb.append("M").append(cx).append(",").append(cy); sb.append("l").append(0).append(",").append(-radius); for(var i = 0; i0); var cx = marginLeft+gridWidth/2, cy = marginTop+gridHeight/2; var radius = Math.min(gridWidth, gridHeight)/2; var vLineCount = this.GetVLineCount(), circleCount = this.GetHLineCount(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var yLabels = this._yLabels; var labelElem = svgDoc.getElementById("yLabelPrototype"); labelElem = labelElem.cloneNode(true); var textHeight = this._addRadarYLabelAt(rootElem, labelElem, circleCount-1, cx,marginTop, textHeight, this._formatValue(maxValue)); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } labelElem = labelElem.cloneNode(true); this._addRadarYLabelAt(rootElem, labelElem, circleCount, cx, cy, textHeight, this._formatValue(minValue)); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } // horizontal lines for (var i = 0; i< circleCount-1; ++i) { var newRadius = (i+1)*radius/circleCount; var value = ((maxValue-minValue)*(i+1)/circleCount) + minValue; labelElem = labelElem.cloneNode(true); this._addRadarYLabelAt(rootElem, labelElem, i, cx, radius-newRadius+marginTop, textHeight, this._formatValue(value)); if(animate) { labelElems.push(labelElem); labelElem.setAttribute("fill-opacity","0"); } } } ApacheRadarChart.prototype._addRadarYLabelAt = function( rootElem, labelElem, index, x, y, textHeight, value) { this._yLabels[index] = labelElem; labelElem.firstChild.data = value; rootElem.appendChild(labelElem); if(textHeight == null) { var rect = labelElem.getBBox(); textHeight = rect.height; } // readjust to right align var labelMargin = ApacheChart._TEXT_MARGIN, textLength = labelElem.getBBox().width; dx = x-textLength-labelMargin; labelElem.setAttribute("x", dx); labelElem.setAttribute("y", y+textHeight/2); return textHeight; } ApacheRadarChart.prototype._drawRadar = function() { var svgDoc = this._svgDoc, rootElem = this._rootElement, model = this._model, margins = this._margins; var dataElems = this._dataElems, animate = (this._animDuration>0), dotElems; var marginLeft = margins.left, marginTop = margins.top; var gridWidth = (this._width - marginLeft - margins.right); var gridHeight = (this._height - marginTop - margins.bottom); var cx = marginLeft+gridWidth/2, cy = marginTop+gridHeight/2; var radius = Math.min(gridWidth, gridHeight)/2; var isRadarArea =(this._type == ApacheChart.TYPE_RADAR_AREA); var protoName = isRadarArea?"areaPathPrototype":"linePathPrototype"; var lineDot,pathElem = svgDoc.getElementById(protoName); var seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length; var seriesColors = model.getSeriesColors(), yValues = model.getYValues(); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var defaultTransform = "scale(0.00001,0.00001)"; var gradientsUsed = this._gradientsUsed; var yValueCount = yValues.length; var dx, dy; if(!isRadarArea) lineDot = svgDoc.getElementById("lineDotPrototype"); for (var i = 0; i< seriesCount; ++i) { var sb = new ApacheChartBuffer(); pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); dotElems = dataElems["dots"+i] = []; } for (var j = 0; j < yValueCount; ++j) { var yPoint = radius*(yValues[j][i]-minValue)/(maxValue-minValue); var dx= cx + yPoint*Math.sin((j)*2*Math.PI/yValueCount), dy = cy - yPoint*Math.cos((j)*2*Math.PI/yValueCount); if(j == 0) { sb.append("M").append(dx).append(",").append(dy); } else { sb.append("L").append(dx).append(",").append(dy); } if(!isRadarArea) { lineDot = lineDot.cloneNode(false); lineDot.setAttribute("cx", dx); lineDot.setAttribute("cy", dy); if(gradientsUsed) lineDot.setAttribute("fill", "url(#gradient"+i+")"); else lineDot.setAttribute("fill", seriesColors[i]); lineDot.setAttribute("stroke", seriesColors[i]); if(animate) { dotElems.push(lineDot); lineDot.setAttribute("transform",defaultTransform); } rootElem.appendChild(lineDot); } } sb.append("Z"); if(isRadarArea) { if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); } else { pathElem.setAttribute("fill", "none"); } pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mousemove",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); pathElem.setAttribute("d", sb.toString()); rootElem.appendChild(pathElem); } } ApacheRadarChart.prototype.isPointInPolygon = function(xs, ys, x, y) { var i, j, npol = ys.length, inside = false; for (i = 0, j = npol-1; i < npol; j = i++) { if ((((ys[i]<=y) && (y(marginLeft + gridWidth) || clientY < marginTop || clientY>(marginTop + gridHeight)) { return null; } if(!isRadarArea) seriesIndex = parseInt(evtTarget.getAttribute("seriesIndex")); if(!seriesXs) seriesXs = []; if(!seriesYs) seriesYs = []; var dx1, dy1, dx2, dy2, seriesIndices = [], seriesValues = []; for (var i = 0; i< seriesCount; ++i) { if(!isRadarArea && (seriesIndex != i)) continue; for (var j = 0; j < yValueCount; ++j) { var nextYVal = (j != yValueCount -1)? yValues[j+1][i]: yValues[0][i]; var yPoint = radius*(yValues[j][i]-minValue)/(maxValue-minValue); var yPoint2 = radius*(nextYVal-minValue)/(maxValue-minValue); var angle = j*2*Math.PI/yValueCount, nextAngle = (j+1)*2*Math.PI/yValueCount; var dx1 = cx + yPoint*Math.sin(angle), dy1 = cy - yPoint*Math.cos(angle); dx2 = cx + yPoint2*Math.sin(nextAngle); dy2 = cy - yPoint2*Math.cos(nextAngle); if(this.isPointInPolygon([cx, dx1, dx2], [cy, dy1, dy2], clientX, clientY)) { // find the point on the radar that matches the current mouse // using the angle of the current mouse point var mousePtAngle = Math.atan2(cy - clientY, clientX - cx); if(mousePtAngle <= Math.PI/2) mousePtAngle = Math.PI/2 - mousePtAngle; else mousePtAngle = 3*Math.PI/2 + (Math.PI - mousePtAngle); var ratio = (mousePtAngle - angle)/(nextAngle - angle); value = yValues[j][i] + (nextYVal-yValues[j][i])*ratio; seriesValues.push(value); seriesIndices.push(i); seriesYs.push(dy1+(dy2-dy1)*ratio); seriesXs.push(dx1+(dx2-dx1)*ratio); break; } } } return new ApacheChartEvent(seriesIndices,null, seriesValues,null); } ApacheRadarChart.prototype.ShowToolTip = function(e) { // Hide any existing tooltips this.HideToolTip(); var seriesXs = [], seriesYs = []; var chartEvent = this.GetChartEvent(e, seriesXs, seriesYs); if(chartEvent == null || chartEvent.getYValues().length == 0) { return; } this._displayToolTips(chartEvent.getYValues(), chartEvent.getSeriesIndices(), seriesYs, seriesXs, e); } ApacheRadarChart.prototype._displayToolTips = function( seriesValues, seriesIndices, seriesYs, seriesXs, e) { var svgDoc = this._svgDoc, rootElem = this._rootElement; var model = this._model, seriesLabels = model.getSeriesLabels(), seriesCount = seriesLabels.length, seriesColors = model.getSeriesColors(); var tooltipCount = seriesIndices.length, toolTips = this._toolTips; var dx, dy; for(var i = 0; i this._width) { dx -= ttBBox.width; circleElem.setAttribute("cx", boundingRectElem.getBBox().width); } else { circleElem.setAttribute("cx",0); } if(dy - ttBBox.height < 0) { dy += ttBBox.height; circleElem.setAttribute("cy",0); } else { circleElem.setAttribute("cy",boundingRectElem.getBBox().height); } boundingRectElem.setAttribute("stroke", seriesColors[seriesIndex]); circleElem.setAttribute("stroke",seriesColors[seriesIndex]); toolTip.setAttribute("transform","translate("+dx+","+dy+")"); } } ApacheRadarChart.prototype.HideToolTip = function(e) { var tooltips = this._toolTips, tooltipCount = tooltips.length; for(var i = 0; i= yValueCount) break; var groupLabel = (iGroup == -1)?null:groupLabels[iGroup]; var fnlContainer = rootElem.cloneNode(false); rootElem.appendChild(fnlContainer); var newHeight = this.DrawGroupLabelTitle(groupLabel, rootElem, labelElem.cloneNode(true), dx, dy, quadWidth, quadHeight); var newWidth = quadWidth - 2*ApacheChart._TEXT_MARGIN; if(this._isPerspective) { // Top ring has a height of 1/6 that of the width. So we need to compensate for half of it. newHeight -= newWidth/6; this._drawPerspectiveFunnel(fnlContainer, newWidth, newHeight, iGroup); fnlContainer.setAttribute("transform", "translate("+(dx+ApacheChart._TEXT_MARGIN)+","+(dy+newWidth/12)+")"); } else { this._drawFunnel(fnlContainer, newWidth, newHeight, iGroup); fnlContainer.setAttribute("transform", "translate("+(dx+ApacheChart._TEXT_MARGIN)+","+dy+")"); } dx +=quadWidth; } dx=margins.left; dy +=quadHeight+vGap; } } ApacheFunnelChart.prototype.ComputeMinMaxValues = function() { } ApacheFunnelChart.prototype.DrawGroupLabels = function() { } ApacheFunnelChart.prototype.LayoutGroupLabels = function() { } ApacheFunnelChart.prototype.DrawGrid = function() { } ApacheFunnelChart.prototype.DrawYValueLabels = function() { } ApacheFunnelChart.prototype.LayoutYValueLabels = function() { } ApacheFunnelChart.prototype._drawFunnel = function( fnlContainer, quadWidth, quadHeight, iGroup) { var svgDoc = this._svgDoc, model = this._model, yValues = model.getYValues(); var groupLabels = model.getGroupLabels(), seriesColors = model.getSeriesColors(); if(iGroup == -1) iGroup = 0; // Number of segments var nSeg = yValues[iGroup].length; var total = 0; for (var i = 0; i < nSeg; ++i) { total += yValues[iGroup][i]; } var dataElems = this._dataElems, animate = (this._animDuration>0) var pathElem = svgDoc.getElementById("funnelPathPrototype"); var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(0.00001, 0.00001)"; var x = 0, y = 0, slope = (quadWidth/2)/quadHeight, dx = quadWidth, dy, nextX, nextY; for (var i = nSeg-1; i >= 0; --i) { pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } var valueRatio = (yValues[iGroup][i])/(total); var sb = new ApacheChartBuffer(); sb.append("M").append(x).append(",").append(y); sb.append("L").append(dx).append(",").append(y); dy = (quadHeight)*valueRatio; nextY = y + dy; nextX = quadWidth/2 - slope*(quadHeight-(nextY) ); dx = quadWidth - nextX; if(i != 0) { sb.append("L").append(dx).append(",").append(nextY); sb.append("L").append(nextX).append(",").append(nextY); sb.append("Z"); } else { var startTipY = (dy/3<=ApacheFunnelChart._MAX_FUNNEL_TIP)?y+(dy - dy/3): quadHeight-ApacheFunnelChart._MAX_FUNNEL_TIP; nextX = quadWidth/2 - slope*(quadHeight-(startTipY) ); dx = quadWidth - nextX; sb.append("L").append(dx).append(",").append(startTipY); sb.append("L").append(dx).append(",").append(quadHeight); sb.append("L").append(nextX).append(",").append(quadHeight); sb.append("L").append(nextX).append(",").append(startTipY); sb.append("Z"); } pathElem.setAttribute("d", sb.toString()); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("yValueIndex", iGroup); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); fnlContainer.appendChild(pathElem); y = nextY; x = nextX; } } ApacheFunnelChart.prototype._drawPerspectiveFunnel = function( fnlContainer, quadWidth, quadHeight, iGroup) { var svgDoc = this._svgDoc, model = this._model, yValues = model.getYValues(); var groupLabels = model.getGroupLabels(), seriesColors = model.getSeriesColors(); if(iGroup == -1) iGroup = 0; // Number of segments var nSeg = yValues[iGroup].length; var total = 0; for (var i = 0; i < nSeg; ++i) { total += yValues[iGroup][i]; } var dataElems = this._dataElems, animate = (this._animDuration>0) var pathElem = svgDoc.getElementById("funnelPathPrototype"); var gradientsUsed = this._gradientsUsed; var defaultTransform = "scale(0.00001, 0.00001)"; var x = 0, y = 0, slope = (quadWidth/2)/quadHeight, dx = quadWidth, dy, nextX, oldDx, nextY; // the ring height is 1/12 of the width var rx = dx/2, ry = dx/24, oldRx, oldRy; for (var i = nSeg-1; i >= 0; --i) { pathElem = pathElem.cloneNode(false); if(animate) { dataElems.push(pathElem); pathElem.setAttribute("transform",defaultTransform); } var valueRatio = (yValues[iGroup][i])/(total); var sb = new ApacheChartBuffer(); sb.append("M").append(x).append(",").append(y); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(dx).append(",").append(y); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(x).append(",").append(y); oldDx = dx; oldRx = rx; oldRy = ry; dy = (quadHeight)*valueRatio; nextY = y + dy; nextX = quadWidth/2 - slope*(quadHeight-(nextY) ); dx = quadWidth - nextX; rx = (dx-nextX)/2; ry = rx/12; if(i != 0) { sb.append("L").append(nextX).append(",").append(nextY); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(dx).append(",").append(nextY); sb.append("L").append(oldDx).append(",").append(y); } else { var startTipY = (dy/3<=ApacheFunnelChart._MAX_FUNNEL_TIP)?y+(dy - dy/3): quadHeight-ApacheFunnelChart._MAX_FUNNEL_TIP; nextX = quadWidth/2 - slope*(quadHeight-(startTipY) ); dx = quadWidth - nextX; rx = (dx-nextX)/2; ry = rx/12; sb.append("L").append(nextX).append(",").append(startTipY); sb.append("L").append(nextX).append(",").append(quadHeight); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(dx).append(",").append(quadHeight); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(nextX).append(",").append(quadHeight); sb.append("A").append(rx).append(",").append(ry); sb.append(" 0 1,0 ").append(dx).append(",").append(quadHeight); sb.append("L").append(dx).append(",").append(startTipY); sb.append("L").append(oldDx).append(",").append(y); } pathElem.setAttribute("d", sb.toString()); if(gradientsUsed) pathElem.setAttribute("fill", "url(#gradient"+i+")"); else pathElem.setAttribute("fill", seriesColors[i]); pathElem.setAttribute("stroke", seriesColors[i]); pathElem.setAttribute("stroke-width", 1); pathElem.setAttribute("yValueIndex", iGroup); pathElem.setAttribute("seriesIndex", i); if(this._tooltipsVisible) { pathElem.addEventListener("mouseover",this.ShowToolTipCallback,false); pathElem.addEventListener("mouseout",this.HideToolTipCallback,false); } pathElem.addEventListener("click",this.ClickCallback,false); fnlContainer.appendChild(pathElem); y = nextY; x = nextX; } } ApacheFunnelChart.prototype.GetToolTipLocation = function(e, ttBBox) { var evtTarget = e.target; var targetBBox = evtTarget.getBBox(); var ctm = evtTarget.parentNode.getCTM(); return {x:(ctm.e+targetBBox.x+targetBBox.width/2), y:(ctm.f+targetBBox.y+targetBBox.height/2 - ttBBox.height)}; } ApacheFunnelChart._MAX_FUNNEL_TIP = 16; //////////////////////////////////////////////////////////////////// // Circular gauge chart subclass //////////////////////////////////////////////////////////////////// function ApacheGaugeChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheChart, ApacheGaugeChart); ApacheGaugeChart.prototype.Init = function( type, model, svgEmbedId, isPerspective, legendPosition) { ApacheGaugeChart.superclass.Init.call(this, type, model, svgEmbedId, isPerspective, legendPosition); //this._markerTextGroup = undefined; //this._animCx; //this._animCy; } ApacheGaugeChart.prototype.SetDataAnimStep = function(ratio) { var animElems = this._dataElems, animCount = animElems.length; var model = this._model, yValues = model.getYValues(); for(var i = 0; i < animCount; ++i) { // For Dial Chart only one value is applicable var yValue = yValues[i][0]; this.SetIndicatorPosition(yValue, animElems[i], ratio); } } ApacheGaugeChart.prototype.SetIndicatorPosition = function(yValue, indicator, ratio) { var theta, cx = this._animCx, cy = this._animCy; var model = this._model, minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var valueRatio = ratio*(yValue - minValue)/(maxValue-minValue); theta = Math.PI/6 + valueRatio*(5*Math.PI/3); if(theta < Math.PI/2) theta += 3*Math.PI/2; else theta -= Math.PI/2; theta *= 180/Math.PI; indicator.setAttribute("transform", "rotate("+theta+" "+cx+" "+cy+")"); } ApacheGaugeChart.prototype.DrawChartData = function() { if(this._yMinorGridCount<0) this._yMinorGridCount = 4; var rootElem = this._rootElement; // calculate the number of rows and columns var model = this._model, yValues = model.getYValues(), yValueCount = yValues.length var groupLabels = model.getGroupLabels(), groupCount = groupLabels?groupLabels.length:1; var nCols = Math.ceil(Math.sqrt(yValueCount)), nRows = Math.round(Math.sqrt(yValueCount)); var margins = this._margins, dx=margins.left, dy=margins.top; var quadWidth = (this._width - margins.left - margins.right)/nCols; var vGap = 2*ApacheChart._TEXT_MARGIN; var quadHeight = (this._height - margins.top - margins.bottom - (nRows-1)*vGap)/nRows; var labelElem = this._svgDoc.getElementById("groupLabelPrototype"); for(var i = 0; i= yValueCount) break; var groupLabel = (iGroup == -1)?null:groupLabels[iGroup]; var gaugeContainer = rootElem.cloneNode(false); rootElem.appendChild(gaugeContainer); if(groupLabel) labelElem = labelElem.cloneNode(true); var newHeight = this.DrawGroupLabelTitle(groupLabel, rootElem, labelElem, dx, dy, quadWidth, quadHeight); var newWidth = quadWidth - 2*ApacheChart._TEXT_MARGIN; this.DrawDial(gaugeContainer, newWidth, newHeight, iGroup); gaugeContainer.setAttribute("transform", "translate("+(dx+ApacheChart._TEXT_MARGIN)+","+dy+")"); if(groupLabel) { var gBBox = gaugeContainer.getBBox(), gHeight = gBBox.height; if(gHeight < newHeight-vGap) { var newY = parseInt(labelElem.getAttribute("y")); newY -= (newHeight-gHeight)/2-vGap; labelElem.setAttribute("y", newY); } } dx +=quadWidth; } dx=margins.left; dy +=quadHeight+vGap; } } ApacheGaugeChart.prototype.DrawLegend = function() { // Legend does not make sense for a gauge } ApacheGaugeChart.prototype.ComputeMinMaxValues = function() { } ApacheGaugeChart.prototype.DrawGroupLabels = function() { } ApacheGaugeChart.prototype.LayoutGroupLabels = function() { } ApacheGaugeChart.prototype.DrawGrid = function() { } ApacheGaugeChart.prototype.DrawYValueLabels = function() { } ApacheGaugeChart.prototype.LayoutYValueLabels = function() { } ApacheGaugeChart.prototype.DrawDial = function( gaugeContainer, quadWidth, quadHeight, iGroup) { var svgDoc = this._svgDoc, model = this._model; if(iGroup == -1) iGroup = 0; var dataElems = this._dataElems, animate = (this._animDuration>0); var templateName = this.GetGaugeTemplateName(); var gauge = svgDoc.getElementById(templateName).cloneNode(true); gaugeContainer.appendChild(gauge); var gaugeBBox = gauge.getBBox(), gaugeWidth = gaugeBBox.width, gaugeR = gaugeWidth/2; var gElem = this._markerTextGroup; var groups = gauge.getElementsByTagName("g"); var indicator = groups.item(groups.length-1); indicator.setAttribute("yValueIndex", iGroup); indicator.setAttribute("seriesIndex", 0); if(this._tooltipsVisible) { indicator.addEventListener("mouseover",this.ShowToolTipCallback,false); indicator.addEventListener("mouseout",this.HideToolTipCallback,false); } indicator.addEventListener("click",this.ClickCallback,false); if(this._animCx == null) { this._animCx = indicator.getAttribute("_pivotCenterX"); this._animCy = indicator.getAttribute("_pivotCenterY"); } if(animate) dataElems.push(indicator); else { // If there is no animation lets move the indicators to the last position this.SetIndicatorPosition(this._model.getYValues()[iGroup][0], indicator, 1); } if(gElem!=null) { gElem = gElem.cloneNode(true); gauge.appendChild(gElem); } else { this.CreateTextMarkerGroup(gauge, gaugeR); } this.ScaleGauge(gauge, quadWidth, quadHeight, gaugeWidth, gaugeBBox.height); } ApacheGaugeChart.prototype.GetGaugeTemplateName = function() { return "circularGauge"; } ApacheGaugeChart.prototype.CreateTextMarkerGroup = function(gauge, gaugeR) { var svgDoc = this._svgDoc, model = this._model; var gElem = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); gauge.appendChild(gElem); this._markerTextGroup = gElem; var majorMarker = svgDoc.getElementById("gaugeMarkerMajor"), majorMarkerCount = this._yMajorGridCount, minorMarker = svgDoc.getElementById("gaugeMarkerMinor"), minorMarkerCount = this._yMinorGridCount, textElem = svgDoc.getElementById("gaugeTextPrototype"); var paths = gauge.getElementsByTagName("path"); var markerContainerR = parseInt(gauge.getAttribute("_markerRadius")); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var x, y, angle, textMargin; for(var i=0, theta = Math.PI/6; i<=majorMarkerCount; ++i, theta += (5*Math.PI/3)/majorMarkerCount) { var adjustedTheta; if(theta < Math.PI/2) adjustedTheta = theta +3*Math.PI/2; else adjustedTheta = theta - Math.PI/2; x = gaugeR-markerContainerR*(Math.cos(adjustedTheta)); y = gaugeR-markerContainerR*(Math.sin(adjustedTheta)); var angle = adjustedTheta*180/Math.PI; var marker = majorMarker.cloneNode(true); gElem.appendChild(marker); marker.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+") rotate("+angle.toFixed(0)+" 0 0)"); var value = minValue + i*(maxValue-minValue)/(majorMarkerCount); textElem = textElem.cloneNode(true); textElem.firstChild.data = this._formatValue(value); gElem.appendChild(textElem); var textBBox = textElem.getBBox(); if(i == 0) { textMargin = textBBox.height/2; } x = gaugeR-(markerContainerR-textMargin)*(Math.cos(adjustedTheta)); y = gaugeR-(markerContainerR-textMargin)*(Math.sin(adjustedTheta)); if(theta >= 5*Math.PI/6 && theta <= 7*Math.PI/6) { y += textBBox.height; x -= textBBox.width/2 } else { y += textBBox.height/2; if(theta < Math.PI) x += 2*ApacheChart._TEXT_MARGIN; else x -= textBBox.width + 2*ApacheChart._TEXT_MARGIN; } textElem.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+")"); } for(var i=(minorMarkerCount+1), theta = Math.PI/6+(5*Math.PI/3)/(majorMarkerCount*minorMarkerCount); i<=(majorMarkerCount+1)*minorMarkerCount; ++i, theta += (5*Math.PI/3)/(majorMarkerCount*minorMarkerCount)) { if(i%minorMarkerCount == 0) continue; var adjustedTheta; if(theta < Math.PI/2) adjustedTheta = theta +3*Math.PI/2; else adjustedTheta = theta - Math.PI/2; var x = gaugeR-markerContainerR*(Math.cos(adjustedTheta)); var y = gaugeR-markerContainerR*(Math.sin(adjustedTheta)); var angle = adjustedTheta*180/Math.PI; var marker = minorMarker.cloneNode(true); gElem.appendChild(marker); marker.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+") rotate("+angle.toFixed(0)+" 0 0)"); } } ApacheGaugeChart.prototype.ScaleGauge = function(gauge, quadWidth, quadHeight, gaugeWidth, gaugeHeight) { var minSide = Math.min(quadWidth, quadHeight); var tx = (minSide == quadWidth)?0: (quadWidth-minSide)/2, ty = (minSide == quadHeight)?0: (quadHeight-minSide)/2; var scale = minSide/gaugeWidth; gauge.setAttribute("transform","translate("+tx+","+ty+") scale("+scale+","+scale+")"); } ApacheGaugeChart.prototype.GetChartEvent = function(e) { var evtTarget = e.target; while(evtTarget != null && evtTarget.tagName != "g") evtTarget = evtTarget.parentNode; if(evtTarget == null) return null; var i = parseInt(evtTarget.getAttribute("yValueIndex")); var model = this._model, yValues = model.getYValues(); return new ApacheChartEvent(null,[i], [yValues[i][0]], null); } ApacheGaugeChart.prototype.FillToolTipData = function(boundingRectElem, circleElem, e) { var chartEvent = this.GetChartEvent(e); if(chartEvent == null) return; var value = chartEvent.getYValues()[0]; var groupLabels = this._model.getGroupLabels(); var textElem = boundingRectElem.nextSibling.nextSibling; textElem.firstChild.data = ""+this._formatValue(value); var labelWidth = textElem.getBBox().width; //We do not need the next label textElem = textElem.nextSibling.nextSibling; textElem.firstChild.data = ""; var rectWidth = labelWidth+2*ApacheChart._TEXT_MARGIN; boundingRectElem.setAttribute("width",rectWidth); boundingRectElem.setAttribute("stroke", seriesColors[0]); circleElem.setAttribute("r",0); } ApacheGaugeChart.prototype.GetToolTipLocation = function(e, ttBBox) { return {x:(e.clientX+20), y:(e.clientY+20)}; } //////////////////////////////////////////////////////////////////// // Semi-Circular gauge chart subclass //////////////////////////////////////////////////////////////////// function ApacheSemiGaugeChart( type, model, svgEmbedId, isPerspective, legendPosition) { this.Init(type, model, svgEmbedId, isPerspective, legendPosition); } ApacheChartObj.Inherit(ApacheGaugeChart, ApacheSemiGaugeChart); ApacheSemiGaugeChart.prototype.SetIndicatorPosition = function(yValue, indicator, ratio) { var theta, cx = this._animCx, cy = this._animCy; var model = this._model, minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var valueRatio = ratio*(yValue - minValue)/(maxValue-minValue); theta = valueRatio*Math.PI; theta *= 180/Math.PI; indicator.setAttribute("transform", "rotate("+theta+" "+cx+" "+cy+")"); } ApacheSemiGaugeChart.prototype.GetGaugeTemplateName = function() { return "semiGauge"; } ApacheSemiGaugeChart.prototype.CreateTextMarkerGroup = function(gauge, gaugeR) { var svgDoc = this._svgDoc, model = this._model; gElem = svgDoc.createElementNS("http://www.w3.org/2000/svg", "g"); gauge.appendChild(gElem); this._markerTextGroup = gElem; var majorMarker = svgDoc.getElementById("gaugeMarkerMajor"), majorMarkerCount = this._yMajorGridCount, minorMarker = svgDoc.getElementById("gaugeMarkerMinor"), minorMarkerCount = this._yMinorGridCount, textElem = svgDoc.getElementById("gaugeTextPrototype"); var paths = gauge.getElementsByTagName("path"); var markerContainerR = parseInt(gauge.getAttribute("_markerRadius")); var minValue = model.getMinYValue(), maxValue = model.getMaxYValue(); var x, y, angle, textMargin; for(var i=0; i<=majorMarkerCount; ++i) { var theta = i*Math.PI/majorMarkerCount; x = gaugeR-markerContainerR*(Math.cos(theta)); y = gaugeR-markerContainerR*(Math.sin(theta)); var angle = theta*180/Math.PI; var marker = majorMarker.cloneNode(true); gElem.appendChild(marker); marker.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+") rotate("+angle.toFixed(0)+" 0 0)"); var value = minValue + i*(maxValue-minValue)/(majorMarkerCount); textElem = textElem.cloneNode(true); textElem.firstChild.data = this._formatValue(value); gElem.appendChild(textElem); var textBBox = textElem.getBBox(); if(i == 0) { textMargin = textBBox.height/2; } x = gaugeR-(markerContainerR-textMargin)*(Math.cos(theta)); y = gaugeR-(markerContainerR-textMargin)*(Math.sin(theta)); if(theta >= Math.PI/3 && theta <= 2*Math.PI/3) { y += textBBox.height; x -= textBBox.width/2 } else { y += textBBox.height/2; if(theta < Math.PI/2) x += 2*ApacheChart._TEXT_MARGIN; else x -= textBBox.width + 2*ApacheChart._TEXT_MARGIN; } textElem.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+")"); } for(var i=1; i<=(majorMarkerCount)*minorMarkerCount; ++i) { if(i%minorMarkerCount == 0) continue; var theta = i*Math.PI/(majorMarkerCount*minorMarkerCount); var x = gaugeR-markerContainerR*(Math.cos(theta)); var y = gaugeR-markerContainerR*(Math.sin(theta)); var angle = theta*180/Math.PI; var marker = minorMarker.cloneNode(true); gElem.appendChild(marker); marker.setAttribute("transform", "translate("+x.toFixed(0)+","+y.toFixed(0)+") rotate("+angle.toFixed(0)+" 0 0)"); } } ApacheSemiGaugeChart.prototype.ScaleGauge = function( gauge, quadWidth, quadHeight, gaugeWidth, gaugeHeight) { var sx = quadWidth/gaugeWidth, sy = quadHeight/gaugeHeight; var scale = Math.min(sx, sy); var tx = (quadWidth<=gaugeWidth)?0:(quadWidth-gaugeWidth)/2, ty = (quadHeight<=gaugeHeight)?0:(quadHeight-gaugeHeight)/2; gauge.setAttribute("transform","translate("+tx+","+ty+") scale("+scale+","+scale+")"); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy