pagyra-js 0.0.19 → 0.0.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +55 -0
- package/dist/assets/fonts/licenses/selawik/SIL Open Font License.txt +43 -0
- package/dist/assets/fonts/ttf/arimo/Arimo-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/arimo/Arimo-BoldItalic.ttf +0 -0
- package/dist/assets/fonts/ttf/arimo/Arimo-Italic.ttf +0 -0
- package/dist/assets/fonts/ttf/arimo/Arimo-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/cinzeldecorative/CinzelDecorative-Black.ttf +0 -0
- package/dist/assets/fonts/ttf/cinzeldecorative/CinzelDecorative-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/cinzeldecorative/CinzelDecorative-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/dejavu/DejaVuSans.ttf +0 -0
- package/dist/assets/fonts/ttf/firecode/FiraCode-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/firecode/FiraCode-Light.ttf +0 -0
- package/dist/assets/fonts/ttf/firecode/FiraCode-Medium.ttf +0 -0
- package/dist/assets/fonts/ttf/firecode/FiraCode-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/firecode/FiraCode-SemiBold.ttf +0 -0
- package/dist/assets/fonts/ttf/notoemoji/NotoEmoji-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/notoemoji/NotoEmoji-Light.ttf +0 -0
- package/dist/assets/fonts/ttf/notoemoji/NotoEmoji-Medium.ttf +0 -0
- package/dist/assets/fonts/ttf/notoemoji/NotoEmoji-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/notoemoji/NotoEmoji-SemiBold.ttf +0 -0
- package/dist/assets/fonts/ttf/notosans/NotoSans-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/roboto/Roboto-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/roboto/Roboto-BoldItalic.ttf +0 -0
- package/dist/assets/fonts/ttf/roboto/Roboto-Italic.ttf +0 -0
- package/dist/assets/fonts/ttf/roboto/Roboto-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/selawik/selawk.ttf +0 -0
- package/dist/assets/fonts/ttf/selawik/selawkb.ttf +0 -0
- package/dist/assets/fonts/ttf/selawik/selawkl.ttf +0 -0
- package/dist/assets/fonts/ttf/selawik/selawksb.ttf +0 -0
- package/dist/assets/fonts/ttf/selawik/selawksl.ttf +0 -0
- package/dist/assets/fonts/ttf/stixtwomath/STIXTwoMath-Regular.ttf +0 -0
- package/dist/assets/fonts/ttf/tinos/Tinos-Bold.ttf +0 -0
- package/dist/assets/fonts/ttf/tinos/Tinos-BoldItalic.ttf +0 -0
- package/dist/assets/fonts/ttf/tinos/Tinos-Italic.ttf +0 -0
- package/dist/assets/fonts/ttf/tinos/Tinos-Regular.ttf +0 -0
- package/dist/assets/fonts/woff/lato/lato-latin-400-italic.woff +0 -0
- package/dist/assets/fonts/woff/lato/lato-latin-400-normal.woff +0 -0
- package/dist/assets/fonts/woff/lato/lato-latin-700-italic.woff +0 -0
- package/dist/assets/fonts/woff/lato/lato-latin-700-normal.woff +0 -0
- package/dist/assets/fonts/woff2/caveat/Caveat-Bold.woff2 +0 -0
- package/dist/assets/fonts/woff2/caveat/Caveat-Regular.woff2 +0 -0
- package/dist/assets/fonts/woff2/lato/lato-latin-400-italic.woff2 +0 -0
- package/dist/assets/fonts/woff2/lato/lato-latin-400-normal.woff2 +0 -0
- package/dist/assets/fonts/woff2/lato/lato-latin-700-italic.woff2 +0 -0
- package/dist/assets/fonts/woff2/lato/lato-latin-700-normal.woff2 +0 -0
- package/dist/browser/pagyra.min.js +34 -34
- package/dist/browser/pagyra.min.js.map +4 -4
- package/dist/playground/server.js +2 -0
- package/dist/src/css/compute-style/base-options.d.ts +7 -0
- package/dist/src/css/compute-style/base-options.js +24 -0
- package/dist/src/css/compute-style/declarations.d.ts +10 -0
- package/dist/src/css/compute-style/declarations.js +77 -0
- package/dist/src/css/compute-style/decoration.d.ts +8 -0
- package/dist/src/css/compute-style/decoration.js +55 -0
- package/dist/src/css/compute-style/defaults.d.ts +3 -0
- package/dist/src/css/compute-style/defaults.js +34 -0
- package/dist/src/css/compute-style/display.d.ts +3 -0
- package/dist/src/css/compute-style/display.js +85 -0
- package/dist/src/css/compute-style/float.d.ts +2 -0
- package/dist/src/css/compute-style/float.js +13 -0
- package/dist/src/css/compute-style/font.d.ts +12 -0
- package/dist/src/css/compute-style/font.js +57 -0
- package/dist/src/css/compute-style/overrides.d.ts +3 -0
- package/dist/src/css/compute-style/overrides.js +241 -0
- package/dist/src/css/compute-style.d.ts +2 -0
- package/dist/src/css/compute-style.js +34 -487
- package/dist/src/css/enums.d.ts +4 -0
- package/dist/src/css/enums.js +5 -0
- package/dist/src/css/layout-property-resolver.js +30 -18
- package/dist/src/css/length.d.ts +26 -2
- package/dist/src/css/length.js +48 -0
- package/dist/src/css/parsers/background-parser.js +1 -1
- package/dist/src/css/parsers/calc-parser.d.ts +2 -0
- package/dist/src/css/parsers/calc-parser.js +310 -0
- package/dist/src/css/parsers/content-parser.d.ts +2 -1
- package/dist/src/css/parsers/content-parser.js +7 -2
- package/dist/src/css/parsers/dimension-parser.js +37 -18
- package/dist/src/css/parsers/display-flex-parser.d.ts +4 -0
- package/dist/src/css/parsers/display-flex-parser.js +97 -0
- package/dist/src/css/parsers/filter-parser.d.ts +14 -0
- package/dist/src/css/parsers/filter-parser.js +255 -0
- package/dist/src/css/parsers/grid-parser-extended.d.ts +1 -0
- package/dist/src/css/parsers/grid-parser-extended.js +40 -1
- package/dist/src/css/parsers/grid-parser.d.ts +5 -2
- package/dist/src/css/parsers/grid-parser.js +71 -7
- package/dist/src/css/parsers/length-parser.d.ts +8 -3
- package/dist/src/css/parsers/length-parser.js +45 -2
- package/dist/src/css/parsers/margin-block-parser.js +3 -3
- package/dist/src/css/parsers/margin-parser.js +3 -3
- package/dist/src/css/parsers/padding-block-parser.js +3 -3
- package/dist/src/css/parsers/padding-inline-parser.js +3 -3
- package/dist/src/css/parsers/padding-parser.js +6 -6
- package/dist/src/css/parsers/position-parser.js +2 -22
- package/dist/src/css/parsers/register-parsers.js +29 -2
- package/dist/src/css/parsers/word-break-parser.d.ts +2 -0
- package/dist/src/css/parsers/word-break-parser.js +23 -0
- package/dist/src/css/properties/grid.d.ts +16 -2
- package/dist/src/css/properties/layout.d.ts +3 -1
- package/dist/src/css/properties/layout.js +1 -1
- package/dist/src/css/properties/misc.d.ts +5 -0
- package/dist/src/css/properties/typography.d.ts +3 -0
- package/dist/src/css/properties/visual.d.ts +36 -0
- package/dist/src/css/shorthands/box-shorthand.d.ts +2 -2
- package/dist/src/css/style-inheritance.d.ts +2 -1
- package/dist/src/css/style-inheritance.js +1 -0
- package/dist/src/css/style.d.ts +30 -10
- package/dist/src/css/style.js +8 -1
- package/dist/src/css/ua-defaults/base-defaults.d.ts +1 -0
- package/dist/src/css/ua-defaults/base-defaults.js +10 -1
- package/dist/src/css/ua-defaults/element-defaults.js +0 -2
- package/dist/src/html/css/parse-css.d.ts +2 -0
- package/dist/src/html/css/parse-css.js +32 -3
- package/dist/src/html/dom-converter/background-images.d.ts +3 -0
- package/dist/src/html/dom-converter/background-images.js +88 -0
- package/dist/src/html/dom-converter/convert-dom-node.d.ts +5 -0
- package/dist/src/html/dom-converter/convert-dom-node.js +81 -0
- package/dist/src/html/dom-converter/handlers/br-handler.d.ts +2 -0
- package/dist/src/html/dom-converter/handlers/br-handler.js +20 -0
- package/dist/src/html/dom-converter/handlers/form-control-handler.d.ts +2 -0
- package/dist/src/html/dom-converter/handlers/form-control-handler.js +28 -0
- package/dist/src/html/dom-converter/handlers/img-handler.d.ts +2 -0
- package/dist/src/html/dom-converter/handlers/img-handler.js +4 -0
- package/dist/src/html/dom-converter/handlers/index.d.ts +4 -0
- package/dist/src/html/dom-converter/handlers/index.js +19 -0
- package/dist/src/html/dom-converter/handlers/svg-handler.d.ts +2 -0
- package/dist/src/html/dom-converter/handlers/svg-handler.js +32 -0
- package/dist/src/html/dom-converter/handlers/types.d.ts +12 -0
- package/dist/src/html/dom-converter/handlers/types.js +2 -0
- package/dist/src/html/dom-converter/helpers.d.ts +7 -0
- package/dist/src/html/dom-converter/helpers.js +35 -0
- package/dist/src/html/dom-converter/index.d.ts +1 -0
- package/dist/src/html/dom-converter/index.js +1 -0
- package/dist/src/html/dom-converter/pseudo-elements.d.ts +6 -0
- package/dist/src/html/dom-converter/pseudo-elements.js +48 -0
- package/dist/src/html/dom-converter/text.d.ts +15 -0
- package/dist/src/html/dom-converter/text.js +170 -0
- package/dist/src/html/dom-converter.d.ts +1 -5
- package/dist/src/html/dom-converter.js +1 -412
- package/dist/src/html/image-converter.d.ts +5 -0
- package/dist/src/html/image-converter.js +8 -3
- package/dist/src/html-to-pdf/document-css.d.ts +14 -0
- package/dist/src/html-to-pdf/document-css.js +45 -0
- package/dist/src/html-to-pdf/fonts.d.ts +16 -0
- package/dist/src/html-to-pdf/fonts.js +74 -0
- package/dist/src/html-to-pdf/header-footer.d.ts +14 -0
- package/dist/src/html-to-pdf/header-footer.js +101 -0
- package/dist/src/html-to-pdf/html-parser.d.ts +6 -0
- package/dist/src/html-to-pdf/html-parser.js +81 -0
- package/dist/src/html-to-pdf/index.d.ts +3 -0
- package/dist/src/html-to-pdf/index.js +2 -0
- package/dist/src/html-to-pdf/layout-build.d.ts +37 -0
- package/dist/src/html-to-pdf/layout-build.js +73 -0
- package/dist/src/html-to-pdf/prepare-html-render.d.ts +2 -0
- package/dist/src/html-to-pdf/prepare-html-render.js +121 -0
- package/dist/src/html-to-pdf/render-finalize.d.ts +15 -0
- package/dist/src/html-to-pdf/render-finalize.js +27 -0
- package/dist/src/html-to-pdf/render-html-to-pdf.d.ts +3 -0
- package/dist/src/html-to-pdf/render-html-to-pdf.js +25 -0
- package/dist/src/html-to-pdf/resource-loader.d.ts +6 -0
- package/dist/src/html-to-pdf/resource-loader.js +120 -0
- package/dist/src/html-to-pdf/types.d.ts +38 -0
- package/dist/src/html-to-pdf/types.js +2 -0
- package/dist/src/html-to-pdf.d.ts +1 -37
- package/dist/src/html-to-pdf.js +1 -537
- package/dist/src/image/js-png-backend.d.ts +7 -0
- package/dist/src/image/js-png-backend.js +9 -0
- package/dist/src/image/png-backend.d.ts +5 -0
- package/dist/src/image/png-backend.js +1 -0
- package/dist/src/image/png-wasm-loader.d.ts +5 -0
- package/dist/src/image/png-wasm-loader.js +59 -0
- package/dist/src/image/wasm/png_decoder_wasm.d.ts +8 -0
- package/dist/src/image/wasm/png_decoder_wasm.js +24 -0
- package/dist/src/image/wasm/png_decoder_wasm_bg.js +16 -0
- package/dist/src/image/wasm-png-backend.d.ts +6 -0
- package/dist/src/image/wasm-png-backend.js +17 -0
- package/dist/src/layout/counter.d.ts +1 -2
- package/dist/src/layout/counter.js +18 -18
- package/dist/src/layout/inline/inline-utils.d.ts +1 -1
- package/dist/src/layout/inline/inline-utils.js +8 -7
- package/dist/src/layout/inline/layout.js +16 -3
- package/dist/src/layout/inline/run-placer.d.ts +1 -0
- package/dist/src/layout/inline/run-placer.js +2 -10
- package/dist/src/layout/pipeline/out-of-flow-manager.js +25 -1
- package/dist/src/layout/strategies/block.js +35 -24
- package/dist/src/layout/strategies/flex.js +305 -61
- package/dist/src/layout/strategies/form.d.ts +2 -0
- package/dist/src/layout/strategies/form.js +38 -13
- package/dist/src/layout/strategies/grid.js +166 -29
- package/dist/src/layout/strategies/image.js +53 -27
- package/dist/src/layout/strategies/inline.js +26 -21
- package/dist/src/layout/strategies/table.js +26 -18
- package/dist/src/layout/utils/content-measurer.d.ts +1 -1
- package/dist/src/layout/utils/content-measurer.js +8 -7
- package/dist/src/layout/utils/floats.d.ts +1 -0
- package/dist/src/layout/utils/floats.js +14 -12
- package/dist/src/layout/utils/margin.d.ts +4 -4
- package/dist/src/layout/utils/margin.js +20 -16
- package/dist/src/layout/utils/node-math.d.ts +12 -6
- package/dist/src/layout/utils/node-math.js +71 -41
- package/dist/src/layout/utils/sizing.js +2 -1
- package/dist/src/pdf/font-subset/font-registry.d.ts +6 -0
- package/dist/src/pdf/font-subset/font-registry.js +30 -2
- package/dist/src/pdf/header-footer-painter.d.ts +2 -0
- package/dist/src/pdf/header-footer-painter.js +52 -4
- package/dist/src/pdf/header-footer-renderer.js +12 -1
- package/dist/src/pdf/layout-tree-builder.js +5 -1
- package/dist/src/pdf/page-painter.js +13 -0
- package/dist/src/pdf/pagination.js +2 -2
- package/dist/src/pdf/renderer/box-painter.js +28 -3
- package/dist/src/pdf/renderer/page-paint.js +12 -3
- package/dist/src/pdf/renderers/radius-utils.js +31 -38
- package/dist/src/pdf/renderers/shape-renderer.js +1 -1
- package/dist/src/pdf/renderers/shape-utils.js +1 -1
- package/dist/src/pdf/renderers/text-renderer.d.ts +9 -1
- package/dist/src/pdf/renderers/text-renderer.js +36 -2
- package/dist/src/pdf/stacking/build-stacking-contexts.js +1 -2
- package/dist/src/pdf/stacking/resolve-paint-order.d.ts +5 -6
- package/dist/src/pdf/stacking/resolve-paint-order.js +29 -9
- package/dist/src/pdf/stacking/types.d.ts +14 -0
- package/dist/src/pdf/svg/shape-renderer.js +47 -20
- package/dist/src/pdf/types.d.ts +7 -1
- package/dist/src/pdf/utils/border-radius-utils.js +31 -38
- package/dist/src/pdf/utils/color-utils.js +17 -2
- package/dist/src/pdf/utils/filter-utils.d.ts +29 -0
- package/dist/src/pdf/utils/filter-utils.js +85 -0
- package/dist/src/pdf/utils/node-text-run-factory.js +1 -1
- package/dist/src/pdf/utils/text-layout-adjuster.d.ts +0 -8
- package/dist/src/pdf/utils/text-layout-adjuster.js +12 -9
- package/dist/src/shim/css-browser.d.ts +14 -9
- package/dist/src/shim/css-browser.js +50 -39
- package/dist/src/units/units.d.ts +1 -1
- package/dist/tests/css/box-sizing.spec.d.ts +1 -0
- package/dist/tests/css/box-sizing.spec.js +46 -0
- package/dist/tests/css/calc-parser.spec.d.ts +1 -0
- package/dist/tests/css/calc-parser.spec.js +68 -0
- package/dist/tests/css/container-query-units.spec.d.ts +1 -0
- package/dist/tests/css/container-query-units.spec.js +64 -0
- package/dist/tests/css/content-parser.spec.js +13 -0
- package/dist/tests/css/filter-parser.spec.d.ts +1 -0
- package/dist/tests/css/filter-parser.spec.js +116 -0
- package/dist/tests/css/flex-shorthand.spec.d.ts +1 -0
- package/dist/tests/css/flex-shorthand.spec.js +45 -0
- package/dist/tests/css/grid-clamp.spec.d.ts +1 -0
- package/dist/tests/css/grid-clamp.spec.js +82 -0
- package/dist/tests/css/parse-css-pseudo.spec.d.ts +1 -0
- package/dist/tests/css/parse-css-pseudo.spec.js +26 -0
- package/dist/tests/helpers/render-utils.d.ts +18 -2
- package/dist/tests/helpers/render-utils.js +25 -12
- package/dist/tests/html/dom-converter-pseudo-elements.spec.d.ts +1 -0
- package/dist/tests/html/dom-converter-pseudo-elements.spec.js +33 -0
- package/dist/tests/html/dom-converter-text.spec.d.ts +1 -0
- package/dist/tests/html/dom-converter-text.spec.js +67 -0
- package/dist/tests/image/png-backend.spec.d.ts +1 -0
- package/dist/tests/image/png-backend.spec.js +34 -0
- package/dist/tests/layout/box-sizing.spec.d.ts +1 -0
- package/dist/tests/layout/box-sizing.spec.js +75 -0
- package/dist/tests/layout/calc-padding.spec.d.ts +1 -0
- package/dist/tests/layout/calc-padding.spec.js +19 -0
- package/dist/tests/layout/container-query-units.spec.d.ts +1 -0
- package/dist/tests/layout/container-query-units.spec.js +24 -0
- package/dist/tests/layout/flex-auto-height.spec.d.ts +1 -0
- package/dist/tests/layout/flex-auto-height.spec.js +35 -0
- package/dist/tests/layout/flex-wrap-cards.spec.d.ts +1 -0
- package/dist/tests/layout/flex-wrap-cards.spec.js +16 -0
- package/dist/tests/layout/flex-wrap-grow-align-content.spec.d.ts +1 -0
- package/dist/tests/layout/flex-wrap-grow-align-content.spec.js +20 -0
- package/dist/tests/layout/grid-clamp-gap.spec.d.ts +1 -0
- package/dist/tests/layout/grid-clamp-gap.spec.js +22 -0
- package/dist/tests/layout/inline-fragments.spec.js +38 -0
- package/dist/tests/layout/paged-body-margin.spec.d.ts +1 -0
- package/dist/tests/layout/paged-body-margin.spec.js +92 -0
- package/dist/tests/layout/pseudo-counters-generated-content.spec.d.ts +1 -0
- package/dist/tests/layout/pseudo-counters-generated-content.spec.js +51 -0
- package/dist/tests/layout/responsive-clamp-grid-parity.spec.d.ts +1 -0
- package/dist/tests/layout/responsive-clamp-grid-parity.spec.js +75 -0
- package/dist/tests/layout/run-placer-baseline.spec.js +13 -11
- package/dist/tests/pdf/backdrop-filter-noop.spec.d.ts +1 -0
- package/dist/tests/pdf/backdrop-filter-noop.spec.js +140 -0
- package/dist/tests/pdf/filter-drop-shadow.spec.d.ts +1 -0
- package/dist/tests/pdf/filter-drop-shadow.spec.js +74 -0
- package/dist/tests/pdf/filter-opacity.spec.d.ts +1 -0
- package/dist/tests/pdf/filter-opacity.spec.js +30 -0
- package/dist/tests/pdf/font-subset-registry-key.spec.d.ts +1 -0
- package/dist/tests/pdf/font-subset-registry-key.spec.js +66 -0
- package/dist/tests/pdf/header-footer-clip-overflow.spec.d.ts +1 -0
- package/dist/tests/pdf/header-footer-clip-overflow.spec.js +45 -0
- package/dist/tests/pdf/selawik-opt-in.spec.d.ts +1 -0
- package/dist/tests/pdf/selawik-opt-in.spec.js +106 -0
- package/dist/tests/pdf/system-ui-fallback-subset-regression.spec.d.ts +1 -0
- package/dist/tests/pdf/system-ui-fallback-subset-regression.spec.js +39 -0
- package/dist/tests/pdf/text-renderer-fallback.spec.js +55 -0
- package/dist/tests/pdf/text-transform-matrix.spec.js +8 -7
- package/package.json +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { extractOpacityMultiplier } from "../../src/pdf/utils/filter-utils.js";
|
|
3
|
+
describe("extractOpacityMultiplier", () => {
|
|
4
|
+
it("returns 1 for empty filter list", () => {
|
|
5
|
+
expect(extractOpacityMultiplier([])).toBe(1);
|
|
6
|
+
});
|
|
7
|
+
it("returns value for single opacity()", () => {
|
|
8
|
+
const filters = [{ kind: "opacity", value: 0.5 }];
|
|
9
|
+
expect(extractOpacityMultiplier(filters)).toBe(0.5);
|
|
10
|
+
});
|
|
11
|
+
it("multiplies multiple opacity() filters", () => {
|
|
12
|
+
const filters = [
|
|
13
|
+
{ kind: "opacity", value: 0.5 },
|
|
14
|
+
{ kind: "opacity", value: 0.5 },
|
|
15
|
+
];
|
|
16
|
+
expect(extractOpacityMultiplier(filters)).toBe(0.25);
|
|
17
|
+
});
|
|
18
|
+
it("ignores non-opacity filters", () => {
|
|
19
|
+
const filters = [
|
|
20
|
+
{ kind: "blur", value: 5 },
|
|
21
|
+
{ kind: "opacity", value: 0.7 },
|
|
22
|
+
{ kind: "brightness", value: 1.5 },
|
|
23
|
+
];
|
|
24
|
+
expect(extractOpacityMultiplier(filters)).toBeCloseTo(0.7);
|
|
25
|
+
});
|
|
26
|
+
it("clamps result to [0, 1]", () => {
|
|
27
|
+
const filters = [{ kind: "opacity", value: 0 }];
|
|
28
|
+
expect(extractOpacityMultiplier(filters)).toBe(0);
|
|
29
|
+
});
|
|
30
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { PdfFontRegistry } from "../../src/pdf/font-subset/font-registry.js";
|
|
3
|
+
function makeFont(family, glyphIdForA, rawAccessor) {
|
|
4
|
+
const unicodeMap = new Map([[65, glyphIdForA]]);
|
|
5
|
+
return {
|
|
6
|
+
metrics: {
|
|
7
|
+
metrics: {
|
|
8
|
+
unitsPerEm: 1000,
|
|
9
|
+
ascender: 800,
|
|
10
|
+
descender: -200,
|
|
11
|
+
lineGap: 0,
|
|
12
|
+
capHeight: 700,
|
|
13
|
+
xHeight: 500,
|
|
14
|
+
},
|
|
15
|
+
glyphMetrics: new Map([
|
|
16
|
+
[0, { advanceWidth: 500, leftSideBearing: 0 }],
|
|
17
|
+
[glyphIdForA, { advanceWidth: 600, leftSideBearing: 0 }],
|
|
18
|
+
]),
|
|
19
|
+
cmap: {
|
|
20
|
+
getGlyphId: (cp) => unicodeMap.get(cp) ?? 0,
|
|
21
|
+
hasCodePoint: (cp) => unicodeMap.has(cp),
|
|
22
|
+
unicodeMap,
|
|
23
|
+
},
|
|
24
|
+
},
|
|
25
|
+
program: {
|
|
26
|
+
sourceFormat: "ttf",
|
|
27
|
+
unitsPerEm: 1000,
|
|
28
|
+
glyphCount: 2048,
|
|
29
|
+
getRawTableData: rawAccessor,
|
|
30
|
+
},
|
|
31
|
+
css: {
|
|
32
|
+
family,
|
|
33
|
+
weight: 700,
|
|
34
|
+
style: "normal",
|
|
35
|
+
},
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
describe("PdfFontRegistry keying", () => {
|
|
39
|
+
it("separates subsets for different actual fonts even with same css family stack", () => {
|
|
40
|
+
const registry = new PdfFontRegistry();
|
|
41
|
+
const familyStack = 'system-ui, -apple-system, "Segoe UI", Roboto, sans-serif';
|
|
42
|
+
const fontA = makeFont(familyStack, 44, () => null);
|
|
43
|
+
const fontB = makeFont(familyStack, 43, () => null);
|
|
44
|
+
const runA = {
|
|
45
|
+
font: fontA,
|
|
46
|
+
glyphIds: [44],
|
|
47
|
+
positions: [{ x: 0, y: 0 }],
|
|
48
|
+
text: "H",
|
|
49
|
+
fontSize: 14,
|
|
50
|
+
};
|
|
51
|
+
const runB = {
|
|
52
|
+
font: fontB,
|
|
53
|
+
glyphIds: [43],
|
|
54
|
+
positions: [{ x: 0, y: 0 }],
|
|
55
|
+
text: "H",
|
|
56
|
+
fontSize: 14,
|
|
57
|
+
};
|
|
58
|
+
registry.registerGlyphRun(runA);
|
|
59
|
+
registry.registerGlyphRun(runB);
|
|
60
|
+
const subsetA = registry.ensureSubsetFor(fontA);
|
|
61
|
+
const subsetB = registry.ensureSubsetFor(fontB);
|
|
62
|
+
expect(subsetA.subset.name).not.toBe(subsetB.subset.name);
|
|
63
|
+
expect(subsetA.subset.encodeGlyph(44)).toBe(44);
|
|
64
|
+
expect(subsetB.subset.encodeGlyph(43)).toBe(43);
|
|
65
|
+
});
|
|
66
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { paintHeaderFooter } from "../../src/pdf/header-footer-painter.js";
|
|
3
|
+
import { PagePainter } from "../../src/pdf/page-painter.js";
|
|
4
|
+
import { PdfDocument } from "../../src/pdf/primitives/pdf-document.js";
|
|
5
|
+
import { FontRegistry } from "../../src/pdf/font/font-registry.js";
|
|
6
|
+
describe("header/footer clip overflow", () => {
|
|
7
|
+
function createPainter() {
|
|
8
|
+
const doc = new PdfDocument({});
|
|
9
|
+
const fontRegistry = new FontRegistry(doc, { fontFaces: [] });
|
|
10
|
+
const painter = new PagePainter(300, (v) => v, fontRegistry, 0);
|
|
11
|
+
return { painter, fontRegistry };
|
|
12
|
+
}
|
|
13
|
+
it("clips header rendering to max height when clipOverflow=true", async () => {
|
|
14
|
+
const { painter, fontRegistry } = createPainter();
|
|
15
|
+
await paintHeaderFooter(painter, {
|
|
16
|
+
content: "<div style=\"height:120px;background:#000\"></div>",
|
|
17
|
+
maxHeightPt: 24,
|
|
18
|
+
maxHeightPx: 24,
|
|
19
|
+
}, undefined, new Map(), 1, 1, { fontSizePt: 10 }, true, {
|
|
20
|
+
margins: { top: 10, right: 10, bottom: 10, left: 10 },
|
|
21
|
+
pageWidthPx: 200,
|
|
22
|
+
pageHeightPx: 300,
|
|
23
|
+
fontRegistry,
|
|
24
|
+
pageOffsetY: 0,
|
|
25
|
+
clipOverflow: true,
|
|
26
|
+
});
|
|
27
|
+
expect(painter.result().content).toContain("W n");
|
|
28
|
+
});
|
|
29
|
+
it("does not clip header rendering when clipOverflow=false", async () => {
|
|
30
|
+
const { painter, fontRegistry } = createPainter();
|
|
31
|
+
await paintHeaderFooter(painter, {
|
|
32
|
+
content: "<div style=\"height:120px;background:#000\"></div>",
|
|
33
|
+
maxHeightPt: 24,
|
|
34
|
+
maxHeightPx: 24,
|
|
35
|
+
}, undefined, new Map(), 1, 1, { fontSizePt: 10 }, true, {
|
|
36
|
+
margins: { top: 10, right: 10, bottom: 10, left: 10 },
|
|
37
|
+
pageWidthPx: 200,
|
|
38
|
+
pageHeightPx: 300,
|
|
39
|
+
fontRegistry,
|
|
40
|
+
pageOffsetY: 0,
|
|
41
|
+
clipOverflow: false,
|
|
42
|
+
});
|
|
43
|
+
expect(painter.result().content).not.toContain("W n");
|
|
44
|
+
});
|
|
45
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import path from "node:path";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { prepareHtmlRender, renderHtmlToPdf } from "../../src/html-to-pdf.js";
|
|
4
|
+
import { loadBuiltinFontConfig } from "../../src/pdf/font/builtin-fonts.js";
|
|
5
|
+
import { NodeEnvironment } from "../../src/environment/node-environment.js";
|
|
6
|
+
import { PdfDocument } from "../../src/pdf/primitives/pdf-document.js";
|
|
7
|
+
import { FontRegistry } from "../../src/pdf/font/font-registry.js";
|
|
8
|
+
import { TextFontResolver } from "../../src/pdf/renderers/text-font-resolver.js";
|
|
9
|
+
const ASSET_FONTS_DIR = path.resolve(process.cwd(), "assets/fonts");
|
|
10
|
+
const STACK = "'Selawik', 'DejaVu Sans', 'Arimo', system-ui, -apple-system, 'Segoe UI', Roboto, sans-serif";
|
|
11
|
+
const STRESS_TEXT = "HTML→PDF Stress • sem JS • sem fixed/sticky";
|
|
12
|
+
const SELAWIK_CSS = `
|
|
13
|
+
@font-face {
|
|
14
|
+
font-family: 'Selawik';
|
|
15
|
+
src: url('ttf/selawik/selawkl.ttf') format('truetype');
|
|
16
|
+
font-weight: 300;
|
|
17
|
+
font-style: normal;
|
|
18
|
+
}
|
|
19
|
+
@font-face {
|
|
20
|
+
font-family: 'Selawik';
|
|
21
|
+
src: url('ttf/selawik/selawksl.ttf') format('truetype');
|
|
22
|
+
font-weight: 300;
|
|
23
|
+
font-style: normal;
|
|
24
|
+
}
|
|
25
|
+
@font-face {
|
|
26
|
+
font-family: 'Selawik';
|
|
27
|
+
src: url('ttf/selawik/selawk.ttf') format('truetype');
|
|
28
|
+
font-weight: 400;
|
|
29
|
+
font-style: normal;
|
|
30
|
+
}
|
|
31
|
+
@font-face {
|
|
32
|
+
font-family: 'Selawik';
|
|
33
|
+
src: url('ttf/selawik/selawksb.ttf') format('truetype');
|
|
34
|
+
font-weight: 600;
|
|
35
|
+
font-style: normal;
|
|
36
|
+
}
|
|
37
|
+
@font-face {
|
|
38
|
+
font-family: 'Selawik';
|
|
39
|
+
src: url('ttf/selawik/selawkb.ttf') format('truetype');
|
|
40
|
+
font-weight: 700;
|
|
41
|
+
font-style: normal;
|
|
42
|
+
}
|
|
43
|
+
body {
|
|
44
|
+
font-family: ${STACK};
|
|
45
|
+
}
|
|
46
|
+
`;
|
|
47
|
+
function cloneFontConfig(config) {
|
|
48
|
+
return {
|
|
49
|
+
fontFaceDefs: config.fontFaceDefs.map((face) => ({ ...face })),
|
|
50
|
+
defaultStack: [...(config.defaultStack ?? [])],
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
describe("Selawik opt-in integration", () => {
|
|
54
|
+
it("loads Selawik from @font-face and keeps symbol fallback safe", async () => {
|
|
55
|
+
const environment = new NodeEnvironment(ASSET_FONTS_DIR);
|
|
56
|
+
const builtin = await loadBuiltinFontConfig(environment);
|
|
57
|
+
if (!builtin) {
|
|
58
|
+
throw new Error("Builtin font config is required for this test");
|
|
59
|
+
}
|
|
60
|
+
const fontConfig = cloneFontConfig(builtin);
|
|
61
|
+
await prepareHtmlRender({
|
|
62
|
+
html: `<div>${STRESS_TEXT}</div>`,
|
|
63
|
+
css: SELAWIK_CSS,
|
|
64
|
+
pageWidth: 794,
|
|
65
|
+
pageHeight: 1123,
|
|
66
|
+
viewportWidth: 794,
|
|
67
|
+
viewportHeight: 1123,
|
|
68
|
+
margins: { top: 0, right: 0, bottom: 0, left: 0 },
|
|
69
|
+
fontConfig,
|
|
70
|
+
resourceBaseDir: ASSET_FONTS_DIR,
|
|
71
|
+
assetRootDir: ASSET_FONTS_DIR,
|
|
72
|
+
environment,
|
|
73
|
+
});
|
|
74
|
+
const selawikFaces = fontConfig.fontFaceDefs.filter((face) => face.family === "Selawik");
|
|
75
|
+
expect(selawikFaces.length).toBeGreaterThanOrEqual(5);
|
|
76
|
+
expect(selawikFaces.every((face) => face.data instanceof ArrayBuffer)).toBe(true);
|
|
77
|
+
expect(selawikFaces.map((face) => face.weight)).toEqual(expect.arrayContaining([300, 300, 400, 600, 700]));
|
|
78
|
+
const registry = new FontRegistry(new PdfDocument(), { fontFaces: [] });
|
|
79
|
+
await registry.initializeEmbedder(fontConfig);
|
|
80
|
+
const resolver = new TextFontResolver(registry);
|
|
81
|
+
const resolved = await resolver.ensureFontResource({
|
|
82
|
+
fontFamily: STACK,
|
|
83
|
+
fontWeight: 400,
|
|
84
|
+
fontStyle: "normal",
|
|
85
|
+
text: STRESS_TEXT,
|
|
86
|
+
});
|
|
87
|
+
const arrow = "→".codePointAt(0);
|
|
88
|
+
const bullet = "•".codePointAt(0);
|
|
89
|
+
expect(resolved.metrics).toBeDefined();
|
|
90
|
+
expect(resolved.metrics.cmap.getGlyphId(arrow)).toBeGreaterThan(0);
|
|
91
|
+
expect(resolved.metrics.cmap.getGlyphId(bullet)).toBeGreaterThan(0);
|
|
92
|
+
const pdfBytes = await renderHtmlToPdf({
|
|
93
|
+
html: `<div>${STRESS_TEXT}</div>`,
|
|
94
|
+
css: SELAWIK_CSS,
|
|
95
|
+
pageWidth: 794,
|
|
96
|
+
pageHeight: 1123,
|
|
97
|
+
viewportWidth: 794,
|
|
98
|
+
viewportHeight: 1123,
|
|
99
|
+
margins: { top: 0, right: 0, bottom: 0, left: 0 },
|
|
100
|
+
resourceBaseDir: ASSET_FONTS_DIR,
|
|
101
|
+
assetRootDir: ASSET_FONTS_DIR,
|
|
102
|
+
environment,
|
|
103
|
+
});
|
|
104
|
+
expect(pdfBytes.byteLength).toBeGreaterThan(1024);
|
|
105
|
+
});
|
|
106
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { PDFParse } from "pdf-parse";
|
|
3
|
+
import { renderHtmlToPdf } from "../../src/html-to-pdf.js";
|
|
4
|
+
describe("system-ui fallback subset regression", () => {
|
|
5
|
+
it("keeps text readable when fallback font differs from initial stack resolution", async () => {
|
|
6
|
+
const html = `
|
|
7
|
+
<!DOCTYPE html>
|
|
8
|
+
<html lang="pt-BR">
|
|
9
|
+
<head>
|
|
10
|
+
<meta charset="utf-8" />
|
|
11
|
+
<style>
|
|
12
|
+
:root { --body-font: system-ui, -apple-system, "Segoe UI", Roboto, sans-serif; }
|
|
13
|
+
body { font-family: var(--body-font); margin: 0; }
|
|
14
|
+
.title { font-weight: 700; font-size: 14pt; }
|
|
15
|
+
</style>
|
|
16
|
+
</head>
|
|
17
|
+
<body>
|
|
18
|
+
<div class="title">HTML→PDF Stress</div>
|
|
19
|
+
<div>Fluxo puro • sem JS • sem fixed/sticky</div>
|
|
20
|
+
</body>
|
|
21
|
+
</html>
|
|
22
|
+
`;
|
|
23
|
+
const pdfBytes = await renderHtmlToPdf({
|
|
24
|
+
html,
|
|
25
|
+
pageWidth: 794,
|
|
26
|
+
pageHeight: 1123,
|
|
27
|
+
viewportWidth: 794,
|
|
28
|
+
viewportHeight: 1123,
|
|
29
|
+
margins: { top: 0, right: 0, bottom: 0, left: 0 },
|
|
30
|
+
});
|
|
31
|
+
const parser = new PDFParse({ data: Buffer.from(pdfBytes) });
|
|
32
|
+
const extracted = await parser.getText();
|
|
33
|
+
await parser.destroy();
|
|
34
|
+
const compact = extracted.text.replace(/\s+/g, " ").trim();
|
|
35
|
+
expect(compact).toContain("HTML→PDF");
|
|
36
|
+
expect(compact.toLowerCase()).toContain("stress");
|
|
37
|
+
expect(compact).not.toContain("GSLK");
|
|
38
|
+
});
|
|
39
|
+
});
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { TextRenderer } from "../../src/pdf/renderers/text-renderer.js";
|
|
2
2
|
import { CoordinateTransformer } from "../../src/pdf/utils/coordinate-transformer.js";
|
|
3
3
|
import { TtfFontMetrics } from "../../src/types/fonts.js";
|
|
4
|
+
import { TextFontResolver } from "../../src/pdf/renderers/text-font-resolver.js";
|
|
4
5
|
describe("text renderer glyph fallback", () => {
|
|
5
6
|
it("rebuilds glyph runs when the resolved font differs from the precomputed one", async () => {
|
|
6
7
|
const marker = "\u25E6";
|
|
@@ -103,4 +104,58 @@ describe("text renderer glyph fallback", () => {
|
|
|
103
104
|
expect(run.glyphs?.glyphIds[0]).toBe(1);
|
|
104
105
|
expect(run.glyphs?.font.metrics.cmap).toBe(presentMetrics.cmap);
|
|
105
106
|
});
|
|
107
|
+
it("tries default stack when requested family cannot render unicode glyph", async () => {
|
|
108
|
+
const arrow = "\u2192";
|
|
109
|
+
const arrowCp = arrow.codePointAt(0);
|
|
110
|
+
const baseMetrics = {
|
|
111
|
+
unitsPerEm: 1000,
|
|
112
|
+
ascender: 0,
|
|
113
|
+
descender: 0,
|
|
114
|
+
lineGap: 0,
|
|
115
|
+
capHeight: 0,
|
|
116
|
+
xHeight: 0,
|
|
117
|
+
};
|
|
118
|
+
const glyphMetrics = new Map([
|
|
119
|
+
[0, { advanceWidth: 500, leftSideBearing: 0 }],
|
|
120
|
+
[1, { advanceWidth: 500, leftSideBearing: 0 }],
|
|
121
|
+
]);
|
|
122
|
+
const missingCmap = {
|
|
123
|
+
getGlyphId: (_codePoint) => 0,
|
|
124
|
+
hasCodePoint: (_codePoint) => false,
|
|
125
|
+
unicodeMap: new Map(),
|
|
126
|
+
};
|
|
127
|
+
const presentCmap = {
|
|
128
|
+
getGlyphId: (_codePoint) => 1,
|
|
129
|
+
hasCodePoint: (_codePoint) => true,
|
|
130
|
+
unicodeMap: new Map([[arrowCp, 1]]),
|
|
131
|
+
};
|
|
132
|
+
const primaryFont = {
|
|
133
|
+
baseFont: "PrimaryFont",
|
|
134
|
+
resourceName: "F1",
|
|
135
|
+
ref: { objectNumber: 1 },
|
|
136
|
+
isBase14: false,
|
|
137
|
+
metrics: new TtfFontMetrics(baseMetrics, glyphMetrics, missingCmap),
|
|
138
|
+
};
|
|
139
|
+
const fallbackFont = {
|
|
140
|
+
baseFont: "FallbackFont",
|
|
141
|
+
resourceName: "F2",
|
|
142
|
+
ref: { objectNumber: 2 },
|
|
143
|
+
isBase14: false,
|
|
144
|
+
metrics: new TtfFontMetrics(baseMetrics, glyphMetrics, presentCmap),
|
|
145
|
+
};
|
|
146
|
+
const fontRegistry = {
|
|
147
|
+
ensureFontResource: async (family) => (family === "FallbackFont" ? fallbackFont : primaryFont),
|
|
148
|
+
getDefaultFontStack: () => ["FallbackFont"],
|
|
149
|
+
ensureSubsetForGlyphRun: () => {
|
|
150
|
+
throw new Error("ensureSubsetForGlyphRun should not be called in this test");
|
|
151
|
+
},
|
|
152
|
+
};
|
|
153
|
+
const resolver = new TextFontResolver(fontRegistry);
|
|
154
|
+
const resolved = await resolver.ensureFontResource({
|
|
155
|
+
fontFamily: "PrimaryFont",
|
|
156
|
+
text: `HTML${arrow}PDF`,
|
|
157
|
+
});
|
|
158
|
+
expect(resolved.baseFont).toBe("FallbackFont");
|
|
159
|
+
expect(resolved.metrics?.cmap.getGlyphId(arrowCp)).toBe(1);
|
|
160
|
+
});
|
|
106
161
|
});
|
|
@@ -3,13 +3,14 @@ const TAN_20 = Math.tan((20 * Math.PI) / 180);
|
|
|
3
3
|
describe("PDF text transforms", () => {
|
|
4
4
|
it("applies skewX to text runs (keeps linear components in CSS space)", async () => {
|
|
5
5
|
const html = `<span style="display:inline-block; font-size:20px; transform: skewX(20deg);">Skewed</span>`;
|
|
6
|
-
const
|
|
7
|
-
const
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
expect(
|
|
12
|
-
expect(
|
|
6
|
+
const tree = await renderTreeForHtml(html);
|
|
7
|
+
const boxes = collectBoxes(tree.root);
|
|
8
|
+
const skewedBox = boxes.find((b) => b.transform !== undefined && b.textRuns.some(r => r.text.includes("Skewed")));
|
|
9
|
+
expect(skewedBox).toBeDefined();
|
|
10
|
+
// transform matrix in CSS coordinates from the style; skewX should set c to tan(20deg)
|
|
11
|
+
expect(Math.abs((skewedBox.transform?.c ?? 0) - TAN_20)).toBeLessThan(1e-3);
|
|
12
|
+
expect(skewedBox.transform?.a ?? 1).toBeCloseTo(1);
|
|
13
|
+
expect(skewedBox.transform?.d ?? 1).toBeCloseTo(1);
|
|
13
14
|
});
|
|
14
15
|
it("keeps text baseline inside table cell", async () => {
|
|
15
16
|
const html = `
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "pagyra-js",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.21",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"files": [
|
|
6
6
|
"dist"
|
|
@@ -37,7 +37,7 @@
|
|
|
37
37
|
"./dist/src/environment/node-environment.js": "./dist/src/environment/node-environment.browser.js"
|
|
38
38
|
},
|
|
39
39
|
"scripts": {
|
|
40
|
-
"build": "tsc && tsx scripts/copy-brotli-vendor.ts && tsx scripts/build-browser-bundle.ts --target dist --sourcemap --minify",
|
|
40
|
+
"build": "tsc && tsx scripts/copy-brotli-vendor.ts && tsx scripts/copy-font-assets.ts && tsx scripts/build-browser-bundle.ts --target dist --sourcemap --minify",
|
|
41
41
|
"check": "tsc --noEmit",
|
|
42
42
|
"start": "npm run playground",
|
|
43
43
|
"clean": "node -e \"require('fs').rmSync('dist',{recursive:true,force:true})\"",
|