diff --git a/bower.json b/bower.json new file mode 100644 index 0000000..e2ca1ad --- /dev/null +++ b/bower.json @@ -0,0 +1,15 @@ +{ + "name": "chartjs-plugin-streaming", + "description": "Chart.js plugin for live streaming data", + "homepage": "https://nagix.github.io/chartjs-plugin-streaming", + "license": "MIT", + "version": "1.7.1", + "main": "./dist/chartjs-plugin-streaming.js", + "ignore": [ + ".codeclimate.yml", + ".gitignore", + ".npmignore", + ".travis.yml", + "scripts" + ] +} \ No newline at end of file diff --git a/dist/chartjs-plugin-streaming.js b/dist/chartjs-plugin-streaming.js new file mode 100644 index 0000000..6a9705d --- /dev/null +++ b/dist/chartjs-plugin-streaming.js @@ -0,0 +1,1060 @@ +/* + * @license + * chartjs-plugin-streaming + * https://github.com/nagix/chartjs-plugin-streaming/ + * Version: 1.7.1 + * + * Copyright 2018 Akihiko Kusanagi + * Released under the MIT license + * https://github.com/nagix/chartjs-plugin-streaming/blob/master/LICENSE.md + */ +(function (global, factory) { + typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('chart.js'), require('moment')) : + typeof define === 'function' && define.amd ? define(['chart.js', 'moment'], factory) : + (global['chartjs-plugin-streaming'] = factory(global.Chart,global.moment)); +}(this, (function (Chart,moment) { 'use strict'; + +Chart = Chart && Chart.hasOwnProperty('default') ? Chart['default'] : Chart; +moment = moment && moment.hasOwnProperty('default') ? moment['default'] : moment; + +'use strict'; + +var helpers = Chart.helpers; + +helpers.cancelAnimFrame = (function() { + if (typeof window !== 'undefined') { + return window.cancelAnimationFrame || + window.webkitCancelAnimationFrame || + window.mozCancelAnimationFrame || + window.oCancelAnimationFrame || + window.msCancelAnimationFrame || + function(id) { + return window.clearTimeout(id); + }; + } +}()); + +helpers.startFrameRefreshTimer = function(context, func) { + if (!context.frameRequestID) { + var frameRefresh = function() { + func(); + context.frameRequestID = helpers.requestAnimFrame.call(window, frameRefresh); + }; + context.frameRequestID = helpers.requestAnimFrame.call(window, frameRefresh); + } +}; + +helpers.stopFrameRefreshTimer = function(context) { + var frameRequestID = context.frameRequestID; + + if (frameRequestID) { + helpers.cancelAnimFrame.call(window, frameRequestID); + delete context.frameRequestID; + } +}; + +'use strict'; + +var scaleService = Chart.scaleService; +var TimeScale = scaleService.getScaleConstructor('time'); + +scaleService.getScaleConstructor = function(type) { + // For backwards compatibility + if (type === 'time') { + type = 'realtime'; + } + return this.constructors.hasOwnProperty(type) ? this.constructors[type] : undefined; +}; + +// Ported from Chart.js 2.7.3 1cd0469. +var MAX_INTEGER = Number.MAX_SAFE_INTEGER || 9007199254740991; + +// Ported from Chart.js 2.7.3 1cd0469. +var INTERVALS = { + millisecond: { + common: true, + size: 1, + steps: [1, 2, 5, 10, 20, 50, 100, 250, 500] + }, + second: { + common: true, + size: 1000, + steps: [1, 2, 5, 10, 15, 30] + }, + minute: { + common: true, + size: 60000, + steps: [1, 2, 5, 10, 15, 30] + }, + hour: { + common: true, + size: 3600000, + steps: [1, 2, 3, 6, 12] + }, + day: { + common: true, + size: 86400000, + steps: [1, 2, 5] + }, + week: { + common: false, + size: 604800000, + steps: [1, 2, 3, 4] + }, + month: { + common: true, + size: 2.628e9, + steps: [1, 2, 3] + }, + quarter: { + common: false, + size: 7.884e9, + steps: [1, 2, 3, 4] + }, + year: { + common: true, + size: 3.154e10 + } +}; + +// Ported from Chart.js 2.7.3 1cd0469. +var UNITS = Object.keys(INTERVALS); + +// Ported from Chart.js 2.7.3 1cd0469. +function momentify(value, options) { + var parser = options.parser; + var format = options.parser || options.format; + + if (typeof parser === 'function') { + return parser(value); + } + + if (typeof value === 'string' && typeof format === 'string') { + return moment(value, format); + } + + if (!(value instanceof moment)) { + value = moment(value); + } + + if (value.isValid()) { + return value; + } + + // Labels are in an incompatible moment format and no `parser` has been provided. + // The user might still use the deprecated `format` option to convert his inputs. + if (typeof format === 'function') { + return format(value); + } + + return value; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function parse(input, scale) { + if (helpers.isNullOrUndef(input)) { + return null; + } + + var options = scale.options.time; + var value = momentify(scale.getRightValue(input), options); + if (!value.isValid()) { + return null; + } + + if (options.round) { + value.startOf(options.round); + } + + return value.valueOf(); +} + +// Ported from Chart.js 2.7.3 1cd0469. +function determineStepSize(min, max, unit, capacity) { + var range = max - min; + var interval = INTERVALS[unit]; + var milliseconds = interval.size; + var steps = interval.steps; + var i, ilen, factor; + + if (!steps) { + return Math.ceil(range / (capacity * milliseconds)); + } + + for (i = 0, ilen = steps.length; i < ilen; ++i) { + factor = steps[i]; + if (Math.ceil(range / (milliseconds * factor)) <= capacity) { + break; + } + } + + return factor; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function determineUnitForAutoTicks(minUnit, min, max, capacity) { + var ilen = UNITS.length; + var i, interval, factor; + + for (i = UNITS.indexOf(minUnit); i < ilen - 1; ++i) { + interval = INTERVALS[UNITS[i]]; + factor = interval.steps ? interval.steps[interval.steps.length - 1] : MAX_INTEGER; + + if (interval.common && Math.ceil((max - min) / (factor * interval.size)) <= capacity) { + return UNITS[i]; + } + } + + return UNITS[ilen - 1]; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function determineUnitForFormatting(ticks, minUnit, min, max) { + var duration = moment.duration(moment(max).diff(moment(min))); + var ilen = UNITS.length; + var i, unit; + + for (i = ilen - 1; i >= UNITS.indexOf(minUnit); i--) { + unit = UNITS[i]; + if (INTERVALS[unit].common && duration.as(unit) >= ticks.length) { + return unit; + } + } + + return UNITS[minUnit ? UNITS.indexOf(minUnit) : 0]; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function determineMajorUnit(unit) { + for (var i = UNITS.indexOf(unit) + 1, ilen = UNITS.length; i < ilen; ++i) { + if (INTERVALS[UNITS[i]].common) { + return UNITS[i]; + } + } +} + +// Ported from Chart.js 2.7.3 1cd0469. Modified for realtime scale. +function generate(min, max, capacity, options, refresh) { + var timeOpts = options.time; + var minor = timeOpts.unit || determineUnitForAutoTicks(timeOpts.minUnit, min, max, capacity); + var major = determineMajorUnit(minor); + var stepSize = helpers.valueOrDefault(timeOpts.stepSize, timeOpts.unitStepSize); + var weekday = minor === 'week' ? timeOpts.isoWeekday : false; + // For realtime scale: Major ticks are always enabled. + var majorTicksEnabled = true; + var interval = INTERVALS[minor]; + var first = moment(min); + // For realtime scale: Add refresh interval for scroll margin. + var last = moment(max + refresh); + var ticks = []; + var time; + + if (!stepSize) { + stepSize = determineStepSize(min, max, minor, capacity); + } + + // For 'week' unit, handle the first day of week option + if (weekday) { + first = first.isoWeekday(weekday); + last = last.isoWeekday(weekday); + } + + // Align first/last ticks on unit + first = first.startOf(weekday ? 'day' : minor); + last = last.startOf(weekday ? 'day' : minor); + + // Make sure that the last tick include max + if (last < max + refresh) { + last.add(1, minor); + } + + time = moment(first); + + if (majorTicksEnabled && major && !weekday && !timeOpts.round) { + // Align the first tick on the previous `minor` unit aligned on the `major` unit: + // we first aligned time on the previous `major` unit then add the number of full + // stepSize there is between first and the previous major time. + time.startOf(major); + time.add(~~((first - time) / (interval.size * stepSize)) * stepSize, minor); + } + + for (; time < last; time.add(stepSize, minor)) { + ticks.push(+time); + } + + ticks.push(+time); + + return ticks; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function ticksFromTimestamps(values, majorUnit) { + var ticks = []; + var i, ilen, value, major; + + for (i = 0, ilen = values.length; i < ilen; ++i) { + value = values[i]; + major = majorUnit ? value === +moment(value).startOf(majorUnit) : false; + + ticks.push({ + value: value, + major: major + }); + } + + return ticks; +} + +// Ported from Chart.js 2.7.3 1cd0469. +function determineLabelFormat(data, timeOpts) { + var i, momentDate, hasTime; + var ilen = data.length; + + // find the label with the most parts (milliseconds, minutes, etc.) + // format all labels with the same level of detail as the most specific label + for (i = 0; i < ilen; i++) { + momentDate = momentify(data[i], timeOpts); + if (momentDate.millisecond() !== 0) { + return 'MMM D, YYYY h:mm:ss.SSS a'; + } + if (momentDate.second() !== 0 || momentDate.minute() !== 0 || momentDate.hour() !== 0) { + hasTime = true; + } + } + if (hasTime) { + return 'MMM D, YYYY h:mm:ss a'; + } + return 'MMM D, YYYY'; +} + +function resolveOption(scale, key) { + var realtimeOpts = scale.options.realtime; + var streamingOpts = scale.chart.options.plugins.streaming; + return helpers.valueOrDefault(realtimeOpts[key], streamingOpts[key]); +} + +var datasetPropertyKeys = [ + 'pointBackgroundColor', + 'pointBorderColor', + 'pointBorderWidth', + 'pointRadius', + 'pointStyle', + 'pointHitRadius', + 'pointHoverBackgroundColor', + 'pointHoverBorderColor', + 'pointHoverBorderWidth', + 'pointHoverRadius', + 'backgroundColor', + 'borderColor', + 'borderWidth', + 'hoverBackgroundColor', + 'hoverBorderColor', + 'hoverBorderWidth', + 'hoverRadius', + 'hitRadius', + 'radius' +]; + +function refreshData(scale) { + var chart = scale.chart; + var id = scale.id; + var duration = resolveOption(scale, 'duration'); + var delay = resolveOption(scale, 'delay'); + var ttl = resolveOption(scale, 'ttl'); + var pause = resolveOption(scale, 'pause'); + var onRefresh = resolveOption(scale, 'onRefresh'); + var max = scale.max; + var min = Date.now() - (isNaN(ttl) ? duration + delay : ttl); + var meta, data, length, i, start, count, removalRange; + + if (onRefresh) { + onRefresh(chart); + } + + // Remove old data + chart.data.datasets.forEach(function(dataset, datasetIndex) { + meta = chart.getDatasetMeta(datasetIndex); + if (id === meta.xAxisID || id === meta.yAxisID) { + data = dataset.data; + length = data.length; + + if (pause) { + // If the scale is paused, preserve the visible data points + for (i = 0; i < length; ++i) { + if (!(scale._getTimeForIndex(i, datasetIndex) < max)) { + break; + } + } + start = i + 2; + } else { + start = 0; + } + + for (i = start; i < length; ++i) { + if (!(scale._getTimeForIndex(i, datasetIndex) <= min)) { + break; + } + } + count = i - start; + if (isNaN(ttl)) { + // Keep the last two data points outside the range not to affect the existing bezier curve + count = Math.max(count - 2, 0); + } + + data.splice(start, count); + datasetPropertyKeys.forEach(function(key) { + if (dataset.hasOwnProperty(key) && helpers.isArray(dataset[key])) { + dataset[key].splice(start, count); + } + }); + if (typeof data[0] !== 'object') { + removalRange = { + start: start, + count: count + }; + } + } + }); + if (removalRange) { + chart.data.labels.splice(removalRange.start, removalRange.count); + } + + chart.update({ + preservation: true + }); +} + +function stopDataRefreshTimer(scale) { + var realtime = scale.realtime; + var refreshTimerID = realtime.refreshTimerID; + + if (refreshTimerID) { + clearInterval(refreshTimerID); + delete realtime.refreshTimerID; + delete realtime.refreshInterval; + } +} + +function startDataRefreshTimer(scale) { + var realtime = scale.realtime; + var interval = resolveOption(scale, 'refresh'); + + realtime.refreshTimerID = setInterval(function() { + var newInterval = resolveOption(scale, 'refresh'); + + refreshData(scale); + if (realtime.refreshInterval !== newInterval && !isNaN(newInterval)) { + stopDataRefreshTimer(scale); + startDataRefreshTimer(scale); + } + }, interval); + realtime.refreshInterval = interval; +} + +var transitionKeys = { + x: { + data: ['x', 'controlPointPreviousX', 'controlPointNextX'], + dataset: ['x'], + tooltip: ['x', 'caretX'] + }, + y: { + data: ['y', 'controlPointPreviousY', 'controlPointNextY'], + dataset: ['y'], + tooltip: ['y', 'caretY'] + } +}; + +function transition(element, keys, translate) { + var start = element._start || {}; + var view = element._view || {}; + var model = element._model || {}; + var i, ilen; + + for (i = 0, ilen = keys.length; i < ilen; ++i) { + var key = keys[i]; + if (start.hasOwnProperty(key)) { + start[key] -= translate; + } + if (view.hasOwnProperty(key) && view !== start) { + view[key] -= translate; + } + if (model.hasOwnProperty(key) && model !== view) { + model[key] -= translate; + } + } +} + +function scroll(scale) { + var chart = scale.chart; + var realtime = scale.realtime; + var duration = resolveOption(scale, 'duration'); + var delay = resolveOption(scale, 'delay'); + var id = scale.id; + var tooltip = chart.tooltip; + var activeTooltip = tooltip._active; + var now = Date.now(); + var length, keys, offset, meta, elements, i, ilen; + + if (scale.isHorizontal()) { + length = scale.width; + keys = transitionKeys.x; + } else { + length = scale.height; + keys = transitionKeys.y; + } + offset = length * (now - realtime.head) / duration; + + // Shift all the elements leftward or upward + helpers.each(chart.data.datasets, function(dataset, datasetIndex) { + meta = chart.getDatasetMeta(datasetIndex); + if (id === meta.xAxisID || id === meta.yAxisID) { + elements = meta.data || []; + + for (i = 0, ilen = elements.length; i < ilen; ++i) { + transition(elements[i], keys.data, offset); + } + + if (meta.dataset) { + transition(meta.dataset, keys.dataset, offset); + } + } + }); + + // Shift tooltip leftward or upward + if (activeTooltip && activeTooltip[0]) { + meta = chart.getDatasetMeta(activeTooltip[0]._datasetIndex); + if (id === meta.xAxisID || id === meta.yAxisID) { + transition(tooltip, keys.tooltip, offset); + } + } + + scale.max = scale._table[1].time = now - delay; + scale.min = scale._table[0].time = scale.max - duration; + + realtime.head = now; +} + +var defaultConfig = { + position: 'bottom', + distribution: 'linear', + bounds: 'data', + + time: { + parser: false, // false == a pattern string from http://momentjs.com/docs/#/parsing/string-format/ or a custom callback that converts its argument to a moment + format: false, // DEPRECATED false == date objects, moment object, callback or a pattern string from http://momentjs.com/docs/#/parsing/string-format/ + unit: false, // false == automatic or override with week, month, year, etc. + round: false, // none, or override with week, month, year, etc. + displayFormat: false, // DEPRECATED + isoWeekday: false, // override week start day - see http://momentjs.com/docs/#/get-set/iso-weekday/ + minUnit: 'millisecond', + + // defaults to unit's corresponding unitFormat below or override using pattern string from http://momentjs.com/docs/#/displaying/format/ + displayFormats: { + millisecond: 'h:mm:ss.SSS a', // 11:20:01.123 AM, + second: 'h:mm:ss a', // 11:20:01 AM + minute: 'h:mm a', // 11:20 AM + hour: 'hA', // 5PM + day: 'MMM D', // Sep 4 + week: 'll', // Week 46, or maybe "[W]WW - YYYY" ? + month: 'MMM YYYY', // Sept 2015 + quarter: '[Q]Q - YYYY', // Q3 + year: 'YYYY' // 2015 + }, + }, + realtime: {}, + ticks: { + autoSkip: false, + source: 'auto', + major: { + enabled: true + } + } +}; + +var RealTimeScale = TimeScale.extend({ + initialize: function() { + var me = this; + + TimeScale.prototype.initialize.apply(me, arguments); + + // For backwards compatibility + if (me.options.type === 'time' && !me.chart.options.plugins.streaming) { + return; + } + + me.realtime = me.realtime || {}; + + startDataRefreshTimer(me); + }, + + update: function() { + var me = this; + var realtime = me.realtime; + + // For backwards compatibility + if (me.options.type === 'time' && !me.chart.options.plugins.streaming) { + return TimeScale.prototype.update.apply(me, arguments); + } + + if (resolveOption(me, 'pause')) { + helpers.stopFrameRefreshTimer(realtime); + } else { + helpers.startFrameRefreshTimer(realtime, function() { + scroll(me); + }); + realtime.head = Date.now(); + } + + return TimeScale.prototype.update.apply(me, arguments); + }, + + buildTicks: function() { + var me = this; + var options = me.options; + + // For backwards compatibility + if (options.type === 'time' && !me.chart.options.plugins.streaming) { + return TimeScale.prototype.buildTicks.apply(me, arguments); + } + + var timeOpts = options.time; + var duration = resolveOption(me, 'duration'); + var delay = resolveOption(me, 'delay'); + var refresh = resolveOption(me, 'refresh'); + var max = me.realtime.head - delay; + var min = max - duration; + var timestamps = []; + + switch (options.ticks.source) { + case 'data': + timestamps = me._timestamps.data; + break; + case 'labels': + timestamps = me._timestamps.labels; + break; + case 'auto': + default: + timestamps = generate(min, max, me.getLabelCapacity(min), options, refresh); + } + + me.min = min; + me.max = max; + + // PRIVATE + me._unit = timeOpts.unit || determineUnitForFormatting(timestamps, timeOpts.minUnit, me.min, me.max); + me._majorUnit = determineMajorUnit(me._unit); + // realtime scale only supports linear distribution. + me._table = [{time: min, pos: 0}, {time: max, pos: 1}]; + // offset is always disabled. + me._offsets = {left: 0, right: 0}; + me._labelFormat = determineLabelFormat(me._timestamps.data, timeOpts); + + return ticksFromTimestamps(timestamps, me._majorUnit); + }, + + fit: function() { + var me = this; + var options = me.options; + + TimeScale.prototype.fit.apply(me, arguments); + + // For backwards compatibility + if (options.type === 'time' && !me.chart.options.plugins.streaming) { + return; + } + + if (options.ticks.display && options.display && me.isHorizontal()) { + me.paddingLeft = 3; + me.paddingRight = 3; + me.handleMargins(); + } + }, + + draw: function(chartArea) { + var me = this; + var chart = me.chart; + + // For backwards compatibility + if (me.options.type === 'time' && !chart.options.plugins.streaming) { + TimeScale.prototype.draw.apply(me, arguments); + return; + } + + var context = me.ctx; + var clipArea = me.isHorizontal() ? + { + left: chartArea.left, + top: 0, + right: chartArea.right, + bottom: chart.height + } : { + left: 0, + top: chartArea.top, + right: chart.width, + bottom: chartArea.bottom + }; + + // Clip and draw the scale + helpers.canvas.clipArea(context, clipArea); + TimeScale.prototype.draw.apply(me, arguments); + helpers.canvas.unclipArea(context); + }, + + destroy: function() { + var me = this; + + // For backwards compatibility + if (me.options.type === 'time' && !me.chart.options.plugins.streaming) { + return; + } + + helpers.stopFrameRefreshTimer(me.realtime); + stopDataRefreshTimer(me); + }, + + /* + * @private + */ + _getTimeForIndex: function(index, datasetIndex) { + var me = this; + var timestamps = me._timestamps; + var time = timestamps.datasets[datasetIndex][index]; + var value; + + if (helpers.isNullOrUndef(time)) { + value = me.chart.data.datasets[datasetIndex].data[index]; + if (helpers.isObject(value)) { + time = parse(me.getRightValue(value), me); + } else { + time = parse(timestamps.labels[index], me); + } + } + + return time; + } +}); + +scaleService.registerScaleType('realtime', RealTimeScale, defaultConfig); + +'use strict'; + +Chart.defaults.global.plugins.streaming = { + duration: 10000, + delay: 0, + frameRate: 30, + refresh: 1000, + onRefresh: null, + pause: false, + ttl: undefined +}; + +// Dispach mouse event for scroll +function generateMouseMoveEvent(chart) { + var event = chart.streaming.lastMouseEvent; + var newEvent; + + if (event) { + if (typeof MouseEvent === 'function') { + newEvent = new MouseEvent('mousemove', event); + } else { + newEvent = document.createEvent('MouseEvents'); + newEvent.initMouseEvent( + 'mousemove', event.bubbles, event.cancelable, event.view, event.detail, + event.screenX, event.screenY, event.clientX, event.clientY, event.ctrlKey, + event.altKey, event.shiftKey, event.metaKey, event.button, event.relatedTarget + ); + } + chart.canvas.dispatchEvent(newEvent); + } +} + +/** + * Update the chart data keeping the current animation but suppressing a new one + * @param chart {Chart} chart to update + */ +function updateChartData(chart) { + var animationOpts = chart.options.animation; + var datasets = chart.data.datasets; + var newControllers = chart.buildOrUpdateControllers(); + + datasets.forEach(function(dataset, datasetIndex) { + chart.getDatasetMeta(datasetIndex).controller.buildOrUpdateElements(); + }); + chart.updateLayout(); + if (animationOpts && animationOpts.duration) { + helpers.each(newControllers, function(controller) { + controller.reset(); + }); + } + chart.updateDatasets(); + + if (chart.animating) { + // If the chart is animating, keep it until the duration is over + Chart.animationService.animations.forEach(function(animation) { + if (animation.chart === chart) { + chart.render({ + duration: (animation.numSteps - animation.currentStep) * 16.66 + }); + } + }); + } else { + // If the chart is not animating, make sure that all elements are at the final positions + datasets.forEach(function(dataset, datasetIndex) { + chart.getDatasetMeta(datasetIndex).controller.transition(1); + }); + } + + if (chart.tooltip._active) { + chart.tooltip.update(true); + } + + generateMouseMoveEvent(chart); +} + +var update = Chart.prototype.update; + +Chart.prototype.update = function(config) { + if (config && config.preservation) { + updateChartData(this); + } else { + update.apply(this, arguments); + } +}; + +// Draw chart at frameRate +function drawChart(chart) { + var streaming = chart.streaming; + var frameRate = chart.options.plugins.streaming.frameRate; + var frameDuration = 1000 / (Math.max(frameRate, 0) || 30); + var next = streaming.lastDrawn + frameDuration || 0; + var now = Date.now(); + + if (next <= now) { + // Draw only when animation is inactive + if (!chart.animating && !chart.tooltip._start) { + chart.draw(); + } + generateMouseMoveEvent(chart); + streaming.lastDrawn = (next + frameDuration > now) ? next : now; + } +} + +var streamingPlugin$1 = { + id: 'streaming', + + beforeInit: function(chart) { + var streaming = chart.streaming = chart.streaming || {}; + var canvas = streaming.canvas = chart.canvas; + var mouseEventListener = streaming.mouseEventListener = function(event) { + streaming.lastMouseEvent = event; + }; + + canvas.addEventListener('mousedown', mouseEventListener); + canvas.addEventListener('mouseup', mouseEventListener); + }, + + afterInit: function(chart) { + if (chart.resetZoom) { + Chart.Zoom.updateResetZoom(chart); + } + }, + + beforeUpdate: function(chart) { + var chartOpts = chart.options; + var scalesOpts = chartOpts.scales; + + if (scalesOpts) { + scalesOpts.xAxes.concat(scalesOpts.yAxes).forEach(function(scaleOpts) { + if (scaleOpts.type === 'realtime' || scaleOpts.type === 'time') { + // Allow Bézier control to be outside the chart + chartOpts.elements.line.capBezierPoints = false; + } + }); + } + return true; + }, + + afterUpdate: function(chart, options) { + var streaming = chart.streaming; + var pause = true; + + // if all scales are paused, stop refreshing frames + helpers.each(chart.scales, function(scale) { + if (scale instanceof RealTimeScale) { + pause &= helpers.valueOrDefault(scale.options.realtime.pause, options.pause); + } + }); + if (pause) { + helpers.stopFrameRefreshTimer(streaming); + } else { + helpers.startFrameRefreshTimer(streaming, function() { + drawChart(chart); + }); + } + }, + + beforeDatasetDraw: function(chart, args) { + var meta = args.meta; + var chartArea = chart.chartArea; + var clipArea = { + left: 0, + top: 0, + right: chart.width, + bottom: chart.height + }; + if (meta.xAxisID && meta.controller.getScaleForId(meta.xAxisID) instanceof RealTimeScale) { + clipArea.left = chartArea.left; + clipArea.right = chartArea.right; + } + if (meta.yAxisID && meta.controller.getScaleForId(meta.yAxisID) instanceof RealTimeScale) { + clipArea.top = chartArea.top; + clipArea.bottom = chartArea.bottom; + } + helpers.canvas.clipArea(chart.ctx, clipArea); + return true; + }, + + afterDatasetDraw: function(chart) { + helpers.canvas.unclipArea(chart.ctx); + }, + + beforeEvent: function(chart, event) { + var streaming = chart.streaming; + + if (event.type === 'mousemove') { + // Save mousemove event for reuse + streaming.lastMouseEvent = event.native; + } else if (event.type === 'mouseout') { + // Remove mousemove event + delete streaming.lastMouseEvent; + } + return true; + }, + + destroy: function(chart) { + var streaming = chart.streaming; + var canvas = streaming.canvas; + var mouseEventListener = streaming.mouseEventListener; + + helpers.stopFrameRefreshTimer(streaming); + + canvas.removeEventListener('mousedown', mouseEventListener); + canvas.removeEventListener('mouseup', mouseEventListener); + + helpers.each(chart.scales, function(scale) { + if (scale instanceof RealTimeScale) { + scale.destroy(); + } + }); + } +}; + +'use strict'; + +// Ported from chartjs-plugin-zoom 0.6.6 b0c3b20 +var zoomNS = Chart.Zoom = Chart.Zoom || {}; + +// Ported from chartjs-plugin-zoom 0.6.6 b0c3b20 +zoomNS.zoomFunctions = zoomNS.zoomFunctions || {}; +zoomNS.panFunctions = zoomNS.panFunctions || {}; + +// Ported from chartjs-plugin-zoom 0.6.6 b0c3b20 +function rangeMaxLimiter(zoomPanOptions, newMax) { + if (zoomPanOptions.scaleAxes && zoomPanOptions.rangeMax && + !helpers.isNullOrUndef(zoomPanOptions.rangeMax[zoomPanOptions.scaleAxes])) { + var rangeMax = zoomPanOptions.rangeMax[zoomPanOptions.scaleAxes]; + if (newMax > rangeMax) { + newMax = rangeMax; + } + } + return newMax; +} + +// Ported from chartjs-plugin-zoom 0.6.6 b0c3b20 +function rangeMinLimiter(zoomPanOptions, newMin) { + if (zoomPanOptions.scaleAxes && zoomPanOptions.rangeMin && + !helpers.isNullOrUndef(zoomPanOptions.rangeMin[zoomPanOptions.scaleAxes])) { + var rangeMin = zoomPanOptions.rangeMin[zoomPanOptions.scaleAxes]; + if (newMin < rangeMin) { + newMin = rangeMin; + } + } + return newMin; +} + +function zoomRealTimeScale(scale, zoom, center, zoomOptions) { + var realtimeOpts = scale.options.realtime; + var streamingOpts = scale.chart.options.plugins.streaming; + var duration = helpers.valueOrDefault(realtimeOpts.duration, streamingOpts.duration); + var delay = helpers.valueOrDefault(realtimeOpts.delay, streamingOpts.delay); + var newDuration = duration * (2 - zoom); + var maxPercent, limitedDuration; + + if (scale.isHorizontal()) { + maxPercent = (scale.right - center.x) / (scale.right - scale.left); + } else { + maxPercent = (scale.bottom - center.y) / (scale.bottom - scale.top); + } + if (zoom < 1) { + limitedDuration = rangeMaxLimiter(zoomOptions, newDuration); + } else { + limitedDuration = rangeMinLimiter(zoomOptions, newDuration); + } + realtimeOpts.duration = limitedDuration; + realtimeOpts.delay = delay + maxPercent * (duration - limitedDuration); +} + +function panRealTimeScale(scale, delta, panOptions) { + var realtimeOpts = scale.options.realtime; + var streamingOpts = scale.chart.options.plugins.streaming; + var delay = helpers.valueOrDefault(realtimeOpts.delay, streamingOpts.delay); + var newDelay = delay + (scale.getValueForPixel(delta) - scale.getValueForPixel(0)); + + if (delta > 0) { + realtimeOpts.delay = rangeMaxLimiter(panOptions, newDelay); + } else { + realtimeOpts.delay = rangeMinLimiter(panOptions, newDelay); + } +} + +zoomNS.zoomFunctions.realtime = zoomRealTimeScale; +zoomNS.panFunctions.realtime = panRealTimeScale; + +function updateResetZoom(chart) { + chart.resetZoom = function() { + helpers.each(chart.scales, function(scale) { + var timeOptions = scale.options.time; + var realtimeOptions = scale.options.realtime; + var tickOptions = scale.options.ticks; + + if (timeOptions) { + timeOptions.min = scale.originalOptions.time.min; + timeOptions.max = scale.originalOptions.time.max; + } + + if (realtimeOptions) { + realtimeOptions.duration = scale.originalOptions.realtime.duration; + realtimeOptions.delay = scale.originalOptions.realtime.delay; + } + + if (tickOptions) { + tickOptions.min = scale.originalOptions.ticks.min; + tickOptions.max = scale.originalOptions.ticks.max; + } + }); + + chart.update({ + duration: 0 + }); + }; +} + +zoomNS.updateResetZoom = updateResetZoom; + +'use strict'; + +Chart.plugins.register(streamingPlugin$1); + +return streamingPlugin$1; + +}))); diff --git a/dist/chartjs-plugin-streaming.min.js b/dist/chartjs-plugin-streaming.min.js new file mode 100644 index 0000000..7733cc4 --- /dev/null +++ b/dist/chartjs-plugin-streaming.min.js @@ -0,0 +1,11 @@ +/* + * @license + * chartjs-plugin-streaming + * https://github.com/nagix/chartjs-plugin-streaming/ + * Version: 1.7.1 + * + * Copyright 2018 Akihiko Kusanagi + * Released under the MIT license + * https://github.com/nagix/chartjs-plugin-streaming/blob/master/LICENSE.md + */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("chart.js"),require("moment")):"function"==typeof define&&define.amd?define(["chart.js","moment"],t):e["chartjs-plugin-streaming"]=t(e.Chart,e.moment)}(this,function(r,h){"use strict";r=r&&r.hasOwnProperty("default")?r.default:r,h=h&&h.hasOwnProperty("default")?h.default:h;var w=r.helpers;w.cancelAnimFrame=function(){if("undefined"!=typeof window)return window.cancelAnimationFrame||window.webkitCancelAnimationFrame||window.mozCancelAnimationFrame||window.oCancelAnimationFrame||window.msCancelAnimationFrame||function(e){return window.clearTimeout(e)}}(),w.startFrameRefreshTimer=function(e,t){if(!e.frameRequestID){var a=function(){t(),e.frameRequestID=w.requestAnimFrame.call(window,a)};e.frameRequestID=w.requestAnimFrame.call(window,a)}},w.stopFrameRefreshTimer=function(e){var t=e.frameRequestID;t&&(w.cancelAnimFrame.call(window,t),delete e.frameRequestID)};var e=r.scaleService,u=e.getScaleConstructor("time");e.getScaleConstructor=function(e){return"time"===e&&(e="realtime"),this.constructors.hasOwnProperty(e)?this.constructors[e]:void 0};var v=Number.MAX_SAFE_INTEGER||9007199254740991,g={millisecond:{common:!0,size:1,steps:[1,2,5,10,20,50,100,250,500]},second:{common:!0,size:1e3,steps:[1,2,5,10,15,30]},minute:{common:!0,size:6e4,steps:[1,2,5,10,15,30]},hour:{common:!0,size:36e5,steps:[1,2,3,6,12]},day:{common:!0,size:864e5,steps:[1,2,5]},week:{common:!1,size:6048e5,steps:[1,2,3,4]},month:{common:!0,size:2628e6,steps:[1,2,3]},quarter:{common:!1,size:7884e6,steps:[1,2,3,4]},year:{common:!0,size:3154e7}},y=Object.keys(g);function m(e,t){var a=t.parser,n=t.parser||t.format;return"function"==typeof a?a(e):"string"==typeof e&&"string"==typeof n?h(e,n):(e instanceof h||(e=h(e)),e.isValid()?e:"function"==typeof n?n(e):e)}function o(e,t){if(w.isNullOrUndef(e))return null;var a=t.options.time,n=m(t.getRightValue(e),a);return n.isValid()?(a.round&&n.startOf(a.round),n.valueOf()):null}function x(e){for(var t=y.indexOf(e)+1,a=y.length;t=y.indexOf(t);i--)if(r=y[i],g[r].common&&o.as(r)>=e.length)return r;return y[t?y.indexOf(t):0]}(l,a.minUnit,e.min,e.max),e._majorUnit=x(e._unit),e._table=[{time:s,pos:0},{time:o,pos:1}],e._offsets={left:0,right:0},e._labelFormat=function(e,t){var a,n,i,r=e.length;for(a=0;a