Форк Rambox
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

/**
* @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();
}
});