Skip to content

Commit b57b257

Browse files
JNK90Jan-Niklaas KochHopding
authored
Scale page content (Hopding#991)
* add function scaleContent to scale the content of a page by x and y * scale annots * refactor * run linter * refactor: repalce .forEach() and .asArray() * add integration tests :web * add integration tests :node * Update src/api/PDFPage.ts Co-authored-by: Andrew Dillon <[email protected]> * Update src/api/PDFPage.ts Co-authored-by: Andrew Dillon <[email protected]> * Update src/api/PDFPage.ts Co-authored-by: Andrew Dillon <[email protected]> * Update src/api/PDFPage.ts Co-authored-by: Andrew Dillon <[email protected]> * refactorings, add scale method * revert scratchpad test * remove integration tests 'test19' and 'test20' and add them into 'test3', extend 'test3' for all plattforms * fix merge Co-authored-by: Jan-Niklaas Koch <[email protected]> Co-authored-by: Andrew Dillon <[email protected]>
1 parent 5cb111a commit b57b257

File tree

10 files changed

+186
-34
lines changed

10 files changed

+186
-34
lines changed

apps/deno/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,7 @@ const assets = {
134134
with_xfa_fields: readPdf('with_xfa_fields.pdf'),
135135
fancy_fields: readPdf('fancy_fields.pdf'),
136136
form_to_flatten: readPdf('form_to_flatten.pdf'),
137+
with_annots: readPdf('with_annots.pdf'),
137138
},
138139
};
139140

apps/deno/tests/test3.ts

+18-6
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,19 @@ export default async (assets: Assets) => {
3737

3838
const page0 = pdfDoc.insertPage(0, [305, 250]);
3939
const page1 = pdfDoc.getPage(1);
40-
const page2 = pdfDoc.addPage([305, 250]);
40+
41+
const docWithAnnots = await PDFDocument.load(pdfs.with_annots);
42+
const [page2] = await pdfDoc.copyPages(docWithAnnots, [0]);
43+
page2.scaleContent(0.5, 0.5);
44+
pdfDoc.addPage(page2);
45+
const [page3] = await pdfDoc.copyPages(docWithAnnots, [0]);
46+
page3.scaleAnnotations(0.5, 0.5);
47+
pdfDoc.addPage(page3);
48+
const [page4] = await pdfDoc.copyPages(docWithAnnots, [0]);
49+
page4.scale(0.5, 0.5);
50+
pdfDoc.addPage(page4);
51+
52+
const page5 = pdfDoc.addPage([305, 250]);
4153

4254
const hotPink = rgb(1, 0, 1);
4355
const red = rgb(1, 0, 0);
@@ -76,25 +88,25 @@ export default async (assets: Assets) => {
7688
ySkew: degrees(15),
7789
});
7890

79-
page2.setFontSize(24);
80-
page2.drawText('This is the last page!', {
91+
page5.setFontSize(24);
92+
page5.drawText('This is the last page!', {
8193
x: 30,
8294
y: 215,
8395
font: helveticaFont,
8496
color: hotPink,
8597
});
86-
page2.drawLine({
98+
page5.drawLine({
8799
start: { x: 30, y: 205 },
88100
end: { x: 30 + lastPageTextWidth, y: 205 },
89101
color: hotPink,
90102
thickness: 5,
91103
});
92-
page2.drawImage(cmykImage, {
104+
page5.drawImage(cmykImage, {
93105
...cmykDims,
94106
x: 30,
95107
y: 30,
96108
});
97-
page2.drawLine({
109+
page5.drawLine({
98110
start: { x: 30, y: 240 },
99111
end: { x: 30 + lastPageTextWidth, y: 240 },
100112
color: hotPink,

apps/node/index.ts

+4-3
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,7 @@ const assets = {
132132
with_xfa_fields: readPdf('with_xfa_fields.pdf'),
133133
fancy_fields: readPdf('fancy_fields.pdf'),
134134
form_to_flatten: readPdf('form_to_flatten.pdf'),
135+
with_annots: readPdf('with_annots.pdf'),
135136
},
136137
};
137138

@@ -161,9 +162,9 @@ const main = async () => {
161162

162163
// prettier-ignore
163164
const allTests = [
164-
test1, test2, test3, test4, test5, test6, test7, test8, test9, test10,
165-
test11, test12, test13, test14, test15, test16, test17, test18
166-
];
165+
test1, test2, test3, test4, test5, test6, test7, test8, test9, test10,
166+
test11, test12, test13, test14, test15, test16, test17, test18,
167+
];
167168

168169
const tests = testIdx ? [allTests[testIdx - 1]] : allTests;
169170

apps/node/tests/test3.ts

+18-6
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,19 @@ export default async (assets: Assets) => {
3434

3535
const page0 = pdfDoc.insertPage(0, [305, 250]);
3636
const page1 = pdfDoc.getPage(1);
37-
const page2 = pdfDoc.addPage([305, 250]);
37+
38+
const docWithAnnots = await PDFDocument.load(assets.pdfs.with_annots);
39+
const [page2] = await pdfDoc.copyPages(docWithAnnots, [0]);
40+
page2.scaleContent(0.5, 0.5);
41+
pdfDoc.addPage(page2);
42+
const [page3] = await pdfDoc.copyPages(docWithAnnots, [0]);
43+
page3.scaleAnnotations(0.5, 0.5);
44+
pdfDoc.addPage(page3);
45+
const [page4] = await pdfDoc.copyPages(docWithAnnots, [0]);
46+
page4.scale(0.5, 0.5);
47+
pdfDoc.addPage(page4);
48+
49+
const page5 = pdfDoc.addPage([305, 250]);
3850

3951
const hotPink = rgb(1, 0, 1);
4052
const red = rgb(1, 0, 0);
@@ -73,25 +85,25 @@ export default async (assets: Assets) => {
7385
ySkew: degrees(15),
7486
});
7587

76-
page2.setFontSize(24);
77-
page2.drawText('This is the last page!', {
88+
page5.setFontSize(24);
89+
page5.drawText('This is the last page!', {
7890
x: 30,
7991
y: 215,
8092
font: helveticaFont,
8193
color: hotPink,
8294
});
83-
page2.drawLine({
95+
page5.drawLine({
8496
start: { x: 30, y: 205 },
8597
end: { x: 30 + lastPageTextWidth, y: 205 },
8698
color: hotPink,
8799
thickness: 5,
88100
});
89-
page2.drawImage(cmykImage, {
101+
page5.drawImage(cmykImage, {
90102
...cmykDims,
91103
x: 30,
92104
y: 30,
93105
});
94-
page2.drawLine({
106+
page5.drawLine({
95107
start: { x: 30, y: 240 },
96108
end: { x: 30 + lastPageTextWidth, y: 240 },
97109
color: hotPink,

apps/rn/src/tests/test3.js

+19-6
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,20 @@ export default async () => {
4141

4242
const page0 = pdfDoc.insertPage(0, [305, 250]);
4343
const page1 = pdfDoc.getPage(1);
44-
const page2 = pdfDoc.addPage([305, 250]);
44+
45+
const existingPdfBytes = await fetchBinaryAsset('pdfs/with_annots.pdf');
46+
const docWithAnnots = await PDFDocument.load(existingPdfBytes);
47+
const [page2] = await pdfDoc.copyPages(docWithAnnots, [0]);
48+
page2.scaleContent(0.5, 0.5);
49+
pdfDoc.addPage(page2);
50+
const [page3] = await pdfDoc.copyPages(docWithAnnots, [0]);
51+
page3.scaleAnnotations(0.5, 0.5);
52+
pdfDoc.addPage(page3);
53+
const [page4] = await pdfDoc.copyPages(docWithAnnots, [0]);
54+
page4.scale(0.5, 0.5);
55+
pdfDoc.addPage(page4);
56+
57+
const page5 = pdfDoc.addPage([305, 250]);
4558

4659
const hotPink = rgb(1, 0, 1);
4760
const red = rgb(1, 0, 0);
@@ -80,25 +93,25 @@ export default async () => {
8093
ySkew: degrees(15),
8194
});
8295

83-
page2.setFontSize(24);
84-
page2.drawText('This is the last page!', {
96+
page5.setFontSize(24);
97+
page5.drawText('This is the last page!', {
8598
x: 30,
8699
y: 215,
87100
font: helveticaFont,
88101
color: hotPink,
89102
});
90-
page2.drawLine({
103+
page5.drawLine({
91104
start: { x: 30, y: 205 },
92105
end: { x: 30 + lastPageTextWidth, y: 205 },
93106
color: hotPink,
94107
thickness: 5,
95108
});
96-
page2.drawImage(cmykImage, {
109+
page5.drawImage(cmykImage, {
97110
...cmykDims,
98111
x: 30,
99112
y: 30,
100113
});
101-
page2.drawLine({
114+
page5.drawLine({
102115
start: { x: 30, y: 240 },
103116
end: { x: 30 + lastPageTextWidth, y: 240 },
104117
color: hotPink,

apps/web/test3.html

+19-6
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,20 @@
8282

8383
const page0 = pdfDoc.insertPage(0, [305, 250]);
8484
const page1 = pdfDoc.getPage(1);
85-
const page2 = pdfDoc.addPage([305, 250]);
85+
86+
const existingPdfBytes = await fetchBinaryAsset('pdfs/with_annots.pdf');
87+
const docWithAnnots = await PDFDocument.load(existingPdfBytes);
88+
const [page2] = await pdfDoc.copyPages(docWithAnnots, [0]);
89+
page2.scaleContent(0.5, 0.5);
90+
pdfDoc.addPage(page2);
91+
const [page3] = await pdfDoc.copyPages(docWithAnnots, [0]);
92+
page3.scaleAnnotations(0.5, 0.5);
93+
pdfDoc.addPage(page3);
94+
const [page4] = await pdfDoc.copyPages(docWithAnnots, [0]);
95+
page4.scale(0.5, 0.5);
96+
pdfDoc.addPage(page4);
97+
98+
const page5 = pdfDoc.addPage([305, 250]);
8699

87100
const hotPink = rgb(1, 0, 1);
88101
const red = rgb(1, 0, 0);
@@ -124,25 +137,25 @@
124137
ySkew: degrees(15),
125138
});
126139

127-
page2.setFontSize(24);
128-
page2.drawText('This is the last page!', {
140+
page5.setFontSize(24);
141+
page5.drawText('This is the last page!', {
129142
x: 30,
130143
y: 215,
131144
font: helveticaFont,
132145
color: hotPink,
133146
});
134-
page2.drawLine({
147+
page5.drawLine({
135148
start: { x: 30, y: 205 },
136149
end: { x: 30 + lastPageTextWidth, y: 205 },
137150
color: hotPink,
138151
thickness: 5,
139152
});
140-
page2.drawImage(cmykImage, {
153+
page5.drawImage(cmykImage, {
141154
...cmykDims,
142155
x: 30,
143156
y: 30,
144157
});
145-
page2.drawLine({
158+
page5.drawLine({
146159
start: { x: 30, y: 240 },
147160
end: { x: 30 + lastPageTextWidth, y: 240 },
148161
color: hotPink,

assets/pdfs/with_annots.pdf

74.1 KB
Binary file not shown.

scratchpad/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { PDFDocument } from 'src/index';
77
// This data can be obtained in a number of different ways
88
// If your running in a Node environment, you could use fs.readFile()
99
// In the browser, you could make a fetch() call and use res.arrayBuffer()
10-
const existingPdfBytes = fs.readFileSync('assets/pdfs/with_viewer_prefs.pdf');
10+
const existingPdfBytes = fs.readFileSync('assets/pdfs/with_annots.pdf');
1111

1212
// Load a PDFDocument without updating its existing metadata
1313
const pdfDoc = await PDFDocument.load(existingPdfBytes);

src/api/PDFPage.ts

+96-6
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import {
1313
pushGraphicsState,
1414
translate,
1515
LineCapStyle,
16+
scale,
1617
} from 'src/api/operators';
1718
import PDFDocument from 'src/api/PDFDocument';
1819
import PDFEmbeddedPage from 'src/api/PDFEmbeddedPage';
@@ -39,6 +40,8 @@ import {
3940
PDFOperator,
4041
PDFPageLeaf,
4142
PDFRef,
43+
PDFDict,
44+
PDFArray,
4245
} from 'src/core';
4346
import {
4447
addRandomSuffix,
@@ -566,6 +569,77 @@ export default class PDFPage {
566569
this.node.wrapContentStreams(startRef, endRef);
567570
}
568571

572+
/**
573+
* Scale the size, content and annotations of a page.
574+
* ```js
575+
* p.scale(0.5, 0.5);
576+
* ```
577+
* @param x The factor by wich the width for the page should be scaled (e.g. 0.5 is 50%)
578+
* @param y The factor by wich the height for the page should be scaled (e.g. 0.5 is 50%)
579+
*/
580+
scale(x: number, y: number): void {
581+
assertIs(x, 'x', ['number']);
582+
assertIs(y, 'y', ['number']);
583+
this.setSize(this.getWidth() * x, this.getHeight() * y);
584+
this.scaleContent(x, y);
585+
this.scaleAnnotations(x, y);
586+
}
587+
588+
/**
589+
* Scale the content of a page. This is useful after resizing an exisiting page.
590+
* This scales only the content not the annotations. See also: [[scaleAnnotations]]
591+
* ```js
592+
* // bisect the size of the page
593+
* p.setSize(p.getWidth() / 2, p.getHeight() / 2);
594+
*
595+
* // scale the content of the page down by 50% in x and y
596+
* page.scaleContent(0.5, 0.5);
597+
* ```
598+
* @param x The factor by wich the x-axis for the content should be scaled (e.g. 0.5 is 50%)
599+
* @param y The factor by wich the y-axis for the content should be scaled (e.g. 0.5 is 50%)
600+
*/
601+
scaleContent(x: number, y: number): void {
602+
assertIs(x, 'x', ['number']);
603+
assertIs(y, 'y', ['number']);
604+
605+
this.node.normalize();
606+
this.getContentStream();
607+
608+
const start = this.createContentStream(pushGraphicsState(), scale(x, y));
609+
const startRef = this.doc.context.register(start);
610+
611+
const end = this.createContentStream(popGraphicsState());
612+
const endRef = this.doc.context.register(end);
613+
614+
this.node.wrapContentStreams(startRef, endRef);
615+
}
616+
617+
/**
618+
* Scale the annotations of a page. This is useful if you want to scale a page with comments or other annotations.
619+
* ```js
620+
* // scale the content of the page down by 50% in x and y
621+
* page.scaleContent(0.5, 0.5);
622+
*
623+
* // scale the content of the page down by 50% in x and y
624+
* page.scaleannotations(0.5, 0.5);
625+
* ```
626+
* See also: [[scaleContent]]
627+
* @param x The factor by wich the x-axis for the annotations should be scaled (e.g. 0.5 is 50%)
628+
* @param y The factor by wich the y-axis for the annotations should be scaled (e.g. 0.5 is 50%)
629+
*/
630+
scaleAnnotations(x: number, y: number) {
631+
assertIs(x, 'x', ['number']);
632+
assertIs(y, 'y', ['number']);
633+
const annots = this.node.Annots();
634+
if (!annots) return;
635+
636+
// loop annotations
637+
for (let idx = 0; idx < annots.size(); idx++) {
638+
const annot = annots.lookup(idx);
639+
if (annot instanceof PDFDict) this.scaleAnnot(annot, x, y);
640+
}
641+
}
642+
569643
/**
570644
* Reset the x and y coordinates of this page to `(0, 0)`. This operation is
571645
* often useful after calling [[translateContent]]. For example:
@@ -1058,16 +1132,16 @@ export default class PDFPage {
10581132

10591133
// prettier-ignore
10601134
const xScale = (
1061-
options.width !== undefined ? options.width / embeddedPage.width
1062-
: options.xScale !== undefined ? options.xScale
1063-
: 1
1135+
options.width !== undefined ? options.width / embeddedPage.width
1136+
: options.xScale !== undefined ? options.xScale
1137+
: 1
10641138
);
10651139

10661140
// prettier-ignore
10671141
const yScale = (
1068-
options.height !== undefined ? options.height / embeddedPage.height
1069-
: options.yScale !== undefined ? options.yScale
1070-
: 1
1142+
options.height !== undefined ? options.height / embeddedPage.height
1143+
: options.yScale !== undefined ? options.yScale
1144+
: 1
10711145
);
10721146

10731147
const contentStream = this.getContentStream();
@@ -1500,4 +1574,20 @@ export default class PDFPage {
15001574

15011575
return key;
15021576
}
1577+
1578+
private scaleAnnot(annot: PDFDict, x: number, y: number) {
1579+
const selectors = ['RD', 'CL', 'Vertices', 'QuadPoints', 'L', 'Rect'];
1580+
for (let sel of selectors) {
1581+
const list = annot.get(PDFName.of(sel));
1582+
if (list instanceof PDFArray) list.scalePDFNumbers(x, y);
1583+
}
1584+
1585+
const pdfNameInkList = annot.get(PDFName.of('InkList')) as PDFArray;
1586+
1587+
for (let index = 0; index < pdfNameInkList?.size(); index++) {
1588+
const arr = pdfNameInkList.get(index);
1589+
if (arr instanceof PDFArray) arr.scalePDFNumbers(x, y);
1590+
}
1591+
}
1592+
15031593
}

0 commit comments

Comments
 (0)