highcharts-rails 4.0.3 → 4.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.markdown +36 -0
- data/app/assets/javascripts/highcharts.js +147 -106
- data/app/assets/javascripts/highcharts/adapters/standalone-framework.js +17 -4
- data/app/assets/javascripts/highcharts/highcharts-3d.js +68 -37
- data/app/assets/javascripts/highcharts/highcharts-more.js +3 -4
- data/app/assets/javascripts/highcharts/modules/canvas-tools.js +1 -1
- data/app/assets/javascripts/highcharts/modules/data.js +476 -73
- data/app/assets/javascripts/highcharts/modules/drilldown.js +43 -23
- data/app/assets/javascripts/highcharts/modules/exporting.js +1 -1
- data/app/assets/javascripts/highcharts/modules/funnel.js +1 -1
- data/app/assets/javascripts/highcharts/modules/heatmap.js +18 -6
- data/app/assets/javascripts/highcharts/modules/no-data-to-display.js +2 -2
- data/app/assets/javascripts/highcharts/modules/solid-gauge.js +1 -1
- data/lib/highcharts/version.rb +1 -1
- metadata +1 -1
@@ -1,5 +1,5 @@
|
|
1
1
|
/**
|
2
|
-
* @license Highcharts JS v4.0.
|
2
|
+
* @license Highcharts JS v4.0.4 (2014-09-02)
|
3
3
|
*
|
4
4
|
* Standalone Highcharts Framework
|
5
5
|
*
|
@@ -15,6 +15,7 @@ var UNDEFINED,
|
|
15
15
|
emptyArray = [],
|
16
16
|
timers = [],
|
17
17
|
timerId,
|
18
|
+
animSetters = {},
|
18
19
|
Fx;
|
19
20
|
|
20
21
|
Math.easeInOutSine = function (t, b, c, d) {
|
@@ -184,6 +185,7 @@ function augment(obj) {
|
|
184
185
|
|
185
186
|
|
186
187
|
return {
|
188
|
+
|
187
189
|
/**
|
188
190
|
* Initialize the adapter. This is run once as Highcharts is first run.
|
189
191
|
*/
|
@@ -292,10 +294,14 @@ return {
|
|
292
294
|
elem = this.elem,
|
293
295
|
elemelem = elem.element; // if destroyed, it is null
|
294
296
|
|
297
|
+
// Animation setter defined from outside
|
298
|
+
if (animSetters[this.prop]) {
|
299
|
+
animSetters[this.prop](this);
|
300
|
+
|
295
301
|
// Animating a path definition on SVGElement
|
296
|
-
if (paths && elemelem) {
|
302
|
+
} else if (paths && elemelem) {
|
297
303
|
elem.attr('d', pathAnim.step(paths[0], paths[1], this.now, this.toD));
|
298
|
-
|
304
|
+
|
299
305
|
// Other animations on SVGElement
|
300
306
|
} else if (elem.attr) {
|
301
307
|
if (elemelem) {
|
@@ -440,7 +446,7 @@ return {
|
|
440
446
|
}
|
441
447
|
|
442
448
|
if (!end) {
|
443
|
-
end =
|
449
|
+
end = prop[name];
|
444
450
|
}
|
445
451
|
fx.custom(start, end, unit);
|
446
452
|
}
|
@@ -454,6 +460,13 @@ return {
|
|
454
460
|
return window.getComputedStyle(el, undefined).getPropertyValue(prop);
|
455
461
|
},
|
456
462
|
|
463
|
+
/**
|
464
|
+
* Add an animation setter for a specific property
|
465
|
+
*/
|
466
|
+
addAnimSetter: function (prop, fn) {
|
467
|
+
animSetters[prop] = fn;
|
468
|
+
},
|
469
|
+
|
457
470
|
/**
|
458
471
|
* Downloads a script and executes a callback when done.
|
459
472
|
* @param {String} scriptLocation
|
@@ -2,7 +2,7 @@
|
|
2
2
|
// @compilation_level SIMPLE_OPTIMIZATIONS
|
3
3
|
|
4
4
|
/**
|
5
|
-
* @license Highcharts JS v4.0.
|
5
|
+
* @license Highcharts JS v4.0.4 (2014-09-02)
|
6
6
|
*
|
7
7
|
* (c) 2009-2013 Torstein Hønsi
|
8
8
|
*
|
@@ -75,6 +75,11 @@ function perspective(points, angle2, angle1, origin) {
|
|
75
75
|
////// HELPER METHODS //////
|
76
76
|
var dFactor = (4 * (Math.sqrt(2) - 1) / 3) / (PI / 2);
|
77
77
|
|
78
|
+
function defined(obj) {
|
79
|
+
return obj !== undefined && obj !== null;
|
80
|
+
}
|
81
|
+
|
82
|
+
|
78
83
|
function curveTo(cx, cy, rx, ry, start, end, dx, dy) {
|
79
84
|
var result = [];
|
80
85
|
if ((end > start) && (end - start > PI / 2 + 0.0001)) {
|
@@ -150,7 +155,7 @@ Highcharts.SVGRenderer.prototype.cuboid = function (shapeArgs) {
|
|
150
155
|
};
|
151
156
|
|
152
157
|
result.attr = function (args) {
|
153
|
-
if (args.shapeArgs || args.x) {
|
158
|
+
if (args.shapeArgs || defined(args.x)) {
|
154
159
|
var shapeArgs = args.shapeArgs || args;
|
155
160
|
var paths = this.renderer.cuboidPath(shapeArgs);
|
156
161
|
this.front.attr({d: paths[0], zIndex: paths[3]});
|
@@ -164,7 +169,7 @@ Highcharts.SVGRenderer.prototype.cuboid = function (shapeArgs) {
|
|
164
169
|
};
|
165
170
|
|
166
171
|
result.animate = function (args, duration, complete) {
|
167
|
-
if (args.x && args.y) {
|
172
|
+
if (defined(args.x) && defined(args.y)) {
|
168
173
|
var paths = this.renderer.cuboidPath(args);
|
169
174
|
this.front.attr({zIndex: paths[3]}).animate({d: paths[0]}, duration, complete);
|
170
175
|
this.top.attr({zIndex: paths[4]}).animate({d: paths[1]}, duration, complete);
|
@@ -333,7 +338,7 @@ Highcharts.SVGRenderer.prototype.arc3d = function (shapeArgs) {
|
|
333
338
|
};
|
334
339
|
|
335
340
|
result.animate = function (args, duration, complete) {
|
336
|
-
if (args.end || args.start) {
|
341
|
+
if (defined(args.end) || defined(args.start)) {
|
337
342
|
this._shapeArgs = this.shapeArgs;
|
338
343
|
|
339
344
|
Highcharts.SVGElement.prototype.animate.call(this, {
|
@@ -348,6 +353,10 @@ Highcharts.SVGRenderer.prototype.arc3d = function (shapeArgs) {
|
|
348
353
|
end = fx.end,
|
349
354
|
pos = fx.pos,
|
350
355
|
sA = Highcharts.merge(start, {
|
356
|
+
x: start.x + ((end.x - start.x) * pos),
|
357
|
+
y: start.y + ((end.y - start.y) * pos),
|
358
|
+
r: start.r + ((end.r - start.r) * pos),
|
359
|
+
innerR: start.innerR + ((end.innerR - start.innerR) * pos),
|
351
360
|
start: start.start + ((end.start - start.start) * pos),
|
352
361
|
end: start.end + ((end.end - start.end) * pos)
|
353
362
|
});
|
@@ -518,7 +527,20 @@ defaultChartOptions.chart.options3d = {
|
|
518
527
|
back: { size: 1, color: 'rgba(255,255,255,0)' }
|
519
528
|
}
|
520
529
|
};
|
521
|
-
|
530
|
+
|
531
|
+
Highcharts.wrap(Highcharts.Chart.prototype, 'init', function (proceed) {
|
532
|
+
var args = [].slice.call(arguments, 1),
|
533
|
+
plotOptions,
|
534
|
+
pieOptions;
|
535
|
+
|
536
|
+
if (args[0].chart.options3d && args[0].chart.options3d.enabled) {
|
537
|
+
plotOptions = args[0].plotOptions || {};
|
538
|
+
pieOptions = plotOptions.pie || {};
|
539
|
+
|
540
|
+
pieOptions.borderColor = Highcharts.pick(pieOptions.borderColor, undefined);
|
541
|
+
}
|
542
|
+
proceed.apply(this, args);
|
543
|
+
});
|
522
544
|
|
523
545
|
Highcharts.wrap(Highcharts.Chart.prototype, 'setChartSize', function (proceed) {
|
524
546
|
proceed.apply(this, [].slice.call(arguments, 1));
|
@@ -851,20 +873,21 @@ Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'translate', function (
|
|
851
873
|
z += (seriesOptions.groupZPadding || 1);
|
852
874
|
|
853
875
|
Highcharts.each(series.data, function (point) {
|
854
|
-
|
855
|
-
|
856
|
-
|
857
|
-
point.shapeType = 'cuboid';
|
858
|
-
shapeArgs.alpha = alpha;
|
859
|
-
shapeArgs.beta = beta;
|
860
|
-
shapeArgs.z = z;
|
861
|
-
shapeArgs.origin = origin;
|
862
|
-
shapeArgs.depth = depth;
|
876
|
+
if (point.y !== null) {
|
877
|
+
var shapeArgs = point.shapeArgs,
|
878
|
+
tooltipPos = point.tooltipPos;
|
863
879
|
|
864
|
-
|
865
|
-
|
866
|
-
|
880
|
+
point.shapeType = 'cuboid';
|
881
|
+
shapeArgs.alpha = alpha;
|
882
|
+
shapeArgs.beta = beta;
|
883
|
+
shapeArgs.z = z;
|
884
|
+
shapeArgs.origin = origin;
|
885
|
+
shapeArgs.depth = depth;
|
867
886
|
|
887
|
+
// Translate the tooltip position in 3d space
|
888
|
+
tooltipPos = perspective([{ x: tooltipPos[0], y: tooltipPos[1], z: z }], alpha, beta, origin)[0];
|
889
|
+
point.tooltipPos = [tooltipPos.x, tooltipPos.y];
|
890
|
+
}
|
868
891
|
});
|
869
892
|
});
|
870
893
|
|
@@ -881,25 +904,29 @@ Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'animate', function (pr
|
|
881
904
|
if (Highcharts.svg) { // VML is too slow anyway
|
882
905
|
if (init) {
|
883
906
|
Highcharts.each(series.data, function (point) {
|
884
|
-
point.
|
885
|
-
|
886
|
-
|
887
|
-
|
888
|
-
if (
|
889
|
-
|
890
|
-
|
891
|
-
|
907
|
+
if (point.y !== null) {
|
908
|
+
point.height = point.shapeArgs.height;
|
909
|
+
point.shapey = point.shapeArgs.y; //#2968
|
910
|
+
point.shapeArgs.height = 1;
|
911
|
+
if (!reversed) {
|
912
|
+
if (point.stackY) {
|
913
|
+
point.shapeArgs.y = point.plotY + yAxis.translate(point.stackY);
|
914
|
+
} else {
|
915
|
+
point.shapeArgs.y = point.plotY + (point.negative ? -point.height : point.height);
|
916
|
+
}
|
892
917
|
}
|
893
918
|
}
|
894
919
|
});
|
895
920
|
|
896
921
|
} else { // run the animation
|
897
922
|
Highcharts.each(series.data, function (point) {
|
898
|
-
point.
|
899
|
-
|
900
|
-
|
901
|
-
|
902
|
-
point.graphic
|
923
|
+
if (point.y !== null) {
|
924
|
+
point.shapeArgs.height = point.height;
|
925
|
+
point.shapeArgs.y = point.shapey; //#2968
|
926
|
+
// null value do not have a graphic
|
927
|
+
if (point.graphic) {
|
928
|
+
point.graphic.animate(point.shapeArgs, series.options.animation);
|
929
|
+
}
|
903
930
|
}
|
904
931
|
});
|
905
932
|
|
@@ -951,14 +978,16 @@ function draw3DPoints(proceed) {
|
|
951
978
|
this.borderWidth = options.borderWidth = options.edgeWidth || 1;
|
952
979
|
|
953
980
|
Highcharts.each(this.data, function (point) {
|
954
|
-
|
981
|
+
if (point.y !== null) {
|
982
|
+
var pointAttr = point.pointAttr;
|
955
983
|
|
956
|
-
|
957
|
-
|
984
|
+
// Set the border color to the fill color to provide a smooth edge
|
985
|
+
this.borderColor = Highcharts.pick(options.edgeColor, pointAttr[''].fill);
|
958
986
|
|
959
|
-
|
960
|
-
|
961
|
-
|
987
|
+
pointAttr[''].stroke = this.borderColor;
|
988
|
+
pointAttr.hover.stroke = Highcharts.pick(states.hover.edgeColor, this.borderColor);
|
989
|
+
pointAttr.select.stroke = Highcharts.pick(states.select.edgeColor, this.borderColor);
|
990
|
+
}
|
962
991
|
});
|
963
992
|
}
|
964
993
|
|
@@ -1108,7 +1137,8 @@ Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'translate', function (pro
|
|
1108
1137
|
});
|
1109
1138
|
|
1110
1139
|
Highcharts.wrap(Highcharts.seriesTypes.pie.prototype.pointClass.prototype, 'haloPath', function (proceed) {
|
1111
|
-
|
1140
|
+
var args = arguments;
|
1141
|
+
return this.series.chart.is3d() ? [] : proceed.call(this, args[1]);
|
1112
1142
|
});
|
1113
1143
|
|
1114
1144
|
Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'drawPoints', function (proceed) {
|
@@ -1119,6 +1149,7 @@ Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'drawPoints', function (pr
|
|
1119
1149
|
|
1120
1150
|
// Set the border color to the fill color to provide a smooth edge
|
1121
1151
|
this.borderWidth = options.borderWidth = options.edgeWidth || 1;
|
1152
|
+
this.borderColor = options.edgeColor = Highcharts.pick(options.edgeColor, options.borderColor, undefined);
|
1122
1153
|
|
1123
1154
|
states.hover.borderColor = Highcharts.pick(states.hover.edgeColor, this.borderColor);
|
1124
1155
|
states.hover.borderWidth = Highcharts.pick(states.hover.edgeWidth, this.borderWidth);
|
@@ -2,7 +2,7 @@
|
|
2
2
|
// @compilation_level SIMPLE_OPTIMIZATIONS
|
3
3
|
|
4
4
|
/**
|
5
|
-
* @license Highcharts JS v4.0.
|
5
|
+
* @license Highcharts JS v4.0.4 (2014-09-02)
|
6
6
|
*
|
7
7
|
* (c) 2009-2014 Torstein Honsi
|
8
8
|
*
|
@@ -1671,11 +1671,10 @@ seriesTypes.waterfall = extendClass(seriesTypes.column, {
|
|
1671
1671
|
*/
|
1672
1672
|
toYData: function (pt) {
|
1673
1673
|
if (pt.isSum) {
|
1674
|
-
return "sum";
|
1674
|
+
return (pt.x === 0 ? null : "sum"); //#3245 Error when first element is Sum or Intermediate Sum
|
1675
1675
|
} else if (pt.isIntermediateSum) {
|
1676
|
-
return "intermediateSum";
|
1676
|
+
return (pt.x === 0 ? null : "intermediateSum"); //#3245
|
1677
1677
|
}
|
1678
|
-
|
1679
1678
|
return pt.y;
|
1680
1679
|
},
|
1681
1680
|
|
@@ -2908,7 +2908,7 @@ if (CanvasRenderingContext2D) {
|
|
2908
2908
|
});
|
2909
2909
|
}
|
2910
2910
|
}/**
|
2911
|
-
* @license Highcharts JS v4.0.
|
2911
|
+
* @license Highcharts JS v4.0.4 (2014-09-02)
|
2912
2912
|
* CanVGRenderer Extension module
|
2913
2913
|
*
|
2914
2914
|
* (c) 2011-2012 Torstein Honsi, Erik Olsson
|
@@ -32,6 +32,11 @@
|
|
32
32
|
* A comma delimited string to be parsed. Related options are startRow, endRow, startColumn
|
33
33
|
* and endColumn to delimit what part of the table is used. The lineDelimiter and
|
34
34
|
* itemDelimiter options define the CSV delimiter formats.
|
35
|
+
*
|
36
|
+
* - dateFormat: String
|
37
|
+
* Which of the predefined date formats in Date.prototype.dateFormats to use to parse date
|
38
|
+
* columns, for example "dd/mm/YYYY" or "YYYY-mm-dd". Defaults to a best guess based on
|
39
|
+
* what format gives valid dates, and prefers ordered dates.
|
35
40
|
*
|
36
41
|
* - endColumn : Integer
|
37
42
|
* In tabular input data, the first row (indexed by 0) to use. Defaults to the last
|
@@ -60,7 +65,8 @@
|
|
60
65
|
* A callback function to access the parsed columns, the two-dimentional input data
|
61
66
|
* array directly, before they are interpreted into series data and categories. See also
|
62
67
|
* the complete callback, that goes in on a later stage where the raw columns are interpreted
|
63
|
-
* into a Highcharts option structure.
|
68
|
+
* into a Highcharts option structure. Return false to stop completion, or call this.complete()
|
69
|
+
* to continue async.
|
64
70
|
*
|
65
71
|
* - parseDate : Function
|
66
72
|
* A callback function to parse string representations of dates into JavaScript timestamps.
|
@@ -69,6 +75,10 @@
|
|
69
75
|
* - rows : Array<Array<Mixed>>
|
70
76
|
* The same as the columns input option, but defining rows intead of columns.
|
71
77
|
*
|
78
|
+
* - seriesMapping : Array<Object>
|
79
|
+
* An array containing object with Point property names along with what column id the
|
80
|
+
* property should be taken from.
|
81
|
+
*
|
72
82
|
* - startColumn : Integer
|
73
83
|
* In tabular input data, the first column (indexed by 0) to use.
|
74
84
|
*
|
@@ -84,13 +94,23 @@
|
|
84
94
|
* endRow, startColumn and endColumn to delimit what part of the table is used.
|
85
95
|
*/
|
86
96
|
|
97
|
+
/*
|
98
|
+
* TODO:
|
99
|
+
* - Handle various date formats
|
100
|
+
* - http://jsfiddle.net/highcharts/114wejdx/
|
101
|
+
* - http://jsfiddle.net/highcharts/ryv67bkq/
|
102
|
+
*/
|
103
|
+
|
87
104
|
// JSLint options:
|
88
|
-
/*global jQuery */
|
105
|
+
/*global jQuery, HighchartsAdapter */
|
89
106
|
|
90
107
|
(function (Highcharts) { // docs
|
91
108
|
|
92
109
|
// Utilities
|
93
|
-
var each = Highcharts.each
|
110
|
+
var each = Highcharts.each,
|
111
|
+
inArray = HighchartsAdapter.inArray,
|
112
|
+
splat = Highcharts.splat,
|
113
|
+
SeriesBuilder;
|
94
114
|
|
95
115
|
|
96
116
|
// The Data constructor
|
@@ -109,6 +129,12 @@
|
|
109
129
|
this.chartOptions = chartOptions;
|
110
130
|
this.columns = options.columns || this.rowsToColumns(options.rows) || [];
|
111
131
|
|
132
|
+
// This is a two-dimensional array holding the raw, trimmed string values
|
133
|
+
// with the same organisation as the columns array. It makes it possible
|
134
|
+
// for example to revert from interpreted timestamps to string-based
|
135
|
+
// categories.
|
136
|
+
this.rawColumns = [];
|
137
|
+
|
112
138
|
// No need to parse or interpret anything
|
113
139
|
if (this.columns.length) {
|
114
140
|
this.dataFound();
|
@@ -135,19 +161,79 @@
|
|
135
161
|
*/
|
136
162
|
getColumnDistribution: function () {
|
137
163
|
var chartOptions = this.chartOptions,
|
164
|
+
options = this.options,
|
165
|
+
xColumns = [],
|
138
166
|
getValueCount = function (type) {
|
139
167
|
return (Highcharts.seriesTypes[type || 'line'].prototype.pointArrayMap || [0]).length;
|
140
168
|
},
|
169
|
+
getPointArrayMap = function (type) {
|
170
|
+
return Highcharts.seriesTypes[type || 'line'].prototype.pointArrayMap;
|
171
|
+
},
|
141
172
|
globalType = chartOptions && chartOptions.chart && chartOptions.chart.type,
|
142
|
-
individualCounts = []
|
173
|
+
individualCounts = [],
|
174
|
+
seriesBuilders = [],
|
175
|
+
seriesIndex,
|
176
|
+
i;
|
143
177
|
|
144
178
|
each((chartOptions && chartOptions.series) || [], function (series) {
|
145
179
|
individualCounts.push(getValueCount(series.type || globalType));
|
146
180
|
});
|
147
181
|
|
182
|
+
// Collect the x-column indexes from seriesMapping
|
183
|
+
each((options && options.seriesMapping) || [], function (mapping) {
|
184
|
+
xColumns.push(mapping.x || 0);
|
185
|
+
});
|
186
|
+
|
187
|
+
// If there are no defined series with x-columns, use the first column as x column
|
188
|
+
if (xColumns.length === 0) {
|
189
|
+
xColumns.push(0);
|
190
|
+
}
|
191
|
+
|
192
|
+
// Loop all seriesMappings and constructs SeriesBuilders from
|
193
|
+
// the mapping options.
|
194
|
+
each((options && options.seriesMapping) || [], function (mapping) {
|
195
|
+
var builder = new SeriesBuilder(),
|
196
|
+
name,
|
197
|
+
numberOfValueColumnsNeeded = individualCounts[seriesIndex] || getValueCount(globalType),
|
198
|
+
seriesArr = (chartOptions && chartOptions.series) || [],
|
199
|
+
series = seriesArr[seriesIndex] || {},
|
200
|
+
pointArrayMap = getPointArrayMap(series.type || globalType) || ['y'];
|
201
|
+
|
202
|
+
// Add an x reader from the x property or from an undefined column
|
203
|
+
// if the property is not set. It will then be auto populated later.
|
204
|
+
builder.addColumnReader(mapping.x, 'x');
|
205
|
+
|
206
|
+
// Add all column mappings
|
207
|
+
for (name in mapping) {
|
208
|
+
if (mapping.hasOwnProperty(name) && name !== 'x') {
|
209
|
+
builder.addColumnReader(mapping[name], name);
|
210
|
+
}
|
211
|
+
}
|
212
|
+
|
213
|
+
// Add missing columns
|
214
|
+
for (i = 0; i < numberOfValueColumnsNeeded; i++) {
|
215
|
+
if (!builder.hasReader(pointArrayMap[i])) {
|
216
|
+
//builder.addNextColumnReader(pointArrayMap[i]);
|
217
|
+
// Create and add a column reader for the next free column index
|
218
|
+
builder.addColumnReader(undefined, pointArrayMap[i]);
|
219
|
+
}
|
220
|
+
}
|
221
|
+
|
222
|
+
seriesBuilders.push(builder);
|
223
|
+
seriesIndex++;
|
224
|
+
});
|
225
|
+
|
226
|
+
var globalPointArrayMap = getPointArrayMap(globalType);
|
227
|
+
if (globalPointArrayMap === undefined) {
|
228
|
+
globalPointArrayMap = ['y'];
|
229
|
+
}
|
230
|
+
|
148
231
|
this.valueCount = {
|
149
232
|
global: getValueCount(globalType),
|
150
|
-
|
233
|
+
xColumns: xColumns,
|
234
|
+
individual: individualCounts,
|
235
|
+
seriesBuilders: seriesBuilders,
|
236
|
+
globalPointArrayMap: globalPointArrayMap
|
151
237
|
};
|
152
238
|
},
|
153
239
|
|
@@ -161,6 +247,9 @@
|
|
161
247
|
this.columns = this.rowsToColumns(this.columns);
|
162
248
|
}
|
163
249
|
|
250
|
+
// Interpret the info about series and columns
|
251
|
+
this.getColumnDistribution();
|
252
|
+
|
164
253
|
// Interpret the values into right types
|
165
254
|
this.parseTypes();
|
166
255
|
|
@@ -168,10 +257,11 @@
|
|
168
257
|
this.findHeaderRow();
|
169
258
|
|
170
259
|
// Handle columns if a handleColumns callback is given
|
171
|
-
this.parsed()
|
260
|
+
if (this.parsed() !== false) {
|
172
261
|
|
173
|
-
|
174
|
-
|
262
|
+
// Complete if a complete callback is given
|
263
|
+
this.complete();
|
264
|
+
}
|
175
265
|
|
176
266
|
},
|
177
267
|
|
@@ -338,11 +428,11 @@
|
|
338
428
|
findHeaderRow: function () {
|
339
429
|
var headerRow = 0;
|
340
430
|
each(this.columns, function (column) {
|
341
|
-
if (typeof column[0] !== 'string') {
|
431
|
+
if (column.isNumeric && typeof column[0] !== 'string') {
|
342
432
|
headerRow = null;
|
343
433
|
}
|
344
434
|
});
|
345
|
-
this.headerRow =
|
435
|
+
this.headerRow = headerRow;
|
346
436
|
},
|
347
437
|
|
348
438
|
/**
|
@@ -357,22 +447,37 @@
|
|
357
447
|
*/
|
358
448
|
parseTypes: function () {
|
359
449
|
var columns = this.columns,
|
450
|
+
rawColumns = this.rawColumns,
|
360
451
|
col = columns.length,
|
361
452
|
row,
|
362
453
|
val,
|
363
454
|
floatVal,
|
364
455
|
trimVal,
|
365
|
-
|
366
|
-
|
456
|
+
isXColumn,
|
457
|
+
dateVal,
|
458
|
+
descending,
|
459
|
+
backup = [],
|
460
|
+
diff,
|
461
|
+
hasHeaderRow,
|
462
|
+
forceCategory,
|
463
|
+
chartOptions = this.chartOptions;
|
464
|
+
|
367
465
|
while (col--) {
|
368
466
|
row = columns[col].length;
|
467
|
+
rawColumns[col] = [];
|
468
|
+
isXColumn = inArray(col, this.valueCount.xColumns) !== -1;
|
469
|
+
forceCategory = isXColumn && chartOptions && chartOptions.xAxis && splat(chartOptions.xAxis)[0].type === 'category';
|
369
470
|
while (row--) {
|
370
|
-
val = columns[col][row];
|
471
|
+
val = backup[row] || columns[col][row];
|
371
472
|
floatVal = parseFloat(val);
|
372
|
-
trimVal = this.trim(val);
|
473
|
+
trimVal = rawColumns[col][row] = this.trim(val);
|
474
|
+
|
475
|
+
// Disable number or date parsing by setting the X axis type to category
|
476
|
+
if (forceCategory) {
|
477
|
+
columns[col][row] = trimVal;
|
373
478
|
|
374
479
|
/*jslint eqeq: true*/
|
375
|
-
if (trimVal == floatVal) { // is numeric
|
480
|
+
} else if (trimVal == floatVal) { // is numeric
|
376
481
|
/*jslint eqeq: false*/
|
377
482
|
columns[col][row] = floatVal;
|
378
483
|
|
@@ -385,16 +490,56 @@
|
|
385
490
|
|
386
491
|
} else { // string, continue to determine if it is a date string or really a string
|
387
492
|
dateVal = this.parseDate(val);
|
388
|
-
|
389
|
-
if (
|
493
|
+
// Only allow parsing of dates if this column is an x-column
|
494
|
+
if (isXColumn && typeof dateVal === 'number' && !isNaN(dateVal)) { // is date
|
495
|
+
backup[row] = val;
|
390
496
|
columns[col][row] = dateVal;
|
391
497
|
columns[col].isDatetime = true;
|
498
|
+
|
499
|
+
// Check if the dates are uniformly descending or ascending. If they
|
500
|
+
// are not, chances are that they are a different time format, so check
|
501
|
+
// for alternative.
|
502
|
+
if (columns[col][row + 1] !== undefined) {
|
503
|
+
diff = dateVal > columns[col][row + 1];
|
504
|
+
if (diff !== descending && descending !== undefined) {
|
505
|
+
if (this.alternativeFormat) {
|
506
|
+
this.dateFormat = this.alternativeFormat;
|
507
|
+
row = columns[col].length;
|
508
|
+
this.alternativeFormat = this.dateFormats[this.dateFormat].alternative;
|
509
|
+
} else {
|
510
|
+
columns[col].unsorted = true;
|
511
|
+
}
|
512
|
+
}
|
513
|
+
descending = diff;
|
514
|
+
}
|
392
515
|
|
393
516
|
} else { // string
|
394
517
|
columns[col][row] = trimVal === '' ? null : trimVal;
|
518
|
+
if (row !== 0 && (columns[col].isDatetime || columns[col].isNumeric)) {
|
519
|
+
columns[col].mixed = true;
|
520
|
+
}
|
395
521
|
}
|
396
522
|
}
|
397
|
-
|
523
|
+
}
|
524
|
+
|
525
|
+
// If strings are intermixed with numbers or dates in a parsed column, it is an indication
|
526
|
+
// that parsing went wrong or the data was not intended to display as numbers or dates and
|
527
|
+
// parsing is too aggressive. Fall back to categories. Demonstrated in the
|
528
|
+
// highcharts/demo/column-drilldown sample.
|
529
|
+
if (isXColumn && columns[col].mixed) {
|
530
|
+
columns[col] = rawColumns[col];
|
531
|
+
}
|
532
|
+
}
|
533
|
+
|
534
|
+
// If the 0 column is date and descending, reverse all columns.
|
535
|
+
// TODO: probably this should apply to xColumns, not 0 column alone.
|
536
|
+
if (columns[0].isDatetime && descending) {
|
537
|
+
hasHeaderRow = typeof columns[0][0] !== 'number';
|
538
|
+
for (col = 0; col < columns.length; col++) {
|
539
|
+
columns[col].reverse();
|
540
|
+
if (hasHeaderRow) {
|
541
|
+
columns[col].unshift(columns[col].pop());
|
542
|
+
}
|
398
543
|
}
|
399
544
|
}
|
400
545
|
},
|
@@ -405,10 +550,37 @@
|
|
405
550
|
*/
|
406
551
|
dateFormats: {
|
407
552
|
'YYYY-mm-dd': {
|
408
|
-
regex:
|
553
|
+
regex: /^([0-9]{4})[\-\/\.]([0-9]{2})[\-\/\.]([0-9]{2})$/,
|
409
554
|
parser: function (match) {
|
410
555
|
return Date.UTC(+match[1], match[2] - 1, +match[3]);
|
411
556
|
}
|
557
|
+
},
|
558
|
+
'dd/mm/YYYY': {
|
559
|
+
regex: /^([0-9]{1,2})[\-\/\.]([0-9]{1,2})[\-\/\.]([0-9]{4})$/,
|
560
|
+
parser: function (match) {
|
561
|
+
return Date.UTC(+match[3], match[2] - 1, +match[1]);
|
562
|
+
},
|
563
|
+
alternative: 'mm/dd/YYYY' // different format with the same regex
|
564
|
+
},
|
565
|
+
'mm/dd/YYYY': {
|
566
|
+
regex: /^([0-9]{1,2})[\-\/\.]([0-9]{1,2})[\-\/\.]([0-9]{4})$/,
|
567
|
+
parser: function (match) {
|
568
|
+
return Date.UTC(+match[3], match[1] - 1, +match[2]);
|
569
|
+
}
|
570
|
+
},
|
571
|
+
'dd/mm/YY': {
|
572
|
+
regex: /^([0-9]{1,2})[\-\/\.]([0-9]{1,2})[\-\/\.]([0-9]{2})$/,
|
573
|
+
parser: function (match) {
|
574
|
+
return Date.UTC(+match[3] + 2000, match[2] - 1, +match[1]);
|
575
|
+
},
|
576
|
+
alternative: 'mm/dd/YY' // different format with the same regex
|
577
|
+
},
|
578
|
+
'mm/dd/YY': {
|
579
|
+
regex: /^([0-9]{1,2})[\-\/\.]([0-9]{1,2})[\-\/\.]([0-9]{2})$/,
|
580
|
+
parser: function (match) {
|
581
|
+
console.log(match)
|
582
|
+
return Date.UTC(+match[3] + 2000, match[1] - 1, +match[2]);
|
583
|
+
}
|
412
584
|
}
|
413
585
|
},
|
414
586
|
|
@@ -420,20 +592,47 @@
|
|
420
592
|
ret,
|
421
593
|
key,
|
422
594
|
format,
|
595
|
+
dateFormat = this.options.dateFormat || this.dateFormat,
|
423
596
|
match;
|
424
597
|
|
425
598
|
if (parseDate) {
|
426
599
|
ret = parseDate(val);
|
427
600
|
}
|
428
|
-
|
601
|
+
|
429
602
|
if (typeof val === 'string') {
|
430
|
-
|
431
|
-
|
603
|
+
// Auto-detect the date format the first time
|
604
|
+
if (!dateFormat) {
|
605
|
+
for (key in this.dateFormats) {
|
606
|
+
format = this.dateFormats[key];
|
607
|
+
match = val.match(format.regex);
|
608
|
+
if (match) {
|
609
|
+
this.dateFormat = dateFormat = key;
|
610
|
+
this.alternativeFormat = format.alternative;
|
611
|
+
ret = format.parser(match);
|
612
|
+
break;
|
613
|
+
}
|
614
|
+
}
|
615
|
+
// Next time, use the one previously found
|
616
|
+
} else {
|
617
|
+
format = this.dateFormats[dateFormat];
|
432
618
|
match = val.match(format.regex);
|
433
619
|
if (match) {
|
434
620
|
ret = format.parser(match);
|
435
621
|
}
|
436
622
|
}
|
623
|
+
// Fall back to Date.parse
|
624
|
+
if (!match) {
|
625
|
+
match = Date.parse(val);
|
626
|
+
// External tools like Date.js and MooTools extend Date object and
|
627
|
+
// returns a date.
|
628
|
+
if (typeof match === 'object' && match !== null && match.getTime) {
|
629
|
+
ret = match.getTime() - match.getTimezoneOffset() * 60000;
|
630
|
+
|
631
|
+
// Timestamp
|
632
|
+
} else if (typeof match === 'number' && !isNaN(match)) {
|
633
|
+
ret = match - (new Date(match)).getTimezoneOffset() * 60000;
|
634
|
+
}
|
635
|
+
}
|
437
636
|
}
|
438
637
|
return ret;
|
439
638
|
},
|
@@ -469,9 +668,40 @@
|
|
469
668
|
*/
|
470
669
|
parsed: function () {
|
471
670
|
if (this.options.parsed) {
|
472
|
-
this.options.parsed.call(this, this.columns);
|
671
|
+
return this.options.parsed.call(this, this.columns);
|
473
672
|
}
|
474
673
|
},
|
674
|
+
|
675
|
+
getFreeIndexes: function (numberOfColumns, seriesBuilders) {
|
676
|
+
var s,
|
677
|
+
i,
|
678
|
+
freeIndexes = [],
|
679
|
+
freeIndexValues = [],
|
680
|
+
referencedIndexes;
|
681
|
+
|
682
|
+
// Add all columns as free
|
683
|
+
for (i = 0; i < numberOfColumns; i = i + 1) {
|
684
|
+
freeIndexes.push(true);
|
685
|
+
}
|
686
|
+
|
687
|
+
// Loop all defined builders and remove their referenced columns
|
688
|
+
for (s = 0; s < seriesBuilders.length; s = s + 1) {
|
689
|
+
referencedIndexes = seriesBuilders[s].getReferencedColumnIndexes();
|
690
|
+
|
691
|
+
for (i = 0; i < referencedIndexes.length; i = i + 1) {
|
692
|
+
freeIndexes[referencedIndexes[i]] = false;
|
693
|
+
}
|
694
|
+
}
|
695
|
+
|
696
|
+
// Collect the values for the free indexes
|
697
|
+
for (i = 0; i < freeIndexes.length; i = i + 1) {
|
698
|
+
if (freeIndexes[i]) {
|
699
|
+
freeIndexValues.push(i);
|
700
|
+
}
|
701
|
+
}
|
702
|
+
|
703
|
+
return freeIndexValues;
|
704
|
+
},
|
475
705
|
|
476
706
|
/**
|
477
707
|
* If a complete callback function is provided in the options, interpret the
|
@@ -480,36 +710,24 @@
|
|
480
710
|
complete: function () {
|
481
711
|
|
482
712
|
var columns = this.columns,
|
483
|
-
|
713
|
+
xColumns = [],
|
484
714
|
type,
|
485
715
|
options = this.options,
|
486
|
-
valueCount,
|
487
716
|
series,
|
488
717
|
data,
|
489
718
|
i,
|
490
719
|
j,
|
720
|
+
r,
|
491
721
|
seriesIndex,
|
492
|
-
chartOptions
|
493
|
-
|
494
|
-
|
495
|
-
|
722
|
+
chartOptions,
|
723
|
+
allSeriesBuilders = [],
|
724
|
+
builder,
|
725
|
+
freeIndexes,
|
726
|
+
typeCol,
|
727
|
+
index;
|
496
728
|
|
497
|
-
|
498
|
-
|
499
|
-
// Use first column for X data or categories?
|
500
|
-
if (columns.length > 1) {
|
501
|
-
firstCol = columns.shift();
|
502
|
-
if (this.headerRow === 0) {
|
503
|
-
firstCol.shift(); // remove the first cell
|
504
|
-
}
|
505
|
-
|
506
|
-
|
507
|
-
if (firstCol.isDatetime) {
|
508
|
-
type = 'datetime';
|
509
|
-
} else if (!firstCol.isNumeric) {
|
510
|
-
type = 'category';
|
511
|
-
}
|
512
|
-
}
|
729
|
+
xColumns.length = columns.length;
|
730
|
+
if (options.complete || options.afterComplete) {
|
513
731
|
|
514
732
|
// Get the names and shift the top row
|
515
733
|
for (i = 0; i < columns.length; i++) {
|
@@ -520,46 +738,84 @@
|
|
520
738
|
|
521
739
|
// Use the next columns for series
|
522
740
|
series = [];
|
523
|
-
|
741
|
+
freeIndexes = this.getFreeIndexes(columns.length, this.valueCount.seriesBuilders);
|
742
|
+
|
743
|
+
// Populate defined series
|
744
|
+
for (seriesIndex = 0; seriesIndex < this.valueCount.seriesBuilders.length; seriesIndex++) {
|
745
|
+
builder = this.valueCount.seriesBuilders[seriesIndex];
|
524
746
|
|
525
|
-
//
|
526
|
-
|
747
|
+
// If the builder can be populated with remaining columns, then add it to allBuilders
|
748
|
+
if (builder.populateColumns(freeIndexes)) {
|
749
|
+
allSeriesBuilders.push(builder);
|
750
|
+
}
|
751
|
+
}
|
752
|
+
|
753
|
+
// Populate dynamic series
|
754
|
+
while (freeIndexes.length > 0) {
|
755
|
+
builder = new SeriesBuilder();
|
756
|
+
builder.addColumnReader(0, 'x');
|
527
757
|
|
528
|
-
//
|
529
|
-
|
758
|
+
// Mark index as used (not free)
|
759
|
+
index = inArray(0, freeIndexes);
|
760
|
+
if (index !== -1) {
|
761
|
+
freeIndexes.splice(index, 1);
|
762
|
+
}
|
530
763
|
|
531
|
-
|
532
|
-
|
533
|
-
|
534
|
-
|
535
|
-
|
536
|
-
|
537
|
-
|
538
|
-
|
539
|
-
|
540
|
-
|
541
|
-
|
542
|
-
|
543
|
-
|
544
|
-
|
545
|
-
|
546
|
-
|
547
|
-
|
548
|
-
|
549
|
-
|
764
|
+
for (i = 0; i < this.valueCount.global; i++) {
|
765
|
+
// Create and add a column reader for the next free column index
|
766
|
+
builder.addColumnReader(undefined, this.valueCount.globalPointArrayMap[i]);
|
767
|
+
}
|
768
|
+
|
769
|
+
// If the builder can be populated with remaining columns, then add it to allBuilders
|
770
|
+
if (builder.populateColumns(freeIndexes)) {
|
771
|
+
allSeriesBuilders.push(builder);
|
772
|
+
}
|
773
|
+
}
|
774
|
+
|
775
|
+
// Get the data-type from the first series x column
|
776
|
+
if (allSeriesBuilders.length > 0 && allSeriesBuilders[0].readers.length > 0) {
|
777
|
+
typeCol = columns[allSeriesBuilders[0].readers[0].columnIndex];
|
778
|
+
if (typeCol !== undefined) {
|
779
|
+
if (typeCol.isDatetime) {
|
780
|
+
type = 'datetime';
|
781
|
+
} else if (!typeCol.isNumeric) {
|
782
|
+
type = 'category';
|
783
|
+
}
|
784
|
+
}
|
785
|
+
}
|
786
|
+
// Axis type is category, then the "x" column should be called "name"
|
787
|
+
if (type === 'category') {
|
788
|
+
for (seriesIndex = 0; seriesIndex < allSeriesBuilders.length; seriesIndex++) {
|
789
|
+
builder = allSeriesBuilders[seriesIndex];
|
790
|
+
for (r = 0; r < builder.readers.length; r++) {
|
791
|
+
if (builder.readers[r].configName === 'x') {
|
792
|
+
builder.readers[r].configName = 'name';
|
550
793
|
}
|
551
794
|
}
|
552
795
|
}
|
796
|
+
}
|
797
|
+
|
798
|
+
// Read data for all builders
|
799
|
+
for (seriesIndex = 0; seriesIndex < allSeriesBuilders.length; seriesIndex++) {
|
800
|
+
builder = allSeriesBuilders[seriesIndex];
|
801
|
+
|
802
|
+
// Iterate down the cells of each column and add data to the series
|
803
|
+
data = [];
|
804
|
+
for (j = 0; j < columns[0].length; j++) { // TODO: which column's length should we use here
|
805
|
+
data[j] = builder.read(columns, j);
|
806
|
+
}
|
553
807
|
|
554
808
|
// Add the series
|
555
809
|
series[seriesIndex] = {
|
556
|
-
name: columns[i].name,
|
557
810
|
data: data
|
558
811
|
};
|
559
|
-
|
560
|
-
|
812
|
+
if (builder.name) {
|
813
|
+
series[seriesIndex].name = builder.name;
|
814
|
+
}
|
561
815
|
}
|
562
|
-
|
816
|
+
|
817
|
+
|
818
|
+
|
563
819
|
// Do the callback
|
564
820
|
chartOptions = {
|
565
821
|
xAxis: {
|
@@ -570,6 +826,7 @@
|
|
570
826
|
if (options.complete) {
|
571
827
|
options.complete(chartOptions);
|
572
828
|
}
|
829
|
+
|
573
830
|
// The afterComplete hook is used internally to avoid conflict with the externally
|
574
831
|
// available complete option.
|
575
832
|
if (options.afterComplete) {
|
@@ -592,6 +849,7 @@
|
|
592
849
|
|
593
850
|
if (userOptions && userOptions.data) {
|
594
851
|
Highcharts.data(Highcharts.extend(userOptions.data, {
|
852
|
+
|
595
853
|
afterComplete: function (dataOptions) {
|
596
854
|
var i, series;
|
597
855
|
|
@@ -619,4 +877,149 @@
|
|
619
877
|
}
|
620
878
|
});
|
621
879
|
|
880
|
+
/**
|
881
|
+
* Creates a new SeriesBuilder. A SeriesBuilder consists of a number
|
882
|
+
* of ColumnReaders that reads columns and give them a name.
|
883
|
+
* Ex: A series builder can be constructed to read column 3 as 'x' and
|
884
|
+
* column 7 and 8 as 'y1' and 'y2'.
|
885
|
+
* The output would then be points/rows of the form {x: 11, y1: 22, y2: 33}
|
886
|
+
*
|
887
|
+
* The name of the builder is taken from the second column. In the above
|
888
|
+
* example it would be the column with index 7.
|
889
|
+
* @constructor
|
890
|
+
*/
|
891
|
+
SeriesBuilder = function () {
|
892
|
+
this.readers = [];
|
893
|
+
this.pointIsArray = true;
|
894
|
+
};
|
895
|
+
|
896
|
+
/**
|
897
|
+
* Populates readers with column indexes. A reader can be added without
|
898
|
+
* a specific index and for those readers the index is taken sequentially
|
899
|
+
* from the free columns (this is handled by the ColumnCursor instance).
|
900
|
+
* @returns {boolean}
|
901
|
+
*/
|
902
|
+
SeriesBuilder.prototype.populateColumns = function (freeIndexes) {
|
903
|
+
var builder = this,
|
904
|
+
enoughColumns = true;
|
905
|
+
|
906
|
+
// Loop each reader and give it an index if its missing.
|
907
|
+
// The freeIndexes.shift() will return undefined if there
|
908
|
+
// are no more columns.
|
909
|
+
each(builder.readers, function (reader) {
|
910
|
+
if (reader.columnIndex === undefined) {
|
911
|
+
reader.columnIndex = freeIndexes.shift();
|
912
|
+
}
|
913
|
+
});
|
914
|
+
|
915
|
+
// Now, all readers should have columns mapped. If not
|
916
|
+
// then return false to signal that this series should
|
917
|
+
// not be added.
|
918
|
+
each(builder.readers, function (reader) {
|
919
|
+
if (reader.columnIndex === undefined) {
|
920
|
+
enoughColumns = false;
|
921
|
+
}
|
922
|
+
});
|
923
|
+
|
924
|
+
return enoughColumns;
|
925
|
+
};
|
926
|
+
|
927
|
+
/**
|
928
|
+
* Reads a row from the dataset and returns a point or array depending
|
929
|
+
* on the names of the readers.
|
930
|
+
* @param columns
|
931
|
+
* @param rowIndex
|
932
|
+
* @returns {Array | Object}
|
933
|
+
*/
|
934
|
+
SeriesBuilder.prototype.read = function (columns, rowIndex) {
|
935
|
+
var builder = this,
|
936
|
+
pointIsArray = builder.pointIsArray,
|
937
|
+
point = pointIsArray ? [] : {},
|
938
|
+
columnIndexes;
|
939
|
+
|
940
|
+
// Loop each reader and ask it to read its value.
|
941
|
+
// Then, build an array or point based on the readers names.
|
942
|
+
each(builder.readers, function (reader) {
|
943
|
+
var value = columns[reader.columnIndex][rowIndex];
|
944
|
+
if (pointIsArray) {
|
945
|
+
point.push(value);
|
946
|
+
} else {
|
947
|
+
point[reader.configName] = value;
|
948
|
+
}
|
949
|
+
});
|
950
|
+
|
951
|
+
// The name comes from the first column (excluding the x column)
|
952
|
+
if (this.name === undefined && builder.readers.length >= 2) {
|
953
|
+
columnIndexes = builder.getReferencedColumnIndexes();
|
954
|
+
if (columnIndexes.length >= 2) {
|
955
|
+
// remove the first one (x col)
|
956
|
+
columnIndexes.shift();
|
957
|
+
|
958
|
+
// Sort the remaining
|
959
|
+
columnIndexes.sort();
|
960
|
+
|
961
|
+
// Now use the lowest index as name column
|
962
|
+
this.name = columns[columnIndexes.shift()].name;
|
963
|
+
}
|
964
|
+
}
|
965
|
+
|
966
|
+
return point;
|
967
|
+
};
|
968
|
+
|
969
|
+
/**
|
970
|
+
* Creates and adds ColumnReader from the given columnIndex and configName.
|
971
|
+
* ColumnIndex can be undefined and in that case the reader will be given
|
972
|
+
* an index when columns are populated.
|
973
|
+
* @param columnIndex {Number | undefined}
|
974
|
+
* @param configName
|
975
|
+
*/
|
976
|
+
SeriesBuilder.prototype.addColumnReader = function (columnIndex, configName) {
|
977
|
+
this.readers.push({
|
978
|
+
columnIndex: columnIndex,
|
979
|
+
configName: configName
|
980
|
+
});
|
981
|
+
|
982
|
+
if (!(configName === 'x' || configName === 'y' || configName === undefined)) {
|
983
|
+
this.pointIsArray = false;
|
984
|
+
}
|
985
|
+
};
|
986
|
+
|
987
|
+
/**
|
988
|
+
* Returns an array of column indexes that the builder will use when
|
989
|
+
* reading data.
|
990
|
+
* @returns {Array}
|
991
|
+
*/
|
992
|
+
SeriesBuilder.prototype.getReferencedColumnIndexes = function () {
|
993
|
+
var i,
|
994
|
+
referencedColumnIndexes = [],
|
995
|
+
columnReader;
|
996
|
+
|
997
|
+
for (i = 0; i < this.readers.length; i = i + 1) {
|
998
|
+
columnReader = this.readers[i];
|
999
|
+
if (columnReader.columnIndex !== undefined) {
|
1000
|
+
referencedColumnIndexes.push(columnReader.columnIndex);
|
1001
|
+
}
|
1002
|
+
}
|
1003
|
+
|
1004
|
+
return referencedColumnIndexes;
|
1005
|
+
};
|
1006
|
+
|
1007
|
+
/**
|
1008
|
+
* Returns true if the builder has a reader for the given configName.
|
1009
|
+
* @param configName
|
1010
|
+
* @returns {boolean}
|
1011
|
+
*/
|
1012
|
+
SeriesBuilder.prototype.hasReader = function (configName) {
|
1013
|
+
var i, columnReader;
|
1014
|
+
for (i = 0; i < this.readers.length; i = i + 1) {
|
1015
|
+
columnReader = this.readers[i];
|
1016
|
+
if (columnReader.configName === configName) {
|
1017
|
+
return true;
|
1018
|
+
}
|
1019
|
+
}
|
1020
|
+
// Else return undefined
|
1021
|
+
};
|
1022
|
+
|
1023
|
+
|
1024
|
+
|
622
1025
|
}(Highcharts));
|