From c7780e1a325fe910af6500597de1d91531f75017 Mon Sep 17 00:00:00 2001 From: nagix Date: Tue, 21 Aug 2018 02:57:44 +0000 Subject: [PATCH] Release 1.6.0 --- bower.json | 15 + dist/chartjs-plugin-streaming.js | 975 +++++++++++++++++++++++++++ dist/chartjs-plugin-streaming.min.js | 11 + 3 files changed, 1001 insertions(+) create mode 100644 bower.json create mode 100644 dist/chartjs-plugin-streaming.js create mode 100644 dist/chartjs-plugin-streaming.min.js diff --git a/bower.json b/bower.json new file mode 100644 index 0000000..a0c126c --- /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.6.0", + "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..d8a4190 --- /dev/null +++ b/dist/chartjs-plugin-streaming.js @@ -0,0 +1,975 @@ +/* + * @license + * chartjs-plugin-streaming + * https://github.com/nagix/chartjs-plugin-streaming/ + * Version: 1.6.0 + * + * 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 realTimeScale = function(Chart$$1, moment$$1) { + + var helpers = Chart$$1.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); + }; + } + }()); + + var realTimeScaleDefaultConfig = { + 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 + }, + }, + ticks: { + autoSkip: false, + source: 'auto', + major: { + enabled: true + } + } + }; + + // Ported from Chart.js 2.7.2 37ec838. + var MAX_INTEGER = Number.MAX_SAFE_INTEGER || 9007199254740991; + + // Ported from Chart.js 2.7.2 37ec838. + 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, 30] + }, + minute: { + common: true, + size: 60000, + steps: [1, 2, 5, 10, 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.2 37ec838. + var UNITS = Object.keys(INTERVALS); + + // Ported from Chart.js 2.7.2 37ec838. + 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$$1(value, format); + } + + if (!(value instanceof moment$$1)) { + value = moment$$1(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.2 37ec838. + 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.2 37ec838. + 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.2 37ec838. + function determineUnitForFormatting(ticks, minUnit, min, max) { + var duration = moment$$1.duration(moment$$1(max).diff(moment$$1(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.2 37ec838. + 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.2 37ec838. 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$$1(min); + // For realtime scale: Add refresh interval for scroll margin. + var last = moment$$1(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$$1(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.2 37ec838. + 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$$1(value).startOf(majorUnit) : false; + + ticks.push({ + value: value, + major: major + }); + } + + return ticks; + } + + // Ported from Chart.js 2.7.2 37ec838. + 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'; + } + + var hidden, visibilityChange; + if (typeof document.hidden !== 'undefined') { // Opera 12.10 and Firefox 18 and later support + hidden = 'hidden'; + visibilityChange = 'visibilitychange'; + } else if (typeof document.msHidden !== 'undefined') { + hidden = 'msHidden'; + visibilityChange = 'msvisibilitychange'; + } else if (typeof document.webkitHidden !== 'undefined') { + hidden = 'webkitHidden'; + visibilityChange = 'webkitvisibilitychange'; + } + + 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; + } + } + } + + var TimeScale = Chart$$1.scaleService.getScaleConstructor('time'); + + Chart$$1.scaleService.getScaleConstructor = function(type) { + // For backwards compatibility + if (type === 'time') { + type = 'realtime'; + } + return this.constructors.hasOwnProperty(type) ? this.constructors[type] : undefined; + }; + + var RealTimeScale = TimeScale.extend({ + initialize: function() { + var me = this; + var chart = me.chart; + + TimeScale.prototype.initialize.apply(me, arguments); + + // For backwards compatibility + if (me.options.type === 'time' && !chart.options.plugins.streaming) { + return; + } + + var requestAnimFrame = helpers.requestAnimFrame; + var realtime = me.realtime = me.realtime || {}; + var lastDrawn = 0; + var visibilityChangeListener = function() { + if (!document[hidden]) { + realtime.head = Date.now(); + chart.update({ + duration: 0 + }); + } + }; + document.addEventListener(visibilityChange, visibilityChangeListener, false); + realtime.visibilityChangeListener = visibilityChangeListener; + + var frameRefresh = function() { + var valueOrDefault = helpers.valueOrDefault; + var realtimeOpts = me.options.realtime || {}; + var streamingOpts = chart.options.plugins.streaming || {}; + var duration = valueOrDefault(realtimeOpts.duration, streamingOpts.duration); + var delay = valueOrDefault(realtimeOpts.delay, streamingOpts.delay); + var frameRate = valueOrDefault(realtimeOpts.frameRate, streamingOpts.frameRate); + var id = me.id; + var tooltip = chart.tooltip; + var activeTooltip = tooltip._active; + var frameDuration = 1000 / (Math.max(frameRate, 0) || 30); + var keys, length, meta; + + if (me.isHorizontal()) { + length = me.width; + keys = transitionKeys.x; + } else { + length = me.height; + keys = transitionKeys.y; + } + + var now = Date.now(); + var 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) { + var elements = meta.data || []; + var i, ilen; + + 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); + } + } + + me.max = me._table[1].time = now - delay; + me.min = me._table[0].time = me.max - duration; + + if (lastDrawn + frameDuration <= now) { + // Draw only when animation is inactive + if (!chart.animating && !tooltip._start) { + chart.draw(); + } + chart.streaming.onDraw(chart); + lastDrawn += frameDuration; + if (lastDrawn + frameDuration <= now) { + lastDrawn = now; + } + } + + realtime.head = now; + realtime.frameRequestID = requestAnimFrame.call(window, frameRefresh); + }; + + realtime.head = Date.now(); + realtime.frameRequestID = requestAnimFrame.call(window, frameRefresh); + }, + + update: function() { + var me = this; + var options = me.options; + var chart = me.chart; + + // For backwards compatibility + if (options.type === 'time' && !chart.options.plugins.streaming) { + return TimeScale.prototype.update.apply(me, arguments); + } + + var frameRequestID = me.realtime.frameRequestID; + var streamingOpts = chart.options.plugins.streaming || {}; + var pause = streamingOpts.pause; + + if (!frameRequestID && !pause) { + me.initialize(); + } else if (frameRequestID && pause) { + me.destroy(); + } + + return TimeScale.prototype.update.apply(me, arguments); + }, + + buildTicks: function() { + var me = this; + var options = me.options; + var chart = me.chart; + + // For backwards compatibility + if (options.type === 'time' && !chart.options.plugins.streaming) { + return TimeScale.prototype.buildTicks.apply(me, arguments); + } + + var valueOrDefault = helpers.valueOrDefault; + var timeOpts = options.time; + var realtimeOpts = options.realtime || {}; + var streamingOpts = chart.options.plugins.streaming || {}; + var duration = valueOrDefault(realtimeOpts.duration, streamingOpts.duration); + var delay = valueOrDefault(realtimeOpts.delay, streamingOpts.delay); + var refresh = valueOrDefault(streamingOpts.refresh, 0); + 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 realtime = this.realtime; + var visibilityChangeListener = realtime.visibilityChangeListener; + var frameRequestID = realtime.frameRequestID; + + if (visibilityChangeListener) { + document.removeEventListener(visibilityChange, visibilityChangeListener, false); + delete realtime.visibilityChangeListener; + } + if (frameRequestID) { + helpers.cancelAnimFrame.call(window, frameRequestID); + delete realtime.frameRequestID; + } + } + }); + + Chart$$1.scaleService.registerScaleType('realtime', RealTimeScale, realTimeScaleDefaultConfig); +}; + +'use strict'; + +var streamingPlugin = function(Chart$$1) { + + var helpers = Chart$$1.helpers; + + Chart$$1.defaults.global.plugins.streaming = { + duration: 10000, + delay: 0, + frameRate: 30, + refresh: 1000, + onRefresh: null, + pause: false, + ttl: undefined + }; + + var realTimeScale = Chart$$1.scaleService.getScaleConstructor('realtime'); + + // 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); + } + } + + var datasetPropertyKeys = [ + 'pointBackgroundColor', + 'pointBorderColor', + 'pointBorderWidth', + 'pointRadius', + 'pointStyle', + 'pointHitRadius', + 'pointHoverBackgroundColor', + 'pointHoverBorderColor', + 'pointHoverBorderWidth', + 'pointHoverRadius', + 'backgroundColor', + 'borderColor', + 'borderWidth', + 'hoverBackgroundColor', + 'hoverBorderColor', + 'hoverBorderWidth', + 'hoverRadius', + 'hitRadius', + 'radius' + ]; + + function removeOldData(scale, lower, ttl, dataset, datasetIndex) { + var data = dataset.data; + var backlog = 2; + var i, ilen; + + if (!isNaN(ttl)) { + lower = scale.getPixelForValue(Date.now() - ttl); + backlog = 0; + } + + for (i = backlog, ilen = data.length; i < ilen; ++i) { + if (!(scale.getPixelForValue(null, i, datasetIndex) <= lower)) { + break; + } + } + // Keep the last two data points outside the range not to affect the existing bezier curve + data.splice(0, i - backlog); + datasetPropertyKeys.forEach(function(key) { + if (dataset.hasOwnProperty(key) && helpers.isArray(dataset[key])) { + dataset[key].splice(0, i - backlog); + } + }); + if (typeof data[0] !== 'object') { + return i - backlog; + } + } + + /** + * 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$$1.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); + } + + function onRefresh(chart) { + var streamingOpts = chart.options.plugins.streaming || {}; + var ttl = streamingOpts.ttl; + var meta, scale, numToRemove; + + if (streamingOpts.onRefresh) { + streamingOpts.onRefresh(chart); + } + + // Remove old data + chart.data.datasets.forEach(function(dataset, datasetIndex) { + meta = chart.getDatasetMeta(datasetIndex); + if (meta.xAxisID) { + scale = meta.controller.getScaleForId(meta.xAxisID); + if (scale instanceof realTimeScale) { + numToRemove = removeOldData(scale, scale.left, ttl, dataset, datasetIndex); + } + } + if (meta.yAxisID) { + scale = meta.controller.getScaleForId(meta.yAxisID); + if (scale instanceof realTimeScale) { + numToRemove = removeOldData(scale, scale.top, ttl, dataset, datasetIndex); + } + } + }); + if (numToRemove) { + chart.data.labels.splice(0, numToRemove); + } + + updateChartData(chart); + } + + function clearRefreshTimer(chart) { + var streaming = chart.streaming; + var refreshTimerID = streaming.refreshTimerID; + + if (refreshTimerID) { + clearInterval(refreshTimerID); + delete streaming.refreshTimerID; + delete streaming.refresh; + } + } + + function setRefreshTimer(chart, refresh) { + var streaming = chart.streaming; + + streaming.refreshTimerID = setInterval(function() { + var streamingOpts = chart.options.plugins.streaming || {}; + var newRefresh = streamingOpts.refresh; + + onRefresh(chart); + if (streaming.refresh !== newRefresh && !isNaN(newRefresh)) { + clearRefreshTimer(chart); + setRefreshTimer(chart, newRefresh); + } + }, refresh); + streaming.refresh = refresh; + } + + return { + id: 'streaming', + + beforeInit: function(chart) { + var canvas = chart.canvas; + var streaming = chart.streaming = chart.streaming || {}; + var mouseEventListener = function(event) { + streaming.lastMouseEvent = event; + }; + + canvas.addEventListener('mousedown', mouseEventListener); + canvas.addEventListener('mouseup', mouseEventListener); + streaming.mouseEventListener = mouseEventListener; + streaming.onDraw = generateMouseMoveEvent; + }, + + afterInit: function(chart, options) { + setRefreshTimer(chart, options.refresh); + }, + + 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; + }, + + 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 canvas = chart.canvas; + var listener = chart.streaming.mouseButtonListener; + + canvas.removeEventListener('mousedown', listener); + canvas.removeEventListener('mouseup', listener); + + clearRefreshTimer(chart); + helpers.each(chart.scales, function(scale) { + if (scale instanceof realTimeScale) { + scale.destroy(); + } + }); + } + }; +}; + +'use strict'; + +var zoomPlugin = function(Chart$$1) { + + var helpers = Chart$$1.helpers; + + // Ported from chartjs-plugin-zoom 0.6.3 9e7bc34 + var zoomNS = Chart$$1.Zoom = Chart$$1.Zoom || {}; + + // Ported from chartjs-plugin-zoom 0.6.3 9e7bc34 + zoomNS.zoomFunctions = zoomNS.zoomFunctions || {}; + zoomNS.panFunctions = zoomNS.panFunctions || {}; + + // Ported from chartjs-plugin-zoom 0.6.3 9e7bc34 + 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.3 9e7bc34 + 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 options = scale.options; + var realtimeOpts = options.realtime = 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 options = scale.options; + var realtimeOpts = options.realtime = 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; +}; + +'use strict'; + +realTimeScale(Chart, moment); + +var plugin = streamingPlugin(Chart); +Chart.plugins.register(plugin); +zoomPlugin(Chart); + +return plugin; + +}))); diff --git a/dist/chartjs-plugin-streaming.min.js b/dist/chartjs-plugin-streaming.min.js new file mode 100644 index 0000000..ec0c3bb --- /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.6.0 + * + * 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(e,t){"use strict";!function(e,h){var b=e.helpers;b.cancelAnimFrame=function(){if("undefined"!=typeof window)return window.cancelAnimationFrame||window.webkitCancelAnimationFrame||window.mozCancelAnimationFrame||window.oCancelAnimationFrame||window.msCancelAnimationFrame||function(e){return window.clearTimeout(e)}}();var t,i,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,30]},minute:{common:!0,size:6e4,steps:[1,2,5,10,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 x(e){for(var t=y.indexOf(e)+1,n=y.length;t=y.indexOf(t);a--)if(o=y[a],g[o].common&&r.as(o)>=e.length)return o;return y[t?y.indexOf(t):0]}(m,a.minUnit,e.min,e.max),e._majorUnit=x(e._unit),e._table=[{time:d,pos:0},{time:c,pos:1}],e._offsets={left:0,right:0},e._labelFormat=function(e,t){var n,i,a,o,r,s,l,u=e.length;for(n=0;n