]> git.ipfire.org Git - thirdparty/Chart.js.git/commitdiff
Migrate to chartjs-test-utils (#8237)
authorJukka Kurkela <jukka.kurkela@gmail.com>
Sun, 27 Dec 2020 19:29:42 +0000 (21:29 +0200)
committerGitHub <noreply@github.com>
Sun, 27 Dec 2020 19:29:42 +0000 (14:29 -0500)
package-lock.json
package.json
test/context.js [deleted file]
test/fixture.js [deleted file]
test/index.js
test/matchers.js [deleted file]
test/spriting.js [deleted file]
test/utils.js [deleted file]

index 1fc98110446767757a1fc4fcfdf64738ddbc6ce3..8e5cbe590e7b3d9feb6119ed350d9d9c50b67e9d 100644 (file)
       "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",
index 3873cf84b9bba243f52279976338c28a74dce508..4e386c1c7a6ce572b68c315dcb46b8ef40b0564f 100644 (file)
@@ -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 (file)
index 63c872c..0000000
+++ /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 (file)
index 334b682..0000000
+++ /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
-};
index 4ebaae9b0e13a9962d95c365356ed6d337a9e040..0a764c3db42763db2331869167094d132a3eadc9 100644 (file)
@@ -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 (file)
index f7d41da..0000000
+++ /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) + '%)<br/>' +
-                       '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 = '<div style="margin: 8px; height: 32px">' + values.label + '</div>';
-               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 (file)
index 9de02dd..0000000
+++ /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 (file)
index ee44388..0000000
+++ /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
-};