1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2024-06-01 05:41:31 +00:00
8bitworkshop/tools/fontgen/uifont.js
2018-08-31 12:17:33 -04:00

333 lines
11 KiB
JavaScript

var font;
var params = {
bpp:1,
wbytes:1,
width:8,
height:8,
lochar:32,
hichar:95,
rotate:false,
xflip:false,
yflip:false,
msbfirst:false,
};
var errors;
var previewCanvas = $("#previewCanvas");
var codeTextarea = $("#codeTextarea");
var paramsForm = $("#paramsForm");
function refreshPreset(item) {
console.log(item);
w2ui.toolbar.set('bpp', {value:'1'});
w2ui.toolbar.set('width', {value:'8'});
w2ui.toolbar.set('height', {value:'8'});
w2ui.toolbar.set('lochar', {value:'32'});
w2ui.toolbar.set('hichar', {value:'95'});
}
function parseBDF(text) {
var chars = {};
var chord;
var ch;
var lines = text.split(/\r?\n/);
var bounds = [0,0,0,0];
for (var i=0; i<lines.length; i++) {
var l = lines[i];
var toks = l.split(/\s+/);
if (toks.length == 0) continue;
switch (toks[0]) {
case 'ENCODING':
chord = parseInt(toks[1]);
ch = {ord:chord};
break;
case 'BITMAP':
ch.bytes = [];
break;
case 'BBX':
if (ch) {
ch.bbx = toks.slice(1).map(function(s) { return parseInt(s); });
bounds[0] = Math.min(bounds[0], ch.bbx[2]);
bounds[1] = Math.min(bounds[1], ch.bbx[3]);
bounds[2] = Math.max(bounds[2], ch.bbx[2]+ch.bbx[0]);
bounds[3] = Math.max(bounds[3], ch.bbx[3]+ch.bbx[1]);
}
break;
case 'ENDCHAR':
chars[chord] = ch;
ch = null;
break;
default:
if (ch && ch.bytes && toks.length == 1) {
var n = parseInt(toks[0], 16);
n <<= (6-toks[0].length) * 4;
ch.bytes.push(n);
}
break;
}
}
return {chars:chars,bounds:bounds,pixbounds:[0,0,0,0]};
}
function loadFont(rec) {
font = {};
var path = 'bitmap-fonts/bitmap/' + rec.path;
$.get(path, function(text) {
font = parseBDF(text);
font.rec = rec;
font.path = path;
console.log(font);
previewFont();
codeTextarea.text(encodeFont());
});
}
function renderGlyph(glyph, putPixel) {
var w = glyph.bbx[0];
var h = glyph.bbx[1];
var dx = glyph.bbx[2];
var dy = glyph.bbx[3];
for (var y=0; y<glyph.bytes.length; y++) {
for (var x=0; x<w; x++) {
if (glyph.bytes[glyph.bytes.length-y-1] & (0x800000 >> x)) {
var xx = x+dx;
var yy = y+dy;
font.pixbounds[0] = Math.min(font.pixbounds[0], xx);
font.pixbounds[1] = Math.min(font.pixbounds[1], yy);
font.pixbounds[2] = Math.max(font.pixbounds[2], xx);
font.pixbounds[3] = Math.max(font.pixbounds[3], yy);
putPixel(xx, yy);
}
}
}
return {w:w+dx,h:h+dy};
}
function drawChar(x0, y0, chord) {
var ctx = previewCanvas[0].getContext('2d');
ctx.fillStyle = "black";
var glyph = font.chars[chord];
if (glyph) {
return renderGlyph(glyph, function(x,y) {
ctx.fillRect( x0+x, y0-y, 1, 1 );
});
} else {
return {w:8,h:8}; // TODO: avgwidth, height
}
}
function drawString(x, y, str) {
for (var i=0; i<str.length; i++) {
x += drawChar(x, y, str.charCodeAt(i)).w+1;
}
}
var TEST_SENTENCES = [
"0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ",
"!@#$%^&*()+ -- == [] {} <>?,./;':\"",
"THE QUICK BROWN FOX JUMPS OVER THE LAZY DOG'S BOWL",
"the quick brown fox jumps over the lazy dog's bowl",
//"Pack my box with five dozen liquor jugs.",
];
function previewFont() {
var ctx = previewCanvas[0].getContext('2d');
ctx.fillStyle = "white";
ctx.fillRect( 0, 0, 1024, 1024 );
var x = 8;
var y = 8;
//console.log(font);
TEST_SENTENCES.forEach(function(s) {
y += font.rec.pixelsize + 1;
drawString(x, y, s);
});
}
function encodeGlyph(glyph, bytes) {
var abort = false;
renderGlyph(glyph, function(x,y) {
//x -= font.pixbounds[0];
//y -= font.pixbounds[1];
if (params.rotate) {
var y2 = x; var x2 = y; x = x2; y = y2;
}
if (params.yflip) { y = params.height-1-y; }
if (params.xflip) { x = params.width-1-x; }
var xoutrange = (x < 0 || x >= params.width);
var youtrange = (y < 0 || y >= params.height);
if (xoutrange || youtrange) {
if (!abort) {
errors.push((xoutrange?"X":"Y") + " out of range on character " + String.fromCharCode(glyph.ord) + " " + x + "," + y);
abort = true;
}
}
var bpb = 8 / (params.bpp||1);
var ofs = Math.floor(x/bpb) + (params.height-1-y)*params.wbytes;
var bit = x % bpb;
if (params.msbfirst) { bit = 7-bit; }
bytes[ofs] |= 1<<bit;
});
}
function encodeFont() {
var s = '/* ' + JSON.stringify(params) + JSON.stringify(font.bounds) + JSON.stringify(font.pixbounds) + ' */\n';
s += "#define LOCHAR " + params.lochar + "\n";
s += "#define HICHAR " + params.hichar + "\n";
s += "#define FONT_HEIGHT " + params.height + "\n";
s += "const char FONT[HICHAR-LOCHAR+1][FONT_HEIGHT] = {\n";
errors = [];
for (var chord=params.lochar; chord<=params.hichar; chord++) {
var glyph = font.chars[chord];
var bytes = new Uint8Array(params.wbytes*params.height);
if (glyph) {
encodeGlyph(glyph, bytes);
s += "{ ";
for (var i=0; i<bytes.length; i++) {
s += "0x" + bytes[i].toString(16) + ",";
}
s += " },";
s += "\n";
}
}
s += "};\n";
while (errors.length) {
s = "/* " + errors.pop() + " */\n" + s;
}
return s;
}
/////
var FONTRECS = [];
var li = 0;
for (var line of FONTLIST.split("\n")) {
var ltoks = line.split("|");
var ftoks = ltoks[1].split("-");
var rec = {
recid: ++li,
path: ltoks[0],
foundry: ftoks[1],
family: ftoks[2],
weight: ftoks[3].toLowerCase(),
slant: ftoks[4].toUpperCase(),
setwidth: ftoks[5],
addstyle: ftoks[6],
pixelsize: parseInt(ftoks[7]),
pointsize: parseInt(ftoks[8]),
resx: parseInt(ftoks[9]),
resy: parseInt(ftoks[10]),
spacing: ftoks[11].toUpperCase(),
avgwidth: ftoks[12]/10,
registry: ftoks[13],
encoding: ftoks[14],
};
FONTRECS.push(rec);
}
function toolbarHTMLItem(id, title, maxchars) {
return function(item) {
var html = '<div style="padding: 3px 5px;">' + title + '&nbsp;' +
'<input size=' + maxchars + ' maxlength=' + maxchars +
' onchange="var el = w2ui.toolbar.set(\'' + id + '\', { value: this.value });" '+
' value="'+ (item.value || '') + '"/></div>';
return html;
};
}
$().w2toolbar({
name: 'toolbar',
items: [
{ type: 'menu-radio', id: 'preset', caption: 'Presets', img: 'icon-folder',
text: function(item) { refreshPreset(item); return item.caption; },
items: [
{ text: 'Generic 8x8', id: '8x8' },
{ text: 'Atari 2600', id: 'vcs' },
{ text: 'Midway 8080', id: 'mw8080bw' },
{ text: 'NES', id: 'nes' }
]},
{ type: 'check', id: 'rotate', caption: 'Rotate' },
{ type: 'check', id: 'yflip', caption: 'Flip' },
{ type: 'menu-radio', id: 'bpp', caption: 'BPP', img: 'fas fa-star', items: [
{ text: '1' },
{ text: '2' },
{ text: '4' },
{ text: '8' }
]},
{ type: 'html', id: 'width', html: toolbarHTMLItem('width','Width:',2) },
{ type: 'html', id: 'height', html: toolbarHTMLItem('height','Height:',2) },
{ type: 'break', id: 'break1' },
{ type: 'menu-radio', id: 'charsel', caption: 'Characters', img: 'icon-folder', items: [
//{ text: 'ISO (256 chars)', value:'0-255' },
{ text: 'ASCII (upper+lower)', id:'32-95' },
{ text: 'ASCII (upper only)', id:'32-127' }
]},
{ type: 'html', id: 'lochar', html: toolbarHTMLItem('lochar','First:',3) },
{ type: 'html', id: 'hichar', html: toolbarHTMLItem('hichar','Last:',3) },
/*
{ type: 'spacer' },
{ type: 'check', id: 'item1', caption: 'Check', img: 'icon-page', checked: true },
{ type: 'break', id: 'break0' },
{ type: 'radio', id: 'item3', group: '1', caption: 'Radio 1', icon: 'fa-star', checked: true },
{ type: 'radio', id: 'item4', group: '1', caption: 'Radio 2', icon: 'fa-star-empty' },
{ type: 'button', id: 'item5', caption: 'Item 5', icon: 'fa-home' }
*/
]
});
$().w2grid({
name : 'fontGrid',
show : {
toolbar : true
},
multiSearch: true,
searches: [
{ field: 'weight', caption: 'Weight', type: 'list', options: { items: ['medium','bold','narrow']} },
{ field: 'slant', caption: 'Slant', type: 'list', options: { items: ['R','O']} },
{ field: 'spacing', caption: 'Spacing', type: 'list', options: { items: ['C','M','P']} },
{ field: 'pixelsize', caption: 'Pixel Size', type: 'int' },
{ field: 'avgwidth', caption: 'Avg. Width', type: 'int' },
],
columns: [
{ field: 'foundry', caption: 'Foundry', size: '20%', sortable:true },
{ field: 'family', caption: 'Family', size: '20%', sortable:true },
{ field: 'weight', caption: 'Weight', size: '10%', sortable:true },
{ field: 'slant', caption: 'Slant', size: '10%', sortable:true },
{ field: 'pixelsize', caption: 'Pixel Size', size: '10%', sortable:true },
{ field: 'avgwidth', caption: 'Avg. Width', size: '10%', sortable:true },
{ field: 'spacing', caption: 'Spacing', size: '10%', sortable:true },
],
records: FONTRECS,
onClick: function (event) {
var record = this.get(event.recid);
loadFont(record);
},
});
var pstyle = 'background-color: #F5F6F7; border: 1px solid #dfdfdf; padding: 5px;';
$('#layout').w2layout({
name: 'layout',
panels: [
{ type: 'top', size: 50, resizable: false, style: pstyle, content: 'top' },
{ type: 'left', size: 200, resizable: true, style: pstyle, content: paramsForm },
{ type: 'main', style: pstyle, content: codeTextarea },
{ type: 'preview', size: '50%', resizable: true, style: pstyle, content: previewCanvas },
{ type: 'right', size: 200, resizable: true, style: pstyle, content: 'right' },
{ type: 'bottom', size: 200, resizable: true, style: pstyle, content: 'bottom' }
]
});
w2ui['layout'].content('top', w2ui['toolbar']);
w2ui['layout'].content('bottom', w2ui['fontGrid']);
w2ui.toolbar.set('preset', {selected:'8x8'});
w2ui.toolbar.set('bpp', {value:'1'});
w2ui.toolbar.set('width', {value:'8'});
w2ui.toolbar.set('height', {value:'8'});
w2ui.toolbar.set('lochar', {value:'32'});
w2ui.toolbar.set('hichar', {value:'95'});