snap-report-viewer 0.1.0 → 0.1.2
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.
|
@@ -3,16 +3,65 @@ import * as i0 from "@angular/core";
|
|
|
3
3
|
export class PrintService {
|
|
4
4
|
constructor() {
|
|
5
5
|
this.printStyleId = 'rv-print-styles';
|
|
6
|
+
this.hiddenElements = [];
|
|
7
|
+
this.ancestorElements = [];
|
|
6
8
|
}
|
|
7
9
|
print(viewerElement, pageSetup) {
|
|
8
|
-
this.
|
|
10
|
+
this.isolateForPrint(viewerElement);
|
|
11
|
+
this.injectPrintStyles(pageSetup);
|
|
12
|
+
// Cleanup after print dialog closes
|
|
13
|
+
const cleanup = () => {
|
|
14
|
+
this.cleanupIsolation(viewerElement);
|
|
15
|
+
this.cleanupPrintStyles();
|
|
16
|
+
window.removeEventListener('afterprint', cleanup);
|
|
17
|
+
};
|
|
18
|
+
window.addEventListener('afterprint', cleanup);
|
|
9
19
|
window.print();
|
|
10
20
|
}
|
|
11
|
-
|
|
21
|
+
/**
|
|
22
|
+
* Walk from the viewer element up to <body>, hiding all siblings
|
|
23
|
+
* and marking all ancestors so their layout can be reset for print.
|
|
24
|
+
*/
|
|
25
|
+
isolateForPrint(viewerElement) {
|
|
26
|
+
this.hiddenElements = [];
|
|
27
|
+
this.ancestorElements = [];
|
|
28
|
+
let current = viewerElement;
|
|
29
|
+
while (current && current !== document.body) {
|
|
30
|
+
const parent = current.parentElement;
|
|
31
|
+
if (parent) {
|
|
32
|
+
// Mark the ancestor (the parent in the path) for layout reset
|
|
33
|
+
parent.setAttribute('data-rv-print-ancestor', '');
|
|
34
|
+
this.ancestorElements.push(parent);
|
|
35
|
+
for (let i = 0; i < parent.children.length; i++) {
|
|
36
|
+
const sibling = parent.children[i];
|
|
37
|
+
if (sibling !== current && sibling.tagName !== 'STYLE' && sibling.tagName !== 'SCRIPT' && sibling.tagName !== 'LINK') {
|
|
38
|
+
sibling.setAttribute('data-rv-print-hide', '');
|
|
39
|
+
this.hiddenElements.push(sibling);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
current = parent;
|
|
44
|
+
}
|
|
45
|
+
// Also mark body for layout reset
|
|
46
|
+
document.body.setAttribute('data-rv-print-ancestor', '');
|
|
47
|
+
this.ancestorElements.push(document.body);
|
|
48
|
+
// Mark internal viewer chrome (toolbar, sidebars) for hiding
|
|
49
|
+
viewerElement.setAttribute('data-rv-printing', 'true');
|
|
50
|
+
}
|
|
51
|
+
cleanupIsolation(viewerElement) {
|
|
52
|
+
for (const el of this.hiddenElements) {
|
|
53
|
+
el.removeAttribute('data-rv-print-hide');
|
|
54
|
+
}
|
|
55
|
+
for (const el of this.ancestorElements) {
|
|
56
|
+
el.removeAttribute('data-rv-print-ancestor');
|
|
57
|
+
}
|
|
58
|
+
this.hiddenElements = [];
|
|
59
|
+
this.ancestorElements = [];
|
|
60
|
+
viewerElement.removeAttribute('data-rv-printing');
|
|
61
|
+
}
|
|
62
|
+
injectPrintStyles(pageSetup) {
|
|
12
63
|
// Remove existing print styles if any
|
|
13
64
|
this.cleanupPrintStyles();
|
|
14
|
-
// Add a data attribute to identify the viewer for print
|
|
15
|
-
viewerElement.setAttribute('data-rv-printing', 'true');
|
|
16
65
|
const pageRule = this.buildPageRule(pageSetup);
|
|
17
66
|
const style = document.createElement('style');
|
|
18
67
|
style.id = this.printStyleId;
|
|
@@ -20,12 +69,35 @@ export class PrintService {
|
|
|
20
69
|
@media print {
|
|
21
70
|
${pageRule}
|
|
22
71
|
|
|
23
|
-
/* Hide
|
|
24
|
-
|
|
72
|
+
/* Hide all sibling elements along the ancestor path */
|
|
73
|
+
[data-rv-print-hide] {
|
|
25
74
|
display: none !important;
|
|
26
75
|
}
|
|
27
76
|
|
|
28
|
-
/*
|
|
77
|
+
/* Reset all ancestor containers so they don't offset the content */
|
|
78
|
+
[data-rv-print-ancestor] {
|
|
79
|
+
display: block !important;
|
|
80
|
+
margin: 0 !important;
|
|
81
|
+
padding: 0 !important;
|
|
82
|
+
width: 100% !important;
|
|
83
|
+
max-width: 100% !important;
|
|
84
|
+
height: auto !important;
|
|
85
|
+
position: static !important;
|
|
86
|
+
overflow: visible !important;
|
|
87
|
+
float: none !important;
|
|
88
|
+
columns: auto !important;
|
|
89
|
+
column-gap: 0 !important;
|
|
90
|
+
gap: 0 !important;
|
|
91
|
+
transform: none !important;
|
|
92
|
+
grid-template-columns: none !important;
|
|
93
|
+
grid-template-rows: none !important;
|
|
94
|
+
flex: none !important;
|
|
95
|
+
border: none !important;
|
|
96
|
+
box-shadow: none !important;
|
|
97
|
+
background: transparent !important;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/* Hide toolbar and sidebars inside the viewer */
|
|
29
101
|
rv-viewer-toolbar,
|
|
30
102
|
.rv-sidebar-wrapper,
|
|
31
103
|
rv-toc-sidebar,
|
|
@@ -55,6 +127,8 @@ export class PrintService {
|
|
|
55
127
|
overflow: visible !important;
|
|
56
128
|
padding: 0 !important;
|
|
57
129
|
background: white !important;
|
|
130
|
+
display: block !important;
|
|
131
|
+
align-items: initial !important;
|
|
58
132
|
}
|
|
59
133
|
|
|
60
134
|
.rv-page-wrapper {
|
|
@@ -74,13 +148,6 @@ export class PrintService {
|
|
|
74
148
|
}
|
|
75
149
|
`;
|
|
76
150
|
document.head.appendChild(style);
|
|
77
|
-
// Cleanup after print
|
|
78
|
-
const cleanup = () => {
|
|
79
|
-
this.cleanupPrintStyles();
|
|
80
|
-
viewerElement.removeAttribute('data-rv-printing');
|
|
81
|
-
window.removeEventListener('afterprint', cleanup);
|
|
82
|
-
};
|
|
83
|
-
window.addEventListener('afterprint', cleanup);
|
|
84
151
|
}
|
|
85
152
|
buildPageRule(pageSetup) {
|
|
86
153
|
if (!pageSetup) {
|
|
@@ -130,4 +197,4 @@ export class PrintService {
|
|
|
130
197
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrintService, decorators: [{
|
|
131
198
|
type: Injectable
|
|
132
199
|
}] });
|
|
133
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"print.service.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/services/print.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { PageSetup, PAGE_SIZES } from '../models/report-template.model';\r\n\r\n@Injectable()\r\nexport class PrintService {\r\n  private printStyleId = 'rv-print-styles';\r\n\r\n  print(viewerElement: HTMLElement, pageSetup?: PageSetup): void {\r\n    this.injectPrintStyles(viewerElement, pageSetup);\r\n    window.print();\r\n  }\r\n\r\n  private injectPrintStyles(viewerElement: HTMLElement, pageSetup?: PageSetup): void {\r\n    // Remove existing print styles if any\r\n    this.cleanupPrintStyles();\r\n\r\n    // Add a data attribute to identify the viewer for print\r\n    viewerElement.setAttribute('data-rv-printing', 'true');\r\n\r\n    const pageRule = this.buildPageRule(pageSetup);\r\n\r\n    const style = document.createElement('style');\r\n    style.id = this.printStyleId;\r\n    style.textContent = `\r\n      @media print {\r\n        ${pageRule}\r\n\r\n        /* Hide everything except the viewer */\r\n        body > *:not(:has([data-rv-printing])) {\r\n          display: none !important;\r\n        }\r\n\r\n        /* Hide toolbar and sidebars */\r\n        rv-viewer-toolbar,\r\n        .rv-sidebar-wrapper,\r\n        rv-toc-sidebar,\r\n        rv-thumbnail-sidebar,\r\n        .rv-search-highlight {\r\n          display: none !important;\r\n        }\r\n\r\n        /* Reset viewer layout for print */\r\n        [data-rv-printing] {\r\n          position: static !important;\r\n          width: 100% !important;\r\n          height: auto !important;\r\n          overflow: visible !important;\r\n        }\r\n\r\n        .rv-viewer-root {\r\n          height: auto !important;\r\n        }\r\n\r\n        .rv-viewer-body {\r\n          height: auto !important;\r\n          overflow: visible !important;\r\n        }\r\n\r\n        .rv-page-container {\r\n          overflow: visible !important;\r\n          padding: 0 !important;\r\n          background: white !important;\r\n        }\r\n\r\n        .rv-page-wrapper {\r\n          transform: none !important;\r\n          page-break-after: always;\r\n          box-shadow: none !important;\r\n          margin: 0 !important;\r\n        }\r\n\r\n        .rv-page-wrapper:last-child {\r\n          page-break-after: avoid;\r\n        }\r\n\r\n        .rv-page {\r\n          box-shadow: none !important;\r\n        }\r\n      }\r\n    `;\r\n    document.head.appendChild(style);\r\n\r\n    // Cleanup after print\r\n    const cleanup = () => {\r\n      this.cleanupPrintStyles();\r\n      viewerElement.removeAttribute('data-rv-printing');\r\n      window.removeEventListener('afterprint', cleanup);\r\n    };\r\n    window.addEventListener('afterprint', cleanup);\r\n  }\r\n\r\n  private buildPageRule(pageSetup?: PageSetup): string {\r\n    if (!pageSetup) {\r\n      return '@page { size: A4 portrait; margin: 25.4mm; }';\r\n    }\r\n\r\n    // Build the size value\r\n    const sizeValue = this.getPageSizeCSS(pageSetup);\r\n\r\n    // Build the margin value\r\n    const margins = pageSetup.margins;\r\n    const marginValue = `${margins.top}mm ${margins.right}mm ${margins.bottom}mm ${margins.left}mm`;\r\n\r\n    return `@page { size: ${sizeValue}; margin: ${marginValue}; }`;\r\n  }\r\n\r\n  private getPageSizeCSS(pageSetup: PageSetup): string {\r\n    const orientation = pageSetup.orientation === 'Landscape' ? 'landscape' : 'portrait';\r\n\r\n    if (pageSetup.size === 'Custom') {\r\n      // For custom sizes, use explicit width and height in mm\r\n      const width = pageSetup.width || 210;\r\n      const height = pageSetup.height || 297;\r\n      // CSS @page size with explicit dimensions doesn't use orientation keyword;\r\n      // instead we swap width/height for landscape\r\n      if (orientation === 'landscape') {\r\n        return `${Math.max(width, height)}mm ${Math.min(width, height)}mm`;\r\n      }\r\n      return `${Math.min(width, height)}mm ${Math.max(width, height)}mm`;\r\n    }\r\n\r\n    // Named page sizes: A3, A4, A5, letter, legal + orientation\r\n    const sizeMap: Record<string, string> = {\r\n      'A2': 'A3',  // CSS @page doesn't support A2, fall back to closest\r\n      'A3': 'A3',\r\n      'A4': 'A4',\r\n      'A5': 'A5',\r\n      'Letter': 'letter',\r\n      'Legal': 'legal',\r\n    };\r\n\r\n    const cssSize = sizeMap[pageSetup.size] || 'A4';\r\n    return `${cssSize} ${orientation}`;\r\n  }\r\n\r\n  private cleanupPrintStyles(): void {\r\n    const existing = document.getElementById(this.printStyleId);\r\n    if (existing) {\r\n      existing.remove();\r\n    }\r\n  }\r\n}\r\n"]}
|
|
200
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"print.service.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/services/print.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { PageSetup, PAGE_SIZES } from '../models/report-template.model';\r\n\r\n@Injectable()\r\nexport class PrintService {\r\n  private printStyleId = 'rv-print-styles';\r\n  private hiddenElements: HTMLElement[] = [];\r\n  private ancestorElements: HTMLElement[] = [];\r\n\r\n  print(viewerElement: HTMLElement, pageSetup?: PageSetup): void {\r\n    this.isolateForPrint(viewerElement);\r\n    this.injectPrintStyles(pageSetup);\r\n\r\n    // Cleanup after print dialog closes\r\n    const cleanup = () => {\r\n      this.cleanupIsolation(viewerElement);\r\n      this.cleanupPrintStyles();\r\n      window.removeEventListener('afterprint', cleanup);\r\n    };\r\n    window.addEventListener('afterprint', cleanup);\r\n\r\n    window.print();\r\n  }\r\n\r\n  /**\r\n   * Walk from the viewer element up to <body>, hiding all siblings\r\n   * and marking all ancestors so their layout can be reset for print.\r\n   */\r\n  private isolateForPrint(viewerElement: HTMLElement): void {\r\n    this.hiddenElements = [];\r\n    this.ancestorElements = [];\r\n    let current: HTMLElement | null = viewerElement;\r\n\r\n    while (current && current !== document.body) {\r\n      const parent: HTMLElement | null = current.parentElement;\r\n      if (parent) {\r\n        // Mark the ancestor (the parent in the path) for layout reset\r\n        parent.setAttribute('data-rv-print-ancestor', '');\r\n        this.ancestorElements.push(parent);\r\n\r\n        for (let i = 0; i < parent.children.length; i++) {\r\n          const sibling = parent.children[i] as HTMLElement;\r\n          if (sibling !== current && sibling.tagName !== 'STYLE' && sibling.tagName !== 'SCRIPT' && sibling.tagName !== 'LINK') {\r\n            sibling.setAttribute('data-rv-print-hide', '');\r\n            this.hiddenElements.push(sibling);\r\n          }\r\n        }\r\n      }\r\n      current = parent;\r\n    }\r\n\r\n    // Also mark body for layout reset\r\n    document.body.setAttribute('data-rv-print-ancestor', '');\r\n    this.ancestorElements.push(document.body);\r\n\r\n    // Mark internal viewer chrome (toolbar, sidebars) for hiding\r\n    viewerElement.setAttribute('data-rv-printing', 'true');\r\n  }\r\n\r\n  private cleanupIsolation(viewerElement: HTMLElement): void {\r\n    for (const el of this.hiddenElements) {\r\n      el.removeAttribute('data-rv-print-hide');\r\n    }\r\n    for (const el of this.ancestorElements) {\r\n      el.removeAttribute('data-rv-print-ancestor');\r\n    }\r\n    this.hiddenElements = [];\r\n    this.ancestorElements = [];\r\n    viewerElement.removeAttribute('data-rv-printing');\r\n  }\r\n\r\n  private injectPrintStyles(pageSetup?: PageSetup): void {\r\n    // Remove existing print styles if any\r\n    this.cleanupPrintStyles();\r\n\r\n    const pageRule = this.buildPageRule(pageSetup);\r\n\r\n    const style = document.createElement('style');\r\n    style.id = this.printStyleId;\r\n    style.textContent = `\r\n      @media print {\r\n        ${pageRule}\r\n\r\n        /* Hide all sibling elements along the ancestor path */\r\n        [data-rv-print-hide] {\r\n          display: none !important;\r\n        }\r\n\r\n        /* Reset all ancestor containers so they don't offset the content */\r\n        [data-rv-print-ancestor] {\r\n          display: block !important;\r\n          margin: 0 !important;\r\n          padding: 0 !important;\r\n          width: 100% !important;\r\n          max-width: 100% !important;\r\n          height: auto !important;\r\n          position: static !important;\r\n          overflow: visible !important;\r\n          float: none !important;\r\n          columns: auto !important;\r\n          column-gap: 0 !important;\r\n          gap: 0 !important;\r\n          transform: none !important;\r\n          grid-template-columns: none !important;\r\n          grid-template-rows: none !important;\r\n          flex: none !important;\r\n          border: none !important;\r\n          box-shadow: none !important;\r\n          background: transparent !important;\r\n        }\r\n\r\n        /* Hide toolbar and sidebars inside the viewer */\r\n        rv-viewer-toolbar,\r\n        .rv-sidebar-wrapper,\r\n        rv-toc-sidebar,\r\n        rv-thumbnail-sidebar,\r\n        .rv-search-highlight {\r\n          display: none !important;\r\n        }\r\n\r\n        /* Reset viewer layout for print */\r\n        [data-rv-printing] {\r\n          position: static !important;\r\n          width: 100% !important;\r\n          height: auto !important;\r\n          overflow: visible !important;\r\n        }\r\n\r\n        .rv-viewer-root {\r\n          height: auto !important;\r\n        }\r\n\r\n        .rv-viewer-body {\r\n          height: auto !important;\r\n          overflow: visible !important;\r\n        }\r\n\r\n        .rv-page-container {\r\n          overflow: visible !important;\r\n          padding: 0 !important;\r\n          background: white !important;\r\n          display: block !important;\r\n          align-items: initial !important;\r\n        }\r\n\r\n        .rv-page-wrapper {\r\n          transform: none !important;\r\n          page-break-after: always;\r\n          box-shadow: none !important;\r\n          margin: 0 !important;\r\n        }\r\n\r\n        .rv-page-wrapper:last-child {\r\n          page-break-after: avoid;\r\n        }\r\n\r\n        .rv-page {\r\n          box-shadow: none !important;\r\n        }\r\n      }\r\n    `;\r\n    document.head.appendChild(style);\r\n  }\r\n\r\n  private buildPageRule(pageSetup?: PageSetup): string {\r\n    if (!pageSetup) {\r\n      return '@page { size: A4 portrait; margin: 25.4mm; }';\r\n    }\r\n\r\n    // Build the size value\r\n    const sizeValue = this.getPageSizeCSS(pageSetup);\r\n\r\n    // Build the margin value\r\n    const margins = pageSetup.margins;\r\n    const marginValue = `${margins.top}mm ${margins.right}mm ${margins.bottom}mm ${margins.left}mm`;\r\n\r\n    return `@page { size: ${sizeValue}; margin: ${marginValue}; }`;\r\n  }\r\n\r\n  private getPageSizeCSS(pageSetup: PageSetup): string {\r\n    const orientation = pageSetup.orientation === 'Landscape' ? 'landscape' : 'portrait';\r\n\r\n    if (pageSetup.size === 'Custom') {\r\n      // For custom sizes, use explicit width and height in mm\r\n      const width = pageSetup.width || 210;\r\n      const height = pageSetup.height || 297;\r\n      // CSS @page size with explicit dimensions doesn't use orientation keyword;\r\n      // instead we swap width/height for landscape\r\n      if (orientation === 'landscape') {\r\n        return `${Math.max(width, height)}mm ${Math.min(width, height)}mm`;\r\n      }\r\n      return `${Math.min(width, height)}mm ${Math.max(width, height)}mm`;\r\n    }\r\n\r\n    // Named page sizes: A3, A4, A5, letter, legal + orientation\r\n    const sizeMap: Record<string, string> = {\r\n      'A2': 'A3',  // CSS @page doesn't support A2, fall back to closest\r\n      'A3': 'A3',\r\n      'A4': 'A4',\r\n      'A5': 'A5',\r\n      'Letter': 'letter',\r\n      'Legal': 'legal',\r\n    };\r\n\r\n    const cssSize = sizeMap[pageSetup.size] || 'A4';\r\n    return `${cssSize} ${orientation}`;\r\n  }\r\n\r\n  private cleanupPrintStyles(): void {\r\n    const existing = document.getElementById(this.printStyleId);\r\n    if (existing) {\r\n      existing.remove();\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -2280,16 +2280,65 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
2280
2280
|
class PrintService {
|
|
2281
2281
|
constructor() {
|
|
2282
2282
|
this.printStyleId = 'rv-print-styles';
|
|
2283
|
+
this.hiddenElements = [];
|
|
2284
|
+
this.ancestorElements = [];
|
|
2283
2285
|
}
|
|
2284
2286
|
print(viewerElement, pageSetup) {
|
|
2285
|
-
this.
|
|
2287
|
+
this.isolateForPrint(viewerElement);
|
|
2288
|
+
this.injectPrintStyles(pageSetup);
|
|
2289
|
+
// Cleanup after print dialog closes
|
|
2290
|
+
const cleanup = () => {
|
|
2291
|
+
this.cleanupIsolation(viewerElement);
|
|
2292
|
+
this.cleanupPrintStyles();
|
|
2293
|
+
window.removeEventListener('afterprint', cleanup);
|
|
2294
|
+
};
|
|
2295
|
+
window.addEventListener('afterprint', cleanup);
|
|
2286
2296
|
window.print();
|
|
2287
2297
|
}
|
|
2288
|
-
|
|
2298
|
+
/**
|
|
2299
|
+
* Walk from the viewer element up to <body>, hiding all siblings
|
|
2300
|
+
* and marking all ancestors so their layout can be reset for print.
|
|
2301
|
+
*/
|
|
2302
|
+
isolateForPrint(viewerElement) {
|
|
2303
|
+
this.hiddenElements = [];
|
|
2304
|
+
this.ancestorElements = [];
|
|
2305
|
+
let current = viewerElement;
|
|
2306
|
+
while (current && current !== document.body) {
|
|
2307
|
+
const parent = current.parentElement;
|
|
2308
|
+
if (parent) {
|
|
2309
|
+
// Mark the ancestor (the parent in the path) for layout reset
|
|
2310
|
+
parent.setAttribute('data-rv-print-ancestor', '');
|
|
2311
|
+
this.ancestorElements.push(parent);
|
|
2312
|
+
for (let i = 0; i < parent.children.length; i++) {
|
|
2313
|
+
const sibling = parent.children[i];
|
|
2314
|
+
if (sibling !== current && sibling.tagName !== 'STYLE' && sibling.tagName !== 'SCRIPT' && sibling.tagName !== 'LINK') {
|
|
2315
|
+
sibling.setAttribute('data-rv-print-hide', '');
|
|
2316
|
+
this.hiddenElements.push(sibling);
|
|
2317
|
+
}
|
|
2318
|
+
}
|
|
2319
|
+
}
|
|
2320
|
+
current = parent;
|
|
2321
|
+
}
|
|
2322
|
+
// Also mark body for layout reset
|
|
2323
|
+
document.body.setAttribute('data-rv-print-ancestor', '');
|
|
2324
|
+
this.ancestorElements.push(document.body);
|
|
2325
|
+
// Mark internal viewer chrome (toolbar, sidebars) for hiding
|
|
2326
|
+
viewerElement.setAttribute('data-rv-printing', 'true');
|
|
2327
|
+
}
|
|
2328
|
+
cleanupIsolation(viewerElement) {
|
|
2329
|
+
for (const el of this.hiddenElements) {
|
|
2330
|
+
el.removeAttribute('data-rv-print-hide');
|
|
2331
|
+
}
|
|
2332
|
+
for (const el of this.ancestorElements) {
|
|
2333
|
+
el.removeAttribute('data-rv-print-ancestor');
|
|
2334
|
+
}
|
|
2335
|
+
this.hiddenElements = [];
|
|
2336
|
+
this.ancestorElements = [];
|
|
2337
|
+
viewerElement.removeAttribute('data-rv-printing');
|
|
2338
|
+
}
|
|
2339
|
+
injectPrintStyles(pageSetup) {
|
|
2289
2340
|
// Remove existing print styles if any
|
|
2290
2341
|
this.cleanupPrintStyles();
|
|
2291
|
-
// Add a data attribute to identify the viewer for print
|
|
2292
|
-
viewerElement.setAttribute('data-rv-printing', 'true');
|
|
2293
2342
|
const pageRule = this.buildPageRule(pageSetup);
|
|
2294
2343
|
const style = document.createElement('style');
|
|
2295
2344
|
style.id = this.printStyleId;
|
|
@@ -2297,12 +2346,35 @@ class PrintService {
|
|
|
2297
2346
|
@media print {
|
|
2298
2347
|
${pageRule}
|
|
2299
2348
|
|
|
2300
|
-
/* Hide
|
|
2301
|
-
|
|
2349
|
+
/* Hide all sibling elements along the ancestor path */
|
|
2350
|
+
[data-rv-print-hide] {
|
|
2302
2351
|
display: none !important;
|
|
2303
2352
|
}
|
|
2304
2353
|
|
|
2305
|
-
/*
|
|
2354
|
+
/* Reset all ancestor containers so they don't offset the content */
|
|
2355
|
+
[data-rv-print-ancestor] {
|
|
2356
|
+
display: block !important;
|
|
2357
|
+
margin: 0 !important;
|
|
2358
|
+
padding: 0 !important;
|
|
2359
|
+
width: 100% !important;
|
|
2360
|
+
max-width: 100% !important;
|
|
2361
|
+
height: auto !important;
|
|
2362
|
+
position: static !important;
|
|
2363
|
+
overflow: visible !important;
|
|
2364
|
+
float: none !important;
|
|
2365
|
+
columns: auto !important;
|
|
2366
|
+
column-gap: 0 !important;
|
|
2367
|
+
gap: 0 !important;
|
|
2368
|
+
transform: none !important;
|
|
2369
|
+
grid-template-columns: none !important;
|
|
2370
|
+
grid-template-rows: none !important;
|
|
2371
|
+
flex: none !important;
|
|
2372
|
+
border: none !important;
|
|
2373
|
+
box-shadow: none !important;
|
|
2374
|
+
background: transparent !important;
|
|
2375
|
+
}
|
|
2376
|
+
|
|
2377
|
+
/* Hide toolbar and sidebars inside the viewer */
|
|
2306
2378
|
rv-viewer-toolbar,
|
|
2307
2379
|
.rv-sidebar-wrapper,
|
|
2308
2380
|
rv-toc-sidebar,
|
|
@@ -2332,6 +2404,8 @@ class PrintService {
|
|
|
2332
2404
|
overflow: visible !important;
|
|
2333
2405
|
padding: 0 !important;
|
|
2334
2406
|
background: white !important;
|
|
2407
|
+
display: block !important;
|
|
2408
|
+
align-items: initial !important;
|
|
2335
2409
|
}
|
|
2336
2410
|
|
|
2337
2411
|
.rv-page-wrapper {
|
|
@@ -2351,13 +2425,6 @@ class PrintService {
|
|
|
2351
2425
|
}
|
|
2352
2426
|
`;
|
|
2353
2427
|
document.head.appendChild(style);
|
|
2354
|
-
// Cleanup after print
|
|
2355
|
-
const cleanup = () => {
|
|
2356
|
-
this.cleanupPrintStyles();
|
|
2357
|
-
viewerElement.removeAttribute('data-rv-printing');
|
|
2358
|
-
window.removeEventListener('afterprint', cleanup);
|
|
2359
|
-
};
|
|
2360
|
-
window.addEventListener('afterprint', cleanup);
|
|
2361
2428
|
}
|
|
2362
2429
|
buildPageRule(pageSetup) {
|
|
2363
2430
|
if (!pageSetup) {
|