Merge pull request #5163 from yurydelendik/fontrefact
Refactoring of OpenType/TrueType font construction
This commit is contained in:
commit
29d116f769
@ -15,7 +15,7 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
/* globals FONT_IDENTITY_MATRIX, FontType, warn, GlyphsUnicode, error, string32,
|
/* globals FONT_IDENTITY_MATRIX, FontType, warn, GlyphsUnicode, error, string32,
|
||||||
readUint32, stringToArray, Stream, FontRendererFactory, shadow,
|
readUint32, Stream, FontRendererFactory, shadow, stringToBytes,
|
||||||
bytesToString, info, assert, IdentityCMap, Name, CMapFactory, PDFJS,
|
bytesToString, info, assert, IdentityCMap, Name, CMapFactory, PDFJS,
|
||||||
isNum, Lexer, isArray, ISOAdobeCharset, ExpertCharset,
|
isNum, Lexer, isArray, ISOAdobeCharset, ExpertCharset,
|
||||||
ExpertSubsetCharset, Util */
|
ExpertSubsetCharset, Util */
|
||||||
@ -2202,20 +2202,20 @@ var IdentityToUnicodeMap = (function IdentityToUnicodeMapClosure() {
|
|||||||
error('should not access .length');
|
error('should not access .length');
|
||||||
},
|
},
|
||||||
|
|
||||||
forEach: function(callback) {
|
forEach: function (callback) {
|
||||||
for (var i = this.firstChar, ii = this.lastChar; i <= ii; i++) {
|
for (var i = this.firstChar, ii = this.lastChar; i <= ii; i++) {
|
||||||
callback(i, i);
|
callback(i, i);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
get: function(i) {
|
get: function (i) {
|
||||||
if (this.firstChar <= i && i <= this.lastChar) {
|
if (this.firstChar <= i && i <= this.lastChar) {
|
||||||
return String.fromCharCode(i);
|
return String.fromCharCode(i);
|
||||||
}
|
}
|
||||||
return undefined;
|
return undefined;
|
||||||
},
|
},
|
||||||
|
|
||||||
charCodeOf: function(v) {
|
charCodeOf: function (v) {
|
||||||
error('should not call .charCodeOf');
|
error('should not call .charCodeOf');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -2223,6 +2223,147 @@ var IdentityToUnicodeMap = (function IdentityToUnicodeMapClosure() {
|
|||||||
return IdentityToUnicodeMap;
|
return IdentityToUnicodeMap;
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
var OpenTypeFileBuilder = (function OpenTypeFileBuilderClosure() {
|
||||||
|
function writeInt16(dest, offset, num) {
|
||||||
|
dest[offset] = (num >> 8) & 0xFF;
|
||||||
|
dest[offset + 1] = num & 0xFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
function writeInt32(dest, offset, num) {
|
||||||
|
dest[offset] = (num >> 24) & 0xFF;
|
||||||
|
dest[offset + 1] = (num >> 16) & 0xFF;
|
||||||
|
dest[offset + 2] = (num >> 8) & 0xFF;
|
||||||
|
dest[offset + 3] = num & 0xFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
function writeData(dest, offset, data) {
|
||||||
|
var i, ii;
|
||||||
|
if (data instanceof Uint8Array) {
|
||||||
|
dest.set(data, offset);
|
||||||
|
} else if (typeof data === 'string') {
|
||||||
|
for (i = 0, ii = data.length; i < ii; i++) {
|
||||||
|
dest[offset++] = data.charCodeAt(i) & 0xFF;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// treating everything else as array
|
||||||
|
for (i = 0, ii = data.length; i < ii; i++) {
|
||||||
|
dest[offset++] = data[i] & 0xFF;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function OpenTypeFileBuilder(sfnt) {
|
||||||
|
this.sfnt = sfnt;
|
||||||
|
this.tables = Object.create(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
OpenTypeFileBuilder.getSearchParams =
|
||||||
|
function OpenTypeFileBuilder_getSearchParams(entriesCount, entrySize) {
|
||||||
|
var maxPower2 = 1, log2 = 0;
|
||||||
|
while ((maxPower2 ^ entriesCount) > maxPower2) {
|
||||||
|
maxPower2 <<= 1;
|
||||||
|
log2++;
|
||||||
|
}
|
||||||
|
var searchRange = maxPower2 * entrySize;
|
||||||
|
return {
|
||||||
|
range: searchRange,
|
||||||
|
entry: log2,
|
||||||
|
rangeShift: entrySize * entriesCount - searchRange
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
var OTF_HEADER_SIZE = 12;
|
||||||
|
var OTF_TABLE_ENTRY_SIZE = 16;
|
||||||
|
|
||||||
|
OpenTypeFileBuilder.prototype = {
|
||||||
|
toArray: function OpenTypeFileBuilder_toArray() {
|
||||||
|
var sfnt = this.sfnt;
|
||||||
|
|
||||||
|
// Tables needs to be written by ascendant alphabetic order
|
||||||
|
var tables = this.tables;
|
||||||
|
var tablesNames = Object.keys(tables);
|
||||||
|
tablesNames.sort();
|
||||||
|
var numTables = tablesNames.length;
|
||||||
|
|
||||||
|
var i, j, jj, table, tableName;
|
||||||
|
// layout the tables data
|
||||||
|
var offset = OTF_HEADER_SIZE + numTables * OTF_TABLE_ENTRY_SIZE;
|
||||||
|
var tableOffsets = [offset];
|
||||||
|
for (i = 0; i < numTables; i++) {
|
||||||
|
table = tables[tablesNames[i]];
|
||||||
|
var paddedLength = ((table.length + 3) & ~3) >>> 0;
|
||||||
|
offset += paddedLength;
|
||||||
|
tableOffsets.push(offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
var file = new Uint8Array(offset);
|
||||||
|
// write the table data first (mostly for checksum)
|
||||||
|
for (i = 0; i < numTables; i++) {
|
||||||
|
table = tables[tablesNames[i]];
|
||||||
|
writeData(file, tableOffsets[i], table);
|
||||||
|
}
|
||||||
|
|
||||||
|
// sfnt version (4 bytes)
|
||||||
|
if (sfnt === 'true') {
|
||||||
|
// Windows hates the Mac TrueType sfnt version number
|
||||||
|
sfnt = string32(0x00010000);
|
||||||
|
}
|
||||||
|
file[0] = sfnt.charCodeAt(0) & 0xFF;
|
||||||
|
file[1] = sfnt.charCodeAt(1) & 0xFF;
|
||||||
|
file[2] = sfnt.charCodeAt(2) & 0xFF;
|
||||||
|
file[3] = sfnt.charCodeAt(3) & 0xFF;
|
||||||
|
|
||||||
|
// numTables (2 bytes)
|
||||||
|
writeInt16(file, 4, numTables);
|
||||||
|
|
||||||
|
var searchParams = OpenTypeFileBuilder.getSearchParams(numTables, 16);
|
||||||
|
|
||||||
|
// searchRange (2 bytes)
|
||||||
|
writeInt16(file, 6, searchParams.range);
|
||||||
|
// entrySelector (2 bytes)
|
||||||
|
writeInt16(file, 8, searchParams.entry);
|
||||||
|
// rangeShift (2 bytes)
|
||||||
|
writeInt16(file, 10, searchParams.rangeShift);
|
||||||
|
|
||||||
|
offset = OTF_HEADER_SIZE;
|
||||||
|
// writing table entries
|
||||||
|
for (i = 0; i < numTables; i++) {
|
||||||
|
tableName = tablesNames[i];
|
||||||
|
file[offset] = tableName.charCodeAt(0) & 0xFF;
|
||||||
|
file[offset + 1] = tableName.charCodeAt(1) & 0xFF;
|
||||||
|
file[offset + 2] = tableName.charCodeAt(2) & 0xFF;
|
||||||
|
file[offset + 3] = tableName.charCodeAt(3) & 0xFF;
|
||||||
|
|
||||||
|
// checksum
|
||||||
|
var checksum = 0;
|
||||||
|
for (j = tableOffsets[i], jj = tableOffsets[i + 1]; j < jj; j += 4) {
|
||||||
|
var quad = (file[j] << 24) + (file[j + 1] << 16) +
|
||||||
|
(file[j + 2] << 8) + file[j + 3];
|
||||||
|
checksum = (checksum + quad) | 0;
|
||||||
|
}
|
||||||
|
writeInt32(file, offset + 4, checksum);
|
||||||
|
|
||||||
|
// offset
|
||||||
|
writeInt32(file, offset + 8, tableOffsets[i]);
|
||||||
|
// length
|
||||||
|
writeInt32(file, offset + 12, tables[tableName].length);
|
||||||
|
|
||||||
|
offset += OTF_TABLE_ENTRY_SIZE;
|
||||||
|
}
|
||||||
|
return file;
|
||||||
|
},
|
||||||
|
|
||||||
|
addTable: function OpenTypeFileBuilder_addTable(tag, data) {
|
||||||
|
if (tag in this.tables) {
|
||||||
|
throw new Error('Table ' + tag + ' already exists');
|
||||||
|
}
|
||||||
|
this.tables[tag] = data;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return OpenTypeFileBuilder;
|
||||||
|
})();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 'Font' is the class the outside world should use, it encapsulate all the font
|
* 'Font' is the class the outside world should use, it encapsulate all the font
|
||||||
* decoding logics whatever type it is (assuming the font type is supported).
|
* decoding logics whatever type it is (assuming the font type is supported).
|
||||||
@ -2432,21 +2573,6 @@ var Font = (function FontClosure() {
|
|||||||
return (b0 << 24) + (b1 << 16) + (b2 << 8) + b3;
|
return (b0 << 24) + (b1 << 16) + (b2 << 8) + b3;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getMaxPower2(number) {
|
|
||||||
var maxPower = 0;
|
|
||||||
var value = number;
|
|
||||||
while (value >= 2) {
|
|
||||||
value /= 2;
|
|
||||||
maxPower++;
|
|
||||||
}
|
|
||||||
|
|
||||||
value = 2;
|
|
||||||
for (var i = 1; i < maxPower; i++) {
|
|
||||||
value *= 2;
|
|
||||||
}
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
function string16(value) {
|
function string16(value) {
|
||||||
return String.fromCharCode((value >> 8) & 0xff, value & 0xff);
|
return String.fromCharCode((value >> 8) & 0xff, value & 0xff);
|
||||||
}
|
}
|
||||||
@ -2457,89 +2583,6 @@ var Font = (function FontClosure() {
|
|||||||
return String.fromCharCode((value >> 8) & 0xff, value & 0xff);
|
return String.fromCharCode((value >> 8) & 0xff, value & 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
function createOpenTypeHeader(sfnt, file, numTables) {
|
|
||||||
// Windows hates the Mac TrueType sfnt version number
|
|
||||||
if (sfnt === 'true') {
|
|
||||||
sfnt = string32(0x00010000);
|
|
||||||
}
|
|
||||||
|
|
||||||
// sfnt version (4 bytes)
|
|
||||||
var header = sfnt;
|
|
||||||
|
|
||||||
// numTables (2 bytes)
|
|
||||||
header += string16(numTables);
|
|
||||||
|
|
||||||
// searchRange (2 bytes)
|
|
||||||
var tablesMaxPower2 = getMaxPower2(numTables);
|
|
||||||
var searchRange = tablesMaxPower2 * 16;
|
|
||||||
header += string16(searchRange);
|
|
||||||
|
|
||||||
// entrySelector (2 bytes)
|
|
||||||
header += string16(Math.log(tablesMaxPower2) / Math.log(2));
|
|
||||||
|
|
||||||
// rangeShift (2 bytes)
|
|
||||||
header += string16(numTables * 16 - searchRange);
|
|
||||||
|
|
||||||
file.file += header;
|
|
||||||
file.virtualOffset += header.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
function createTableEntry(file, tag, data) {
|
|
||||||
// offset
|
|
||||||
var offset = file.virtualOffset;
|
|
||||||
|
|
||||||
// length
|
|
||||||
var length = data.length;
|
|
||||||
|
|
||||||
// Per spec tables must be 4-bytes align so add padding as needed.
|
|
||||||
var paddedLength = length;
|
|
||||||
while (paddedLength & 3) {
|
|
||||||
paddedLength++;
|
|
||||||
}
|
|
||||||
var i;
|
|
||||||
var padding = paddedLength - length;
|
|
||||||
if (padding !== 0) {
|
|
||||||
// Padding is required. |data| can be an Array, Uint8Array, or
|
|
||||||
// Uint16Array. In the latter two cases we need to create slightly larger
|
|
||||||
// typed arrays and copy the old contents in. Fortunately that's not a
|
|
||||||
// common case.
|
|
||||||
var data2;
|
|
||||||
if (data instanceof Array) {
|
|
||||||
for (i = 0; i < padding; i++) {
|
|
||||||
data.push(0);
|
|
||||||
}
|
|
||||||
} else if (data instanceof Uint8Array) {
|
|
||||||
data2 = new Uint8Array(paddedLength);
|
|
||||||
data2.set(data);
|
|
||||||
data = data2;
|
|
||||||
} else if (data instanceof Uint16Array) {
|
|
||||||
data2 = new Uint16Array(paddedLength);
|
|
||||||
data2.set(data);
|
|
||||||
data = data2;
|
|
||||||
} else {
|
|
||||||
error('bad array kind in createTableEntry');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (file.virtualOffset & 3) {
|
|
||||||
file.virtualOffset++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// checksum
|
|
||||||
var checksum = 0, n = data.length;
|
|
||||||
for (i = 0; i < n; i += 4) {
|
|
||||||
checksum = (checksum + int32(data[i], data[i + 1], data[i + 2],
|
|
||||||
data[i + 3])) | 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
var tableEntry = (tag + string32(checksum) +
|
|
||||||
string32(offset) + string32(length));
|
|
||||||
file.file += tableEntry;
|
|
||||||
file.virtualOffset += data.length;
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
function isTrueTypeFile(file) {
|
function isTrueTypeFile(file) {
|
||||||
var header = file.peekBytes(4);
|
var header = file.peekBytes(4);
|
||||||
return readUint32(header, 0) === 0x00010000;
|
return readUint32(header, 0) === 0x00010000;
|
||||||
@ -2676,10 +2719,7 @@ var Font = (function FontClosure() {
|
|||||||
}
|
}
|
||||||
var trailingRangesCount = ranges[i][1] < 0xFFFF ? 1 : 0;
|
var trailingRangesCount = ranges[i][1] < 0xFFFF ? 1 : 0;
|
||||||
var segCount = bmpLength + trailingRangesCount;
|
var segCount = bmpLength + trailingRangesCount;
|
||||||
var segCount2 = segCount * 2;
|
var searchParams = OpenTypeFileBuilder.getSearchParams(segCount, 2);
|
||||||
var searchRange = getMaxPower2(segCount) * 2;
|
|
||||||
var searchEntry = Math.log(segCount) / Math.log(2);
|
|
||||||
var rangeShift = 2 * segCount - searchRange;
|
|
||||||
|
|
||||||
// Fill up the 4 parallel arrays describing the segments.
|
// Fill up the 4 parallel arrays describing the segments.
|
||||||
var startCount = '';
|
var startCount = '';
|
||||||
@ -2730,10 +2770,10 @@ var Font = (function FontClosure() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var format314 = '\x00\x00' + // language
|
var format314 = '\x00\x00' + // language
|
||||||
string16(segCount2) +
|
string16(2 * segCount) +
|
||||||
string16(searchRange) +
|
string16(searchParams.range) +
|
||||||
string16(searchEntry) +
|
string16(searchParams.entry) +
|
||||||
string16(rangeShift) +
|
string16(searchParams.rangeShift) +
|
||||||
endCount + '\x00\x00' + startCount +
|
endCount + '\x00\x00' + startCount +
|
||||||
idDeltas + idRangeOffsets + glyphsIds;
|
idDeltas + idRangeOffsets + glyphsIds;
|
||||||
|
|
||||||
@ -2771,10 +2811,9 @@ var Font = (function FontClosure() {
|
|||||||
string32(format31012.length / 12); // nGroups
|
string32(format31012.length / 12); // nGroups
|
||||||
}
|
}
|
||||||
|
|
||||||
return stringToArray(cmap +
|
return cmap + '\x00\x04' + // format
|
||||||
'\x00\x04' + // format
|
string16(format314.length + 4) + // length
|
||||||
string16(format314.length + 4) + // length
|
format314 + header31012 + format31012;
|
||||||
format314 + header31012 + format31012);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function validateOS2Table(os2) {
|
function validateOS2Table(os2) {
|
||||||
@ -3268,7 +3307,7 @@ var Font = (function FontClosure() {
|
|||||||
for (i = 0; i < numMissing; i++) {
|
for (i = 0; i < numMissing; i++) {
|
||||||
entries += '\x00\x00';
|
entries += '\x00\x00';
|
||||||
}
|
}
|
||||||
metrics.data = stringToArray(entries);
|
metrics.data = entries;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3854,7 +3893,7 @@ var Font = (function FontClosure() {
|
|||||||
|
|
||||||
var tables = { 'OS/2': null, cmap: null, head: null, hhea: null,
|
var tables = { 'OS/2': null, cmap: null, head: null, hhea: null,
|
||||||
hmtx: null, maxp: null, name: null, post: null };
|
hmtx: null, maxp: null, name: null, post: null };
|
||||||
var table, tableData;
|
var table;
|
||||||
for (var i = 0; i < numTables; i++) {
|
for (var i = 0; i < numTables; i++) {
|
||||||
table = readTableEntry(font);
|
table = readTableEntry(font);
|
||||||
if (VALID_TABLES.indexOf(table.tag) < 0) {
|
if (VALID_TABLES.indexOf(table.tag) < 0) {
|
||||||
@ -3928,24 +3967,6 @@ var Font = (function FontClosure() {
|
|||||||
delete tables['cvt '];
|
delete tables['cvt '];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tables needs to be written by ascendant alphabetic order
|
|
||||||
var tablesNames = Object.keys(tables);
|
|
||||||
tablesNames.sort();
|
|
||||||
|
|
||||||
numTables = tablesNames.length;
|
|
||||||
|
|
||||||
// header and new offsets. Table entry information is appended to the
|
|
||||||
// end of file. The virtualOffset represents where to put the actual
|
|
||||||
// data of a particular table;
|
|
||||||
var ttf = {
|
|
||||||
file: '',
|
|
||||||
virtualOffset: numTables * (4 * 4)
|
|
||||||
};
|
|
||||||
|
|
||||||
// The new numbers of tables will be the last one plus the num
|
|
||||||
// of missing tables
|
|
||||||
createOpenTypeHeader(header.version, ttf, numTables);
|
|
||||||
|
|
||||||
// Ensure the hmtx table contains the advance width and
|
// Ensure the hmtx table contains the advance width and
|
||||||
// sidebearings information for numGlyphs in the maxp table
|
// sidebearings information for numGlyphs in the maxp table
|
||||||
sanitizeMetrics(font, tables.hhea, tables.hmtx, numGlyphs);
|
sanitizeMetrics(font, tables.hhea, tables.hmtx, numGlyphs);
|
||||||
@ -4108,9 +4129,8 @@ var Font = (function FontClosure() {
|
|||||||
|
|
||||||
tables['OS/2'] = {
|
tables['OS/2'] = {
|
||||||
tag: 'OS/2',
|
tag: 'OS/2',
|
||||||
data: stringToArray(createOS2Table(properties,
|
data: createOS2Table(properties, newMapping.charCodeToGlyphId,
|
||||||
newMapping.charCodeToGlyphId,
|
override)
|
||||||
override))
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4118,7 +4138,7 @@ var Font = (function FontClosure() {
|
|||||||
if (!tables.post) {
|
if (!tables.post) {
|
||||||
tables.post = {
|
tables.post = {
|
||||||
tag: 'post',
|
tag: 'post',
|
||||||
data: stringToArray(createPostTable(properties))
|
data: createPostTable(properties)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4139,47 +4159,22 @@ var Font = (function FontClosure() {
|
|||||||
if (!tables.name) {
|
if (!tables.name) {
|
||||||
tables.name = {
|
tables.name = {
|
||||||
tag: 'name',
|
tag: 'name',
|
||||||
data: stringToArray(createNameTable(this.name))
|
data: createNameTable(this.name)
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
// ... using existing 'name' table as prototype
|
// ... using existing 'name' table as prototype
|
||||||
var namePrototype = readNameTable(tables.name);
|
var namePrototype = readNameTable(tables.name);
|
||||||
tables.name.data = stringToArray(createNameTable(name, namePrototype));
|
tables.name.data = createNameTable(name, namePrototype);
|
||||||
}
|
}
|
||||||
|
|
||||||
// rewrite the tables but tweak offsets
|
var builder = new OpenTypeFileBuilder(header.version);
|
||||||
for (i = 0; i < numTables; i++) {
|
for (var tableTag in tables) {
|
||||||
table = tables[tablesNames[i]];
|
builder.addTable(tableTag, tables[tableTag].data);
|
||||||
table.data = createTableEntry(ttf, table.tag, table.data);
|
|
||||||
}
|
}
|
||||||
|
return builder.toArray();
|
||||||
// Add the table datas
|
|
||||||
for (i = 0; i < numTables; i++) {
|
|
||||||
table = tables[tablesNames[i]];
|
|
||||||
tableData = table.data;
|
|
||||||
ttf.file += bytesToString(new Uint8Array(tableData));
|
|
||||||
|
|
||||||
// 4-byte aligned data
|
|
||||||
while (ttf.file.length & 3) {
|
|
||||||
ttf.file += String.fromCharCode(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return stringToArray(ttf.file);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
convert: function Font_convert(fontName, font, properties) {
|
convert: function Font_convert(fontName, font, properties) {
|
||||||
// The offsets object holds at the same time a representation of where
|
|
||||||
// to write the table entry information about a table and another offset
|
|
||||||
// representing the offset where to draw the actual data of a particular
|
|
||||||
// table
|
|
||||||
var otf = {
|
|
||||||
file: '',
|
|
||||||
virtualOffset: 9 * (4 * 4)
|
|
||||||
};
|
|
||||||
|
|
||||||
createOpenTypeHeader('\x4F\x54\x54\x4F', otf, 9);
|
|
||||||
|
|
||||||
// TODO: Check the charstring widths to determine this.
|
// TODO: Check the charstring widths to determine this.
|
||||||
properties.fixedPitch = false;
|
properties.fixedPitch = false;
|
||||||
|
|
||||||
@ -4259,20 +4254,16 @@ var Font = (function FontClosure() {
|
|||||||
|
|
||||||
var unitsPerEm = 1 / (properties.fontMatrix || FONT_IDENTITY_MATRIX)[0];
|
var unitsPerEm = 1 / (properties.fontMatrix || FONT_IDENTITY_MATRIX)[0];
|
||||||
|
|
||||||
var fields = {
|
var builder = new OpenTypeFileBuilder('\x4F\x54\x54\x4F');
|
||||||
// PostScript Font Program
|
// PostScript Font Program
|
||||||
'CFF ': font.data,
|
builder.addTable('CFF ', font.data);
|
||||||
|
// OS/2 and Windows Specific metrics
|
||||||
// OS/2 and Windows Specific metrics
|
builder.addTable('OS/2', createOS2Table(properties,
|
||||||
'OS/2': stringToArray(createOS2Table(properties,
|
newMapping.charCodeToGlyphId));
|
||||||
newMapping.charCodeToGlyphId)),
|
// Character to glyphs mapping
|
||||||
|
builder.addTable('cmap', createCmapTable(newMapping.charCodeToGlyphId));
|
||||||
// Character to glyphs mapping
|
// Font header
|
||||||
'cmap': createCmapTable(newMapping.charCodeToGlyphId),
|
builder.addTable('head',
|
||||||
|
|
||||||
// Font header
|
|
||||||
'head': (function fontFieldsHead() {
|
|
||||||
return stringToArray(
|
|
||||||
'\x00\x01\x00\x00' + // Version number
|
'\x00\x01\x00\x00' + // Version number
|
||||||
'\x00\x00\x10\x00' + // fontRevision
|
'\x00\x00\x10\x00' + // fontRevision
|
||||||
'\x00\x00\x00\x00' + // checksumAdjustement
|
'\x00\x00\x00\x00' + // checksumAdjustement
|
||||||
@ -4290,11 +4281,9 @@ var Font = (function FontClosure() {
|
|||||||
'\x00\x00' + // fontDirectionHint
|
'\x00\x00' + // fontDirectionHint
|
||||||
'\x00\x00' + // indexToLocFormat
|
'\x00\x00' + // indexToLocFormat
|
||||||
'\x00\x00'); // glyphDataFormat
|
'\x00\x00'); // glyphDataFormat
|
||||||
})(),
|
|
||||||
|
|
||||||
// Horizontal header
|
// Horizontal header
|
||||||
'hhea': (function fontFieldsHhea() {
|
builder.addTable('hhea',
|
||||||
return stringToArray(
|
|
||||||
'\x00\x01\x00\x00' + // Version number
|
'\x00\x01\x00\x00' + // Version number
|
||||||
safeString16(properties.ascent) + // Typographic Ascent
|
safeString16(properties.ascent) + // Typographic Ascent
|
||||||
safeString16(properties.descent) + // Typographic Descent
|
safeString16(properties.descent) + // Typographic Descent
|
||||||
@ -4313,10 +4302,9 @@ var Font = (function FontClosure() {
|
|||||||
'\x00\x00' + // -reserved-
|
'\x00\x00' + // -reserved-
|
||||||
'\x00\x00' + // metricDataFormat
|
'\x00\x00' + // metricDataFormat
|
||||||
string16(numGlyphs)); // Number of HMetrics
|
string16(numGlyphs)); // Number of HMetrics
|
||||||
})(),
|
|
||||||
|
|
||||||
// Horizontal metrics
|
// Horizontal metrics
|
||||||
'hmtx': (function fontFieldsHmtx() {
|
builder.addTable('hmtx', (function fontFieldsHmtx() {
|
||||||
var charstrings = font.charstrings;
|
var charstrings = font.charstrings;
|
||||||
var hmtx = '\x00\x00\x00\x00'; // Fake .notdef
|
var hmtx = '\x00\x00\x00\x00'; // Fake .notdef
|
||||||
for (var i = 1, ii = numGlyphs; i < ii; i++) {
|
for (var i = 1, ii = numGlyphs; i < ii; i++) {
|
||||||
@ -4326,33 +4314,21 @@ var Font = (function FontClosure() {
|
|||||||
var width = 'width' in charstring ? charstring.width : 0;
|
var width = 'width' in charstring ? charstring.width : 0;
|
||||||
hmtx += string16(width) + string16(0);
|
hmtx += string16(width) + string16(0);
|
||||||
}
|
}
|
||||||
return stringToArray(hmtx);
|
return hmtx;
|
||||||
})(),
|
})());
|
||||||
|
|
||||||
// Maximum profile
|
// Maximum profile
|
||||||
'maxp': (function fontFieldsMaxp() {
|
builder.addTable('maxp',
|
||||||
return stringToArray(
|
|
||||||
'\x00\x00\x50\x00' + // Version number
|
'\x00\x00\x50\x00' + // Version number
|
||||||
string16(numGlyphs)); // Num of glyphs
|
string16(numGlyphs)); // Num of glyphs
|
||||||
})(),
|
|
||||||
|
|
||||||
// Naming tables
|
// Naming tables
|
||||||
'name': stringToArray(createNameTable(fontName)),
|
builder.addTable('name', createNameTable(fontName));
|
||||||
|
|
||||||
// PostScript informations
|
// PostScript informations
|
||||||
'post': stringToArray(createPostTable(properties))
|
builder.addTable('post', createPostTable(properties));
|
||||||
};
|
|
||||||
|
|
||||||
var field;
|
return builder.toArray();
|
||||||
for (field in fields) {
|
|
||||||
fields[field] = createTableEntry(otf, field, fields[field]);
|
|
||||||
}
|
|
||||||
for (field in fields) {
|
|
||||||
var table = fields[field];
|
|
||||||
otf.file += bytesToString(new Uint8Array(table));
|
|
||||||
}
|
|
||||||
|
|
||||||
return stringToArray(otf.file);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -6902,7 +6878,7 @@ var CFFCompiler = (function CFFCompilerClosure() {
|
|||||||
compileNameIndex: function CFFCompiler_compileNameIndex(names) {
|
compileNameIndex: function CFFCompiler_compileNameIndex(names) {
|
||||||
var nameIndex = new CFFIndex();
|
var nameIndex = new CFFIndex();
|
||||||
for (var i = 0, ii = names.length; i < ii; ++i) {
|
for (var i = 0, ii = names.length; i < ii; ++i) {
|
||||||
nameIndex.add(stringToArray(names[i]));
|
nameIndex.add(stringToBytes(names[i]));
|
||||||
}
|
}
|
||||||
return this.compileIndex(nameIndex);
|
return this.compileIndex(nameIndex);
|
||||||
},
|
},
|
||||||
@ -7027,7 +7003,7 @@ var CFFCompiler = (function CFFCompilerClosure() {
|
|||||||
compileStringIndex: function CFFCompiler_compileStringIndex(strings) {
|
compileStringIndex: function CFFCompiler_compileStringIndex(strings) {
|
||||||
var stringIndex = new CFFIndex();
|
var stringIndex = new CFFIndex();
|
||||||
for (var i = 0, ii = strings.length; i < ii; ++i) {
|
for (var i = 0, ii = strings.length; i < ii; ++i) {
|
||||||
stringIndex.add(stringToArray(strings[i]));
|
stringIndex.add(stringToBytes(strings[i]));
|
||||||
}
|
}
|
||||||
return this.compileIndex(stringIndex);
|
return this.compileIndex(stringIndex);
|
||||||
},
|
},
|
||||||
|
@ -434,15 +434,6 @@ function bytesToString(bytes) {
|
|||||||
return strBuf.join('');
|
return strBuf.join('');
|
||||||
}
|
}
|
||||||
|
|
||||||
function stringToArray(str) {
|
|
||||||
var length = str.length;
|
|
||||||
var array = new Uint16Array(length);
|
|
||||||
for (var i = 0; i < length; ++i) {
|
|
||||||
array[i] = str.charCodeAt(i);
|
|
||||||
}
|
|
||||||
return array;
|
|
||||||
}
|
|
||||||
|
|
||||||
function stringToBytes(str) {
|
function stringToBytes(str) {
|
||||||
var length = str.length;
|
var length = str.length;
|
||||||
var bytes = new Uint8Array(length);
|
var bytes = new Uint8Array(length);
|
||||||
|
Loading…
Reference in New Issue
Block a user