Chart.js/test/spriting.js

102 lines
18 KiB
JavaScript
Raw Normal View History

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;
}
}