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