From f0fd28af66c15a74cc3b8c7a38fc68a20346adb8 Mon Sep 17 00:00:00 2001 From: Jukka Kurkela Date: Fri, 14 Feb 2020 19:33:08 +0200 Subject: [PATCH] Allow "spriting" text in fixtures (#7073) Allow "blit" text drawing in fixtures --- .../scale.category/ticks-from-data.js | 30 ++++++ .../scale.category/ticks-from-data.png | Bin 0 -> 11594 bytes test/specs/scale.category.tests.js | 29 +---- test/spriting.js | 102 ++++++++++++++++++ test/utils.js | 11 +- 5 files changed, 143 insertions(+), 29 deletions(-) create mode 100644 test/fixtures/scale.category/ticks-from-data.js create mode 100644 test/fixtures/scale.category/ticks-from-data.png create mode 100644 test/spriting.js diff --git a/test/fixtures/scale.category/ticks-from-data.js b/test/fixtures/scale.category/ticks-from-data.js new file mode 100644 index 000000000..a7f067962 --- /dev/null +++ b/test/fixtures/scale.category/ticks-from-data.js @@ -0,0 +1,30 @@ +module.exports = { + threshold: 0.01, + config: { + type: 'horizontalBar', + data: { + datasets: [{ + data: [10, 5, 0, 25, 78] + }], + labels: ['tick1', 'tick2', 'tick3', 'tick4', 'tick5'] + }, + options: { + legend: false, + title: false, + elements: { + rectangle: { + backgroundColor: '#AAAAAA80', + borderColor: '#80808080', + borderWidth: {bottom: 6, left: 15, top: 6, right: 15} + } + }, + scales: { + x: {display: false}, + y: {display: true} + } + } + }, + options: { + spriteText: true + } +}; diff --git a/test/fixtures/scale.category/ticks-from-data.png b/test/fixtures/scale.category/ticks-from-data.png new file mode 100644 index 0000000000000000000000000000000000000000..cab649d6b1de0eb34f494e344c4dfe855910b3e6 GIT binary patch literal 11594 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 literal 0 HcmV?d00001 diff --git a/test/specs/scale.category.tests.js b/test/specs/scale.category.tests.js index 4a0ff03a0..00b7cfd9a 100644 --- a/test/specs/scale.category.tests.js +++ b/test/specs/scale.category.tests.js @@ -9,6 +9,8 @@ function getValues(scale) { } describe('Category scale tests', function() { + describe('auto', jasmine.fixture.specs('scale.category')); + it('Should register the constructor with the scale service', function() { var Constructor = Chart.scaleService.getScaleConstructor('category'); expect(Constructor).not.toBe(undefined); @@ -57,33 +59,6 @@ describe('Category scale tests', function() { expect(defaultConfig.ticks.callback).toEqual(jasmine.any(Function)); }); - it('Should generate ticks from the data labels', function() { - var scaleID = 'myScale'; - - var mockData = { - datasets: [{ - yAxisID: scaleID, - data: [10, 5, 0, 25, 78] - }], - labels: ['tick1', 'tick2', 'tick3', 'tick4', 'tick5'] - }; - - var config = Chart.helpers.clone(Chart.scaleService.getScaleDefaults('category')); - config.position = 'bottom'; - var Constructor = Chart.scaleService.getScaleConstructor('category'); - var scale = new Constructor({ - ctx: {}, - options: config, - chart: { - data: mockData - }, - id: scaleID - }); - - scale.determineDataLimits(); - scale.ticks = scale.buildTicks(); - expect(getValues(scale)).toEqual(mockData.labels); - }); it('Should generate ticks from the data xLabels', function() { var scaleID = 'myScale'; diff --git a/test/spriting.js b/test/spriting.js new file mode 100644 index 000000000..981f0dd3e --- /dev/null +++ b/test/spriting.js @@ -0,0 +1,102 @@ +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 (index = character code) +function getChar(index) { + const i = index - startIndex; + const size = 17; + const x = Math.min(i % columns, columns - 1); + const y = Math.min(Math.floor(i / columns), rows - 1); + return {sx: x * size, sy: y * size, w: fontWidth[i], 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))]; + } + } + 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 index db0515769..2145cbe0d 100644 --- a/test/utils.js +++ b/test/utils.js @@ -1,3 +1,5 @@ +import {spritingOn, spritingOff} from './spriting'; + function createCanvas(w, h) { var canvas = document.createElement('canvas'); canvas.width = w; @@ -62,7 +64,11 @@ function acquireChart(config, options) { window.document.body.appendChild(wrapper); try { - chart = new Chart(canvas.getContext('2d'), config); + var ctx = canvas.getContext('2d'); + if (options.spriteText) { + spritingOn(ctx); + } + chart = new Chart(ctx, config); } catch (e) { window.document.body.removeChild(wrapper); throw e; @@ -77,6 +83,7 @@ function acquireChart(config, options) { } function releaseChart(chart) { + spritingOff(chart.ctx); chart.destroy(); var wrapper = (chart.$test || {}).wrapper; @@ -146,7 +153,7 @@ function triggerMouseEvent(chart, type, el) { node.dispatchEvent(event); } -module.exports = { +export default { injectCSS: injectCSS, createCanvas: createCanvas, acquireChart: acquireChart,