@@ -13,6 +13,7 @@ import {
13
13
pushGraphicsState ,
14
14
translate ,
15
15
LineCapStyle ,
16
+ scale ,
16
17
} from 'src/api/operators' ;
17
18
import PDFDocument from 'src/api/PDFDocument' ;
18
19
import PDFEmbeddedPage from 'src/api/PDFEmbeddedPage' ;
@@ -39,6 +40,8 @@ import {
39
40
PDFOperator ,
40
41
PDFPageLeaf ,
41
42
PDFRef ,
43
+ PDFDict ,
44
+ PDFArray ,
42
45
} from 'src/core' ;
43
46
import {
44
47
addRandomSuffix ,
@@ -566,6 +569,77 @@ export default class PDFPage {
566
569
this . node . wrapContentStreams ( startRef , endRef ) ;
567
570
}
568
571
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
+
569
643
/**
570
644
* Reset the x and y coordinates of this page to `(0, 0)`. This operation is
571
645
* often useful after calling [[translateContent]]. For example:
@@ -1058,16 +1132,16 @@ export default class PDFPage {
1058
1132
1059
1133
// prettier-ignore
1060
1134
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
1064
1138
) ;
1065
1139
1066
1140
// prettier-ignore
1067
1141
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
1071
1145
) ;
1072
1146
1073
1147
const contentStream = this . getContentStream ( ) ;
@@ -1500,4 +1574,20 @@ export default class PDFPage {
1500
1574
1501
1575
return key ;
1502
1576
}
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
+
1503
1593
}
0 commit comments