/** * @class Ext.sparkline.Line * * Plots a line graph based upon the input {@link #values} array. * * See {@link Ext.sparkline.Base the base class} for a simple example. */Ext.define('Ext.sparkline.Line', { extend: 'Ext.sparkline.Base', requires: [ 'Ext.sparkline.RangeMap' ], alias: 'widget.sparklineline', config: { /** * @cfg {String} [fillColor=#def] The hex value for fill color. */ fillColor: '#def', /** * @cfg {String} [spotColor=#f80] The colour of the final value marker. Set to false or an empty string to hide it. */ spotColor: '#f80', /** * @cfg {String} [highlightSpotColor=#5f5] The colour of value marker spots when mouseovered. */ highlightSpotColor: '#5f5', /** * @cfg {String} [highlightLineColor=#f22] The colour of value line shown when the graph is mouseovered. */ highlightLineColor: '#f22', /** * @cfg {Number} [spotRadius=1.5] The pixel radius of min, max and final value dots. */ spotRadius: 1.5, /** * @cfg {String} [minSpotColor=#f80] The colour of the mimimum value marker. Set to false or an empty string to hide it. */ minSpotColor: '#f80', /** * @cfg {String} [maxSpotColor=#f80] The colour of the maximum value marker. Set to false or an empty string to hide it. */ maxSpotColor: '#f80', /** * @cfg {Number} [lineWidth=1] The pixel width of the line plotted. */ lineWidth: 1, /** * @cfg {Number} [normalRangeMin] See {@link #normalRangeMax} The minimum value to overlay a "normal range bar" over the graph using the {@link #normalRangeColor}. */ normalRangeMin: null, /** * @cfg {Number} [normalRangeMax] See {@link #normalRangeMin} The maximum value to overlay a "normal range bar" over the graph using the {@link #normalRangeColor}. */ normalRangeMax: null, /** * @cfg {String} [normalRangeColor=#ccc] See {@link #normalRangeMin} and {@link #normalRangeMax} The color of the undererlayed "normal range bar". */ normalRangeColor: '#ccc', /** * @cfg {Boolean} [drawNormalOnTop=false] Configure as `true` to draw the normal range overlaying the chart. */ drawNormalOnTop: false, /** * @cfg {Number} [chartRangeMin] The minimum value to use for the range of Y values of the chart - Defaults to the minimum value supplied. */ chartRangeMin: null, /** * @cfg {Number} [chartRangeMax] The maximum value to use for the range of Y values of the chart - Defaults to the minimum value supplied. */ chartRangeMax: null, /** * @cfg {Number} [chartRangeMinX] The minimum value to use for the X value of the chart. */ chartRangeMinX: null, /** * @cfg {Number} [chartRangeMaxX] The maximum value to use for the X value of the chart. */ chartRangeMaxX: null, /** * @cfg {Object} [valueSpots] An object which uses range specifiers as keys to indicate spot color values * for range of values. A range specifier is of the form `[number]:[number]` indicating start and end range. * Omitting aither means an open ended range. For example to render green spots on all values less than 50 * and red on values higher than 50 use: * * { * // Open ended range, with max value 49 * ":49": "green", * * // Open ended range, with min value 50 * "50:": "red" * } */ valueSpots: null }, tipTpl: '● {prefix}{y}{suffix}', applyValueSpots: function(valueSpots) { if (valueSpots && !valueSpots.get) { valueSpots = new Ext.sparkline.RangeMap(valueSpots); } this.applyConfigChange(); return valueSpots; }, onUpdate: function () { this.vertices = []; this.regionMap = []; this.xvalues = []; this.yvalues = []; this.yminmax = []; }, getRegion: function(x, y) { var i, regionMap = this.regionMap; // maps regions to value positions for (i = regionMap.length; i--;) { if (regionMap[i] !== null && x >= regionMap[i][0] && x <= regionMap[i][1]) { return regionMap[i][2]; } } return undefined; }, getRegionFields: function(region) { return { isNull: this.yvalues[region] === null, x: this.xvalues[region], y: this.yvalues[region], color: this.getLineColor(), fillColor: this.getFillColor(), offset: region }; }, renderHighlight: function(region) { var me = this, canvas = me.canvas, vertex = me.vertices[region], spotRadius = me.getSpotRadius(), highlightSpotColor = me.getHighlightSpotColor(), highlightLineColor = me.getHighlightLineColor(); if (!vertex) { return; } if (spotRadius && highlightSpotColor) { canvas.drawCircle(vertex[0], vertex[1], spotRadius, null, highlightSpotColor).append(); } if (highlightLineColor) { canvas.drawLine(vertex[0], me.canvasTop, vertex[0], me.canvasTop + me.getHeight(), highlightLineColor).append(); } }, scanValues: function () { var me = this, values = me.values, valcount = values.length, xvalues = me.xvalues, yvalues = me.yvalues, yminmax = me.yminmax, i, val, isStr, isArray, sp; for (i = 0; i < valcount; i++) { val = values[i]; isStr = typeof(values[i]) === 'string'; isArray = typeof(values[i]) === 'object' && values[i] instanceof Array; sp = isStr && values[i].split(':'); if (isStr && sp.length === 2) { // x:y xvalues.push(Number(sp[0])); yvalues.push(Number(sp[1])); yminmax.push(Number(sp[1])); } else if (isArray) { xvalues.push(val[0]); yvalues.push(val[1]); yminmax.push(val[1]); } else { xvalues.push(i); if (values[i] === null || values[i] === 'null') { yvalues.push(null); } else { yvalues.push(Number(val)); yminmax.push(Number(val)); } } } if (me.xvalues) { xvalues = me.xvalues; } me.maxy = me.maxyorg = Math.max.apply(Math, yminmax); me.miny = me.minyorg = Math.min.apply(Math, yminmax); me.maxx = Math.max.apply(Math, xvalues); me.minx = Math.min.apply(Math, xvalues); me.xvalues = xvalues; me.yvalues = yvalues; me.yminmax = yminmax; }, processRangeOptions: function () { var me = this, normalRangeMin = me.getNormalRangeMin(), normalRangeMax = me.getNormalRangeMax(), chartRangeMin = me.getChartRangeMin(), chartRangeMinX = me.getChartRangeMinX(), chartRangeMax = me.getChartRangeMax(), chartRangeMaxX = me.getChartRangeMaxX(); if (normalRangeMin != null) { if (normalRangeMin < me.miny) { me.miny = normalRangeMin; } if (normalRangeMax > me.maxy) { me.maxy = normalRangeMax; } } if (chartRangeMin != null && (me.chartRangeClip || chartRangeMin < me.miny)) { me.miny = chartRangeMin; } if (chartRangeMax != null && (me.chartRangeClip || chartRangeMax > me.maxy)) { this.maxy = chartRangeMax; } if (chartRangeMinX != null && (me.chartRangeClipX || chartRangeMinX < me.minx)) { me.minx = chartRangeMinX; } if (chartRangeMaxX != null && (me.chartRangeClipX || chartRangeMaxX > me.maxx)) { me.maxx = chartRangeMaxX; } }, drawNormalRange: function (canvasLeft, canvasTop, canvasHeight, canvasWidth, rangey) { var normalRangeMin = this.getNormalRangeMin(), normalRangeMax = this.getNormalRangeMax(), ytop = canvasTop + Math.round(canvasHeight - (canvasHeight * ((normalRangeMax - this.miny) / rangey))), height = Math.round((canvasHeight * (normalRangeMax - normalRangeMin)) / rangey); this.canvas.drawRect(canvasLeft, ytop, canvasWidth, height, undefined, this.normalRangeColor).append(); }, renderGraph: function () { var me = this, canvas = me.canvas, canvasWidth = me.getWidth(), canvasHeight = me.getHeight(), vertices = me.vertices, spotRadius = me.getSpotRadius(), regionMap = me.regionMap, rangeX, Y, yvallast, canvasTop, canvasLeft, vertex, path, paths, x, y, xNext, xPos, xPosNext, last, next, yValCount, lineShapes, fillShapes, plen, valueSpots = me.getValueSpots(), hlSpotsEnabled, color, xValues, yValues, i, spotColor = me.getSpotColor(), minSpotColor = me.getMinSpotColor(), maxSpotColor = me.getMaxSpotColor(), normalRangeMin = me.getNormalRangeMin(), drawNormalOnTop = me.getDrawNormalOnTop(); if (!me.callParent()) { return; } me.scanValues(); me.processRangeOptions(); xValues = me.xvalues; yValues = me.yvalues; if (!me.yminmax.length || me.yvalues.length < 2) { // empty or all null valuess return; } canvasTop = canvasLeft = 0; rangeX = me.maxx - me.minx === 0 ? 1 : me.maxx - me.minx; Y = me.maxy - me.miny === 0 ? 1 : me.maxy - me.miny; yvallast = me.yvalues.length - 1; if (spotRadius && (canvasWidth < (spotRadius * 4) || canvasHeight < (spotRadius * 4))) { spotRadius = 0; } if (spotRadius) { // adjust the canvas size as required so that spots will fit hlSpotsEnabled = me.getHighlightSpotColor() && !me.disableInteraction; if (hlSpotsEnabled || minSpotColor || (spotColor && yValues[yvallast] === me.miny)) { canvasHeight -= Math.ceil(spotRadius); } if (hlSpotsEnabled || maxSpotColor || (spotColor && yValues[yvallast] === me.maxy)) { canvasHeight -= Math.ceil(spotRadius); canvasTop += Math.ceil(spotRadius); } if (hlSpotsEnabled || ((minSpotColor || maxSpotColor) && (yValues[0] === me.miny || yValues[0] === me.maxy))) { canvasLeft += Math.ceil(spotRadius); canvasWidth -= Math.ceil(spotRadius); } if (hlSpotsEnabled || spotColor || (minSpotColor || maxSpotColor && (yValues[yvallast] === me.miny || yValues[yvallast] === me.maxy))) { canvasWidth -= Math.ceil(spotRadius); } } canvasHeight--; if (normalRangeMin != null && !drawNormalOnTop) { me.drawNormalRange(canvasLeft, canvasTop, canvasHeight, canvasWidth, Y); } path = []; paths = [path]; last = next = null; yValCount = yValues.length; for (i = 0; i < yValCount; i++) { x = xValues[i]; xNext = xValues[i + 1]; y = yValues[i]; xPos = canvasLeft + Math.round((x - me.minx) * (canvasWidth / rangeX)); xPosNext = i < yValCount - 1 ? canvasLeft + Math.round((xNext - me.minx) * (canvasWidth / rangeX)) : canvasWidth; next = xPos + ((xPosNext - xPos) / 2); regionMap[i] = [last || 0, next, i]; last = next; if (y === null) { if (i) { if (yValues[i - 1] !== null) { path = []; paths.push(path); } vertices.push(null); } } else { if (y < me.miny) { y = me.miny; } if (y > me.maxy) { y = me.maxy; } if (!path.length) { // previous value was null path.push([xPos, canvasTop + canvasHeight]); } vertex = [xPos, canvasTop + Math.round(canvasHeight - (canvasHeight * ((y - this.miny) / Y)))]; path.push(vertex); vertices.push(vertex); } } lineShapes = []; fillShapes = []; plen = paths.length; for (i = 0; i < plen; i++) { path = paths[i]; if (path.length) { if (me.fillColor) { path.push([path[path.length - 1][0], (canvasTop + canvasHeight)]); fillShapes.push(path.slice(0)); path.pop(); } // if there's only a single point in this path, then we want to display it // as a vertical line which means we keep path[0] as is if (path.length > 2) { // else we want the first value path[0] = [path[0][0], path[1][1]]; } lineShapes.push(path); } } // draw the fill first, then optionally the normal range, then the line on top of that plen = fillShapes.length; for (i = 0; i < plen; i++) { canvas.drawShape(fillShapes[i], me.fillColor, me.fillColor).append(); } if (normalRangeMin != null && drawNormalOnTop) { me.drawNormalRange(canvasLeft, canvasTop, canvasHeight, canvasWidth, Y); } plen = lineShapes.length; for (i = 0; i < plen; i++) { canvas.drawShape(lineShapes[i], me.getLineColor(), null, me.getLineWidth()).append(); } if (spotRadius && valueSpots) { if (valueSpots.get == null) { valueSpots = new Ext.sparkline.RangeMap(valueSpots); } for (i = 0; i < yValCount; i++) { color = valueSpots.get(yValues[i]); if (color) { canvas.drawCircle(canvasLeft + Math.round((xValues[i] - me.minx) * (canvasWidth / rangeX)), canvasTop + Math.round(canvasHeight - (canvasHeight * ((yValues[i] - me.miny) / Y))), spotRadius, null, color).append(); } } } if (spotRadius && spotColor && yValues[yvallast] != null) { canvas.drawCircle(canvasLeft + Math.round((xValues[xValues.length - 1] - me.minx) * (canvasWidth / rangeX)), canvasTop + Math.round(canvasHeight - (canvasHeight * ((yValues[yvallast] - me.miny) / Y))), spotRadius, null, spotColor).append(); } if (me.maxy !== me.minyorg) { if (spotRadius && minSpotColor) { x = xValues[Ext.Array.indexOf(yValues, me.minyorg)]; canvas.drawCircle(canvasLeft + Math.round((x - me.minx) * (canvasWidth / rangeX)), canvasTop + Math.round(canvasHeight - (canvasHeight * ((me.minyorg - me.miny) / Y))), spotRadius, null, minSpotColor).append(); } if (spotRadius && maxSpotColor) { x = xValues[Ext.Array.indexOf(yValues, me.maxyorg)]; canvas.drawCircle(canvasLeft + Math.round((x - me.minx) * (canvasWidth / rangeX)), canvasTop + Math.round(canvasHeight - (canvasHeight * ((me.maxyorg - me.miny) / Y))), spotRadius, null, maxSpotColor).append(); } } me.canvasTop = canvasTop; // If mouse is over, apply the highlight if (me.currentPageXY && me.el.getRegion().contains(me.currentPageXY)) { me.updateDisplay(); } canvas.render(); }});