2014-12-01 17:41:39 +00:00
|
|
|
/*global define,Float32Array*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prepares data to be rendered in a GL Plot. Handles
|
|
|
|
* the conversion from data API to displayable buffers.
|
|
|
|
*/
|
|
|
|
define(
|
|
|
|
function () {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
function identity(x) { return x; }
|
|
|
|
|
2014-12-02 22:38:03 +00:00
|
|
|
/**
|
|
|
|
* The PlotPreparer is responsible for handling data sets and
|
|
|
|
* preparing them to be rendered. It creates a WebGL-plottable
|
|
|
|
* Float32Array for each trace, and tracks the boundaries of the
|
|
|
|
* data sets (since this is convenient to do during the same pass).
|
|
|
|
* @constructor
|
|
|
|
* @param {Telemetry[]} datas telemetry data objects
|
|
|
|
* @param {string} domain the key to use when looking up domain values
|
|
|
|
* @param {string} range the key to use when looking up range values
|
|
|
|
*/
|
2014-12-01 21:27:30 +00:00
|
|
|
function PlotPreparer(datas, domain, range) {
|
2014-12-01 17:41:39 +00:00
|
|
|
var index,
|
|
|
|
vertices = [],
|
|
|
|
max = [Number.NEGATIVE_INFINITY, Number.NEGATIVE_INFINITY],
|
|
|
|
min = [Number.POSITIVE_INFINITY, Number.POSITIVE_INFINITY],
|
|
|
|
x,
|
|
|
|
y,
|
|
|
|
domainOffset = Number.POSITIVE_INFINITY,
|
|
|
|
buffers;
|
|
|
|
|
2014-12-02 18:11:46 +00:00
|
|
|
// Remove any undefined data sets
|
|
|
|
datas = (datas || []).filter(identity);
|
2014-12-01 17:41:39 +00:00
|
|
|
|
2014-12-02 22:38:03 +00:00
|
|
|
// Do a first pass to determine the domain offset.
|
|
|
|
// This will be use to reduce the magnitude of domain values
|
|
|
|
// in the buffer, to minimize loss-of-precision when
|
|
|
|
// converting to a 32-bit float.
|
2014-12-02 18:11:46 +00:00
|
|
|
datas.forEach(function (data) {
|
2014-12-01 17:41:39 +00:00
|
|
|
domainOffset = Math.min(data.getDomainValue(0, domain), domainOffset);
|
|
|
|
});
|
|
|
|
|
2014-12-02 22:38:03 +00:00
|
|
|
// Assemble buffers, and track bounds of the data present
|
2014-12-01 17:41:39 +00:00
|
|
|
datas.forEach(function (data, i) {
|
|
|
|
vertices.push([]);
|
2014-12-02 22:38:03 +00:00
|
|
|
for (index = 0; index < data.getPointCount(); index += 1) {
|
2014-12-02 00:35:38 +00:00
|
|
|
x = data.getDomainValue(index, domain);
|
2014-12-01 17:41:39 +00:00
|
|
|
y = data.getRangeValue(index, range);
|
2014-12-02 00:35:38 +00:00
|
|
|
vertices[i].push(x - domainOffset);
|
2014-12-01 17:41:39 +00:00
|
|
|
vertices[i].push(y);
|
|
|
|
min[0] = Math.min(min[0], x);
|
|
|
|
min[1] = Math.min(min[1], y);
|
|
|
|
max[0] = Math.max(max[0], x);
|
|
|
|
max[1] = Math.max(max[1], y);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-12-02 22:38:03 +00:00
|
|
|
// If range is empty, add some padding
|
2014-12-01 17:41:39 +00:00
|
|
|
if (max[1] === min[1]) {
|
|
|
|
max[1] = max[1] + 1.0;
|
|
|
|
min[1] = min[1] - 1.0;
|
|
|
|
}
|
|
|
|
|
2014-12-02 22:38:03 +00:00
|
|
|
// Convert to Float32Array
|
2014-12-01 17:41:39 +00:00
|
|
|
buffers = vertices.map(function (v) { return new Float32Array(v); });
|
|
|
|
|
|
|
|
return {
|
2014-12-02 22:38:03 +00:00
|
|
|
/**
|
|
|
|
* Get the dimensions which bound all data in the provided
|
|
|
|
* data sets. This is given as a two-element array where the
|
|
|
|
* first element is domain, and second is range.
|
|
|
|
* @returns {number[]} the dimensions which bound this data set
|
|
|
|
*/
|
2014-12-01 17:41:39 +00:00
|
|
|
getDimensions: function () {
|
|
|
|
return [max[0] - min[0], max[1] - min[1]];
|
|
|
|
},
|
2014-12-02 22:38:03 +00:00
|
|
|
/**
|
|
|
|
* Get the origin of this data set's boundary.
|
|
|
|
* This is given as a two-element array where the
|
|
|
|
* first element is domain, and second is range.
|
|
|
|
* The domain value here is not adjusted by the domain offset.
|
|
|
|
* @returns {number[]} the origin of this data set's boundary
|
|
|
|
*/
|
2014-12-01 17:41:39 +00:00
|
|
|
getOrigin: function () {
|
|
|
|
return min;
|
|
|
|
},
|
2014-12-02 22:38:03 +00:00
|
|
|
/**
|
|
|
|
* Get the domain offset; this offset will have been subtracted
|
|
|
|
* from all domain values in all buffers returned by this
|
|
|
|
* preparer, in order to minimize loss-of-precision due to
|
|
|
|
* conversion to the 32-bit float format needed by WebGL.
|
|
|
|
* @returns {number} the domain offset
|
|
|
|
*/
|
2014-12-01 17:41:39 +00:00
|
|
|
getDomainOffset: function () {
|
|
|
|
return domainOffset;
|
|
|
|
},
|
2014-12-02 22:38:03 +00:00
|
|
|
/**
|
|
|
|
* Get all renderable buffers for this data set. This will
|
|
|
|
* be returned as an array which can be correlated back to
|
|
|
|
* the provided telemetry data objects (from the constructor
|
|
|
|
* call) by index.
|
|
|
|
*
|
|
|
|
* Internally, these are flattened; each buffer contains a
|
|
|
|
* sequence of alternating domain and range values.
|
|
|
|
*
|
|
|
|
* All domain values in all buffers will have been adjusted
|
|
|
|
* from their original values by subtraction of the domain
|
|
|
|
* offset; this minimizes loss-of-precision resulting from
|
|
|
|
* the conversion to 32-bit floats, which may otherwise
|
|
|
|
* cause aliasing artifacts (particularly for timestamps)
|
|
|
|
*
|
|
|
|
* @returns {Float32Array[]} the buffers for these traces
|
|
|
|
*/
|
2014-12-01 17:41:39 +00:00
|
|
|
getBuffers: function () {
|
|
|
|
return buffers;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2014-12-01 21:27:30 +00:00
|
|
|
return PlotPreparer;
|
2014-12-01 17:41:39 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
);
|