2013-05-16 05:57:27 +09:00
|
|
|
/* Copyright 2012 Mozilla Foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2017-12-13 22:51:45 +09:00
|
|
|
import {
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
bytesToString,
|
|
|
|
FONT_IDENTITY_MATRIX,
|
|
|
|
FormatError,
|
|
|
|
unreachable,
|
|
|
|
warn,
|
2020-01-02 20:00:16 +09:00
|
|
|
} from "../shared/util.js";
|
|
|
|
import { CFFParser } from "./cff_parser.js";
|
|
|
|
import { getGlyphsUnicode } from "./glyphlist.js";
|
|
|
|
import { StandardEncoding } from "./encodings.js";
|
|
|
|
import { Stream } from "./stream.js";
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2022-01-16 07:41:18 +09:00
|
|
|
// TODO: use DataView and its methods.
|
|
|
|
|
|
|
|
function getUint32(data, offset) {
|
2021-05-06 05:20:10 +09:00
|
|
|
return (
|
2022-01-16 07:41:18 +09:00
|
|
|
((data[offset] << 24) |
|
|
|
|
(data[offset + 1] << 16) |
|
|
|
|
(data[offset + 2] << 8) |
|
|
|
|
data[offset + 3]) >>>
|
|
|
|
0
|
2021-05-06 05:20:10 +09:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-16 07:41:18 +09:00
|
|
|
function getUint16(data, offset) {
|
2021-05-06 05:20:10 +09:00
|
|
|
return (data[offset] << 8) | data[offset + 1];
|
|
|
|
}
|
|
|
|
|
2022-01-16 07:41:18 +09:00
|
|
|
function getInt16(data, offset) {
|
|
|
|
return ((data[offset] << 24) | (data[offset + 1] << 16)) >> 16;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getInt8(data, offset) {
|
|
|
|
return (data[offset] << 24) >> 24;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getFloat214(data, offset) {
|
|
|
|
return getInt16(data, offset) / 16384;
|
|
|
|
}
|
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
function getSubroutineBias(subrs) {
|
|
|
|
const numSubrs = subrs.length;
|
|
|
|
let bias = 32768;
|
|
|
|
if (numSubrs < 1240) {
|
|
|
|
bias = 107;
|
|
|
|
} else if (numSubrs < 33900) {
|
|
|
|
bias = 1131;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
return bias;
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseCmap(data, start, end) {
|
|
|
|
const offset =
|
2022-01-16 07:41:18 +09:00
|
|
|
getUint16(data, start + 2) === 1
|
|
|
|
? getUint32(data, start + 8)
|
|
|
|
: getUint32(data, start + 16);
|
|
|
|
const format = getUint16(data, start + offset);
|
2021-05-06 05:20:10 +09:00
|
|
|
let ranges, p, i;
|
|
|
|
if (format === 4) {
|
2022-01-16 07:41:18 +09:00
|
|
|
getUint16(data, start + offset + 2); // length
|
|
|
|
const segCount = getUint16(data, start + offset + 6) >> 1;
|
2021-05-06 05:20:10 +09:00
|
|
|
p = start + offset + 14;
|
|
|
|
ranges = [];
|
|
|
|
for (i = 0; i < segCount; i++, p += 2) {
|
2022-01-16 07:41:18 +09:00
|
|
|
ranges[i] = { end: getUint16(data, p) };
|
2020-01-14 23:28:37 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
p += 2;
|
|
|
|
for (i = 0; i < segCount; i++, p += 2) {
|
2022-01-16 07:41:18 +09:00
|
|
|
ranges[i].start = getUint16(data, p);
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
for (i = 0; i < segCount; i++, p += 2) {
|
2022-01-16 07:41:18 +09:00
|
|
|
ranges[i].idDelta = getUint16(data, p);
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
for (i = 0; i < segCount; i++, p += 2) {
|
2022-01-16 07:41:18 +09:00
|
|
|
let idOffset = getUint16(data, p);
|
2021-05-06 05:20:10 +09:00
|
|
|
if (idOffset === 0) {
|
|
|
|
continue;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
ranges[i].ids = [];
|
|
|
|
for (let j = 0, jj = ranges[i].end - ranges[i].start + 1; j < jj; j++) {
|
2022-01-16 07:41:18 +09:00
|
|
|
ranges[i].ids[j] = getUint16(data, p + idOffset);
|
2021-05-06 05:20:10 +09:00
|
|
|
idOffset += 2;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
return ranges;
|
|
|
|
} else if (format === 12) {
|
2022-01-16 07:41:18 +09:00
|
|
|
const groups = getUint32(data, start + offset + 12);
|
2021-05-06 05:20:10 +09:00
|
|
|
p = start + offset + 16;
|
|
|
|
ranges = [];
|
|
|
|
for (i = 0; i < groups; i++) {
|
2022-01-16 07:41:18 +09:00
|
|
|
start = getUint32(data, p);
|
2021-05-06 05:20:10 +09:00
|
|
|
ranges.push({
|
2022-01-16 07:41:18 +09:00
|
|
|
start,
|
|
|
|
end: getUint32(data, p + 4),
|
|
|
|
idDelta: getUint32(data, p + 8) - start,
|
2021-05-06 05:20:10 +09:00
|
|
|
});
|
|
|
|
p += 12;
|
|
|
|
}
|
|
|
|
return ranges;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
throw new FormatError(`unsupported cmap: ${format}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseCff(data, start, end, seacAnalysisEnabled) {
|
|
|
|
const properties = {};
|
|
|
|
const parser = new CFFParser(
|
|
|
|
new Stream(data, start, end - start),
|
|
|
|
properties,
|
|
|
|
seacAnalysisEnabled
|
|
|
|
);
|
|
|
|
const cff = parser.parse();
|
|
|
|
return {
|
|
|
|
glyphs: cff.charStrings.objects,
|
|
|
|
subrs:
|
|
|
|
cff.topDict.privateDict &&
|
|
|
|
cff.topDict.privateDict.subrsIndex &&
|
|
|
|
cff.topDict.privateDict.subrsIndex.objects,
|
|
|
|
gsubrs: cff.globalSubrIndex && cff.globalSubrIndex.objects,
|
|
|
|
isCFFCIDFont: cff.isCIDFont,
|
|
|
|
fdSelect: cff.fdSelect,
|
|
|
|
fdArray: cff.fdArray,
|
|
|
|
};
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
function parseGlyfTable(glyf, loca, isGlyphLocationsLong) {
|
|
|
|
let itemSize, itemDecode;
|
|
|
|
if (isGlyphLocationsLong) {
|
|
|
|
itemSize = 4;
|
2022-01-16 07:41:18 +09:00
|
|
|
itemDecode = getUint32;
|
2021-05-06 05:20:10 +09:00
|
|
|
} else {
|
|
|
|
itemSize = 2;
|
2022-01-16 07:41:18 +09:00
|
|
|
itemDecode = (data, offset) => 2 * getUint16(data, offset);
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
const glyphs = [];
|
|
|
|
let startOffset = itemDecode(loca, 0);
|
|
|
|
for (let j = itemSize; j < loca.length; j += itemSize) {
|
|
|
|
const endOffset = itemDecode(loca, j);
|
|
|
|
glyphs.push(glyf.subarray(startOffset, endOffset));
|
|
|
|
startOffset = endOffset;
|
|
|
|
}
|
|
|
|
return glyphs;
|
|
|
|
}
|
|
|
|
|
|
|
|
function lookupCmap(ranges, unicode) {
|
|
|
|
const code = unicode.codePointAt(0);
|
|
|
|
let gid = 0,
|
|
|
|
l = 0,
|
|
|
|
r = ranges.length - 1;
|
|
|
|
while (l < r) {
|
|
|
|
const c = (l + r + 1) >> 1;
|
|
|
|
if (code < ranges[c].start) {
|
|
|
|
r = c - 1;
|
2013-05-16 05:57:27 +09:00
|
|
|
} else {
|
2021-05-06 05:20:10 +09:00
|
|
|
l = c;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
if (ranges[l].start <= code && code <= ranges[l].end) {
|
|
|
|
gid =
|
|
|
|
(ranges[l].idDelta +
|
|
|
|
(ranges[l].ids ? ranges[l].ids[code - ranges[l].start] : code)) &
|
|
|
|
0xffff;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
return {
|
|
|
|
charCode: code,
|
|
|
|
glyphId: gid,
|
|
|
|
};
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
function compileGlyf(code, cmds, font) {
|
|
|
|
function moveTo(x, y) {
|
|
|
|
cmds.push({ cmd: "moveTo", args: [x, y] });
|
|
|
|
}
|
|
|
|
function lineTo(x, y) {
|
|
|
|
cmds.push({ cmd: "lineTo", args: [x, y] });
|
|
|
|
}
|
|
|
|
function quadraticCurveTo(xa, ya, x, y) {
|
|
|
|
cmds.push({ cmd: "quadraticCurveTo", args: [xa, ya, x, y] });
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
let i = 0;
|
2022-01-16 07:41:18 +09:00
|
|
|
const numberOfContours = getInt16(code, i);
|
2021-05-06 05:20:10 +09:00
|
|
|
let flags;
|
|
|
|
let x = 0,
|
|
|
|
y = 0;
|
|
|
|
i += 10;
|
|
|
|
if (numberOfContours < 0) {
|
|
|
|
// composite glyph
|
|
|
|
do {
|
2022-01-16 07:41:18 +09:00
|
|
|
flags = getUint16(code, i);
|
|
|
|
const glyphIndex = getUint16(code, i + 2);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 4;
|
|
|
|
let arg1, arg2;
|
|
|
|
if (flags & 0x01) {
|
2022-01-16 07:41:18 +09:00
|
|
|
if (flags & 0x02) {
|
|
|
|
arg1 = getInt16(code, i);
|
|
|
|
arg2 = getInt16(code, i + 2);
|
|
|
|
} else {
|
|
|
|
arg1 = getUint16(code, i);
|
|
|
|
arg2 = getUint16(code, i + 2);
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
i += 4;
|
2021-05-06 05:20:10 +09:00
|
|
|
} else {
|
2022-01-16 07:41:18 +09:00
|
|
|
if (flags & 0x02) {
|
|
|
|
arg1 = getInt8(code, i++);
|
|
|
|
arg2 = getInt8(code, i++);
|
|
|
|
} else {
|
|
|
|
arg1 = code[i++];
|
|
|
|
arg2 = code[i++];
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
if (flags & 0x02) {
|
|
|
|
x = arg1;
|
|
|
|
y = arg2;
|
|
|
|
} else {
|
|
|
|
x = 0;
|
2022-01-16 07:41:18 +09:00
|
|
|
y = 0;
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
let scaleX = 1,
|
|
|
|
scaleY = 1,
|
|
|
|
scale01 = 0,
|
|
|
|
scale10 = 0;
|
|
|
|
if (flags & 0x08) {
|
2022-01-16 07:41:18 +09:00
|
|
|
scaleX = scaleY = getFloat214(code, i);
|
2013-05-16 05:57:27 +09:00
|
|
|
i += 2;
|
2021-05-06 05:20:10 +09:00
|
|
|
} else if (flags & 0x40) {
|
2022-01-16 07:41:18 +09:00
|
|
|
scaleX = getFloat214(code, i);
|
|
|
|
scaleY = getFloat214(code, i + 2);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 4;
|
|
|
|
} else if (flags & 0x80) {
|
2022-01-16 07:41:18 +09:00
|
|
|
scaleX = getFloat214(code, i);
|
|
|
|
scale01 = getFloat214(code, i + 2);
|
|
|
|
scale10 = getFloat214(code, i + 4);
|
|
|
|
scaleY = getFloat214(code, i + 6);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 8;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
const subglyph = font.glyphs[glyphIndex];
|
|
|
|
if (subglyph) {
|
2022-01-16 07:41:18 +09:00
|
|
|
// TODO: the transform should be applied only if there is a scale:
|
|
|
|
// https://github.com/freetype/freetype/blob/edd4fedc5427cf1cf1f4b045e53ff91eb282e9d4/src/truetype/ttgload.c#L1205
|
2021-05-24 20:20:19 +09:00
|
|
|
cmds.push(
|
|
|
|
{ cmd: "save" },
|
|
|
|
{
|
|
|
|
cmd: "transform",
|
|
|
|
args: [scaleX, scale01, scale10, scaleY, x, y],
|
|
|
|
}
|
|
|
|
);
|
2022-01-16 07:41:18 +09:00
|
|
|
|
|
|
|
if (!(flags & 0x02)) {
|
|
|
|
// TODO: we must use arg1 and arg2 to make something similar to:
|
|
|
|
// https://github.com/freetype/freetype/blob/edd4fedc5427cf1cf1f4b045e53ff91eb282e9d4/src/truetype/ttgload.c#L1209
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
compileGlyf(subglyph, cmds, font);
|
|
|
|
cmds.push({ cmd: "restore" });
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
} while (flags & 0x20);
|
|
|
|
} else {
|
|
|
|
// simple glyph
|
|
|
|
const endPtsOfContours = [];
|
|
|
|
let j, jj;
|
|
|
|
for (j = 0; j < numberOfContours; j++) {
|
2022-01-16 07:41:18 +09:00
|
|
|
endPtsOfContours.push(getUint16(code, i));
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 2;
|
|
|
|
}
|
2022-01-16 07:41:18 +09:00
|
|
|
const instructionLength = getUint16(code, i);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 2 + instructionLength; // skipping the instructions
|
2022-06-09 19:53:39 +09:00
|
|
|
const numberOfPoints = endPtsOfContours.at(-1) + 1;
|
2021-05-06 05:20:10 +09:00
|
|
|
const points = [];
|
|
|
|
while (points.length < numberOfPoints) {
|
|
|
|
flags = code[i++];
|
|
|
|
let repeat = 1;
|
|
|
|
if (flags & 0x08) {
|
|
|
|
repeat += code[i++];
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
while (repeat-- > 0) {
|
|
|
|
points.push({ flags });
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
for (j = 0; j < numberOfPoints; j++) {
|
|
|
|
switch (points[j].flags & 0x12) {
|
|
|
|
case 0x00:
|
2022-01-16 07:41:18 +09:00
|
|
|
x += getInt16(code, i);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 2;
|
|
|
|
break;
|
|
|
|
case 0x02:
|
|
|
|
x -= code[i++];
|
|
|
|
break;
|
|
|
|
case 0x12:
|
|
|
|
x += code[i++];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
points[j].x = x;
|
|
|
|
}
|
|
|
|
for (j = 0; j < numberOfPoints; j++) {
|
|
|
|
switch (points[j].flags & 0x24) {
|
|
|
|
case 0x00:
|
2022-01-16 07:41:18 +09:00
|
|
|
y += getInt16(code, i);
|
2021-05-06 05:20:10 +09:00
|
|
|
i += 2;
|
|
|
|
break;
|
|
|
|
case 0x04:
|
|
|
|
y -= code[i++];
|
|
|
|
break;
|
|
|
|
case 0x24:
|
|
|
|
y += code[i++];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
points[j].y = y;
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
let startPoint = 0;
|
|
|
|
for (i = 0; i < numberOfContours; i++) {
|
|
|
|
const endPoint = endPtsOfContours[i];
|
|
|
|
// contours might have implicit points, which is located in the middle
|
|
|
|
// between two neighboring off-curve points
|
|
|
|
const contour = points.slice(startPoint, endPoint + 1);
|
|
|
|
if (contour[0].flags & 1) {
|
|
|
|
contour.push(contour[0]); // using start point at the contour end
|
2022-06-09 19:53:39 +09:00
|
|
|
} else if (contour.at(-1).flags & 1) {
|
2021-05-06 05:20:10 +09:00
|
|
|
// first is off-curve point, trying to use one from the end
|
2022-06-09 19:53:39 +09:00
|
|
|
contour.unshift(contour.at(-1));
|
2021-05-06 05:20:10 +09:00
|
|
|
} else {
|
|
|
|
// start and end are off-curve points, creating implicit one
|
|
|
|
const p = {
|
|
|
|
flags: 1,
|
2022-06-09 19:53:39 +09:00
|
|
|
x: (contour[0].x + contour.at(-1).x) / 2,
|
|
|
|
y: (contour[0].y + contour.at(-1).y) / 2,
|
2021-05-06 05:20:10 +09:00
|
|
|
};
|
|
|
|
contour.unshift(p);
|
|
|
|
contour.push(p);
|
|
|
|
}
|
|
|
|
moveTo(contour[0].x, contour[0].y);
|
|
|
|
for (j = 1, jj = contour.length; j < jj; j++) {
|
|
|
|
if (contour[j].flags & 1) {
|
|
|
|
lineTo(contour[j].x, contour[j].y);
|
|
|
|
} else if (contour[j + 1].flags & 1) {
|
|
|
|
quadraticCurveTo(
|
|
|
|
contour[j].x,
|
|
|
|
contour[j].y,
|
|
|
|
contour[j + 1].x,
|
|
|
|
contour[j + 1].y
|
|
|
|
);
|
|
|
|
j++;
|
2013-05-16 05:57:27 +09:00
|
|
|
} else {
|
2021-05-06 05:20:10 +09:00
|
|
|
quadraticCurveTo(
|
|
|
|
contour[j].x,
|
|
|
|
contour[j].y,
|
|
|
|
(contour[j].x + contour[j + 1].x) / 2,
|
|
|
|
(contour[j].y + contour[j + 1].y) / 2
|
|
|
|
);
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
startPoint = endPoint + 1;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
function compileCharString(charStringCode, cmds, font, glyphId) {
|
|
|
|
function moveTo(x, y) {
|
|
|
|
cmds.push({ cmd: "moveTo", args: [x, y] });
|
|
|
|
}
|
|
|
|
function lineTo(x, y) {
|
|
|
|
cmds.push({ cmd: "lineTo", args: [x, y] });
|
|
|
|
}
|
|
|
|
function bezierCurveTo(x1, y1, x2, y2, x, y) {
|
|
|
|
cmds.push({ cmd: "bezierCurveTo", args: [x1, y1, x2, y2, x, y] });
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
const stack = [];
|
|
|
|
let x = 0,
|
|
|
|
y = 0;
|
|
|
|
let stems = 0;
|
|
|
|
|
|
|
|
function parse(code) {
|
|
|
|
let i = 0;
|
|
|
|
while (i < code.length) {
|
|
|
|
let stackClean = false;
|
|
|
|
let v = code[i++];
|
|
|
|
let xa, xb, ya, yb, y1, y2, y3, n, subrCode;
|
|
|
|
switch (v) {
|
|
|
|
case 1: // hstem
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 3: // vstem
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 4: // vmoveto
|
|
|
|
y += stack.pop();
|
|
|
|
moveTo(x, y);
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 5: // rlineto
|
|
|
|
while (stack.length > 0) {
|
2013-05-16 05:57:27 +09:00
|
|
|
x += stack.shift();
|
|
|
|
y += stack.shift();
|
|
|
|
lineTo(x, y);
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 6: // hlineto
|
|
|
|
while (stack.length > 0) {
|
|
|
|
x += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
if (stack.length === 0) {
|
|
|
|
break;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
y += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 7: // vlineto
|
|
|
|
while (stack.length > 0) {
|
|
|
|
y += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
if (stack.length === 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
x += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 8: // rrcurveto
|
|
|
|
while (stack.length > 0) {
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb + stack.shift();
|
2013-05-16 05:57:27 +09:00
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 10: // callsubr
|
|
|
|
n = stack.pop();
|
|
|
|
subrCode = null;
|
|
|
|
if (font.isCFFCIDFont) {
|
|
|
|
const fdIndex = font.fdSelect.getFDIndex(glyphId);
|
|
|
|
if (fdIndex >= 0 && fdIndex < font.fdArray.length) {
|
|
|
|
const fontDict = font.fdArray[fdIndex];
|
|
|
|
let subrs;
|
|
|
|
if (fontDict.privateDict && fontDict.privateDict.subrsIndex) {
|
|
|
|
subrs = fontDict.privateDict.subrsIndex.objects;
|
|
|
|
}
|
|
|
|
if (subrs) {
|
|
|
|
// Add subroutine bias.
|
|
|
|
n += getSubroutineBias(subrs);
|
|
|
|
subrCode = subrs[n];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
warn("Invalid fd index for glyph index.");
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
} else {
|
|
|
|
subrCode = font.subrs[n + font.subrsBias];
|
|
|
|
}
|
|
|
|
if (subrCode) {
|
|
|
|
parse(subrCode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 11: // return
|
|
|
|
return;
|
|
|
|
case 12:
|
|
|
|
v = code[i++];
|
|
|
|
switch (v) {
|
|
|
|
case 34: // flex
|
|
|
|
xa = x + stack.shift();
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xb = xa + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
y1 = y + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
bezierCurveTo(xa, y, xb, y1, x, y1);
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xa = x + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
bezierCurveTo(xa, y1, xb, y, x, y);
|
|
|
|
break;
|
|
|
|
case 35: // flex
|
|
|
|
xa = x + stack.shift();
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
2013-05-16 05:57:27 +09:00
|
|
|
x = xb + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
y = yb + stack.shift();
|
2013-05-16 05:57:27 +09:00
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xa = x + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
ya = y + stack.shift();
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
x = xb + stack.shift();
|
2013-05-16 05:57:27 +09:00
|
|
|
y = yb + stack.shift();
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
2021-05-06 05:20:10 +09:00
|
|
|
stack.pop(); // fd
|
|
|
|
break;
|
|
|
|
case 36: // hflex1
|
|
|
|
xa = x + stack.shift();
|
|
|
|
y1 = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
y2 = y1 + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
bezierCurveTo(xa, y1, xb, y2, x, y2);
|
|
|
|
xa = x + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
y3 = y2 + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
bezierCurveTo(xa, y2, xb, y3, x, y);
|
|
|
|
break;
|
|
|
|
case 37: // flex1
|
|
|
|
const x0 = x,
|
|
|
|
y0 = y;
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xa = x + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
ya = y + stack.shift();
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
x = xb + stack.shift();
|
2013-05-16 05:57:27 +09:00
|
|
|
y = yb + stack.shift();
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
2021-05-06 05:20:10 +09:00
|
|
|
xa = x + stack.shift();
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
2021-05-06 05:20:10 +09:00
|
|
|
x = xb;
|
|
|
|
y = yb;
|
|
|
|
if (Math.abs(x - x0) > Math.abs(y - y0)) {
|
|
|
|
x += stack.shift();
|
|
|
|
} else {
|
|
|
|
y += stack.shift();
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
2021-05-06 05:20:10 +09:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new FormatError(`unknown operator: 12 ${v}`);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 14: // endchar
|
|
|
|
if (stack.length >= 4) {
|
|
|
|
const achar = stack.pop();
|
|
|
|
const bchar = stack.pop();
|
|
|
|
y = stack.pop();
|
|
|
|
x = stack.pop();
|
2021-05-24 20:20:19 +09:00
|
|
|
cmds.push({ cmd: "save" }, { cmd: "translate", args: [x, y] });
|
2021-05-06 05:20:10 +09:00
|
|
|
let cmap = lookupCmap(
|
|
|
|
font.cmap,
|
|
|
|
String.fromCharCode(font.glyphNameMap[StandardEncoding[achar]])
|
|
|
|
);
|
|
|
|
compileCharString(
|
|
|
|
font.glyphs[cmap.glyphId],
|
|
|
|
cmds,
|
|
|
|
font,
|
|
|
|
cmap.glyphId
|
|
|
|
);
|
|
|
|
cmds.push({ cmd: "restore" });
|
|
|
|
|
|
|
|
cmap = lookupCmap(
|
|
|
|
font.cmap,
|
|
|
|
String.fromCharCode(font.glyphNameMap[StandardEncoding[bchar]])
|
|
|
|
);
|
|
|
|
compileCharString(
|
|
|
|
font.glyphs[cmap.glyphId],
|
|
|
|
cmds,
|
|
|
|
font,
|
|
|
|
cmap.glyphId
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 18: // hstemhm
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 19: // hintmask
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
i += (stems + 7) >> 3;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 20: // cntrmask
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
i += (stems + 7) >> 3;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 21: // rmoveto
|
|
|
|
y += stack.pop();
|
|
|
|
x += stack.pop();
|
|
|
|
moveTo(x, y);
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 22: // hmoveto
|
|
|
|
x += stack.pop();
|
|
|
|
moveTo(x, y);
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 23: // vstemhm
|
|
|
|
stems += stack.length >> 1;
|
|
|
|
stackClean = true;
|
|
|
|
break;
|
|
|
|
case 24: // rcurveline
|
|
|
|
while (stack.length > 2) {
|
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb + stack.shift();
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
}
|
|
|
|
x += stack.shift();
|
|
|
|
y += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
break;
|
|
|
|
case 25: // rlinecurve
|
|
|
|
while (stack.length > 6) {
|
|
|
|
x += stack.shift();
|
|
|
|
y += stack.shift();
|
|
|
|
lineTo(x, y);
|
|
|
|
}
|
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb + stack.shift();
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
break;
|
|
|
|
case 26: // vvcurveto
|
|
|
|
if (stack.length % 2) {
|
|
|
|
x += stack.shift();
|
|
|
|
}
|
|
|
|
while (stack.length > 0) {
|
|
|
|
xa = x;
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb;
|
|
|
|
y = yb + stack.shift();
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 27: // hhcurveto
|
|
|
|
if (stack.length % 2) {
|
|
|
|
y += stack.shift();
|
|
|
|
}
|
|
|
|
while (stack.length > 0) {
|
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y;
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb;
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 28:
|
|
|
|
stack.push(((code[i] << 24) | (code[i + 1] << 16)) >> 16);
|
|
|
|
i += 2;
|
|
|
|
break;
|
|
|
|
case 29: // callgsubr
|
|
|
|
n = stack.pop() + font.gsubrsBias;
|
|
|
|
subrCode = font.gsubrs[n];
|
|
|
|
if (subrCode) {
|
|
|
|
parse(subrCode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 30: // vhcurveto
|
|
|
|
while (stack.length > 0) {
|
|
|
|
xa = x;
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb + (stack.length === 1 ? stack.shift() : 0);
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
if (stack.length === 0) {
|
|
|
|
break;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
|
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y;
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
y = yb + stack.shift();
|
|
|
|
x = xb + (stack.length === 1 ? stack.shift() : 0);
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 31: // hvcurveto
|
|
|
|
while (stack.length > 0) {
|
|
|
|
xa = x + stack.shift();
|
|
|
|
ya = y;
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
y = yb + stack.shift();
|
|
|
|
x = xb + (stack.length === 1 ? stack.shift() : 0);
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
if (stack.length === 0) {
|
|
|
|
break;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
|
|
|
|
xa = x;
|
|
|
|
ya = y + stack.shift();
|
|
|
|
xb = xa + stack.shift();
|
|
|
|
yb = ya + stack.shift();
|
|
|
|
x = xb + stack.shift();
|
|
|
|
y = yb + (stack.length === 1 ? stack.shift() : 0);
|
|
|
|
bezierCurveTo(xa, ya, xb, yb, x, y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (v < 32) {
|
|
|
|
throw new FormatError(`unknown operator: ${v}`);
|
|
|
|
}
|
|
|
|
if (v < 247) {
|
|
|
|
stack.push(v - 139);
|
|
|
|
} else if (v < 251) {
|
|
|
|
stack.push((v - 247) * 256 + code[i++] + 108);
|
|
|
|
} else if (v < 255) {
|
|
|
|
stack.push(-(v - 251) * 256 - code[i++] - 108);
|
|
|
|
} else {
|
|
|
|
stack.push(
|
|
|
|
((code[i] << 24) |
|
|
|
|
(code[i + 1] << 16) |
|
|
|
|
(code[i + 2] << 8) |
|
|
|
|
code[i + 3]) /
|
|
|
|
65536
|
|
|
|
);
|
|
|
|
i += 4;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (stackClean) {
|
|
|
|
stack.length = 0;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
parse(charStringCode);
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
const NOOP = [];
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
class CompiledFont {
|
|
|
|
constructor(fontMatrix) {
|
|
|
|
if (this.constructor === CompiledFont) {
|
|
|
|
unreachable("Cannot initialize CompiledFont.");
|
2018-07-09 05:16:05 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
this.fontMatrix = fontMatrix;
|
2018-07-09 05:16:05 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
this.compiledGlyphs = Object.create(null);
|
|
|
|
this.compiledCharCodeToGlyphId = Object.create(null);
|
|
|
|
}
|
|
|
|
|
|
|
|
getPathJs(unicode) {
|
2021-05-16 01:21:18 +09:00
|
|
|
const { charCode, glyphId } = lookupCmap(this.cmap, unicode);
|
|
|
|
let fn = this.compiledGlyphs[glyphId];
|
2021-05-06 05:20:10 +09:00
|
|
|
if (!fn) {
|
2021-05-16 01:21:18 +09:00
|
|
|
try {
|
|
|
|
fn = this.compileGlyph(this.glyphs[glyphId], glyphId);
|
|
|
|
this.compiledGlyphs[glyphId] = fn;
|
|
|
|
} catch (ex) {
|
|
|
|
// Avoid attempting to re-compile a corrupt glyph.
|
|
|
|
this.compiledGlyphs[glyphId] = NOOP;
|
|
|
|
|
|
|
|
if (this.compiledCharCodeToGlyphId[charCode] === undefined) {
|
|
|
|
this.compiledCharCodeToGlyphId[charCode] = glyphId;
|
|
|
|
}
|
|
|
|
throw ex;
|
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
2021-05-16 01:21:18 +09:00
|
|
|
if (this.compiledCharCodeToGlyphId[charCode] === undefined) {
|
|
|
|
this.compiledCharCodeToGlyphId[charCode] = glyphId;
|
2018-07-09 05:16:05 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
return fn;
|
|
|
|
}
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
compileGlyph(code, glyphId) {
|
|
|
|
if (!code || code.length === 0 || code[0] === 14) {
|
|
|
|
return NOOP;
|
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
let fontMatrix = this.fontMatrix;
|
|
|
|
if (this.isCFFCIDFont) {
|
|
|
|
// Top DICT's FontMatrix can be ignored because CFFCompiler always
|
|
|
|
// removes it and copies to FDArray DICTs.
|
|
|
|
const fdIndex = this.fdSelect.getFDIndex(glyphId);
|
|
|
|
if (fdIndex >= 0 && fdIndex < this.fdArray.length) {
|
|
|
|
const fontDict = this.fdArray[fdIndex];
|
|
|
|
fontMatrix = fontDict.getByName("FontMatrix") || FONT_IDENTITY_MATRIX;
|
|
|
|
} else {
|
|
|
|
warn("Invalid fd index for glyph index.");
|
2018-04-10 22:44:42 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
2018-04-10 22:44:42 +09:00
|
|
|
|
2021-05-24 20:20:19 +09:00
|
|
|
const cmds = [
|
|
|
|
{ cmd: "save" },
|
|
|
|
{ cmd: "transform", args: fontMatrix.slice() },
|
|
|
|
{ cmd: "scale", args: ["size", "-size"] },
|
|
|
|
];
|
2021-05-06 05:20:10 +09:00
|
|
|
this.compileGlyphImpl(code, cmds, glyphId);
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
cmds.push({ cmd: "restore" });
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
return cmds;
|
|
|
|
}
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
compileGlyphImpl() {
|
|
|
|
unreachable("Children classes should implement this.");
|
2018-07-09 05:16:05 +09:00
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
hasBuiltPath(unicode) {
|
2021-05-16 01:21:18 +09:00
|
|
|
const { charCode, glyphId } = lookupCmap(this.cmap, unicode);
|
2021-05-06 05:20:10 +09:00
|
|
|
return (
|
2021-05-16 01:21:18 +09:00
|
|
|
this.compiledGlyphs[glyphId] !== undefined &&
|
|
|
|
this.compiledCharCodeToGlyphId[charCode] !== undefined
|
2021-05-06 05:20:10 +09:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
class TrueTypeCompiled extends CompiledFont {
|
|
|
|
constructor(glyphs, cmap, fontMatrix) {
|
|
|
|
super(fontMatrix || [0.000488, 0, 0, 0.000488, 0, 0]);
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
this.glyphs = glyphs;
|
|
|
|
this.cmap = cmap;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
compileGlyphImpl(code, cmds) {
|
|
|
|
compileGlyf(code, cmds, this);
|
|
|
|
}
|
|
|
|
}
|
2018-07-09 05:16:05 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
class Type2Compiled extends CompiledFont {
|
|
|
|
constructor(cffInfo, cmap, fontMatrix, glyphNameMap) {
|
|
|
|
super(fontMatrix || [0.001, 0, 0, 0.001, 0, 0]);
|
2018-07-09 05:16:05 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
this.glyphs = cffInfo.glyphs;
|
|
|
|
this.gsubrs = cffInfo.gsubrs || [];
|
|
|
|
this.subrs = cffInfo.subrs || [];
|
|
|
|
this.cmap = cmap;
|
|
|
|
this.glyphNameMap = glyphNameMap || getGlyphsUnicode();
|
2018-07-09 05:16:05 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
this.gsubrsBias = getSubroutineBias(this.gsubrs);
|
|
|
|
this.subrsBias = getSubroutineBias(this.subrs);
|
2018-07-09 05:16:05 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
this.isCFFCIDFont = cffInfo.isCFFCIDFont;
|
|
|
|
this.fdSelect = cffInfo.fdSelect;
|
|
|
|
this.fdArray = cffInfo.fdArray;
|
2018-07-09 05:16:05 +09:00
|
|
|
}
|
2013-08-20 08:33:20 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
compileGlyphImpl(code, cmds, glyphId) {
|
|
|
|
compileCharString(code, cmds, this, glyphId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class FontRendererFactory {
|
|
|
|
static create(font, seacAnalysisEnabled) {
|
|
|
|
const data = new Uint8Array(font.data);
|
|
|
|
let cmap, glyf, loca, cff, indexToLocFormat, unitsPerEm;
|
2022-01-16 07:41:18 +09:00
|
|
|
const numTables = getUint16(data, 4);
|
2021-05-06 05:20:10 +09:00
|
|
|
for (let i = 0, p = 12; i < numTables; i++, p += 16) {
|
|
|
|
const tag = bytesToString(data.subarray(p, p + 4));
|
2022-01-16 07:41:18 +09:00
|
|
|
const offset = getUint32(data, p + 8);
|
|
|
|
const length = getUint32(data, p + 12);
|
2021-05-06 05:20:10 +09:00
|
|
|
switch (tag) {
|
|
|
|
case "cmap":
|
|
|
|
cmap = parseCmap(data, offset, offset + length);
|
|
|
|
break;
|
|
|
|
case "glyf":
|
|
|
|
glyf = data.subarray(offset, offset + length);
|
|
|
|
break;
|
|
|
|
case "loca":
|
|
|
|
loca = data.subarray(offset, offset + length);
|
|
|
|
break;
|
|
|
|
case "head":
|
2022-01-16 07:41:18 +09:00
|
|
|
unitsPerEm = getUint16(data, offset + 18);
|
|
|
|
indexToLocFormat = getUint16(data, offset + 50);
|
2021-05-06 05:20:10 +09:00
|
|
|
break;
|
|
|
|
case "CFF ":
|
|
|
|
cff = parseCff(data, offset, offset + length, seacAnalysisEnabled);
|
|
|
|
break;
|
2013-05-16 05:57:27 +09:00
|
|
|
}
|
2021-05-06 05:20:10 +09:00
|
|
|
}
|
2013-05-16 05:57:27 +09:00
|
|
|
|
2021-05-06 05:20:10 +09:00
|
|
|
if (glyf) {
|
|
|
|
const fontMatrix = !unitsPerEm
|
|
|
|
? font.fontMatrix
|
|
|
|
: [1 / unitsPerEm, 0, 0, 1 / unitsPerEm, 0, 0];
|
|
|
|
return new TrueTypeCompiled(
|
|
|
|
parseGlyfTable(glyf, loca, indexToLocFormat),
|
|
|
|
cmap,
|
|
|
|
fontMatrix
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return new Type2Compiled(cff, cmap, font.fontMatrix, font.glyphNameMap);
|
|
|
|
}
|
|
|
|
}
|
2015-11-22 01:32:47 +09:00
|
|
|
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
export { FontRendererFactory };
|