From: Jukka Kurkela Date: Sun, 27 Dec 2020 19:29:42 +0000 (+0200) Subject: Migrate to chartjs-test-utils (#8237) X-Git-Tag: v3.0.0-beta.8~30 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=af0658659031b6b5d8b5b1503671fda36f615c88;p=thirdparty%2FChart.js.git Migrate to chartjs-test-utils (#8237) --- diff --git a/package-lock.json b/package-lock.json index 1fc981104..8e5cbe590 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1177,6 +1177,18 @@ "integrity": "sha512-SiTLtTdjiNVbNoG4Mt4vXyp3bx2uyd8reQt+rkoYNyOeemhSMuNYZGuI3whgSI07WeL4NNVOPtv7EyPMlwjpEQ==", "dev": true }, + "chartjs-test-utils": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/chartjs-test-utils/-/chartjs-test-utils-0.1.1.tgz", + "integrity": "sha512-u+LHaWAu8yAVB/1/oSoKqrBt4OEAptxJylZAKnCReHp4vgdMaXw8xqeL2r3gZeei8l7SZk98wEk6Ldb2PKNrPg==", + "dev": true, + "requires": { + "jasmine": "^3.6.3", + "karma": "^5.2.3", + "karma-jasmine": "^4.0.1", + "pixelmatch": "^5.2.1" + } + }, "chokidar": { "version": "3.4.0", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.0.tgz", diff --git a/package.json b/package.json index 3873cf84b..4e386c1c7 100644 --- a/package.json +++ b/package.json @@ -50,6 +50,7 @@ "@rollup/plugin-json": "^4.1.0", "@rollup/plugin-node-resolve": "^11.0.1", "chartjs-adapter-moment": "^0.1.2", + "chartjs-test-utils": "^0.1.1", "concurrently": "^5.3.0", "coveralls": "^3.1.0", "cross-env": "^7.0.3", diff --git a/test/context.js b/test/context.js deleted file mode 100644 index 63c872c98..000000000 --- a/test/context.js +++ /dev/null @@ -1,162 +0,0 @@ -// Code from https://stackoverflow.com/questions/4406864/html-canvas-unit-testing -const Context = function() { - this._calls = []; // names/args of recorded calls - this._initMethods(); - - this._fillStyle = null; - this._lineCap = null; - this._lineDashOffset = null; - this._lineJoin = null; - this._lineWidth = null; - this._strokeStyle = null; - this._textAlign = null; - this._textBaseline = null; - - // Define properties here so that we can record each time they are set - Object.defineProperties(this, { - fillStyle: { - get: function() { - return this._fillStyle; - }, - set: function(style) { - this._fillStyle = style; - this.record('setFillStyle', [style]); - } - }, - lineCap: { - get: function() { - return this._lineCap; - }, - set: function(cap) { - this._lineCap = cap; - this.record('setLineCap', [cap]); - } - }, - lineDashOffset: { - get: function() { - return this._lineDashOffset; - }, - set: function(offset) { - this._lineDashOffset = offset; - this.record('setLineDashOffset', [offset]); - } - }, - lineJoin: { - get: function() { - return this._lineJoin; - }, - set: function(join) { - this._lineJoin = join; - this.record('setLineJoin', [join]); - } - }, - lineWidth: { - get: function() { - return this._lineWidth; - }, - set: function(width) { - this._lineWidth = width; - this.record('setLineWidth', [width]); - } - }, - strokeStyle: { - get: function() { - return this._strokeStyle; - }, - set: function(style) { - this._strokeStyle = style; - this.record('setStrokeStyle', [style]); - } - }, - textAlign: { - get: function() { - return this._textAlign; - }, - set: function(align) { - this._textAlign = align; - this.record('setTextAlign', [align]); - } - }, - textBaseline: { - get: function() { - return this._textBaseline; - }, - set: function(baseline) { - this._textBaseline = baseline; - this.record('setTextBaseline', [baseline]); - } - } - }); -}; - -Context.prototype._initMethods = function() { - // define methods to test here - // no way to introspect so we have to do some extra work :( - var me = this; - var methods = { - arc: function() {}, - arcTo: function() {}, - beginPath: function() {}, - bezierCurveTo: function() {}, - clearRect: function() {}, - clip: function() {}, - closePath: function() {}, - fill: function() {}, - fillRect: function() {}, - fillText: function() {}, - strokeText: function() {}, - lineTo: function() {}, - measureText: function(text) { - // return the number of characters * fixed size - // Uses fake numbers for the bounding box - return text ? { - actualBoundingBoxAscent: 4, - actualBoundingBoxDescent: 8, - actualBoundingBoxLeft: 15, - actualBoundingBoxRight: 25, - width: text.length * 10 - } : { - actualBoundingBoxAscent: 0, - actualBoundingBoxDescent: 0, - actualBoundingBoxLeft: 0, - actualBoundingBoxRight: 0, - width: 0 - }; - }, - moveTo: function() {}, - quadraticCurveTo: function() {}, - rect: function() {}, - restore: function() {}, - rotate: function() {}, - save: function() {}, - setLineDash: function() {}, - stroke: function() {}, - strokeRect: function() {}, - setTransform: function() {}, - translate: function() {}, - }; - - Object.keys(methods).forEach(function(name) { - me[name] = function() { - me.record(name, arguments); - return methods[name].apply(me, arguments); - }; - }); -}; - -Context.prototype.record = function(methodName, args) { - this._calls.push({ - name: methodName, - args: Array.prototype.slice.call(args) - }); -}; - -Context.prototype.getCalls = function() { - return this._calls; -}; - -Context.prototype.resetCalls = function() { - this._calls = []; -}; - -export default Context; diff --git a/test/fixture.js b/test/fixture.js deleted file mode 100644 index 334b682b9..000000000 --- a/test/fixture.js +++ /dev/null @@ -1,94 +0,0 @@ -/* global __karma__ */ - -'use strict'; - -import utils from './utils'; - -function readFile(url, callback) { - var request = new XMLHttpRequest(); - request.onreadystatechange = function() { - if (request.readyState === 4) { - return callback(request.responseText); - } - }; - - request.open('GET', url, false); - request.send(null); -} - -function loadConfig(url, callback) { - var regex = /\.(json|js)$/i; - var matches = url.match(regex); - var type = matches ? matches[1] : 'json'; - var cfg = null; - - readFile(url, function(content) { - switch (type) { - case 'js': - // eslint-disable-next-line - cfg = new Function('"use strict"; var module = {};' + content + '; return module.exports;')(); - break; - case 'json': - cfg = JSON.parse(content); - break; - default: - } - - callback(cfg); - }); -} - -function specFromFixture(description, inputs) { - var input = inputs.js || inputs.json; - it(input, function(done) { - loadConfig(input, function(json) { - var descr = json.description || (json.description = description); - - var config = json.config; - var options = config.options || (config.options = {}); - - // plugins are disabled by default, except if the path contains 'plugin' or there are instance plugins - if (input.indexOf('plugin') === -1 && config.plugins === undefined) { - options.plugins = options.plugins || false; - } - - var chart = utils.acquireChart(config, json.options); - if (!inputs.png) { - fail(descr + '\r\nMissing PNG comparison file for ' + input); - done(); - } - - utils.readImageData(inputs.png, function(expected) { - expect(chart).toEqualImageData(expected, json); - utils.releaseChart(chart); - done(); - }); - }); - }); -} - -function specsFromFixtures(path) { - var regex = new RegExp('(^/base/test/fixtures/' + path + '.+)\\.(png|json|js)'); - var inputs = {}; - - Object.keys(__karma__.files || {}).forEach(function(file) { - var matches = file.match(regex); - var name = matches && matches[1]; - var type = matches && matches[2]; - - if (name && type) { - inputs[name] = inputs[name] || {}; - inputs[name][type] = file; - } - }); - - return function() { - Object.keys(inputs).forEach(function(key) { - specFromFixture(key, inputs[key]); - }); - }; -} - -export default { - specs: specsFromFixtures -}; diff --git a/test/index.js b/test/index.js index 4ebaae9b0..0a764c3db 100644 --- a/test/index.js +++ b/test/index.js @@ -1,63 +1,27 @@ -import fixture from './fixture'; -import Context from './context'; -import matchers from './matchers'; -import utils from './utils'; +import {acquireChart, releaseChart, createMockContext, afterEvent, waitForResize, injectWrapperCSS, specsFromFixtures, triggerMouseEvent, addMatchers, releaseCharts} from 'chartjs-test-utils'; -(function() { +// force ratio=1 for tests on high-res/retina devices +// fixes https://github.com/chartjs/Chart.js/issues/4515 +window.devicePixelRatio = 1; - // Keep track of all acquired charts to automatically release them after each specs - var charts = {}; +window.acquireChart = acquireChart; +window.afterEvent = afterEvent; +window.releaseChart = releaseChart; +window.waitForResize = waitForResize; +window.createMockContext = createMockContext; - function acquireChart() { - var chart = utils.acquireChart.apply(utils, arguments); - charts[chart.id] = chart; - return chart; - } +injectWrapperCSS(); - function releaseChart(chart) { - utils.releaseChart.apply(utils, arguments); - delete charts[chart.id]; - } +jasmine.fixture = { + specs: specsFromFixtures +}; - function createMockContext() { - return new Context(); - } +jasmine.triggerMouseEvent = triggerMouseEvent; - // force ratio=1 for tests on high-res/retina devices - // fixes https://github.com/chartjs/Chart.js/issues/4515 - window.devicePixelRatio = 1; +beforeEach(function() { + addMatchers(); +}); - window.acquireChart = acquireChart; - window.afterEvent = utils.afterEvent; - window.releaseChart = releaseChart; - window.waitForResize = utils.waitForResize; - window.createMockContext = createMockContext; - - // some style initialization to limit differences between browsers across different platforms. - utils.injectCSS( - '.chartjs-wrapper, .chartjs-wrapper canvas {' + - 'border: 0;' + - 'margin: 0;' + - 'padding: 0;' + - '}' + - '.chartjs-wrapper {' + - 'position: absolute' + - '}'); - - jasmine.fixture = fixture; - jasmine.triggerMouseEvent = utils.triggerMouseEvent; - - beforeEach(function() { - jasmine.addMatchers(matchers); - }); - - afterEach(function() { - // Auto releasing acquired charts - Object.keys(charts).forEach(function(id) { - var chart = charts[id]; - if (!(chart.$test || {}).persistent) { - releaseChart(chart); - } - }); - }); -}()); +afterEach(function() { + releaseCharts(); +}); diff --git a/test/matchers.js b/test/matchers.js deleted file mode 100644 index f7d41da2b..000000000 --- a/test/matchers.js +++ /dev/null @@ -1,220 +0,0 @@ -'use strict'; - -import pixelmatch from 'pixelmatch'; -import utils from './utils'; - -function toPercent(value) { - return Math.round(value * 10000) / 100; -} - -function createImageData(w, h) { - var canvas = utils.createCanvas(w, h); - var context = canvas.getContext('2d'); - return context.getImageData(0, 0, w, h); -} - -function canvasFromImageData(data) { - var canvas = utils.createCanvas(data.width, data.height); - var context = canvas.getContext('2d'); - context.putImageData(data, 0, 0); - return canvas; -} - -function buildPixelMatchPreview(actual, expected, diff, threshold, tolerance, count, description) { - var ratio = count / (actual.width * actual.height); - var wrapper = document.createElement('div'); - wrapper.appendChild(document.createTextNode(description)); - - wrapper.style.cssText = 'display: flex; overflow-y: auto'; - - [ - {data: actual, label: 'Actual'}, - {data: expected, label: 'Expected'}, - {data: diff, label: - 'diff: ' + count + 'px ' + - '(' + toPercent(ratio) + '%)
' + - 'thr: ' + toPercent(threshold) + '%, ' + - 'tol: ' + toPercent(tolerance) + '%' - } - ].forEach(function(values) { - var item = document.createElement('div'); - item.style.cssText = 'text-align: center; font: 12px monospace; line-height: 1.4; margin: 8px'; - item.innerHTML = '
' + values.label + '
'; - item.appendChild(canvasFromImageData(values.data)); - wrapper.appendChild(item); - }); - - // WORKAROUND: https://github.com/karma-runner/karma-jasmine/issues/139 - wrapper.indexOf = function() { - return -1; - }; - - return wrapper; -} - -function toBeCloseToPixel() { - return { - compare: function(actual, expected) { - var result = false; - - if (!isNaN(actual) && !isNaN(expected)) { - var diff = Math.abs(actual - expected); - var A = Math.abs(actual); - var B = Math.abs(expected); - var percentDiff = 0.005; // 0.5% diff - result = (diff <= (A > B ? A : B) * percentDiff) || diff < 2; // 2 pixels is fine - } - - return {pass: result}; - } - }; -} - -function toBeCloseToPoint() { - function rnd(v) { - return Math.round(v * 100) / 100; - } - return { - compare: function(actual, expected) { - return { - pass: rnd(actual.x) === rnd(expected.x) && rnd(actual.y) === rnd(expected.y) - }; - } - }; -} - -function toEqualOneOf() { - return { - compare: function(actual, expecteds) { - var result = false; - for (var i = 0, l = expecteds.length; i < l; i++) { - if (actual === expecteds[i]) { - result = true; - break; - } - } - return { - pass: result - }; - } - }; -} - -function toBeValidChart() { - return { - compare: function(actual) { - var message = null; - - if (!(actual instanceof Chart)) { - message = 'Expected ' + actual + ' to be an instance of Chart'; - } else if (Object.prototype.toString.call(actual.canvas) !== '[object HTMLCanvasElement]') { - message = 'Expected canvas to be an instance of HTMLCanvasElement'; - } else if (Object.prototype.toString.call(actual.ctx) !== '[object CanvasRenderingContext2D]') { - message = 'Expected context to be an instance of CanvasRenderingContext2D'; - } else if (typeof actual.height !== 'number' || !isFinite(actual.height)) { - message = 'Expected height to be a strict finite number'; - } else if (typeof actual.width !== 'number' || !isFinite(actual.width)) { - message = 'Expected width to be a strict finite number'; - } - - return { - message: message ? message : 'Expected ' + actual + ' to be valid chart', - pass: !message - }; - } - }; -} - -function toBeChartOfSize() { - return { - compare: function(actual, expected) { - var res = toBeValidChart().compare(actual); - if (!res.pass) { - return res; - } - - var message = null; - var canvas = actual.ctx.canvas; - var style = getComputedStyle(canvas); - var pixelRatio = actual.options.devicePixelRatio || window.devicePixelRatio; - var dh = parseInt(style.height, 10) || 0; - var dw = parseInt(style.width, 10) || 0; - var rh = canvas.height; - var rw = canvas.width; - var orh = rh / pixelRatio; - var orw = rw / pixelRatio; - - // sanity checks - if (actual.height !== orh) { - message = 'Expected chart height ' + actual.height + ' to be equal to original render height ' + orh; - } else if (actual.width !== orw) { - message = 'Expected chart width ' + actual.width + ' to be equal to original render width ' + orw; - } - - // validity checks - if (dh !== expected.dh) { - message = 'Expected display height ' + dh + ' to be equal to ' + expected.dh; - } else if (dw !== expected.dw) { - message = 'Expected display width ' + dw + ' to be equal to ' + expected.dw; - } else if (rh !== expected.rh) { - message = 'Expected render height ' + rh + ' to be equal to ' + expected.rh; - } else if (rw !== expected.rw) { - message = 'Expected render width ' + rw + ' to be equal to ' + expected.rw; - } - - return { - message: message ? message : 'Expected ' + actual + ' to be a chart of size ' + expected, - pass: !message - }; - } - }; -} - -function toEqualImageData() { - return { - compare: function(actual, expected, opts) { - var message = null; - var debug = opts.debug || false; - var tolerance = opts.tolerance === undefined ? 0.001 : opts.tolerance; - var threshold = opts.threshold === undefined ? 0.1 : opts.threshold; - var ctx, idata, ddata, w, h, count, ratio; - - if (actual instanceof Chart) { - ctx = actual.ctx; - } else if (actual instanceof HTMLCanvasElement) { - ctx = actual.getContext('2d'); - } else if (actual instanceof CanvasRenderingContext2D) { - ctx = actual; - } - - if (ctx) { - h = expected.height; - w = expected.width; - idata = ctx.getImageData(0, 0, w, h); - ddata = createImageData(w, h); - count = pixelmatch(idata.data, expected.data, ddata.data, w, h, {threshold: threshold}); - ratio = count / (w * h); - - if ((ratio > tolerance) || debug) { - message = buildPixelMatchPreview(idata, expected, ddata, threshold, tolerance, count, opts.description); - } - } else { - message = 'Input value is not a valid image source.'; - } - - return { - message: message, - pass: !message - }; - } - }; -} - -export default { - toBeCloseToPixel, - toBeCloseToPoint, - toEqualOneOf, - toBeValidChart, - toBeChartOfSize, - toEqualImageData -}; diff --git a/test/spriting.js b/test/spriting.js deleted file mode 100644 index 9de02dd62..000000000 --- a/test/spriting.js +++ /dev/null @@ -1,101 +0,0 @@ -const characters = new Image(); -// data url for image containing all the characters -characters.src = ''; - -// the data url image size -const imgWidth = 256; -const imgHeight = 256; -// individual characted bounding box -const cellWidth = 17; -const cellHeight = 17; -// char code for [0, 0] -const startIndex = 32; -// number of columns in image -const columns = Math.floor(imgWidth / cellWidth); -// number of rows in image -const rows = Math.floor(imgHeight / cellHeight); -// font height (in pixels) -const fontHeight = 16; -// the font widths by char code, starting at startIndex -const fontWidth = [ - 4, 4, 6, 7, 7, 10, 9, 3, 4, 4, 5, 8, 4, 4, 4, 4, - 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 4, 4, 8, 8, 8, 8, - 13, 9, 9, 9, 9, 8, 8, 10, 9, 4, 7, 9, 8, 11, 9, 10, - 9, 10, 9, 9, 8, 9, 9, 13, 9, 8, 7, 4, 4, 4, 8, 7, - 4, 8, 8, 7, 8, 8, 4, 8, 8, 4, 4, 7, 4, 12, 8, 8, - 8, 8, 5, 6, 4, 8, 7, 11, 8, 7, 7, 5, 3, 5, 8, 10, - 7, 10, 4, 7, 7, 13, 7, 7, 4, 12, 9, 4, 14, 10, 7, 10, - 10, 4, 4, 7, 7, 5, 7, 13, 4, 13, 6, 4, 12, 10, 7, 8, - 4, 4, 7, 7, 7, 7, 3, 7, 4, 10, 5, 7, 8, 4, 10, 7, - 5, 7, 4, 4, 4, 7, 7, 4, 4, 4, 5, 7, 11, 11, 11, 8, - 9, 9, 9, 9, 9, 9, 13, 9, 8, 8, 8, 8, 4, 4, 4, 4, - 9, 9, 10, 10, 10, 10, 10, 8, 10, 9, 9, 9, 9, 8, 9, 8, - 8, 8, 8, 8, 8, 8, 12, 7, 8, 8, 8, 8, 4, 4, 4, 4, - 8, 8, 8, 8, 8, 8, 8, 7, 8, 8, 8, 8, 8, 7, 8, 7 -]; - -// get coordinates and size for one character -function getChar(asciiCode) { - const index = asciiCode - startIndex; - const x = Math.min(index % columns, columns - 1); - const y = Math.min(Math.floor(index / columns), rows - 1); - return {sx: x * cellWidth, sy: y * cellHeight, w: fontWidth[index], h: fontHeight}; -} - -function measureText(text) { - let width = 0; - if (text && text.charCodeAt) { - for (let i = 0; i < text.length; ++i) { - width += fontWidth[Math.min(223, text.charCodeAt(i) - startIndex)]; - } - } - return {width}; -} - -function spriteWrite(text, x, y) { - if (text && text.charCodeAt) { - const align = this.textAlign; - if (align === 'center' || align === 'right') { - const w = measureText(text).width; - x -= align === 'center' ? w / 2 : w; - } - const base = this.textBaseline; - switch (base) { - case 'top': - case 'hanging': - y -= fontHeight; - break; - case 'middle': - case 'alphabetic': - case 'ideaographic': - y -= fontHeight / 2; - break; - default: - break; - } - for (let i = 0; i < text.length; ++i) { - const {sx, sy, w, h} = getChar(text.charCodeAt(i)); - this.drawImage(characters, sx, sy, w, h, x, y, w, h); - x += w; - } - } -} - -export function spritingOn(ctx) { - if (ctx && !ctx._fillText) { - ctx._fillText = ctx.fillText; - ctx._measureText = ctx.measureText; - - ctx.fillText = spriteWrite; - ctx.measureText = measureText; - } -} - -export function spritingOff(ctx) { - if (ctx && ctx._fillText) { - ctx.fillText = ctx._fillText; - ctx.measureText = ctx._measureText; - delete ctx._fillText; - delete ctx._measureText; - } -} diff --git a/test/utils.js b/test/utils.js deleted file mode 100644 index ee443880e..000000000 --- a/test/utils.js +++ /dev/null @@ -1,187 +0,0 @@ -import {spritingOn, spritingOff} from './spriting'; - -function createCanvas(w, h) { - var canvas = document.createElement('canvas'); - canvas.width = w; - canvas.height = h; - return canvas; -} - -function readImageData(url, callback) { - var image = new Image(); - - image.onload = function() { - var h = image.height; - var w = image.width; - var canvas = createCanvas(w, h); - var ctx = canvas.getContext('2d'); - ctx.drawImage(image, 0, 0, w, h); - callback(ctx.getImageData(0, 0, w, h)); - }; - - image.src = url; -} - -/** - * Injects a new canvas (and div wrapper) and creates the associated Chart instance - * using the given config. Additional options allow tweaking elements generation. - * @param {object} config - Chart config. - * @param {object} options - Chart acquisition options. - * @param {object} options.canvas - Canvas attributes. - * @param {object} options.wrapper - Canvas wrapper attributes. - * @param {boolean} options.useOffscreenCanvas - use an OffscreenCanvas instead of the normal HTMLCanvasElement. - * @param {boolean} options.useShadowDOM - use shadowDom - * @param {boolean} options.persistent - If true, the chart will not be released after the spec. - */ -function acquireChart(config, options) { - var wrapper = document.createElement('div'); - var canvas = document.createElement('canvas'); - var chart, key; - - config = config || {}; - options = options || {}; - options.canvas = options.canvas || {height: 512, width: 512}; - options.wrapper = options.wrapper || {class: 'chartjs-wrapper'}; - - for (key in options.canvas) { - if (Object.prototype.hasOwnProperty.call(options.canvas, key)) { - canvas.setAttribute(key, options.canvas[key]); - } - } - - for (key in options.wrapper) { - if (Object.prototype.hasOwnProperty.call(options.wrapper, key)) { - wrapper.setAttribute(key, options.wrapper[key]); - } - } - - // by default, remove chart animation and auto resize - config.options = config.options || {}; - config.options.animation = config.options.animation === undefined ? false : config.options.animation; - config.options.responsive = config.options.responsive === undefined ? false : config.options.responsive; - config.options.locale = config.options.locale || 'en-US'; - - if (options.useShadowDOM) { - if (!wrapper.attachShadow) { - // If shadowDOM is not supported by the browsers, mark test as 'pending'. - return pending(); - } - wrapper.attachShadow({mode: 'open'}).appendChild(canvas); - } else { - wrapper.appendChild(canvas); - } - window.document.body.appendChild(wrapper); - - try { - var ctx; - if (options.useOffscreenCanvas) { - if (!canvas.transferControlToOffscreen) { - // If this browser does not support offscreen canvas, mark the test as 'pending', which will skip the - // test. - // TODO: switch to skip() once it's implemented (https://github.com/jasmine/jasmine/issues/1709), or - // remove if all browsers implement `transferControlToOffscreen` - return pending(); - } - var offscreenCanvas = canvas.transferControlToOffscreen(); - ctx = offscreenCanvas.getContext('2d'); - } else { - ctx = canvas.getContext('2d'); - } - if (options.spriteText) { - spritingOn(ctx); - } - chart = new Chart(ctx, config); - } catch (e) { - window.document.body.removeChild(wrapper); - throw e; - } - - chart.$test = { - persistent: options.persistent, - wrapper: wrapper - }; - - return chart; -} - -function releaseChart(chart) { - spritingOff(chart.ctx); - chart.destroy(); - - var wrapper = (chart.$test || {}).wrapper; - if (wrapper && wrapper.parentNode) { - wrapper.parentNode.removeChild(wrapper); - } -} - -function injectCSS(css) { - // https://stackoverflow.com/q/3922139 - var head = document.getElementsByTagName('head')[0]; - var style = document.createElement('style'); - style.setAttribute('type', 'text/css'); - if (style.styleSheet) { // IE - style.styleSheet.cssText = css; - } else { - style.appendChild(document.createTextNode(css)); - } - head.appendChild(style); -} - -function waitForResize(chart, callback) { - var override = chart.resize; - chart.resize = function() { - chart.resize = override; - override.apply(this, arguments); - callback(); - }; -} - -function afterEvent(chart, type, callback) { - var override = chart._eventHandler; - chart._eventHandler = function(event) { - override.call(this, event); - if (event.type === type) { - chart._eventHandler = override; - // eslint-disable-next-line callback-return - callback(); - } - }; -} - -function _resolveElementPoint(el) { - var point = {x: 0, y: 0}; - if (el) { - if (typeof el.getCenterPoint === 'function') { - point = el.getCenterPoint(); - } else if (el.x !== undefined && el.y !== undefined) { - point = el; - } - } - return point; -} - -function triggerMouseEvent(chart, type, el) { - var node = chart.canvas; - var rect = node.getBoundingClientRect(); - var point = _resolveElementPoint(el); - var event = new MouseEvent(type, { - clientX: rect.left + point.x, - clientY: rect.top + point.y, - cancelable: true, - bubbles: true, - view: window - }); - - node.dispatchEvent(event); -} - -export default { - injectCSS: injectCSS, - createCanvas: createCanvas, - acquireChart: acquireChart, - releaseChart: releaseChart, - readImageData: readImageData, - triggerMouseEvent: triggerMouseEvent, - waitForResize: waitForResize, - afterEvent: afterEvent -};