Iosevka/generator.js
2017-01-25 16:11:51 +08:00

136 lines
4.3 KiB
JavaScript

var fs = require("fs");
var path = require("path");
// var TTFWriter = require('node-sfnt').TTFWriter;
var argv = require("yargs").argv;
var buildGlyphs = require("./buildglyphs.js");
var parameters = require("./parameters");
var toml = require("toml");
var Glyph = require("./support/glyph");
var autoref = require("./support/autoref");
var caryllShapeOps = require("caryll-shapeops");
var c2q = require("otfcc-c2q");
function hasv(obj) {
if (!obj) return false;
for (var k in obj)
if (obj[k]) return true;
return false;
}
// Font building
var font = function () {
var parametersData = toml.parse(fs.readFileSync(path.join(path.dirname(require.main.filename), "parameters.toml"), "utf-8"));
var emptyFont = toml.parse(fs.readFileSync(path.join(path.dirname(require.main.filename), "emptyfont.toml"), "utf-8"));
var para = parameters.build(parametersData, argv._);
var fontUniqueName = para.family + " " + para.style + " " + para.version + " (" + para.codename + ")";
console.log(" Start building font " + fontUniqueName);
var font = buildGlyphs.build.call(emptyFont, para);
console.log(" " + fontUniqueName + " Successfully built.");
font.parameters = para;
font.glyf = font.glyf.sort(function (a, b) {
var pri1 = a.cmpPriority || 0;
var pri2 = b.cmpPriority || 0;
if (a.contours && b.contours && a.contours.length < b.contours.length) return 1;
if (a.contours && b.contours && a.contours.length > b.contours.length) return (-1);
if (a.unicode && a.unicode[0] && !b.unicode || !b.unicode[0]) return (-1);
if (b.unicode && b.unicode[0] && !a.unicode || !a.unicode[0]) return (+1);
if (a.unicode && a.unicode[0] && b.unicode && b.unicode[0] && a.unicode[0] < b.unicode[0]) return (-1);
if (a.unicode && a.unicode[0] && b.unicode && b.unicode[0] && a.unicode[0] > b.unicode[0]) return (+1);
return (a.name < b.name) ? (-1) : (a.name > b.name) ? 1 : 0;
});
return font;
}();
if (argv.charmap) {
(function () {
console.log(" Writing character map -> " + argv.charmap);
fs.writeFileSync(argv.charmap, JSON.stringify(font.glyf.map(function (glyph) {
return [
glyph.name,
glyph.unicode,
glyph.advanceWidth === 0 ? (hasv(glyph.anchors) ? 1 : (glyph.contours && glyph.contours.length) ? 2 : 0) : 0
];
})), "utf8");
})();
}
if (argv.o) {
console.log(" Writing output -> " + argv.o);
var o_glyf = {};
var cmap = {};
var skew = (argv.uprightify ? 1 : 0) * Math.tan((font.post.italicAngle || 0) / 180 * Math.PI);
// autoref
autoref(font.glyf);
// regulate
font.glyf.forEach((g) => {
if (g.contours) {
for (var k = 0; k < g.contours.length; k++) {
var contour = g.contours[k];
for (var p = 0; p < contour.length; p++) {
contour[p].x += contour[p].y * skew;
if (contour[p].on) {
contour[p].x = Math.round(contour[p].x);
}
}
var offJ = null, mx = null;
for (var p = 0; p < contour.length; p++) {
if (contour[p].on) {
if (offJ) {
var origx = contour[p].x;
var rx = Math.round(contour[p].x * 4) / 4;
var origx0 = mx;
var rx0 = contour[offJ - 1].x;
if (origx != origx0) {
for (var poff = offJ; poff < p; poff++) {
contour[poff].x = (contour[poff].x - origx0) / (origx - origx0) * (rx - rx0) + rx0;
}
}
}
mx = contour[p].x;
contour[p].x = Math.round(contour[p].x * 4) / 4;
offJ = p + 1;
}
}
}
var c1 = [];
for (var k = 0; k < g.contours.length; k++) {
c1.push(Glyph.contourToStandardCubic(g.contours[k]));
}
g.contours = c1;
}
});
// overlap removal
font.glyf.forEach((g) => {
if (g.contours) {
g.contours = caryllShapeOps.removeOverlap(g.contours, 1, 2048, true);
}
});
// finalize
font.glyf.forEach((g) => {
if (g.contours) {
Glyph.prototype.cleanup.call(g, 0.25);
g.contours = c2q.contours(g.contours);
for (var k = 0; k < g.contours.length; k++) {
var contour = g.contours[k];
for (var p = 0; p < contour.length; p++) {
contour[p].x -= contour[p].y * skew;
}
}
}
o_glyf[g.name] = g;
if (g.unicode && g.unicode.length) {
cmap[g.unicode[0]] = g.name;
}
});
font.glyf = o_glyf;
font.cmap = cmap;
font.glyfMap = null;
fs.writeFileSync(argv.o, JSON.stringify(font));
}