Files
enviPy-bayer/static/js/ketcher2/node_modules/svg2ttf/index.js
2025-06-23 20:13:54 +02:00

187 lines
5.8 KiB
JavaScript

/*
* Copyright: Vitaly Puzrin
* Author: Sergey Batishchev <snb2003@rambler.ru>
*
* Written for fontello.com project.
*/
'use strict';
var _ = require('lodash');
var SvgPath = require('svgpath');
var ucs2 = require('./lib/ucs2');
var svg = require('./lib/svg');
var sfnt = require('./lib/sfnt');
var VERSION_RE = /^(Version )?(\d+[.]\d+)$/i;
function svg2ttf(svgString, options) {
var font = new sfnt.Font();
var svgFont = svg.load(svgString);
options = options || {};
font.id = options.id || svgFont.id;
font.familyName = options.familyname || svgFont.familyName || svgFont.id;
font.copyright = options.copyright || svgFont.metadata;
font.sfntNames.push({ id: 2, value: options.subfamilyname || svgFont.subfamilyName || 'Regular' }); // subfamily name
font.sfntNames.push({ id: 4, value: options.fullname || svgFont.id }); // full name
var versionString = options.version || 'Version 1.0';
if (typeof versionString !== 'string') {
throw new Error('svg2ttf: version option should be a string');
}
if (!VERSION_RE.test(versionString)) {
throw new Error('svg2ttf: invalid option, version - "' + options.version + '"');
}
versionString = 'Version ' + versionString.match(VERSION_RE)[2];
font.sfntNames.push({ id: 5, value: versionString }); // version ID for TTF name table
font.sfntNames.push({ id: 6, value: (options.fullname || svgFont.id).replace(/[\s\(\)\[\]<>%\/]/g, '').substr(0, 62) }); // Postscript name for the font, required for OSX Font Book
if (typeof options.ts !== 'undefined') {
font.createdDate = font.modifiedDate = new Date(parseInt(options.ts, 10) * 1000);
}
// Try to fill font metrics or guess defaults
//
font.unitsPerEm = svgFont.unitsPerEm || 1000;
font.horizOriginX = svgFont.horizOriginX || 0;
font.horizOriginY = svgFont.horizOriginY || 0;
font.vertOriginX = svgFont.vertOriginX || 0;
font.vertOriginY = svgFont.vertOriginY || 0;
// need to correctly convert text values, use default (400) until compleete
//font.weightClass = svgFont.weightClass;
font.width = svgFont.width || svgFont.unitsPerEm;
font.height = svgFont.height || svgFont.unitsPerEm;
font.descent = !isNaN(svgFont.descent) ? svgFont.descent : -font.vertOriginY;
font.ascent = svgFont.ascent || (font.unitsPerEm - font.vertOriginY);
var glyphs = font.glyphs;
var codePoints = font.codePoints;
var ligatures = font.ligatures;
function addCodePoint(codePoint, glyph) {
if (codePoints[codePoint]) {
// Ignore code points already defined
return false;
}
codePoints[codePoint] = glyph;
return true;
}
// add SVG glyphs to SFNT font
_.forEach(svgFont.glyphs, function (svgGlyph) {
var glyph = new sfnt.Glyph();
glyph.name = svgGlyph.name;
glyph.d = svgGlyph.d;
glyph.height = !isNaN(svgGlyph.height) ? svgGlyph.height : font.height;
glyph.width = !isNaN(svgGlyph.width) ? svgGlyph.width : font.width;
glyphs.push(glyph);
svgGlyph.sfntGlyph = glyph;
_.forEach(svgGlyph.unicode, function (codePoint) {
addCodePoint(codePoint, glyph);
});
});
var missingGlyph;
// add missing glyph to SFNT font
// also, check missing glyph existance and single instance
if (svgFont.missingGlyph) {
missingGlyph = new sfnt.Glyph();
missingGlyph.d = svgFont.missingGlyph.d;
missingGlyph.height = !isNaN(svgFont.missingGlyph.height) ? svgFont.missingGlyph.height : font.height;
missingGlyph.width = !isNaN(svgFont.missingGlyph.width) ? svgFont.missingGlyph.width : font.width;
} else {
missingGlyph = _.find(glyphs, function (glyph) {
return glyph.name === '.notdef';
});
}
if (!missingGlyph) { // no missing glyph and .notdef glyph, we need to create missing glyph
missingGlyph = new sfnt.Glyph();
}
// Create glyphs for all characters used in ligatures
_.forEach(svgFont.ligatures, function (svgLigature) {
var ligature = {
ligature: svgLigature.ligature,
unicode: svgLigature.unicode,
glyph: svgLigature.glyph.sfntGlyph
};
_.forEach(ligature.unicode, function (charPoint) {
// We need to have a distinct glyph for each code point so we can reference it in GSUB
var glyph = new sfnt.Glyph();
var added = addCodePoint(charPoint, glyph);
if (added) {
glyph.name = ucs2.encode([ charPoint ]);
glyphs.push(glyph);
}
});
ligatures.push(ligature);
});
// Missing Glyph needs to have index 0
if (glyphs.indexOf(missingGlyph) !== -1) {
glyphs.splice(glyphs.indexOf(missingGlyph), 1);
}
glyphs.unshift(missingGlyph);
var nextID = 0;
//add IDs
_.forEach(glyphs, function (glyph) {
glyph.id = nextID;
nextID++;
});
_.forEach(glyphs, function (glyph) {
// Calculate accuracy for cubicToQuad transformation
// For glyphs with height and width smaller than 500 use relative 0.06% accuracy,
// for larger glyphs use fixed accuracy 0.3.
var glyphSize = Math.max(glyph.width, glyph.height);
var accuracy = (glyphSize > 500) ? 0.3 : glyphSize * 0.0006;
//SVG transformations
var svgPath = new SvgPath(glyph.d)
.abs()
.unshort()
.unarc()
.iterate(function (segment, index, x, y) {
return svg.cubicToQuad(segment, index, x, y, accuracy);
});
var sfntContours = svg.toSfntCoutours(svgPath);
// Add contours to SFNT font
glyph.contours = _.map(sfntContours, function (sfntContour) {
var contour = new sfnt.Contour();
contour.points = _.map(sfntContour, function (sfntPoint) {
var point = new sfnt.Point();
point.x = sfntPoint.x;
point.y = sfntPoint.y;
point.onCurve = sfntPoint.onCurve;
return point;
});
return contour;
});
});
var ttf = sfnt.toTTF(font);
return ttf;
}
module.exports = svg2ttf;