linuxwindowsinboxwhatsappicloudtweetdeckhipchattelegramhangoutsslackgmailskypefacebook-workplaceoutlookemailmicrosoft-teamsdiscordmessengercustom-servicesmacos
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
449 lines
16 KiB
449 lines
16 KiB
/** |
|
* @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} [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, |
|
|
|
tipTpl: new Ext.XTemplate('● {prefix}{y}{suffix}'), |
|
|
|
/** |
|
* @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 |
|
}, |
|
|
|
applyValueSpots: function(valueSpots) { |
|
if (valueSpots && !valueSpots.get) { |
|
valueSpots = new Ext.sparkline.RangeMap(valueSpots); |
|
} |
|
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(); |
|
} |
|
}); |