From bc8674310be3559465bb9a95c96b60614e5dea4f Mon Sep 17 00:00:00 2001 From: nagix Date: Sun, 2 May 2021 09:32:05 +0000 Subject: [PATCH] Release 2.0.0-beta.1 --- bower.json | 15 + dist/chartjs-plugin-streaming.js | 882 +++++++++++++++++++++++++++ dist/chartjs-plugin-streaming.min.js | 7 + 3 files changed, 904 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..09c284b --- /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": "2.0.0-beta.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..02d12f2 --- /dev/null +++ b/dist/chartjs-plugin-streaming.js @@ -0,0 +1,882 @@ +/*! + * chartjs-plugin-streaming v2.0.0-beta.1 + * https://nagix.github.io/chartjs-plugin-streaming + * (c) 2021 Akihiko Kusanagi + * Released under the MIT license + */ +(function (global, factory) { +typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('chart.js'), require('chart.js/helpers')) : +typeof define === 'function' && define.amd ? define(['chart.js', 'chart.js/helpers'], factory) : +(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.ChartStreaming = factory(global.Chart, global.Chart.helpers)); +}(this, (function (Chart, helpers) { 'use strict'; + +function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } + +var Chart__default = /*#__PURE__*/_interopDefaultLegacy(Chart); + +function clamp(value, lower, upper) { + return Math.min(Math.max(value, lower), upper); +} + +function resolveOption(scale, key) { + const realtimeOpts = scale.options.realtime; + const streamingOpts = scale.chart.options.plugins.streaming; + return helpers.valueOrDefault(realtimeOpts[key], streamingOpts[key]); +} + +/** +* Cancel animation polyfill +*/ +const cancelAnimFrame = (function() { + if (typeof window === 'undefined') { + return helpers.noop; + } + return window.cancelAnimationFrame; +}()); + +function startFrameRefreshTimer(context, func) { + if (!context.frameRequestID) { + const frameRefresh = function() { + func(); + context.frameRequestID = helpers.requestAnimFrame.call(window, frameRefresh); + }; + context.frameRequestID = helpers.requestAnimFrame.call(window, frameRefresh); + } +} + +function stopFrameRefreshTimer(context) { + const frameRequestID = context.frameRequestID; + + if (frameRequestID) { + cancelAnimFrame.call(window, frameRequestID); + delete context.frameRequestID; + } +} + +var StreamingHelper = /*#__PURE__*/Object.freeze({ +__proto__: null, +clamp: clamp, +resolveOption: resolveOption, +cancelAnimFrame: cancelAnimFrame, +startFrameRefreshTimer: startFrameRefreshTimer, +stopFrameRefreshTimer: stopFrameRefreshTimer +}); + +const chartStates = new WeakMap(); + +function getState(chart) { + let state = chartStates.get(chart); + + if (!state) { + state = {originalScaleLimits: {}}; + chartStates.set(chart, state); + } + return state; +} + +function storeOriginalScaleLimits(chart) { + const {originalScaleLimits} = getState(chart); + const scales = chart.scales; + + helpers.each(scales, scale => { + const id = scale.id; + + if (!originalScaleLimits[id]) { + originalScaleLimits[id] = { + duration: resolveOption(scale, 'duration'), + delay: resolveOption(scale, 'delay') + }; + } + }); + helpers.each(originalScaleLimits, (opt, key) => { + if (!scales[key]) { + delete originalScaleLimits[key]; + } + }); + return originalScaleLimits; +} + +function zoomRealTimeScale(scale, zoom, center, limits) { + const {chart, axis} = scale; + const {minDuration = 0, maxDuration = Infinity, minDelay = -Infinity, maxDelay = Infinity} = limits && limits[axis] || {}; + const realtimeOpts = scale.options.realtime; + const duration = resolveOption(scale, 'duration'); + const delay = resolveOption(scale, 'delay'); + const newDuration = clamp(duration * (2 - zoom), minDuration, maxDuration); + let maxPercent, newDelay; + + storeOriginalScaleLimits(chart); + + if (scale.isHorizontal()) { + maxPercent = (scale.right - center.x) / (scale.right - scale.left); + } else { + maxPercent = (scale.bottom - center.y) / (scale.bottom - scale.top); + } + newDelay = delay + maxPercent * (duration - newDuration); + realtimeOpts.duration = newDuration; + realtimeOpts.delay = clamp(newDelay, minDelay, maxDelay); + return newDuration !== scale.max - scale.min; +} + +function panRealTimeScale(scale, delta, limits) { + const {chart, axis} = scale; + const {minDelay = -Infinity, maxDelay = Infinity} = limits && limits[axis] || {}; + const delay = resolveOption(scale, 'delay'); + const newDelay = delay + (scale.getValueForPixel(delta) - scale.getValueForPixel(0)); + + storeOriginalScaleLimits(chart); + + scale.options.realtime.delay = clamp(newDelay, minDelay, maxDelay); + return true; +} + +function resetRealTimeOptions(chart) { + const originalScaleLimits = storeOriginalScaleLimits(chart); + + helpers.each(chart.scales, scale => { + const realtimeOptions = scale.options.realtime; + + if (realtimeOptions) { + const original = originalScaleLimits[scale.id]; + + if (original) { + realtimeOptions.duration = original.duration; + realtimeOptions.delay = original.delay; + } else { + delete realtimeOptions.duration; + delete realtimeOptions.delay; + } + } + }); +} + +const TimeScale = Chart__default['default'].registry.getScale('time'); + +// Ported from Chart.js 2.8.0 35273ee. +const 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.8.0 35273ee. +const UNITS = Object.keys(INTERVALS); + +// Ported from Chart.js 2.8.0 35273ee. +function determineStepSize(min, max, unit, capacity) { + const range = max - min; + const {size: milliseconds, steps} = INTERVALS[unit]; + let factor; + + if (!steps) { + return Math.ceil(range / (capacity * milliseconds)); + } + + for (let 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.8.0 35273ee. +function determineUnitForAutoTicks(minUnit, min, max, capacity) { + const range = max - min; + const ilen = UNITS.length; + + for (let i = UNITS.indexOf(minUnit); i < ilen - 1; ++i) { + const {common, size, steps} = INTERVALS[UNITS[i]]; + const factor = steps ? steps[steps.length - 1] : Number.MAX_SAFE_INTEGER; + + if (common && Math.ceil(range / (factor * size)) <= capacity) { + return UNITS[i]; + } + } + + return UNITS[ilen - 1]; +} + +// Ported from Chart.js 2.8.0 35273ee. +function determineMajorUnit(unit) { + for (let i = UNITS.indexOf(unit) + 1, ilen = UNITS.length; i < ilen; ++i) { + if (INTERVALS[UNITS[i]].common) { + return UNITS[i]; + } + } +} + +// Ported from Chart.js 3.2.0 e1404ac. +function addTick(ticks, time, timestamps) { + if (!timestamps) { + ticks[time] = true; + } else if (timestamps.length) { + const {lo, hi} = helpers._lookup(timestamps, time); + const timestamp = timestamps[lo] >= time ? timestamps[lo] : timestamps[hi]; + ticks[timestamp] = true; + } +} + +function getAxisKey(meta, id) { + return id === meta.xAxisID && 'x' || id === meta.yAxisID && 'y'; +} + +const datasetPropertyKeys = [ + 'pointBackgroundColor', + 'pointBorderColor', + 'pointBorderWidth', + 'pointRadius', + 'pointRotation', + 'pointStyle', + 'pointHitRadius', + 'pointHoverBackgroundColor', + 'pointHoverBorderColor', + 'pointHoverBorderWidth', + 'pointHoverRadius', + 'backgroundColor', + 'borderColor', + 'borderSkipped', + 'borderWidth', + 'hoverBackgroundColor', + 'hoverBorderColor', + 'hoverBorderWidth', + 'hoverRadius', + 'hitRadius', + 'radius', + 'rotation' +]; + +function refreshData(scale) { + const {chart, id, max} = scale; + const duration = resolveOption(scale, 'duration'); + const delay = resolveOption(scale, 'delay'); + const ttl = resolveOption(scale, 'ttl'); + const pause = resolveOption(scale, 'pause'); + const onRefresh = resolveOption(scale, 'onRefresh'); + const min = Date.now() - (isNaN(ttl) ? duration + delay : ttl); + let i, start, count, removalRange; + + helpers.callback(onRefresh, [chart]); + + // Remove old data + chart.data.datasets.forEach((dataset, datasetIndex) => { + const meta = chart.getDatasetMeta(datasetIndex); + const axis = getAxisKey(meta, id); + + if (axis) { + const controller = meta.controller; + const data = dataset.data; + const length = data.length; + + if (pause) { + // If the scale is paused, preserve the visible data points + for (i = 0; i < length; ++i) { + if (!(controller.getParsed(i)[axis] < max)) { + break; + } + } + start = i + 2; + } else { + start = 0; + } + + for (i = start; i < length; ++i) { + if (!(controller.getParsed(i)[axis] <= 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(key => { + if (dataset.hasOwnProperty(key) && helpers.isArray(dataset[key])) { + dataset[key].splice(start, count); + } + }); + helpers.each(dataset.datalabels, value => { + if (helpers.isArray(value)) { + value.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('quiet'); +} + +function stopDataRefreshTimer(scale) { + const realtime = scale.realtime; + const refreshTimerID = realtime.refreshTimerID; + + if (refreshTimerID) { + clearInterval(refreshTimerID); + delete realtime.refreshTimerID; + delete realtime.refreshInterval; + } +} + +function startDataRefreshTimer(scale) { + const realtime = scale.realtime; + const interval = resolveOption(scale, 'refresh'); + + if (realtime.refreshTimerID) { + return; + } + + realtime.refreshTimerID = setInterval(() => { + const newInterval = resolveOption(scale, 'refresh'); + + refreshData(scale); + if (realtime.refreshInterval !== newInterval && !isNaN(newInterval)) { + stopDataRefreshTimer(scale); + startDataRefreshTimer(scale); + } + }, interval); + realtime.refreshInterval = interval; +} + +const 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) { + const animations = element.$animations; + + for (let i = 0, ilen = keys.length; i < ilen; ++i) { + const key = keys[i]; + + if (!isNaN(element[key])) { + element[key] -= translate; + } + } + if (animations) { + for (let i = 0, ilen = keys.length; i < ilen; ++i) { + const value = animations[keys[i]]; + + if (value) { + value._from -= translate; + value._to -= translate; + } + } + } +} + +function scroll(scale) { + const {chart, id, realtime} = scale; + const duration = resolveOption(scale, 'duration'); + const delay = resolveOption(scale, 'delay'); + const isHorizontal = scale.isHorizontal(); + const tooltip = chart.tooltip; + const activeTooltip = tooltip._active; + const now = Date.now(); + let length, keys, offset; + + if (isHorizontal) { + length = scale.width; + keys = transitionKeys.x; + } else { + length = scale.height; + keys = transitionKeys.y; + } + offset = length * (now - realtime.head) / duration; + + if (!!isHorizontal === !!scale.options.reverse) { + offset = -offset; + } + + // Shift all the elements leftward or upward + helpers.each(chart.data.datasets, (dataset, datasetIndex) => { + const meta = chart.getDatasetMeta(datasetIndex); + + if (getAxisKey(meta, id)) { + const {data, dataset: element} = meta; + const elements = data || []; + + for (let i = 0, ilen = elements.length; i < ilen; ++i) { + transition(elements[i], keys.data, offset); + } + + if (element) { + transition(element, keys.dataset, offset); + } + } + }); + + // Shift tooltip leftward or upward + if (activeTooltip && activeTooltip[0]) { + const meta = chart.getDatasetMeta(activeTooltip[0].datasetIndex); + + if (getAxisKey(meta, id)) { + transition(tooltip, keys.tooltip, offset); + } + } + + scale.max = now - delay; + scale.min = scale.max - duration; + + realtime.head = now; +} + +class RealTimeScale extends TimeScale { + + constructor(props) { + super(props); + this.realtime = this.realtime || {}; + } + + init(scaleOpts, opts) { + super.init(scaleOpts, opts); + startDataRefreshTimer(this); + } + + update(maxWidth, maxHeight, margins) { + const me = this; + const {realtime, options} = me; + const {bounds, offset, ticks: ticksOpts} = options; + const {autoSkip, source, major: majorTicksOpts} = ticksOpts; + const majorEnabled = majorTicksOpts.enabled; + + if (resolveOption(me, 'pause')) { + stopFrameRefreshTimer(realtime); + } else { + startFrameRefreshTimer(realtime, () => { + scroll(me); + }); + realtime.head = Date.now(); + } + + options.bounds = undefined; + options.offset = false; + ticksOpts.autoSkip = false; + ticksOpts.source = source === 'auto' ? '' : source; + majorTicksOpts.enabled = true; + + super.update(maxWidth, maxHeight, margins); + + options.bounds = bounds; + options.offset = offset; + ticksOpts.autoSkip = autoSkip; + ticksOpts.source = source; + majorTicksOpts.enabled = majorEnabled; + } + + buildTicks() { + const me = this; + const duration = resolveOption(me, 'duration'); + const delay = resolveOption(me, 'delay'); + const max = me.realtime.head - delay; + const min = max - duration; + const maxArray = [1e15, max]; + const minArray = [-1e15, min]; + + Object.defineProperty(me, 'min', { + get: () => minArray.shift(), + set: helpers.noop + }); + Object.defineProperty(me, 'max', { + get: () => maxArray.shift(), + set: helpers.noop + }); + + const ticks = super.buildTicks(); + + delete me.min; + delete me.max; + me.min = min; + me.max = max; + + return ticks; + } + + calculateLabelRotation() { + const ticksOpts = this.options.ticks; + const maxRotation = ticksOpts.maxRotation; + + ticksOpts.maxRotation = ticksOpts.minRotation || 0; + super.calculateLabelRotation(); + ticksOpts.maxRotation = maxRotation; + } + + fit() { + const me = this; + const options = me.options; + + super.fit(); + + if (options.ticks.display && options.display && me.isHorizontal()) { + me.paddingLeft = 3; + me.paddingRight = 3; + me._handleMargins(); + } + } + + draw(chartArea) { + const me = this; + const {chart, ctx} = me; + const area = me.isHorizontal() ? + { + left: chartArea.left, + top: 0, + right: chartArea.right, + bottom: chart.height + } : { + left: 0, + top: chartArea.top, + right: chart.width, + bottom: chartArea.bottom + }; + + me._gridLineItems = null; + me._labelItems = null; + + // Clip and draw the scale + helpers.clipArea(ctx, area); + super.draw(chartArea); + helpers.unclipArea(ctx); + } + + destroy() { + const me = this; + + stopFrameRefreshTimer(me.realtime); + stopDataRefreshTimer(me); + } + + _generate() { + const me = this; + const adapter = me._adapter; + const duration = resolveOption(me, 'duration'); + const delay = resolveOption(me, 'delay'); + const refresh = resolveOption(me, 'refresh'); + const max = me.realtime.head - delay; + const min = max - duration; + const capacity = me._getLabelCapacity(min); + const {time: timeOpts, ticks: ticksOpts} = me.options; + const minor = timeOpts.unit || determineUnitForAutoTicks(timeOpts.minUnit, min, max, capacity); + const major = determineMajorUnit(minor); + const stepSize = timeOpts.stepSize || determineStepSize(min, max, minor, capacity); + const weekday = minor === 'week' ? timeOpts.isoWeekday : false; + const majorTicksEnabled = ticksOpts.major.enabled; + const hasWeekday = helpers.isNumber(weekday) || weekday === true; + const interval = INTERVALS[minor]; + const ticks = {}; + let first = min; + let time, count; + + // For 'week' unit, handle the first day of week option + if (hasWeekday) { + first = +adapter.startOf(first, 'isoWeek', weekday); + } + + // Align first ticks on unit + first = +adapter.startOf(first, hasWeekday ? 'day' : minor); + + // Prevent browser from freezing in case user options request millions of milliseconds + if (adapter.diff(max, min, minor) > 100000 * stepSize) { + throw new Error(min + ' and ' + max + ' are too far apart with stepSize of ' + stepSize + ' ' + minor); + } + + time = first; + + if (majorTicksEnabled && major && !hasWeekday && !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 = +adapter.startOf(time, major); + time = +adapter.add(time, ~~((first - time) / (interval.size * stepSize)) * stepSize, minor); + } + + const timestamps = ticksOpts.source === 'data' && me.getDataTimestamps(); + for (count = 0; time < max + refresh; time = +adapter.add(time, stepSize, minor), count++) { + addTick(ticks, time, timestamps); + } + + if (time === max + refresh || count === 1) { + addTick(ticks, time, timestamps); + } + + return Object.keys(ticks).sort((a, b) => a - b).map(x => +x); + } +} + +RealTimeScale.id = 'realtime'; + +RealTimeScale.defaults = { + bounds: 'data', + adapters: {}, + time: { + parser: false, // false == a pattern string from or a custom callback that converts its argument to a timestamp + unit: false, // false == automatic or override with week, month, year, etc. + round: false, // none, or override with week, month, year, etc. + isoWeekday: false, // override week start day - see http://momentjs.com/docs/#/get-set/iso-weekday/ + minUnit: 'millisecond', + displayFormats: {} + }, + realtime: {}, + ticks: { + autoSkip: false, + source: 'auto', + major: { + enabled: true + } + } +}; + +Chart__default['default'].defaults.describe('scale.realtime', { + _scriptable: name => name !== 'onRefresh' +}); + +// Draw chart at frameRate +function drawChart(chart) { + const streaming = chart.streaming; + const frameRate = chart.options.plugins.streaming.frameRate; + const frameDuration = 1000 / (Math.max(frameRate, 0) || 30); + const next = streaming.lastDrawn + frameDuration || 0; + const now = Date.now(); + + if (next <= now) { + chart.render(); + if (streaming.lastMouseEvent) { + setTimeout(() => { + const lastMouseEvent = streaming.lastMouseEvent; + if (lastMouseEvent) { + chart._eventHandler(lastMouseEvent); + } + }, 0); + } + streaming.lastDrawn = (next + frameDuration > now) ? next : now; + } +} + +var StreamingPlugin = { + id: 'streaming', + + beforeInit(chart) { + const streaming = chart.streaming = chart.streaming || {}; + const canvas = streaming.canvas = chart.canvas; + const mouseEventListener = streaming.mouseEventListener = event => { + const pos = helpers.getRelativePosition(event, chart); + streaming.lastMouseEvent = { + type: 'mousemove', + chart: chart, + native: event, + x: pos.x, + y: pos.y + }; + }; + + canvas.addEventListener('mousedown', mouseEventListener); + canvas.addEventListener('mouseup', mouseEventListener); + + chart.options.transitions.quiet = { + animation: { + duration: 0 + } + }; + }, + + afterInit(chart) { + const {update, render, resetZoom} = chart; + + chart.update = mode => { + if (mode === 'quiet') { + // Skip the render call in the quiet mode + chart.render = helpers.noop; + update.call(chart, mode); + chart.render = render; + } else { + update.call(chart, mode); + } + }; + + if (resetZoom) { + const zoomPlugin = Chart__default['default'].registry.getPlugin('zoom'); + + zoomPlugin.zoomFunctions.realtime = zoomRealTimeScale; + zoomPlugin.panFunctions.realtime = panRealTimeScale; + chart.resetZoom = transition => { + resetRealTimeOptions(chart); + resetZoom(transition); + }; + } + }, + + beforeUpdate(chart) { + const chartOpts = chart.options; + const scalesOpts = chartOpts.scales; + + if (scalesOpts) { + Object.keys(scalesOpts).forEach(id => { + const scaleOpts = scalesOpts[id]; + + if (scaleOpts.type === 'realtime') { + // Allow Bézier control to be outside the chart + chartOpts.elements.line.capBezierPoints = false; + } + }); + } + return true; + }, + + afterUpdate(chart) { + const {scales, streaming} = chart; + let pause = true; + + // if all scales are paused, stop refreshing frames + helpers.each(scales, scale => { + if (scale instanceof RealTimeScale) { + pause &= resolveOption(scale, 'pause'); + } + }); + if (pause) { + stopFrameRefreshTimer(streaming); + } else { + startFrameRefreshTimer(streaming, () => { + drawChart(chart); + }); + } + }, + + beforeDatasetDraw(chart, args) { + const {ctx, chartArea, width, height} = chart; + const {xAxisID, yAxisID, controller} = args.meta; + const area = { + left: 0, + top: 0, + right: width, + bottom: height + }; + + if (xAxisID && controller.getScaleForId(xAxisID) instanceof RealTimeScale) { + area.left = chartArea.left; + area.right = chartArea.right; + } + if (yAxisID && controller.getScaleForId(yAxisID) instanceof RealTimeScale) { + area.top = chartArea.top; + area.bottom = chartArea.bottom; + } + helpers.clipArea(ctx, area); + return true; + }, + + afterDatasetDraw(chart) { + helpers.unclipArea(chart.ctx); + }, + + beforeEvent(chart, args) { + const streaming = chart.streaming; + const event = args.event; + + if (event.type === 'mousemove') { + // Save mousemove event for reuse + streaming.lastMouseEvent = event; + } else if (event.type === 'mouseout') { + // Remove mousemove event + delete streaming.lastMouseEvent; + } + return true; + }, + + destroy(chart) { + const {scales, streaming} = chart; + const {canvas, mouseEventListener} = streaming; + + stopFrameRefreshTimer(streaming); + + canvas.removeEventListener('mousedown', mouseEventListener); + canvas.removeEventListener('mouseup', mouseEventListener); + + helpers.each(scales, scale => { + if (scale instanceof RealTimeScale) { + scale.destroy(); + } + }); + }, + + defaults: { + duration: 10000, + delay: 0, + frameRate: 30, + refresh: 1000, + onRefresh: null, + pause: false, + ttl: undefined + }, + + descriptors: { + _scriptable: name => name !== 'onRefresh' + } +}; + +Chart__default['default'].helpers.streaming = StreamingHelper; + +Chart__default['default'].register(StreamingPlugin, RealTimeScale); + +return StreamingPlugin; + +}))); diff --git a/dist/chartjs-plugin-streaming.min.js b/dist/chartjs-plugin-streaming.min.js new file mode 100644 index 0000000..a3e0436 --- /dev/null +++ b/dist/chartjs-plugin-streaming.min.js @@ -0,0 +1,7 @@ +/*! + * chartjs-plugin-streaming v2.0.0-beta.1 + * https://nagix.github.io/chartjs-plugin-streaming + * (c) 2021 Akihiko Kusanagi + * Released under the MIT license + */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("chart.js"),require("chart.js/helpers")):"function"==typeof define&&define.amd?define(["chart.js","chart.js/helpers"],t):(e="undefined"!=typeof globalThis?globalThis:e||self).ChartStreaming=t(e.Chart,e.Chart.helpers)}(this,(function(e,t){"use strict";function o(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var a=o(e);function n(e,t,o){return Math.min(Math.max(e,t),o)}function r(e,o){const a=e.options.realtime,n=e.chart.options.plugins.streaming;return t.valueOrDefault(a[o],n[o])}const i="undefined"==typeof window?t.noop:window.cancelAnimationFrame;function s(e,o){if(!e.frameRequestID){const a=function(){o(),e.frameRequestID=t.requestAnimFrame.call(window,a)};e.frameRequestID=t.requestAnimFrame.call(window,a)}}function l(e){const t=e.frameRequestID;t&&(i.call(window,t),delete e.frameRequestID)}var c=Object.freeze({__proto__:null,clamp:n,resolveOption:r,cancelAnimFrame:i,startFrameRefreshTimer:s,stopFrameRefreshTimer:l});const d=new WeakMap;function u(e){const{originalScaleLimits:o}=function(e){let t=d.get(e);return t||(t={originalScaleLimits:{}},d.set(e,t)),t}(e),a=e.scales;return t.each(a,(e=>{const t=e.id;o[t]||(o[t]={duration:r(e,"duration"),delay:r(e,"delay")})})),t.each(o,((e,t)=>{a[t]||delete o[t]})),o}function m(e,t,o,a){const{chart:i,axis:s}=e,{minDuration:l=0,maxDuration:c=1/0,minDelay:d=-1/0,maxDelay:m=1/0}=a&&a[s]||{},f=e.options.realtime,p=r(e,"duration"),h=r(e,"delay"),g=n(p*(2-t),l,c);let y,b;return u(i),y=e.isHorizontal()?(e.right-o.x)/(e.right-e.left):(e.bottom-o.y)/(e.bottom-e.top),b=h+y*(p-g),f.duration=g,f.delay=n(b,d,m),g!==e.max-e.min}function f(e,t,o){const{chart:a,axis:i}=e,{minDelay:s=-1/0,maxDelay:l=1/0}=o&&o[i]||{},c=r(e,"delay")+(e.getValueForPixel(t)-e.getValueForPixel(0));return u(a),e.options.realtime.delay=n(c,s,l),!0}const p=a.default.registry.getScale("time"),h={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}},g=Object.keys(h);function y(e,o,a){if(a){if(a.length){const{lo:n,hi:r}=t._lookup(a,o);e[a[n]>=o?a[n]:a[r]]=!0}}else e[o]=!0}function b(e,t){return t===e.xAxisID?"x":t===e.yAxisID&&"y"}const v=["pointBackgroundColor","pointBorderColor","pointBorderWidth","pointRadius","pointRotation","pointStyle","pointHitRadius","pointHoverBackgroundColor","pointHoverBorderColor","pointHoverBorderWidth","pointHoverRadius","backgroundColor","borderColor","borderSkipped","borderWidth","hoverBackgroundColor","hoverBorderColor","hoverBorderWidth","hoverRadius","hitRadius","radius","rotation"];function x(e){const t=e.realtime,o=t.refreshTimerID;o&&(clearInterval(o),delete t.refreshTimerID,delete t.refreshInterval)}function D(e){const o=e.realtime,a=r(e,"refresh");o.refreshTimerID||(o.refreshTimerID=setInterval((()=>{const a=r(e,"refresh");!function(e){const{chart:o,id:a,max:n}=e,i=r(e,"duration"),s=r(e,"delay"),l=r(e,"ttl"),c=r(e,"pause"),d=r(e,"onRefresh"),u=Date.now()-(isNaN(l)?i+s:l);let m,f,p,h;t.callback(d,[o]),o.data.datasets.forEach(((e,r)=>{const i=o.getDatasetMeta(r),s=b(i,a);if(s){const o=i.controller,a=e.data,r=a.length;if(c){for(m=0;m{e.hasOwnProperty(o)&&t.isArray(e[o])&&e[o].splice(f,p)})),t.each(e.datalabels,(e=>{t.isArray(e)&&e.splice(f,p)})),"object"!=typeof a[0]&&(h={start:f,count:p})}})),h&&o.data.labels.splice(h.start,h.count),o.update("quiet")}(e),o.refreshInterval===a||isNaN(a)||(x(e),D(e))}),a),o.refreshInterval=a)}const w={data:["x","controlPointPreviousX","controlPointNextX"],dataset:["x"],tooltip:["x","caretX"]},R={data:["y","controlPointPreviousY","controlPointNextY"],dataset:["y"],tooltip:["y","caretY"]};function k(e,t,o){const a=e.$animations;for(let a=0,n=t.length;a{!function(e){const{chart:o,id:a,realtime:n}=e,i=r(e,"duration"),s=r(e,"delay"),l=e.isHorizontal(),c=o.tooltip,d=c._active,u=Date.now();let m,f,p;l?(m=e.width,f=w):(m=e.height,f=R),p=m*(u-n.head)/i,!!l==!!e.options.reverse&&(p=-p),t.each(o.data.datasets,((e,t)=>{const n=o.getDatasetMeta(t);if(b(n,a)){const{data:e,dataset:t}=n,o=e||[];for(let e=0,t=o.length;el.shift(),set:t.noop}),Object.defineProperty(e,"max",{get:()=>s.shift(),set:t.noop});const c=super.buildTicks();return delete e.min,delete e.max,e.min=i,e.max=n,c}calculateLabelRotation(){const e=this.options.ticks,t=e.maxRotation;e.maxRotation=e.minRotation||0,super.calculateLabelRotation(),e.maxRotation=t}fit(){const e=this,t=e.options;super.fit(),t.ticks.display&&t.display&&e.isHorizontal()&&(e.paddingLeft=3,e.paddingRight=3,e._handleMargins())}draw(e){const o=this,{chart:a,ctx:n}=o,r=o.isHorizontal()?{left:e.left,top:0,right:e.right,bottom:a.height}:{left:0,top:e.top,right:a.width,bottom:e.bottom};o._gridLineItems=null,o._labelItems=null,t.clipArea(n,r),super.draw(e),t.unclipArea(n)}destroy(){l(this.realtime),x(this)}_generate(){const e=this,o=e._adapter,a=r(e,"duration"),n=r(e,"delay"),i=r(e,"refresh"),s=e.realtime.head-n,l=s-a,c=e._getLabelCapacity(l),{time:d,ticks:u}=e.options,m=d.unit||function(e,t,o,a){const n=o-t,r=g.length;for(let t=g.indexOf(e);t1e5*p)throw new Error(l+" and "+s+" are too far apart with stepSize of "+p+" "+m);R=I,v&&f&&!x&&!d.round&&(R=+o.startOf(R,f),R=+o.add(R,~~((I-R)/(D.size*p))*p,m));const z="data"===u.source&&e.getDataTimestamps();for(k=0;Re-t)).map((e=>+e))}}I.id="realtime",I.defaults={bounds:"data",adapters:{},time:{parser:!1,unit:!1,round:!1,isoWeekday:!1,minUnit:"millisecond",displayFormats:{}},realtime:{},ticks:{autoSkip:!1,source:"auto",major:{enabled:!0}}},a.default.defaults.describe("scale.realtime",{_scriptable:e=>"onRefresh"!==e});var z={id:"streaming",beforeInit(e){const o=e.streaming=e.streaming||{},a=o.canvas=e.canvas,n=o.mouseEventListener=a=>{const n=t.getRelativePosition(a,e);o.lastMouseEvent={type:"mousemove",chart:e,native:a,x:n.x,y:n.y}};a.addEventListener("mousedown",n),a.addEventListener("mouseup",n),e.options.transitions.quiet={animation:{duration:0}}},afterInit(e){const{update:o,render:n,resetZoom:r}=e;if(e.update=a=>{"quiet"===a?(e.render=t.noop,o.call(e,a),e.render=n):o.call(e,a)},r){const o=a.default.registry.getPlugin("zoom");o.zoomFunctions.realtime=m,o.panFunctions.realtime=f,e.resetZoom=o=>{!function(e){const o=u(e);t.each(e.scales,(e=>{const t=e.options.realtime;if(t){const a=o[e.id];a?(t.duration=a.duration,t.delay=a.delay):(delete t.duration,delete t.delay)}}))}(e),r(o)}}},beforeUpdate(e){const t=e.options,o=t.scales;return o&&Object.keys(o).forEach((e=>{"realtime"===o[e].type&&(t.elements.line.capBezierPoints=!1)})),!0},afterUpdate(e){const{scales:o,streaming:a}=e;let n=!0;t.each(o,(e=>{e instanceof I&&(n&=r(e,"pause"))})),n?l(a):s(a,(()=>{!function(e){const t=e.streaming,o=e.options.plugins.streaming.frameRate,a=1e3/(Math.max(o,0)||30),n=t.lastDrawn+a||0,r=Date.now();n<=r&&(e.render(),t.lastMouseEvent&&setTimeout((()=>{const o=t.lastMouseEvent;o&&e._eventHandler(o)}),0),t.lastDrawn=n+a>r?n:r)}(e)}))},beforeDatasetDraw(e,o){const{ctx:a,chartArea:n,width:r,height:i}=e,{xAxisID:s,yAxisID:l,controller:c}=o.meta,d={left:0,top:0,right:r,bottom:i};return s&&c.getScaleForId(s)instanceof I&&(d.left=n.left,d.right=n.right),l&&c.getScaleForId(l)instanceof I&&(d.top=n.top,d.bottom=n.bottom),t.clipArea(a,d),!0},afterDatasetDraw(e){t.unclipArea(e.ctx)},beforeEvent(e,t){const o=e.streaming,a=t.event;return"mousemove"===a.type?o.lastMouseEvent=a:"mouseout"===a.type&&delete o.lastMouseEvent,!0},destroy(e){const{scales:o,streaming:a}=e,{canvas:n,mouseEventListener:r}=a;l(a),n.removeEventListener("mousedown",r),n.removeEventListener("mouseup",r),t.each(o,(e=>{e instanceof I&&e.destroy()}))},defaults:{duration:1e4,delay:0,frameRate:30,refresh:1e3,onRefresh:null,pause:!1,ttl:void 0},descriptors:{_scriptable:e=>"onRefresh"!==e}};return a.default.helpers.streaming=c,a.default.register(z,I),z}));