@ -0,0 +1,209 @@ | |||||
/** | |||||
* @constructor DataStep | |||||
* The class DataStep is an iterator for data for the lineGraph. You provide a start data point and an | |||||
* end data point. The class itself determines the best scale (step size) based on the | |||||
* provided start Date, end Date, and minimumStep. | |||||
* | |||||
* If minimumStep is provided, the step size is chosen as close as possible | |||||
* to the minimumStep but larger than minimumStep. If minimumStep is not | |||||
* provided, the scale is set to 1 DAY. | |||||
* The minimumStep should correspond with the onscreen size of about 6 characters | |||||
* | |||||
* Alternatively, you can set a scale by hand. | |||||
* After creation, you can initialize the class by executing first(). Then you | |||||
* can iterate from the start date to the end date via next(). You can check if | |||||
* the end date is reached with the function hasNext(). After each step, you can | |||||
* retrieve the current date via getCurrent(). | |||||
* The DataStep has scales ranging from milliseconds, seconds, minutes, hours, | |||||
* days, to years. | |||||
* | |||||
* Version: 1.2 | |||||
* | |||||
* @param {Date} [start] The start date, for example new Date(2010, 9, 21) | |||||
* or new Date(2010, 9, 21, 23, 45, 00) | |||||
* @param {Date} [end] The end date | |||||
* @param {Number} [minimumStep] Optional. Minimum step size in milliseconds | |||||
*/ | |||||
function DataStep(start, end, minimumStep, containerHeight) { | |||||
// variables | |||||
this.current = 0; | |||||
this.containerHeight = containerHeight; | |||||
this.autoScale = true; | |||||
this.stepIndex = 0; | |||||
this.step = 1; | |||||
this.scale = 1; | |||||
this.marginStart; | |||||
this.marginEnd; | |||||
this.majorSteps = [1, 2, 5, 10]; | |||||
this.minorSteps = [0.25, 0.5, 1, 2]; | |||||
this.setRange(start,end,minimumStep, containerHeight); | |||||
} | |||||
/** | |||||
* Set a new range | |||||
* If minimumStep is provided, the step size is chosen as close as possible | |||||
* to the minimumStep but larger than minimumStep. If minimumStep is not | |||||
* provided, the scale is set to 1 DAY. | |||||
* The minimumStep should correspond with the onscreen size of about 6 characters | |||||
* @param {Number} [start] The start date and time. | |||||
* @param {Number} [end] The end date and time. | |||||
* @param {Number} [minimumStep] Optional. Minimum step size in milliseconds | |||||
*/ | |||||
DataStep.prototype.setRange = function(start, end, minimumStep, containerHeight) { | |||||
this._start = start; | |||||
this._end = end; | |||||
this.setFirst(); | |||||
if (this.autoScale) { | |||||
this.setMinimumStep(minimumStep, containerHeight); | |||||
} | |||||
}; | |||||
/** | |||||
* Set the range iterator to the start date. | |||||
*/ | |||||
DataStep.prototype.first = function() { | |||||
this.setFirst(); | |||||
}; | |||||
/** | |||||
* Round the current date to the first minor date value | |||||
* This must be executed once when the current date is set to start Date | |||||
*/ | |||||
DataStep.prototype.setFirst = function() { | |||||
var niceStart = this._start - (this.scale * this.minorSteps[this.stepIndex]); | |||||
var niceEnd = this._end + (this.scale * this.minorSteps[this.stepIndex]); | |||||
this.marginEnd = this.roundToMinor(niceEnd); | |||||
this.marginStart = this.roundToMinor(niceStart); | |||||
this.marginRange = this.marginEnd - this.marginStart; | |||||
this.current = this.marginEnd; | |||||
}; | |||||
DataStep.prototype.roundToMinor = function(value) { | |||||
var rounded = value - (value % (this.scale * this.minorSteps[this.stepIndex])); | |||||
if (value % (this.scale * this.minorSteps[this.stepIndex]) > 0.5 * (this.scale * this.minorSteps[this.stepIndex])) { | |||||
return rounded + (this.scale * this.minorSteps[this.stepIndex]); | |||||
} | |||||
else { | |||||
return rounded; | |||||
} | |||||
} | |||||
/** | |||||
* Check if the there is a next step | |||||
* @return {boolean} true if the current date has not passed the end date | |||||
*/ | |||||
DataStep.prototype.hasNext = function () { | |||||
return (this.current >= this.marginStart); | |||||
}; | |||||
/** | |||||
* Do the next step | |||||
*/ | |||||
DataStep.prototype.next = function() { | |||||
var prev = this.current; | |||||
this.current -= this.step; | |||||
// safety mechanism: if current time is still unchanged, move to the end | |||||
if (this.current == prev) { | |||||
this.current = this._end; | |||||
} | |||||
}; | |||||
/** | |||||
* Get the current datetime | |||||
* @return {Date} current The current date | |||||
*/ | |||||
DataStep.prototype.getCurrent = function() { | |||||
return this.current; | |||||
}; | |||||
/** | |||||
* Automatically determine the scale that bests fits the provided minimum step | |||||
* @param {Number} [minimumStep] The minimum step size in milliseconds | |||||
*/ | |||||
DataStep.prototype.setMinimumStep = function(minimumStep, containerHeight) { | |||||
// round to floor | |||||
var size = this._end - this._start; | |||||
var safeSize = size * 1.1; | |||||
var minimumStepValue = minimumStep * (safeSize / containerHeight); | |||||
var orderOfMagnitude = Math.round(Math.log(safeSize)/Math.LN10); | |||||
var minorStepIdx = -1; | |||||
var magnitudefactor = Math.pow(10,orderOfMagnitude); | |||||
var solutionFound = false; | |||||
for (var i = 0; i <= orderOfMagnitude; i++) { | |||||
magnitudefactor = Math.pow(10,i); | |||||
for (var j = 0; j < this.minorSteps.length; j++) { | |||||
var stepSize = magnitudefactor * this.minorSteps[j]; | |||||
if (stepSize >= minimumStepValue) { | |||||
solutionFound = true; | |||||
minorStepIdx = j; | |||||
break; | |||||
} | |||||
} | |||||
if (solutionFound == true) { | |||||
break; | |||||
} | |||||
} | |||||
this.stepIndex = minorStepIdx; | |||||
this.scale = magnitudefactor; | |||||
this.step = magnitudefactor * this.minorSteps[minorStepIdx]; | |||||
}; | |||||
/** | |||||
* Snap a date to a rounded value. | |||||
* The snap intervals are dependent on the current scale and step. | |||||
* @param {Date} date the date to be snapped. | |||||
* @return {Date} snappedDate | |||||
*/ | |||||
DataStep.prototype.snap = function(date) { | |||||
}; | |||||
/** | |||||
* Check if the current value is a major value (for example when the step | |||||
* is DAY, a major value is each first day of the MONTH) | |||||
* @return {boolean} true if current date is major, else false. | |||||
*/ | |||||
DataStep.prototype.isMajor = function() { | |||||
return (this.current % (this.scale * this.majorSteps[this.stepIndex]) == 0); | |||||
}; | |||||
/** | |||||
* Returns formatted text for the minor axislabel, depending on the current | |||||
* date and the scale. For example when scale is MINUTE, the current time is | |||||
* formatted as "hh:mm". | |||||
* @param {Date} [date] custom date. if not provided, current date is taken | |||||
*/ | |||||
DataStep.prototype.getLabelMinor = function() { | |||||
return this.current; | |||||
}; | |||||
/** | |||||
* Returns formatted text for the major axis label, depending on the current | |||||
* date and the scale. For example when scale is MINUTE, the major scale is | |||||
* hours, and the hour will be formatted as "hh". | |||||
* @param {Date} [date] custom date. if not provided, current date is taken | |||||
*/ | |||||
DataStep.prototype.getLabelMajor = function() { | |||||
return this.current; | |||||
}; |
@ -0,0 +1,405 @@ | |||||
/** | |||||
* A horizontal time axis | |||||
* @param {Object} [options] See DataAxis.setOptions for the available | |||||
* options. | |||||
* @constructor DataAxis | |||||
* @extends Component | |||||
*/ | |||||
function DataAxis (options) { | |||||
this.id = util.randomUUID(); | |||||
this.dom = { | |||||
majorLines: [], | |||||
majorTexts: [], | |||||
minorLines: [], | |||||
minorTexts: [], | |||||
redundant: { | |||||
majorLines: [], | |||||
majorTexts: [], | |||||
minorLines: [], | |||||
minorTexts: [] | |||||
} | |||||
}; | |||||
this.props = { | |||||
range: { | |||||
start: 0, | |||||
end: 0, | |||||
minimumStep: 0 | |||||
}, | |||||
lineTop: 0 | |||||
}; | |||||
this.options = options || {}; | |||||
this.defaultOptions = { | |||||
orientation: 'left', // supported: 'left' | |||||
showMinorLabels: true, | |||||
showMajorLabels: true | |||||
}; | |||||
this.range = null; | |||||
this.conversionFactor = 1; | |||||
// create the HTML DOM | |||||
this._create(); | |||||
} | |||||
DataAxis.prototype = new Component(); | |||||
// TODO: comment options | |||||
DataAxis.prototype.setOptions = Component.prototype.setOptions; | |||||
/** | |||||
* Create the HTML DOM for the DataAxis | |||||
*/ | |||||
DataAxis.prototype._create = function _create() { | |||||
this.frame = document.createElement('div'); | |||||
}; | |||||
/** | |||||
* Set a range (start and end) | |||||
* @param {Range | Object} range A Range or an object containing start and end. | |||||
*/ | |||||
DataAxis.prototype.setRange = function (range) { | |||||
if (!(range instanceof Range) && (!range || range.start === undefined || range.end === undefined)) { | |||||
throw new TypeError('Range must be an instance of Range, ' + | |||||
'or an object containing start and end.'); | |||||
} | |||||
this.range = range; | |||||
}; | |||||
/** | |||||
* Get the outer frame of the time axis | |||||
* @return {HTMLElement} frame | |||||
*/ | |||||
DataAxis.prototype.getFrame = function getFrame() { | |||||
return this.frame; | |||||
}; | |||||
/** | |||||
* Repaint the component | |||||
* @return {boolean} Returns true if the component is resized | |||||
*/ | |||||
DataAxis.prototype.repaint = function () { | |||||
var asSize = util.option.asSize; | |||||
var options = this.options; | |||||
var props = this.props; | |||||
var frame = this.frame; | |||||
// update classname | |||||
frame.className = 'dataaxis'; // TODO: add className from options if defined | |||||
// calculate character width and height | |||||
this._calculateCharSize(); | |||||
// TODO: recalculate sizes only needed when parent is resized or options is changed | |||||
var orientation = this.getOption('orientation'); | |||||
var showMinorLabels = this.getOption('showMinorLabels'); | |||||
var showMajorLabels = this.getOption('showMajorLabels'); | |||||
// determine the width and height of the elemens for the axis | |||||
props.minorLabelHeight = showMinorLabels ? props.minorCharHeight : 0; | |||||
props.majorLabelHeight = showMajorLabels ? props.majorCharHeight : 0; | |||||
this.height = this.options.height; | |||||
this.width = frame.offsetWidth; // TODO: only update the width when the frame is resized? | |||||
props.minorLineWidth = this.options.svg.offsetWidth; | |||||
props.minorLineHeight = 1; // TODO: really calculate width | |||||
props.majorLineWidth = this.options.svg.offsetWidth; | |||||
props.majorLineHeight = 1; // TODO: really calculate width | |||||
// take frame offline while updating (is almost twice as fast) | |||||
// TODO: top/bottom positioning should be determined by options set in the Timeline, not here | |||||
if (orientation == 'left') { | |||||
frame.style.top = '0'; | |||||
frame.style.left = '0'; | |||||
frame.style.bottom = ''; | |||||
frame.style.width = this.width + 'px'; | |||||
frame.style.height = this.height + "px"; | |||||
} | |||||
else { // right | |||||
frame.style.top = ''; | |||||
frame.style.bottom = '0'; | |||||
frame.style.left = '0'; | |||||
frame.style.width = this.width + 'px'; | |||||
frame.style.height = this.height + "px"; | |||||
} | |||||
this._repaintLabels(); | |||||
}; | |||||
/** | |||||
* Repaint major and minor text labels and vertical grid lines | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._repaintLabels = function () { | |||||
var orientation = this.getOption('orientation'); | |||||
// calculate range and step (step such that we have space for 7 characters per label) | |||||
var start = this.range.start; | |||||
var end = this.range.end; | |||||
var minimumStep = (this.props.minorCharHeight || 10); //in pixels | |||||
var step = new DataStep(start, end, minimumStep, this.options.svg.offsetHeight); | |||||
this.step = step; | |||||
// Move all DOM elements to a "redundant" list, where they | |||||
// can be picked for re-use, and clear the lists with lines and texts. | |||||
// At the end of the function _repaintLabels, left over elements will be cleaned up | |||||
var dom = this.dom; | |||||
dom.redundant.majorLines = dom.majorLines; | |||||
dom.redundant.majorTexts = dom.majorTexts; | |||||
dom.redundant.minorLines = dom.minorLines; | |||||
dom.redundant.minorTexts = dom.minorTexts; | |||||
dom.majorLines = []; | |||||
dom.majorTexts = []; | |||||
dom.minorLines = []; | |||||
dom.minorTexts = []; | |||||
step.first(); | |||||
var stepPixels = this.options.svg.offsetHeight / ((step.marginRange / step.step) + 1); | |||||
var xFirstMajorLabel = undefined; | |||||
this.valueAtZero = step.marginEnd; | |||||
var marginStartPos = 0; | |||||
var max = 0; | |||||
while (step.hasNext() && max < 1000) { | |||||
var y = max * stepPixels; | |||||
y = y.toPrecision(5) | |||||
var isMajor = step.isMajor(); | |||||
if (this.getOption('showMinorLabels') && isMajor == false) { | |||||
this._repaintMinorText(y, step.getLabelMinor(), orientation); | |||||
} | |||||
if (isMajor && this.getOption('showMajorLabels')) { | |||||
if (y > 0) { | |||||
if (xFirstMajorLabel == undefined) { | |||||
xFirstMajorLabel = y; | |||||
} | |||||
this._repaintMajorText(y, step.getLabelMajor(), orientation); | |||||
} | |||||
this._repaintMajorLine(y, orientation); | |||||
} | |||||
else { | |||||
this._repaintMinorLine(y, orientation); | |||||
} | |||||
step.next(); | |||||
marginStartPos = y; | |||||
max++; | |||||
} | |||||
this.conversionFactor = marginStartPos/step.marginRange; | |||||
console.log(marginStartPos, step.marginRange, this.conversionFactor); | |||||
// create a major label on the left when needed | |||||
if (this.getOption('showMajorLabels')) { | |||||
var leftPoint = this._start; | |||||
var leftText = step.getLabelMajor(leftPoint); | |||||
var widthText = leftText.length * (this.props.majorCharWidth || 10) + 10; // upper bound estimation | |||||
if (xFirstMajorLabel == undefined || widthText < xFirstMajorLabel) { | |||||
this._repaintMajorText(0, leftText, orientation); | |||||
} | |||||
} | |||||
// Cleanup leftover DOM elements from the redundant list | |||||
util.forEach(this.dom.redundant, function (arr) { | |||||
while (arr.length) { | |||||
var elem = arr.pop(); | |||||
if (elem && elem.parentNode) { | |||||
elem.parentNode.removeChild(elem); | |||||
} | |||||
} | |||||
}); | |||||
}; | |||||
DataAxis.prototype.convertValues = function(data) { | |||||
for (var i = 0; i < data.length; i++) { | |||||
data[i].y = this._getPos(data[i].y); | |||||
} | |||||
return data; | |||||
} | |||||
DataAxis.prototype._getPos = function(value) { | |||||
var invertedValue = this.valueAtZero - value; | |||||
var convertedValue = invertedValue * this.conversionFactor; | |||||
return convertedValue | |||||
} | |||||
/** | |||||
* Create a minor label for the axis at position x | |||||
* @param {Number} x | |||||
* @param {String} text | |||||
* @param {String} orientation "top" or "bottom" (default) | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._repaintMinorText = function (x, text, orientation) { | |||||
// reuse redundant label | |||||
var label = this.dom.redundant.minorTexts.shift(); | |||||
if (!label) { | |||||
// create new label | |||||
var content = document.createTextNode(''); | |||||
label = document.createElement('div'); | |||||
label.appendChild(content); | |||||
label.className = 'yAxis minor'; | |||||
this.frame.appendChild(label); | |||||
} | |||||
this.dom.minorTexts.push(label); | |||||
label.childNodes[0].nodeValue = text; | |||||
if (orientation == 'left') { | |||||
label.style.left = '-2px'; | |||||
label.style.textAlign = "right"; | |||||
} | |||||
else { | |||||
label.style.left = '2px'; | |||||
label.style.textAlign = "left"; | |||||
} | |||||
label.style.top = x + 'px'; | |||||
//label.title = title; // TODO: this is a heavy operation | |||||
}; | |||||
/** | |||||
* Create a Major label for the axis at position x | |||||
* @param {Number} x | |||||
* @param {String} text | |||||
* @param {String} orientation "top" or "bottom" (default) | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._repaintMajorText = function (x, text, orientation) { | |||||
// reuse redundant label | |||||
var label = this.dom.redundant.majorTexts.shift(); | |||||
if (!label) { | |||||
// create label | |||||
var content = document.createTextNode(text); | |||||
label = document.createElement('div'); | |||||
label.className = 'yAxis major'; | |||||
label.appendChild(content); | |||||
this.frame.appendChild(label); | |||||
} | |||||
this.dom.majorTexts.push(label); | |||||
label.childNodes[0].nodeValue = text; | |||||
//label.title = title; // TODO: this is a heavy operation | |||||
if (orientation == 'left') { | |||||
label.style.left = '-2px'; | |||||
label.style.textAlign = "right"; | |||||
} | |||||
else { | |||||
label.style.left = '2'; | |||||
label.style.textAlign = "left"; | |||||
} | |||||
label.style.top = x + 'px'; | |||||
}; | |||||
/** | |||||
* Create a minor line for the axis at position y | |||||
* @param {Number} y | |||||
* @param {String} orientation "top" or "bottom" (default) | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._repaintMinorLine = function (y, orientation) { | |||||
// reuse redundant line | |||||
var line = this.dom.redundant.minorLines.shift(); | |||||
if (!line) { | |||||
// create vertical line | |||||
line = document.createElement('div'); | |||||
line.className = 'grid horizontal minor'; | |||||
this.frame.appendChild(line); | |||||
} | |||||
this.dom.minorLines.push(line); | |||||
var props = this.props; | |||||
if (orientation == 'left') { | |||||
line.style.left = (this.width - 15) + 'px'; | |||||
} | |||||
else { | |||||
line.style.left = -1*(this.width - 15) + 'px'; | |||||
} | |||||
line.style.width = props.minorLineWidth + 'px'; | |||||
line.style.top = y + 'px'; | |||||
}; | |||||
/** | |||||
* Create a Major line for the axis at position x | |||||
* @param {Number} x | |||||
* @param {String} orientation "top" or "bottom" (default) | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._repaintMajorLine = function (y, orientation) { | |||||
// reuse redundant line | |||||
var line = this.dom.redundant.majorLines.shift(); | |||||
if (!line) { | |||||
// create vertical line | |||||
line = document.createElement('div'); | |||||
line.className = 'grid horizontal major'; | |||||
this.frame.appendChild(line); | |||||
} | |||||
this.dom.majorLines.push(line); | |||||
var props = this.props; | |||||
if (orientation == 'left') { | |||||
line.style.left = (this.width - 25) + 'px'; | |||||
} | |||||
else { | |||||
line.style.left = -1*(this.width - 25) + 'px'; | |||||
} | |||||
line.style.top = y + 'px'; | |||||
line.style.width = props.majorLineWidth + 'px'; | |||||
}; | |||||
/** | |||||
* Determine the size of text on the axis (both major and minor axis). | |||||
* The size is calculated only once and then cached in this.props. | |||||
* @private | |||||
*/ | |||||
DataAxis.prototype._calculateCharSize = function () { | |||||
// determine the char width and height on the minor axis | |||||
if (!('minorCharHeight' in this.props)) { | |||||
var textMinor = document.createTextNode('0'); | |||||
var measureCharMinor = document.createElement('DIV'); | |||||
measureCharMinor.className = 'text minor measure'; | |||||
measureCharMinor.appendChild(textMinor); | |||||
this.frame.appendChild(measureCharMinor); | |||||
this.props.minorCharHeight = measureCharMinor.clientHeight; | |||||
this.props.minorCharWidth = measureCharMinor.clientWidth; | |||||
this.frame.removeChild(measureCharMinor); | |||||
} | |||||
if (!('majorCharHeight' in this.props)) { | |||||
var textMajor = document.createTextNode('0'); | |||||
var measureCharMajor = document.createElement('DIV'); | |||||
measureCharMajor.className = 'text major measure'; | |||||
measureCharMajor.appendChild(textMajor); | |||||
this.frame.appendChild(measureCharMajor); | |||||
this.props.majorCharHeight = measureCharMajor.clientHeight; | |||||
this.props.majorCharWidth = measureCharMajor.clientWidth; | |||||
this.frame.removeChild(measureCharMajor); | |||||
} | |||||
}; | |||||
/** | |||||
* Snap a date to a rounded value. | |||||
* The snap intervals are dependent on the current scale and step. | |||||
* @param {Date} date the date to be snapped. | |||||
* @return {Date} snappedDate | |||||
*/ | |||||
DataAxis.prototype.snap = function snap (date) { | |||||
return this.step.snap(date); | |||||
}; |
@ -0,0 +1,500 @@ | |||||
/** | |||||
* Created by Alex on 5/6/14. | |||||
*/ | |||||
var UNGROUPED = '__ungrouped__'; // reserved group id for ungrouped items | |||||
/** | |||||
* An ItemSet holds a set of items and ranges which can be displayed in a | |||||
* range. The width is determined by the parent of the ItemSet, and the height | |||||
* is determined by the size of the items. | |||||
* @param {Panel} backgroundPanel Panel which can be used to display the | |||||
* vertical lines of box items. | |||||
* @param {Panel} axisPanel Panel on the axis where the dots of box-items | |||||
* can be displayed. | |||||
* @param {Panel} sidePanel Left side panel holding labels | |||||
* @param {Object} [options] See ItemSet.setOptions for the available options. | |||||
* @constructor ItemSet | |||||
* @extends Panel | |||||
*/ | |||||
function Linegraph(backgroundPanel, axisPanel, sidePanel, options, timeline, sidePanelParent) { | |||||
this.id = util.randomUUID(); | |||||
this.timeline = timeline; | |||||
// one options object is shared by this itemset and all its items | |||||
this.options = options || {}; | |||||
this.backgroundPanel = backgroundPanel; | |||||
this.axisPanel = axisPanel; | |||||
this.sidePanel = sidePanel; | |||||
this.sidePanelParent = sidePanelParent; | |||||
this.itemOptions = Object.create(this.options); | |||||
this.dom = {}; | |||||
this.hammer = null; | |||||
this.itemsData = null; // DataSet | |||||
this.groupsData = null; // DataSet | |||||
this.range = null; // Range or Object {start: number, end: number} | |||||
// listeners for the DataSet of the items | |||||
// this.itemListeners = { | |||||
// 'add': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onAdd(params.items); | |||||
// }, | |||||
// 'update': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onUpdate(params.items); | |||||
// }, | |||||
// 'remove': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onRemove(params.items); | |||||
// } | |||||
// }; | |||||
// | |||||
// // listeners for the DataSet of the groups | |||||
// this.groupListeners = { | |||||
// 'add': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onAddGroups(params.items); | |||||
// }, | |||||
// 'update': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onUpdateGroups(params.items); | |||||
// }, | |||||
// 'remove': function(event, params, senderId) { | |||||
// if (senderId != me.id) me._onRemoveGroups(params.items); | |||||
// } | |||||
// }; | |||||
this.items = {}; // object with an Item for every data item | |||||
this.groups = {}; // Group object for every group | |||||
this.groupIds = []; | |||||
this.selection = []; // list with the ids of all selected nodes | |||||
this.stackDirty = true; // if true, all items will be restacked on next repaint | |||||
this.touchParams = {}; // stores properties while dragging | |||||
// create the HTML DOM | |||||
this.lastStart = 0; | |||||
this._create(); | |||||
var me = this; | |||||
this.timeline.on("rangechange", function() { | |||||
if (me.lastStart != 0) { | |||||
var offset = me.range.start - me.lastStart; | |||||
var range = me.range.end - me.range.start; | |||||
if (me.width != 0) { | |||||
var rangePerPixelInv = me.width/range; | |||||
var xOffset = offset * rangePerPixelInv; | |||||
me.svg.style.left = util.option.asSize(-me.width - xOffset); | |||||
} | |||||
} | |||||
}) | |||||
this.timeline.on("rangechanged", function() { | |||||
me.lastStart = me.range.start; | |||||
me.svg.style.left = util.option.asSize(-me.width); | |||||
me.setData.apply(me); | |||||
}); | |||||
// this.data = new DataView(this.items) | |||||
} | |||||
Linegraph.prototype = new Panel(); | |||||
/** | |||||
* Create the HTML DOM for the ItemSet | |||||
*/ | |||||
Linegraph.prototype._create = function(){ | |||||
var frame = document.createElement('div'); | |||||
frame['timeline-linegraph'] = this; | |||||
this.frame = frame; | |||||
this.frame.className = 'itemset'; | |||||
// create background panel | |||||
var background = document.createElement('div'); | |||||
background.className = 'background'; | |||||
this.backgroundPanel.frame.appendChild(background); | |||||
this.dom.background = background; | |||||
// create foreground panel | |||||
var foreground = document.createElement('div'); | |||||
foreground.className = 'foreground'; | |||||
frame.appendChild(foreground); | |||||
this.dom.foreground = foreground; | |||||
// // create axis panel | |||||
// var axis = document.createElement('div'); | |||||
// axis.className = 'axis'; | |||||
// this.dom.axis = axis; | |||||
// this.axisPanel.frame.appendChild(axis); | |||||
// | |||||
// // create labelset | |||||
// var labelSet = document.createElement('div'); | |||||
// labelSet.className = 'labelset'; | |||||
// this.dom.labelSet = labelSet; | |||||
// this.sidePanel.frame.appendChild(labelSet); | |||||
this.svg = document.createElementNS('http://www.w3.org/2000/svg',"svg"); | |||||
this.svg.style.position = "relative" | |||||
this.svg.style.height = "300px"; | |||||
this.svg.style.display = "block"; | |||||
this.path = document.createElementNS('http://www.w3.org/2000/svg',"path"); | |||||
this.path.setAttributeNS(null, "fill","none"); | |||||
this.path.setAttributeNS(null, "stroke","blue"); | |||||
this.path.setAttributeNS(null, "stroke-width","1"); | |||||
this.path2 = document.createElementNS('http://www.w3.org/2000/svg',"path"); | |||||
this.path2.setAttributeNS(null, "fill","none"); | |||||
this.path2.setAttributeNS(null, "stroke","red"); | |||||
this.path2.setAttributeNS(null, "stroke-width","1"); | |||||
this.path3 = document.createElementNS('http://www.w3.org/2000/svg',"path"); | |||||
this.path3.setAttributeNS(null, "fill","none"); | |||||
this.path3.setAttributeNS(null, "stroke","green"); | |||||
this.path3.setAttributeNS(null, "stroke-width","1"); | |||||
this.dom.foreground.appendChild(this.svg); | |||||
this.svg.appendChild(this.path3); | |||||
this.svg.appendChild(this.path2); | |||||
this.svg.appendChild(this.path); | |||||
// this.yAxisDiv = document.createElement('div'); | |||||
// this.yAxisDiv.style.backgroundColor = 'rgb(220,220,220)'; | |||||
// this.yAxisDiv.style.width = '100px'; | |||||
// this.yAxisDiv.style.height = this.svg.style.height; | |||||
this._createAxis(); | |||||
// this.dom.yAxisDiv = this.yAxisDiv; | |||||
// this.sidePanel.frame.appendChild(this.yAxisDiv); | |||||
this.sidePanel.showPanel.apply(this.sidePanel); | |||||
this.sidePanelParent.showPanel(); | |||||
}; | |||||
Linegraph.prototype._createAxis = function() { | |||||
// panel with time axis | |||||
var dataAxisOptions = { | |||||
range: this.range, | |||||
left: null, | |||||
top: null, | |||||
width: null, | |||||
height: 300, | |||||
svg: this.svg | |||||
}; | |||||
this.yAxis = new DataAxis(dataAxisOptions); | |||||
this.sidePanel.frame.appendChild(this.yAxis.getFrame()); | |||||
} | |||||
Linegraph.prototype.setData = function() { | |||||
if (this.width != 0) { | |||||
var dataview = new DataView(this.timeline.itemsData, | |||||
{filter: function (item) {return (item.value);}}) | |||||
var datapoints = dataview.get(); | |||||
if (datapoints != null) { | |||||
if (datapoints.length > 0) { | |||||
var dataset = this._extractData(datapoints); | |||||
var data = dataset.data; | |||||
console.log("height",data,datapoints, dataset); | |||||
this.yAxis.setRange({start:dataset.range.low,end:dataset.range.high}); | |||||
this.yAxis.repaint(); | |||||
data = this.yAxis.convertValues(data); | |||||
var d, d2, d3; | |||||
d = this._catmullRom(data,0.5); | |||||
d3 = this._catmullRom(data,0); | |||||
d2 = this._catmullRom(data,1); | |||||
// var data2 = []; | |||||
// this.startTime = this.range.start; | |||||
// var min = Date.now() - 3600000 * 24 * 30; | |||||
// var max = Date.now() + 3600000 * 24 * 10; | |||||
// var count = 60; | |||||
// var step = (max-min) / count; | |||||
// | |||||
// var range = this.range.end - this.range.start; | |||||
// var rangePerPixel = range/this.width; | |||||
// var rangePerPixelInv = this.width/range; | |||||
// var xOffset = -this.range.start + this.width*rangePerPixel; | |||||
// | |||||
// for (var i = 0; i < count; i++) { | |||||
// data2.push({x:(min + i*step + xOffset) * rangePerPixelInv, y: 250*(i%2) + 25}) | |||||
// } | |||||
// | |||||
// var d2 = this._catmullRom(data2); | |||||
this.path.setAttributeNS(null, "d",d); | |||||
this.path2.setAttributeNS(null, "d",d2); | |||||
this.path3.setAttributeNS(null, "d",d3); | |||||
} | |||||
} | |||||
} | |||||
} | |||||
/** | |||||
* Set options for the Linegraph. Existing options will be extended/overwritten. | |||||
* @param {Object} [options] The following options are available: | |||||
* {String | function} [className] | |||||
* class name for the itemset | |||||
* {String} [type] | |||||
* Default type for the items. Choose from 'box' | |||||
* (default), 'point', or 'range'. The default | |||||
* Style can be overwritten by individual items. | |||||
* {String} align | |||||
* Alignment for the items, only applicable for | |||||
* ItemBox. Choose 'center' (default), 'left', or | |||||
* 'right'. | |||||
* {String} orientation | |||||
* Orientation of the item set. Choose 'top' or | |||||
* 'bottom' (default). | |||||
* {Number} margin.axis | |||||
* Margin between the axis and the items in pixels. | |||||
* Default is 20. | |||||
* {Number} margin.item | |||||
* Margin between items in pixels. Default is 10. | |||||
* {Number} padding | |||||
* Padding of the contents of an item in pixels. | |||||
* Must correspond with the items css. Default is 5. | |||||
* {Function} snap | |||||
* Function to let items snap to nice dates when | |||||
* dragging items. | |||||
*/ | |||||
Linegraph.prototype.setOptions = function(options) { | |||||
Component.prototype.setOptions.call(this, options); | |||||
}; | |||||
Linegraph.prototype._extractData = function(dataset) { | |||||
var extractedData = []; | |||||
var low = dataset[0].value; | |||||
var high = dataset[0].value; | |||||
var range = this.range.end - this.range.start; | |||||
var rangePerPixel = range/this.width; | |||||
var rangePerPixelInv = this.width/range; | |||||
var xOffset = -this.range.start + this.width*rangePerPixel; | |||||
for (var i = 0; i < dataset.length; i++) { | |||||
var val = new Date(dataset[i].start).getTime(); | |||||
val += xOffset; | |||||
val *= rangePerPixelInv; | |||||
extractedData.push({x:val, y:dataset[i].value}); | |||||
if (low > dataset[i].value) { | |||||
low = dataset[i].value; | |||||
} | |||||
if (high < dataset[i].value) { | |||||
high = dataset[i].value; | |||||
} | |||||
} | |||||
//extractedData.sort(function (a,b) {return a.x - b.x;}) | |||||
return {range:{low:low,high:high},data:extractedData}; | |||||
} | |||||
Linegraph.prototype._catmullRomUniform = function(data) { | |||||
// catmull rom | |||||
var p0, p1, p2, p3, bp1, bp2 | |||||
var d = "M" + Math.round(data[0].x) + "," + Math.round(data[0].y) + " "; | |||||
var normalization = 1/6; | |||||
var length = data.length; | |||||
for (var i = 0; i < length - 1; i++) { | |||||
p0 = (i == 0) ? data[0] : data[i-1]; | |||||
p1 = data[i]; | |||||
p2 = data[i+1]; | |||||
p3 = (i + 2 < length) ? data[i+2] : p2; | |||||
// Catmull-Rom to Cubic Bezier conversion matrix | |||||
// 0 1 0 0 | |||||
// -1/6 1 1/6 0 | |||||
// 0 1/6 1 -1/6 | |||||
// 0 0 1 0 | |||||
// bp0 = { x: p1.x, y: p1.y }; | |||||
bp1 = { x: ((-p0.x + 6*p1.x + p2.x) *normalization), y: ((-p0.y + 6*p1.y + p2.y) *normalization)}; | |||||
bp2 = { x: (( p1.x + 6*p2.x - p3.x) *normalization), y: (( p1.y + 6*p2.y - p3.y) *normalization)}; | |||||
// bp0 = { x: p2.x, y: p2.y }; | |||||
d += "C" + | |||||
Math.round(bp1.x) + "," + | |||||
Math.round(bp1.y) + " " + | |||||
Math.round(bp2.x) + "," + | |||||
Math.round(bp2.y) + " " + | |||||
Math.round(p2.x) + "," + | |||||
Math.round(p2.y) + " "; | |||||
} | |||||
return d; | |||||
}; | |||||
Linegraph.prototype._catmullRom = function(data, alpha) { | |||||
if (alpha == 0 || alpha === undefined) { | |||||
return this._catmullRomUniform(data); | |||||
} | |||||
else { | |||||
var p0, p1, p2, p3, bp1, bp2, d1,d2,d3, A, B, N, M; | |||||
var d3powA, d2powA, d3pow2A, d2pow2A, d1pow2A, d1powA; | |||||
var d = "M" + Math.round(data[0].x) + "," + Math.round(data[0].y) + " "; | |||||
var length = data.length; | |||||
for (var i = 0; i < length - 1; i++) { | |||||
p0 = (i == 0) ? data[0] : data[i-1]; | |||||
p1 = data[i]; | |||||
p2 = data[i+1]; | |||||
p3 = (i + 2 < length) ? data[i+2] : p2; | |||||
d1 = Math.sqrt(Math.pow(p0.x - p1.x,2) + Math.pow(p0.y - p1.y,2)); | |||||
d2 = Math.sqrt(Math.pow(p1.x - p2.x,2) + Math.pow(p1.y - p2.y,2)); | |||||
d3 = Math.sqrt(Math.pow(p2.x - p3.x,2) + Math.pow(p2.y - p3.y,2)); | |||||
// Catmull-Rom to Cubic Bezier conversion matrix | |||||
// | |||||
// A = 2d1^2a + 3d1^a * d2^a + d3^2a | |||||
// B = 2d3^2a + 3d3^a * d2^a + d2^2a | |||||
// | |||||
// [ 0 1 0 0 ] | |||||
// [ -d2^2a/N A/N d1^2a/N 0 ] | |||||
// [ 0 d3^2a/M B/M -d2^2a/M ] | |||||
// [ 0 0 1 0 ] | |||||
// [ 0 1 0 0 ] | |||||
// [ -d2pow2a/N A/N d1pow2a/N 0 ] | |||||
// [ 0 d3pow2a/M B/M -d2pow2a/M ] | |||||
// [ 0 0 1 0 ] | |||||
d3powA = Math.pow(d3, alpha); | |||||
d3pow2A = Math.pow(d3,2*alpha); | |||||
d2powA = Math.pow(d2, alpha); | |||||
d2pow2A = Math.pow(d2,2*alpha); | |||||
d1powA = Math.pow(d1, alpha); | |||||
d1pow2A = Math.pow(d1,2*alpha); | |||||
A = 2*d1pow2A + 3*d1powA * d2powA + d2pow2A; | |||||
B = 2*d3pow2A + 3*d3powA * d2powA + d2pow2A; | |||||
N = 3*d1powA * (d1powA + d2powA); | |||||
if (N > 0) {N = 1 / N;} | |||||
M = 3*d3powA * (d3powA + d2powA); | |||||
if (M > 0) {M = 1 / M;} | |||||
bp1 = { x: ((-d2pow2A * p0.x + A*p1.x + d1pow2A * p2.x) * N), | |||||
y: ((-d2pow2A * p0.y + A*p1.y + d1pow2A * p2.y) * N)}; | |||||
bp2 = { x: (( d3pow2A * p1.x + B*p2.x - d2pow2A * p3.x) * M), | |||||
y: (( d3pow2A * p1.y + B*p2.y - d2pow2A * p3.y) * M)}; | |||||
if (bp1.x == 0 && bp1.y == 0) {bp1 = p1;} | |||||
if (bp2.x == 0 && bp2.y == 0) {bp2 = p2;} | |||||
d += "C" + | |||||
Math.round(bp1.x) + "," + | |||||
Math.round(bp1.y) + " " + | |||||
Math.round(bp2.x) + "," + | |||||
Math.round(bp2.y) + " " + | |||||
Math.round(p2.x) + "," + | |||||
Math.round(p2.y) + " "; | |||||
} | |||||
return d; | |||||
} | |||||
}; | |||||
Linegraph.prototype._linear = function(data) { | |||||
// linear | |||||
var d = ""; | |||||
for (var i = 0; i < data.length; i++) { | |||||
if (i == 0) { | |||||
d += "M" + data[i].x + "," + data[i].y; | |||||
} | |||||
else { | |||||
d += " " + data[i].x + "," + data[i].y; | |||||
} | |||||
} | |||||
return d; | |||||
} | |||||
/** | |||||
* Set range (start and end). | |||||
* @param {Range | Object} range A Range or an object containing start and end. | |||||
*/ | |||||
Linegraph.prototype.setRange = function(range) { | |||||
if (!(range instanceof Range) && (!range || !range.start || !range.end)) { | |||||
throw new TypeError('Range must be an instance of Range, ' + | |||||
'or an object containing start and end.'); | |||||
} | |||||
this.range = range; | |||||
}; | |||||
Linegraph.prototype.repaint = function() { | |||||
var margin = this.options.margin, | |||||
range = this.range, | |||||
asSize = util.option.asSize, | |||||
asString = util.option.asString, | |||||
options = this.options, | |||||
orientation = this.getOption('orientation'), | |||||
resized = false, | |||||
frame = this.frame; | |||||
// TODO: document this feature to specify one margin for both item and axis distance | |||||
if (typeof margin === 'number') { | |||||
margin = { | |||||
item: margin, | |||||
axis: margin | |||||
}; | |||||
} | |||||
// update className | |||||
this.frame.className = 'itemset' + (options.className ? (' ' + asString(options.className)) : ''); | |||||
// check whether zoomed (in that case we need to re-stack everything) | |||||
// TODO: would be nicer to get this as a trigger from Range | |||||
var visibleInterval = this.range.end - this.range.start; | |||||
var zoomed = (visibleInterval != this.lastVisibleInterval) || (this.width != this.lastWidth); | |||||
if (zoomed) this.stackDirty = true; | |||||
this.lastVisibleInterval = visibleInterval; | |||||
this.lastWidth = this.width; | |||||
// reposition frame | |||||
this.frame.style.left = asSize(options.left, ''); | |||||
this.frame.style.right = asSize(options.right, ''); | |||||
this.frame.style.top = asSize((orientation == 'top') ? '0' : ''); | |||||
this.frame.style.bottom = asSize((orientation == 'top') ? '' : '0'); | |||||
this.frame.style.width = asSize(options.width, '100%'); | |||||
// frame.style.height = asSize(height); | |||||
//frame.style.height = asSize('height' in options ? options.height : height); // TODO: reckon with height | |||||
// calculate actual size and position | |||||
this.top = this.frame.offsetTop; | |||||
this.left = this.frame.offsetLeft; | |||||
this.width = this.frame.offsetWidth; | |||||
// this.height = height; | |||||
// check if this component is resized | |||||
resized = this._isResized() || resized; | |||||
if (resized) { | |||||
this.svg.style.width = asSize(3*this.width); | |||||
this.svg.style.left = asSize(-this.width); | |||||
} | |||||
if (zoomed) { | |||||
this.setData(); | |||||
} | |||||
} |
@ -0,0 +1,34 @@ | |||||
.vis.timeline .dataaxis .grid.horizontal { | |||||
position: absolute; | |||||
left: 0; | |||||
width: 100%; | |||||
height: 0; | |||||
border-bottom: 1px solid; | |||||
} | |||||
.vis.timeline .dataaxis .grid.minor { | |||||
border-color: #e5e5e5; | |||||
} | |||||
.vis.timeline .dataaxis .grid.major { | |||||
border-color: #bfbfbf; | |||||
} | |||||
.vis.timeline .dataaxis .yAxis.major { | |||||
font-size:12px; | |||||
width: 100%; | |||||
position: absolute; | |||||
color: #4d4d4d; | |||||
white-space: nowrap; | |||||
} | |||||
.vis.timeline .dataaxis .yAxis.minor{ | |||||
font-size:9px; | |||||
position: absolute; | |||||
width: 100%; | |||||
color: #4d4d4d; | |||||
white-space: nowrap; | |||||
} |