snap-report-viewer 0.1.0
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 +24 -0
- package/esm2022/lib/components/chart-renderer/chart-renderer.component.mjs +75 -0
- package/esm2022/lib/components/complex-table-renderer/complex-table-renderer.component.mjs +155 -0
- package/esm2022/lib/components/component-renderer/component-renderer.component.mjs +158 -0
- package/esm2022/lib/components/container-renderer/container-renderer.component.mjs +53 -0
- package/esm2022/lib/components/image-renderer/image-renderer.component.mjs +67 -0
- package/esm2022/lib/components/line-renderer/line-renderer.component.mjs +94 -0
- package/esm2022/lib/components/list-renderer/list-renderer.component.mjs +96 -0
- package/esm2022/lib/components/qrcode-renderer/qrcode-renderer.component.mjs +67 -0
- package/esm2022/lib/components/shape-renderer/shape-renderer.component.mjs +133 -0
- package/esm2022/lib/components/table-renderer/table-renderer.component.mjs +148 -0
- package/esm2022/lib/components/text-renderer/text-renderer.component.mjs +33 -0
- package/esm2022/lib/constants/page-sizes.mjs +3 -0
- package/esm2022/lib/constants/report-tokens.mjs +21 -0
- package/esm2022/lib/layout/report-band.component.mjs +33 -0
- package/esm2022/lib/layout/report-page.component.mjs +159 -0
- package/esm2022/lib/models/component.model.mjs +3 -0
- package/esm2022/lib/models/report-template.model.mjs +19 -0
- package/esm2022/lib/page-container/page-container.component.mjs +248 -0
- package/esm2022/lib/report-viewer.component.mjs +393 -0
- package/esm2022/lib/services/chart-options-builder.service.mjs +749 -0
- package/esm2022/lib/services/data-resolver.service.mjs +385 -0
- package/esm2022/lib/services/export.service.mjs +82 -0
- package/esm2022/lib/services/formatting.service.mjs +59 -0
- package/esm2022/lib/services/print.service.mjs +133 -0
- package/esm2022/lib/services/search.service.mjs +117 -0
- package/esm2022/lib/services/style-mapper.service.mjs +247 -0
- package/esm2022/lib/services/template-normalizer.service.mjs +213 -0
- package/esm2022/lib/services/template-validator.service.mjs +293 -0
- package/esm2022/lib/services/token-resolver.service.mjs +14 -0
- package/esm2022/lib/services/viewer-state.service.mjs +155 -0
- package/esm2022/lib/sidebar/sidebar-container.component.mjs +86 -0
- package/esm2022/lib/sidebar/thumbnail-sidebar.component.mjs +110 -0
- package/esm2022/lib/sidebar/toc-sidebar.component.mjs +155 -0
- package/esm2022/lib/toolbar/viewer-toolbar.component.mjs +486 -0
- package/esm2022/public-api.mjs +43 -0
- package/esm2022/snap-report-viewer.mjs +5 -0
- package/fesm2022/snap-report-viewer.mjs +5110 -0
- package/fesm2022/snap-report-viewer.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/chart-renderer/chart-renderer.component.d.ts +14 -0
- package/lib/components/complex-table-renderer/complex-table-renderer.component.d.ts +17 -0
- package/lib/components/component-renderer/component-renderer.component.d.ts +14 -0
- package/lib/components/container-renderer/container-renderer.component.d.ts +11 -0
- package/lib/components/image-renderer/image-renderer.component.d.ts +14 -0
- package/lib/components/line-renderer/line-renderer.component.d.ts +10 -0
- package/lib/components/list-renderer/list-renderer.component.d.ts +16 -0
- package/lib/components/qrcode-renderer/qrcode-renderer.component.d.ts +16 -0
- package/lib/components/shape-renderer/shape-renderer.component.d.ts +11 -0
- package/lib/components/table-renderer/table-renderer.component.d.ts +20 -0
- package/lib/components/text-renderer/text-renderer.component.d.ts +13 -0
- package/lib/constants/page-sizes.d.ts +2 -0
- package/lib/constants/report-tokens.d.ts +8 -0
- package/lib/layout/report-band.component.d.ts +10 -0
- package/lib/layout/report-page.component.d.ts +21 -0
- package/lib/models/component.model.d.ts +315 -0
- package/lib/models/report-template.model.d.ts +122 -0
- package/lib/page-container/page-container.component.d.ts +29 -0
- package/lib/report-viewer.component.d.ts +51 -0
- package/lib/services/chart-options-builder.service.d.ts +31 -0
- package/lib/services/data-resolver.service.d.ts +27 -0
- package/lib/services/export.service.d.ts +11 -0
- package/lib/services/formatting.service.d.ts +10 -0
- package/lib/services/print.service.d.ts +12 -0
- package/lib/services/search.service.d.ts +13 -0
- package/lib/services/style-mapper.service.d.ts +14 -0
- package/lib/services/template-normalizer.service.d.ts +24 -0
- package/lib/services/template-validator.service.d.ts +19 -0
- package/lib/services/token-resolver.service.d.ts +6 -0
- package/lib/services/viewer-state.service.d.ts +49 -0
- package/lib/sidebar/sidebar-container.component.d.ts +8 -0
- package/lib/sidebar/thumbnail-sidebar.component.d.ts +15 -0
- package/lib/sidebar/toc-sidebar.component.d.ts +17 -0
- package/lib/toolbar/viewer-toolbar.component.d.ts +17 -0
- package/package.json +43 -0
- package/public-api.d.ts +35 -0
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class PrintService {
|
|
4
|
+
constructor() {
|
|
5
|
+
this.printStyleId = 'rv-print-styles';
|
|
6
|
+
}
|
|
7
|
+
print(viewerElement, pageSetup) {
|
|
8
|
+
this.injectPrintStyles(viewerElement, pageSetup);
|
|
9
|
+
window.print();
|
|
10
|
+
}
|
|
11
|
+
injectPrintStyles(viewerElement, pageSetup) {
|
|
12
|
+
// Remove existing print styles if any
|
|
13
|
+
this.cleanupPrintStyles();
|
|
14
|
+
// Add a data attribute to identify the viewer for print
|
|
15
|
+
viewerElement.setAttribute('data-rv-printing', 'true');
|
|
16
|
+
const pageRule = this.buildPageRule(pageSetup);
|
|
17
|
+
const style = document.createElement('style');
|
|
18
|
+
style.id = this.printStyleId;
|
|
19
|
+
style.textContent = `
|
|
20
|
+
@media print {
|
|
21
|
+
${pageRule}
|
|
22
|
+
|
|
23
|
+
/* Hide everything except the viewer */
|
|
24
|
+
body > *:not(:has([data-rv-printing])) {
|
|
25
|
+
display: none !important;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/* Hide toolbar and sidebars */
|
|
29
|
+
rv-viewer-toolbar,
|
|
30
|
+
.rv-sidebar-wrapper,
|
|
31
|
+
rv-toc-sidebar,
|
|
32
|
+
rv-thumbnail-sidebar,
|
|
33
|
+
.rv-search-highlight {
|
|
34
|
+
display: none !important;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/* Reset viewer layout for print */
|
|
38
|
+
[data-rv-printing] {
|
|
39
|
+
position: static !important;
|
|
40
|
+
width: 100% !important;
|
|
41
|
+
height: auto !important;
|
|
42
|
+
overflow: visible !important;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
.rv-viewer-root {
|
|
46
|
+
height: auto !important;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
.rv-viewer-body {
|
|
50
|
+
height: auto !important;
|
|
51
|
+
overflow: visible !important;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
.rv-page-container {
|
|
55
|
+
overflow: visible !important;
|
|
56
|
+
padding: 0 !important;
|
|
57
|
+
background: white !important;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
.rv-page-wrapper {
|
|
61
|
+
transform: none !important;
|
|
62
|
+
page-break-after: always;
|
|
63
|
+
box-shadow: none !important;
|
|
64
|
+
margin: 0 !important;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
.rv-page-wrapper:last-child {
|
|
68
|
+
page-break-after: avoid;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
.rv-page {
|
|
72
|
+
box-shadow: none !important;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
`;
|
|
76
|
+
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
|
+
}
|
|
85
|
+
buildPageRule(pageSetup) {
|
|
86
|
+
if (!pageSetup) {
|
|
87
|
+
return '@page { size: A4 portrait; margin: 25.4mm; }';
|
|
88
|
+
}
|
|
89
|
+
// Build the size value
|
|
90
|
+
const sizeValue = this.getPageSizeCSS(pageSetup);
|
|
91
|
+
// Build the margin value
|
|
92
|
+
const margins = pageSetup.margins;
|
|
93
|
+
const marginValue = `${margins.top}mm ${margins.right}mm ${margins.bottom}mm ${margins.left}mm`;
|
|
94
|
+
return `@page { size: ${sizeValue}; margin: ${marginValue}; }`;
|
|
95
|
+
}
|
|
96
|
+
getPageSizeCSS(pageSetup) {
|
|
97
|
+
const orientation = pageSetup.orientation === 'Landscape' ? 'landscape' : 'portrait';
|
|
98
|
+
if (pageSetup.size === 'Custom') {
|
|
99
|
+
// For custom sizes, use explicit width and height in mm
|
|
100
|
+
const width = pageSetup.width || 210;
|
|
101
|
+
const height = pageSetup.height || 297;
|
|
102
|
+
// CSS @page size with explicit dimensions doesn't use orientation keyword;
|
|
103
|
+
// instead we swap width/height for landscape
|
|
104
|
+
if (orientation === 'landscape') {
|
|
105
|
+
return `${Math.max(width, height)}mm ${Math.min(width, height)}mm`;
|
|
106
|
+
}
|
|
107
|
+
return `${Math.min(width, height)}mm ${Math.max(width, height)}mm`;
|
|
108
|
+
}
|
|
109
|
+
// Named page sizes: A3, A4, A5, letter, legal + orientation
|
|
110
|
+
const sizeMap = {
|
|
111
|
+
'A2': 'A3', // CSS @page doesn't support A2, fall back to closest
|
|
112
|
+
'A3': 'A3',
|
|
113
|
+
'A4': 'A4',
|
|
114
|
+
'A5': 'A5',
|
|
115
|
+
'Letter': 'letter',
|
|
116
|
+
'Legal': 'legal',
|
|
117
|
+
};
|
|
118
|
+
const cssSize = sizeMap[pageSetup.size] || 'A4';
|
|
119
|
+
return `${cssSize} ${orientation}`;
|
|
120
|
+
}
|
|
121
|
+
cleanupPrintStyles() {
|
|
122
|
+
const existing = document.getElementById(this.printStyleId);
|
|
123
|
+
if (existing) {
|
|
124
|
+
existing.remove();
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrintService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
128
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrintService }); }
|
|
129
|
+
}
|
|
130
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PrintService, decorators: [{
|
|
131
|
+
type: Injectable
|
|
132
|
+
}] });
|
|
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"]}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class SearchService {
|
|
4
|
+
constructor() {
|
|
5
|
+
this.highlights = [];
|
|
6
|
+
}
|
|
7
|
+
search(query, container) {
|
|
8
|
+
this.clearHighlights();
|
|
9
|
+
if (!query || !query.trim())
|
|
10
|
+
return [];
|
|
11
|
+
const matches = [];
|
|
12
|
+
const escapedQuery = this.escapeRegex(query);
|
|
13
|
+
const regex = new RegExp(escapedQuery, 'gi');
|
|
14
|
+
const walker = document.createTreeWalker(container, NodeFilter.SHOW_TEXT, {
|
|
15
|
+
acceptNode: (node) => {
|
|
16
|
+
// Skip nodes inside marks (our own highlights) or script/style tags
|
|
17
|
+
const parent = node.parentElement;
|
|
18
|
+
if (!parent)
|
|
19
|
+
return NodeFilter.FILTER_REJECT;
|
|
20
|
+
const tag = parent.tagName?.toLowerCase();
|
|
21
|
+
if (tag === 'script' || tag === 'style' || tag === 'mark') {
|
|
22
|
+
return NodeFilter.FILTER_REJECT;
|
|
23
|
+
}
|
|
24
|
+
return (node.textContent && node.textContent.trim().length > 0)
|
|
25
|
+
? NodeFilter.FILTER_ACCEPT
|
|
26
|
+
: NodeFilter.FILTER_REJECT;
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
let textNode;
|
|
30
|
+
while ((textNode = walker.nextNode())) {
|
|
31
|
+
const text = textNode.textContent || '';
|
|
32
|
+
let match;
|
|
33
|
+
regex.lastIndex = 0;
|
|
34
|
+
while ((match = regex.exec(text)) !== null) {
|
|
35
|
+
const pageIndex = this.getPageIndex(textNode);
|
|
36
|
+
matches.push({
|
|
37
|
+
textNode,
|
|
38
|
+
startOffset: match.index,
|
|
39
|
+
endOffset: match.index + match[0].length,
|
|
40
|
+
pageIndex,
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
this.applyHighlights(matches);
|
|
45
|
+
return matches;
|
|
46
|
+
}
|
|
47
|
+
highlightCurrentMatch(index) {
|
|
48
|
+
// Reverse the stored order since we apply in reverse
|
|
49
|
+
const reversed = [...this.highlights].reverse();
|
|
50
|
+
reversed.forEach((mark, i) => {
|
|
51
|
+
if (i === index) {
|
|
52
|
+
mark.classList.add('rv-search-active');
|
|
53
|
+
mark.scrollIntoView({ behavior: 'smooth', block: 'center' });
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
mark.classList.remove('rv-search-active');
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
clearHighlights() {
|
|
61
|
+
this.highlights.forEach(mark => {
|
|
62
|
+
const parent = mark.parentNode;
|
|
63
|
+
if (parent) {
|
|
64
|
+
const textNode = document.createTextNode(mark.textContent || '');
|
|
65
|
+
parent.replaceChild(textNode, mark);
|
|
66
|
+
parent.normalize();
|
|
67
|
+
}
|
|
68
|
+
});
|
|
69
|
+
this.highlights = [];
|
|
70
|
+
}
|
|
71
|
+
applyHighlights(matches) {
|
|
72
|
+
// Group matches by text node
|
|
73
|
+
const nodeMap = new Map();
|
|
74
|
+
for (const match of matches) {
|
|
75
|
+
const existing = nodeMap.get(match.textNode) || [];
|
|
76
|
+
existing.push(match);
|
|
77
|
+
nodeMap.set(match.textNode, existing);
|
|
78
|
+
}
|
|
79
|
+
// Process each node. Process offsets in reverse to preserve earlier offsets.
|
|
80
|
+
for (const [textNode, nodeMatches] of nodeMap) {
|
|
81
|
+
const sorted = [...nodeMatches].sort((a, b) => b.startOffset - a.startOffset);
|
|
82
|
+
for (const m of sorted) {
|
|
83
|
+
try {
|
|
84
|
+
const range = document.createRange();
|
|
85
|
+
range.setStart(textNode.parentNode ? textNode : textNode, m.startOffset);
|
|
86
|
+
range.setEnd(textNode.parentNode ? textNode : textNode, m.endOffset);
|
|
87
|
+
const mark = document.createElement('mark');
|
|
88
|
+
mark.className = 'rv-search-highlight';
|
|
89
|
+
range.surroundContents(mark);
|
|
90
|
+
this.highlights.push(mark);
|
|
91
|
+
}
|
|
92
|
+
catch {
|
|
93
|
+
// Range might be invalid if DOM changed; skip silently
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
getPageIndex(node) {
|
|
99
|
+
let current = node;
|
|
100
|
+
while (current) {
|
|
101
|
+
if (current instanceof HTMLElement && current.hasAttribute('data-page-index')) {
|
|
102
|
+
return parseInt(current.getAttribute('data-page-index'), 10);
|
|
103
|
+
}
|
|
104
|
+
current = current.parentNode;
|
|
105
|
+
}
|
|
106
|
+
return 0;
|
|
107
|
+
}
|
|
108
|
+
escapeRegex(str) {
|
|
109
|
+
return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
110
|
+
}
|
|
111
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SearchService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
112
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SearchService }); }
|
|
113
|
+
}
|
|
114
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SearchService, decorators: [{
|
|
115
|
+
type: Injectable
|
|
116
|
+
}] });
|
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search.service.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/services/search.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { SearchMatch } from '../models/report-template.model';\r\n\r\n@Injectable()\r\nexport class SearchService {\r\n  private highlights: HTMLElement[] = [];\r\n\r\n  search(query: string, container: HTMLElement): SearchMatch[] {\r\n    this.clearHighlights();\r\n    if (!query || !query.trim()) return [];\r\n\r\n    const matches: SearchMatch[] = [];\r\n    const escapedQuery = this.escapeRegex(query);\r\n    const regex = new RegExp(escapedQuery, 'gi');\r\n\r\n    const walker = document.createTreeWalker(\r\n      container,\r\n      NodeFilter.SHOW_TEXT,\r\n      {\r\n        acceptNode: (node: Text) => {\r\n          // Skip nodes inside marks (our own highlights) or script/style tags\r\n          const parent = node.parentElement;\r\n          if (!parent) return NodeFilter.FILTER_REJECT;\r\n          const tag = parent.tagName?.toLowerCase();\r\n          if (tag === 'script' || tag === 'style' || tag === 'mark') {\r\n            return NodeFilter.FILTER_REJECT;\r\n          }\r\n          return (node.textContent && node.textContent.trim().length > 0)\r\n            ? NodeFilter.FILTER_ACCEPT\r\n            : NodeFilter.FILTER_REJECT;\r\n        }\r\n      }\r\n    );\r\n\r\n    let textNode: Text | null;\r\n    while ((textNode = walker.nextNode() as Text | null)) {\r\n      const text = textNode.textContent || '';\r\n      let match: RegExpExecArray | null;\r\n      regex.lastIndex = 0;\r\n\r\n      while ((match = regex.exec(text)) !== null) {\r\n        const pageIndex = this.getPageIndex(textNode);\r\n        matches.push({\r\n          textNode,\r\n          startOffset: match.index,\r\n          endOffset: match.index + match[0].length,\r\n          pageIndex,\r\n        });\r\n      }\r\n    }\r\n\r\n    this.applyHighlights(matches);\r\n    return matches;\r\n  }\r\n\r\n  highlightCurrentMatch(index: number): void {\r\n    // Reverse the stored order since we apply in reverse\r\n    const reversed = [...this.highlights].reverse();\r\n    reversed.forEach((mark, i) => {\r\n      if (i === index) {\r\n        mark.classList.add('rv-search-active');\r\n        mark.scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n      } else {\r\n        mark.classList.remove('rv-search-active');\r\n      }\r\n    });\r\n  }\r\n\r\n  clearHighlights(): void {\r\n    this.highlights.forEach(mark => {\r\n      const parent = mark.parentNode;\r\n      if (parent) {\r\n        const textNode = document.createTextNode(mark.textContent || '');\r\n        parent.replaceChild(textNode, mark);\r\n        parent.normalize();\r\n      }\r\n    });\r\n    this.highlights = [];\r\n  }\r\n\r\n  private applyHighlights(matches: SearchMatch[]): void {\r\n    // Group matches by text node\r\n    const nodeMap = new Map<Text, SearchMatch[]>();\r\n    for (const match of matches) {\r\n      const existing = nodeMap.get(match.textNode) || [];\r\n      existing.push(match);\r\n      nodeMap.set(match.textNode, existing);\r\n    }\r\n\r\n    // Process each node. Process offsets in reverse to preserve earlier offsets.\r\n    for (const [textNode, nodeMatches] of nodeMap) {\r\n      const sorted = [...nodeMatches].sort((a, b) => b.startOffset - a.startOffset);\r\n      for (const m of sorted) {\r\n        try {\r\n          const range = document.createRange();\r\n          range.setStart(textNode.parentNode ? textNode : textNode, m.startOffset);\r\n          range.setEnd(textNode.parentNode ? textNode : textNode, m.endOffset);\r\n\r\n          const mark = document.createElement('mark');\r\n          mark.className = 'rv-search-highlight';\r\n          range.surroundContents(mark);\r\n          this.highlights.push(mark);\r\n        } catch {\r\n          // Range might be invalid if DOM changed; skip silently\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  private getPageIndex(node: Node): number {\r\n    let current: Node | null = node;\r\n    while (current) {\r\n      if (current instanceof HTMLElement && current.hasAttribute('data-page-index')) {\r\n        return parseInt(current.getAttribute('data-page-index')!, 10);\r\n      }\r\n      current = current.parentNode;\r\n    }\r\n    return 0;\r\n  }\r\n\r\n  private escapeRegex(str: string): string {\r\n    return str.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,247 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class StyleMapperService {
|
|
4
|
+
mapComponentStyles(style) {
|
|
5
|
+
const result = {};
|
|
6
|
+
// Typography
|
|
7
|
+
if (style.fontFamily)
|
|
8
|
+
result['font-family'] = style.fontFamily;
|
|
9
|
+
if (style.fontSize)
|
|
10
|
+
result['font-size'] = style.fontSize;
|
|
11
|
+
if (style.fontWeight)
|
|
12
|
+
result['font-weight'] = style.fontWeight;
|
|
13
|
+
if (style.fontStyle)
|
|
14
|
+
result['font-style'] = style.fontStyle;
|
|
15
|
+
if (style.textDecoration)
|
|
16
|
+
result['text-decoration'] = style.textDecoration;
|
|
17
|
+
if (style.lineHeight)
|
|
18
|
+
result['line-height'] = style.lineHeight;
|
|
19
|
+
if (style.letterSpacing)
|
|
20
|
+
result['letter-spacing'] = style.letterSpacing;
|
|
21
|
+
// Colors
|
|
22
|
+
if (style.color)
|
|
23
|
+
result['color'] = style.color;
|
|
24
|
+
if (style.backgroundColor)
|
|
25
|
+
result['background-color'] = style.backgroundColor;
|
|
26
|
+
// Alignment
|
|
27
|
+
if (style.textAlign)
|
|
28
|
+
result['text-align'] = style.textAlign;
|
|
29
|
+
// Spacing
|
|
30
|
+
if (style.padding)
|
|
31
|
+
result['padding'] = style.padding;
|
|
32
|
+
if (style.paddingTop)
|
|
33
|
+
result['padding-top'] = style.paddingTop;
|
|
34
|
+
if (style.paddingRight)
|
|
35
|
+
result['padding-right'] = style.paddingRight;
|
|
36
|
+
if (style.paddingBottom)
|
|
37
|
+
result['padding-bottom'] = style.paddingBottom;
|
|
38
|
+
if (style.paddingLeft)
|
|
39
|
+
result['padding-left'] = style.paddingLeft;
|
|
40
|
+
// Borders
|
|
41
|
+
if (style.border)
|
|
42
|
+
result['border'] = style.border;
|
|
43
|
+
if (style.borderWidth)
|
|
44
|
+
result['border-width'] = style.borderWidth;
|
|
45
|
+
if (style.borderStyle)
|
|
46
|
+
result['border-style'] = style.borderStyle;
|
|
47
|
+
if (style.borderColor)
|
|
48
|
+
result['border-color'] = style.borderColor;
|
|
49
|
+
if (style.borderRadius)
|
|
50
|
+
result['border-radius'] = style.borderRadius;
|
|
51
|
+
return result;
|
|
52
|
+
}
|
|
53
|
+
mapTableStyle(component) {
|
|
54
|
+
const result = {};
|
|
55
|
+
const config = component.tableConfig;
|
|
56
|
+
if (config?.showOuterBorder !== false) {
|
|
57
|
+
const width = config?.outerBorderWidth || '1px';
|
|
58
|
+
const color = config?.outerBorderColor || '#cccccc';
|
|
59
|
+
const style = config?.outerBorderStyle || 'solid';
|
|
60
|
+
result['border'] = `${width} ${style} ${color}`;
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
63
|
+
result['border'] = 'none';
|
|
64
|
+
}
|
|
65
|
+
if (config?.tableLayout) {
|
|
66
|
+
result['table-layout'] = config.tableLayout;
|
|
67
|
+
}
|
|
68
|
+
return result;
|
|
69
|
+
}
|
|
70
|
+
mapHeaderStyle(component, colIndex, col) {
|
|
71
|
+
const result = {};
|
|
72
|
+
const config = component.tableConfig;
|
|
73
|
+
if (config?.headerBackground) {
|
|
74
|
+
result['background-color'] = config.headerBackground;
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
result['background-color'] = '#f5f5f5';
|
|
78
|
+
}
|
|
79
|
+
if (config?.headerTextColor) {
|
|
80
|
+
result['color'] = config.headerTextColor;
|
|
81
|
+
}
|
|
82
|
+
if (config?.headerFontWeight) {
|
|
83
|
+
result['font-weight'] = config.headerFontWeight;
|
|
84
|
+
}
|
|
85
|
+
else {
|
|
86
|
+
result['font-weight'] = 'bold';
|
|
87
|
+
}
|
|
88
|
+
if (config?.headerFontSize) {
|
|
89
|
+
result['font-size'] = config.headerFontSize;
|
|
90
|
+
}
|
|
91
|
+
if (config?.headerTextAlign) {
|
|
92
|
+
result['text-align'] = config.headerTextAlign;
|
|
93
|
+
}
|
|
94
|
+
else if (col?.style?.textAlign) {
|
|
95
|
+
result['text-align'] = col.style.textAlign;
|
|
96
|
+
}
|
|
97
|
+
if (config?.headerPadding) {
|
|
98
|
+
result['padding'] = config.headerPadding;
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
result['padding'] = '8px';
|
|
102
|
+
}
|
|
103
|
+
if (config?.headerBorderBottom) {
|
|
104
|
+
result['border-bottom'] = config.headerBorderBottom;
|
|
105
|
+
}
|
|
106
|
+
if (col?.width) {
|
|
107
|
+
result['width'] = col.width;
|
|
108
|
+
}
|
|
109
|
+
const totalCols = component.columns?.length || 0;
|
|
110
|
+
if (config?.showColumnSeparators !== false && colIndex < totalCols - 1) {
|
|
111
|
+
const sepWidth = config?.columnSeparatorWidth || '1px';
|
|
112
|
+
const sepColor = config?.columnSeparatorColor || '#e0e0e0';
|
|
113
|
+
const sepStyle = config?.columnSeparatorStyle || 'solid';
|
|
114
|
+
result['border-right'] = `${sepWidth} ${sepStyle} ${sepColor}`;
|
|
115
|
+
}
|
|
116
|
+
return result;
|
|
117
|
+
}
|
|
118
|
+
mapRowStyle(component, rowIndex, totalRows) {
|
|
119
|
+
const result = {};
|
|
120
|
+
const config = component.tableConfig;
|
|
121
|
+
if (config?.highlightFirstRow && rowIndex === 0) {
|
|
122
|
+
result['background-color'] = config.firstRowBackground || '#e8f5e9';
|
|
123
|
+
}
|
|
124
|
+
else if (config?.highlightLastRow && rowIndex === totalRows - 1) {
|
|
125
|
+
result['background-color'] = config.lastRowBackground || '#fff3e0';
|
|
126
|
+
}
|
|
127
|
+
else if (config?.enableAlternateRows) {
|
|
128
|
+
if (rowIndex % 2 === 0) {
|
|
129
|
+
result['background-color'] = config.evenRowBackground || '#ffffff';
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
result['background-color'] = config.oddRowBackground || '#f9f9f9';
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
if (config?.showRowSeparators !== false && rowIndex < totalRows - 1) {
|
|
136
|
+
const sepWidth = config?.rowSeparatorWidth || '1px';
|
|
137
|
+
const sepColor = config?.rowSeparatorColor || '#e0e0e0';
|
|
138
|
+
const sepStyle = config?.rowSeparatorStyle || 'solid';
|
|
139
|
+
result['border-bottom'] = `${sepWidth} ${sepStyle} ${sepColor}`;
|
|
140
|
+
}
|
|
141
|
+
return result;
|
|
142
|
+
}
|
|
143
|
+
mapCellStyle(component, colIndex, col) {
|
|
144
|
+
const result = {};
|
|
145
|
+
const config = component.tableConfig;
|
|
146
|
+
if (config?.cellPadding) {
|
|
147
|
+
result['padding'] = config.cellPadding;
|
|
148
|
+
}
|
|
149
|
+
else if (col?.style?.padding) {
|
|
150
|
+
result['padding'] = col.style.padding;
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
result['padding'] = '8px';
|
|
154
|
+
}
|
|
155
|
+
if (config?.cellVerticalAlign) {
|
|
156
|
+
result['vertical-align'] = config.cellVerticalAlign;
|
|
157
|
+
}
|
|
158
|
+
if (col?.width) {
|
|
159
|
+
result['width'] = col.width;
|
|
160
|
+
}
|
|
161
|
+
if (col?.style?.textAlign) {
|
|
162
|
+
result['text-align'] = col.style.textAlign;
|
|
163
|
+
}
|
|
164
|
+
if (col?.style?.fontWeight) {
|
|
165
|
+
result['font-weight'] = col.style.fontWeight;
|
|
166
|
+
}
|
|
167
|
+
if (col?.style?.color) {
|
|
168
|
+
result['color'] = col.style.color;
|
|
169
|
+
}
|
|
170
|
+
const totalCols = component.columns?.length || 0;
|
|
171
|
+
if (config?.showColumnSeparators !== false && colIndex < totalCols - 1) {
|
|
172
|
+
const sepWidth = config?.columnSeparatorWidth || '1px';
|
|
173
|
+
const sepColor = config?.columnSeparatorColor || '#e0e0e0';
|
|
174
|
+
const sepStyle = config?.columnSeparatorStyle || 'solid';
|
|
175
|
+
result['border-right'] = `${sepWidth} ${sepStyle} ${sepColor}`;
|
|
176
|
+
}
|
|
177
|
+
return result;
|
|
178
|
+
}
|
|
179
|
+
mapBandStyle(section) {
|
|
180
|
+
const result = {};
|
|
181
|
+
if (section.style?.backgroundColor) {
|
|
182
|
+
result['background-color'] = section.style.backgroundColor;
|
|
183
|
+
}
|
|
184
|
+
if (section.style?.borderBottom) {
|
|
185
|
+
result['border-bottom'] = section.style.borderBottom;
|
|
186
|
+
}
|
|
187
|
+
if (section.style?.borderTop) {
|
|
188
|
+
result['border-top'] = section.style.borderTop;
|
|
189
|
+
}
|
|
190
|
+
return result;
|
|
191
|
+
}
|
|
192
|
+
mapComplexCellStyle(cell, config, isHeader) {
|
|
193
|
+
const result = {};
|
|
194
|
+
// Grid lines
|
|
195
|
+
if (config.showGridLines !== false) {
|
|
196
|
+
const w = config.gridLineWidth || '1px';
|
|
197
|
+
const c = config.gridLineColor || '#e0e0e0';
|
|
198
|
+
const s = config.gridLineStyle || 'solid';
|
|
199
|
+
result['border'] = `${w} ${s} ${c}`;
|
|
200
|
+
}
|
|
201
|
+
// Header/body defaults
|
|
202
|
+
if (isHeader) {
|
|
203
|
+
if (config.headerBackground)
|
|
204
|
+
result['background-color'] = config.headerBackground;
|
|
205
|
+
if (config.headerTextColor)
|
|
206
|
+
result['color'] = config.headerTextColor;
|
|
207
|
+
if (config.headerFontWeight)
|
|
208
|
+
result['font-weight'] = config.headerFontWeight;
|
|
209
|
+
if (config.headerFontSize)
|
|
210
|
+
result['font-size'] = config.headerFontSize;
|
|
211
|
+
if (config.headerTextAlign)
|
|
212
|
+
result['text-align'] = config.headerTextAlign;
|
|
213
|
+
if (config.headerPadding)
|
|
214
|
+
result['padding'] = config.headerPadding;
|
|
215
|
+
}
|
|
216
|
+
else {
|
|
217
|
+
if (config.cellPadding)
|
|
218
|
+
result['padding'] = config.cellPadding;
|
|
219
|
+
if (config.cellTextAlign)
|
|
220
|
+
result['text-align'] = config.cellTextAlign;
|
|
221
|
+
}
|
|
222
|
+
if (config.cellVerticalAlign)
|
|
223
|
+
result['vertical-align'] = config.cellVerticalAlign;
|
|
224
|
+
// Per-cell style overrides
|
|
225
|
+
if (cell.style) {
|
|
226
|
+
if (cell.style.backgroundColor)
|
|
227
|
+
result['background-color'] = cell.style.backgroundColor;
|
|
228
|
+
if (cell.style.color)
|
|
229
|
+
result['color'] = cell.style.color;
|
|
230
|
+
if (cell.style.fontWeight)
|
|
231
|
+
result['font-weight'] = cell.style.fontWeight;
|
|
232
|
+
if (cell.style.fontSize)
|
|
233
|
+
result['font-size'] = cell.style.fontSize;
|
|
234
|
+
if (cell.style.textAlign)
|
|
235
|
+
result['text-align'] = cell.style.textAlign;
|
|
236
|
+
if (cell.style.padding)
|
|
237
|
+
result['padding'] = cell.style.padding;
|
|
238
|
+
}
|
|
239
|
+
return result;
|
|
240
|
+
}
|
|
241
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StyleMapperService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
242
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StyleMapperService }); }
|
|
243
|
+
}
|
|
244
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: StyleMapperService, decorators: [{
|
|
245
|
+
type: Injectable
|
|
246
|
+
}] });
|
|
247
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"style-mapper.service.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/services/style-mapper.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ComponentModel, StyleProperties, TableColumn, ComplexTableCell, ComplexTableConfig } from '../models/component.model';\nimport { ReportSection } from '../models/report-template.model';\n\n@Injectable()\nexport class StyleMapperService {\n\n  mapComponentStyles(style: StyleProperties): Record<string, string> {\n    const result: Record<string, string> = {};\n\n    // Typography\n    if (style.fontFamily) result['font-family'] = style.fontFamily;\n    if (style.fontSize) result['font-size'] = style.fontSize;\n    if (style.fontWeight) result['font-weight'] = style.fontWeight;\n    if (style.fontStyle) result['font-style'] = style.fontStyle;\n    if (style.textDecoration) result['text-decoration'] = style.textDecoration;\n    if (style.lineHeight) result['line-height'] = style.lineHeight;\n    if (style.letterSpacing) result['letter-spacing'] = style.letterSpacing;\n\n    // Colors\n    if (style.color) result['color'] = style.color;\n    if (style.backgroundColor) result['background-color'] = style.backgroundColor;\n\n    // Alignment\n    if (style.textAlign) result['text-align'] = style.textAlign;\n\n    // Spacing\n    if (style.padding) result['padding'] = style.padding;\n    if (style.paddingTop) result['padding-top'] = style.paddingTop;\n    if (style.paddingRight) result['padding-right'] = style.paddingRight;\n    if (style.paddingBottom) result['padding-bottom'] = style.paddingBottom;\n    if (style.paddingLeft) result['padding-left'] = style.paddingLeft;\n\n    // Borders\n    if (style.border) result['border'] = style.border;\n    if (style.borderWidth) result['border-width'] = style.borderWidth;\n    if (style.borderStyle) result['border-style'] = style.borderStyle;\n    if (style.borderColor) result['border-color'] = style.borderColor;\n    if (style.borderRadius) result['border-radius'] = style.borderRadius;\n\n    return result;\n  }\n\n  mapTableStyle(component: ComponentModel): Record<string, string> {\n    const result: Record<string, string> = {};\n    const config = component.tableConfig;\n\n    if (config?.showOuterBorder !== false) {\n      const width = config?.outerBorderWidth || '1px';\n      const color = config?.outerBorderColor || '#cccccc';\n      const style = config?.outerBorderStyle || 'solid';\n      result['border'] = `${width} ${style} ${color}`;\n    } else {\n      result['border'] = 'none';\n    }\n\n    if (config?.tableLayout) {\n      result['table-layout'] = config.tableLayout;\n    }\n\n    return result;\n  }\n\n  mapHeaderStyle(component: ComponentModel, colIndex: number, col?: TableColumn): Record<string, string> {\n    const result: Record<string, string> = {};\n    const config = component.tableConfig;\n\n    if (config?.headerBackground) {\n      result['background-color'] = config.headerBackground;\n    } else {\n      result['background-color'] = '#f5f5f5';\n    }\n\n    if (config?.headerTextColor) {\n      result['color'] = config.headerTextColor;\n    }\n\n    if (config?.headerFontWeight) {\n      result['font-weight'] = config.headerFontWeight;\n    } else {\n      result['font-weight'] = 'bold';\n    }\n\n    if (config?.headerFontSize) {\n      result['font-size'] = config.headerFontSize;\n    }\n\n    if (config?.headerTextAlign) {\n      result['text-align'] = config.headerTextAlign;\n    } else if (col?.style?.textAlign) {\n      result['text-align'] = col.style.textAlign;\n    }\n\n    if (config?.headerPadding) {\n      result['padding'] = config.headerPadding;\n    } else {\n      result['padding'] = '8px';\n    }\n\n    if (config?.headerBorderBottom) {\n      result['border-bottom'] = config.headerBorderBottom;\n    }\n\n    if (col?.width) {\n      result['width'] = col.width;\n    }\n\n    const totalCols = component.columns?.length || 0;\n    if (config?.showColumnSeparators !== false && colIndex < totalCols - 1) {\n      const sepWidth = config?.columnSeparatorWidth || '1px';\n      const sepColor = config?.columnSeparatorColor || '#e0e0e0';\n      const sepStyle = config?.columnSeparatorStyle || 'solid';\n      result['border-right'] = `${sepWidth} ${sepStyle} ${sepColor}`;\n    }\n\n    return result;\n  }\n\n  mapRowStyle(component: ComponentModel, rowIndex: number, totalRows: number): Record<string, string> {\n    const result: Record<string, string> = {};\n    const config = component.tableConfig;\n\n    if (config?.highlightFirstRow && rowIndex === 0) {\n      result['background-color'] = config.firstRowBackground || '#e8f5e9';\n    } else if (config?.highlightLastRow && rowIndex === totalRows - 1) {\n      result['background-color'] = config.lastRowBackground || '#fff3e0';\n    } else if (config?.enableAlternateRows) {\n      if (rowIndex % 2 === 0) {\n        result['background-color'] = config.evenRowBackground || '#ffffff';\n      } else {\n        result['background-color'] = config.oddRowBackground || '#f9f9f9';\n      }\n    }\n\n    if (config?.showRowSeparators !== false && rowIndex < totalRows - 1) {\n      const sepWidth = config?.rowSeparatorWidth || '1px';\n      const sepColor = config?.rowSeparatorColor || '#e0e0e0';\n      const sepStyle = config?.rowSeparatorStyle || 'solid';\n      result['border-bottom'] = `${sepWidth} ${sepStyle} ${sepColor}`;\n    }\n\n    return result;\n  }\n\n  mapCellStyle(component: ComponentModel, colIndex: number, col?: TableColumn): Record<string, string> {\n    const result: Record<string, string> = {};\n    const config = component.tableConfig;\n\n    if (config?.cellPadding) {\n      result['padding'] = config.cellPadding;\n    } else if (col?.style?.padding) {\n      result['padding'] = col.style.padding;\n    } else {\n      result['padding'] = '8px';\n    }\n\n    if (config?.cellVerticalAlign) {\n      result['vertical-align'] = config.cellVerticalAlign;\n    }\n\n    if (col?.width) {\n      result['width'] = col.width;\n    }\n\n    if (col?.style?.textAlign) {\n      result['text-align'] = col.style.textAlign;\n    }\n    if (col?.style?.fontWeight) {\n      result['font-weight'] = col.style.fontWeight;\n    }\n    if (col?.style?.color) {\n      result['color'] = col.style.color;\n    }\n\n    const totalCols = component.columns?.length || 0;\n    if (config?.showColumnSeparators !== false && colIndex < totalCols - 1) {\n      const sepWidth = config?.columnSeparatorWidth || '1px';\n      const sepColor = config?.columnSeparatorColor || '#e0e0e0';\n      const sepStyle = config?.columnSeparatorStyle || 'solid';\n      result['border-right'] = `${sepWidth} ${sepStyle} ${sepColor}`;\n    }\n\n    return result;\n  }\n\n  mapBandStyle(section: ReportSection): Record<string, string> {\n    const result: Record<string, string> = {};\n    if (section.style?.backgroundColor) {\n      result['background-color'] = section.style.backgroundColor;\n    }\n    if (section.style?.borderBottom) {\n      result['border-bottom'] = section.style.borderBottom;\n    }\n    if (section.style?.borderTop) {\n      result['border-top'] = section.style.borderTop;\n    }\n    return result;\n  }\n\n  mapComplexCellStyle(cell: ComplexTableCell, config: ComplexTableConfig, isHeader: boolean): Record<string, string> {\n    const result: Record<string, string> = {};\n\n    // Grid lines\n    if (config.showGridLines !== false) {\n      const w = config.gridLineWidth || '1px';\n      const c = config.gridLineColor || '#e0e0e0';\n      const s = config.gridLineStyle || 'solid';\n      result['border'] = `${w} ${s} ${c}`;\n    }\n\n    // Header/body defaults\n    if (isHeader) {\n      if (config.headerBackground) result['background-color'] = config.headerBackground;\n      if (config.headerTextColor) result['color'] = config.headerTextColor;\n      if (config.headerFontWeight) result['font-weight'] = config.headerFontWeight;\n      if (config.headerFontSize) result['font-size'] = config.headerFontSize;\n      if (config.headerTextAlign) result['text-align'] = config.headerTextAlign;\n      if (config.headerPadding) result['padding'] = config.headerPadding;\n    } else {\n      if (config.cellPadding) result['padding'] = config.cellPadding;\n      if (config.cellTextAlign) result['text-align'] = config.cellTextAlign;\n    }\n\n    if (config.cellVerticalAlign) result['vertical-align'] = config.cellVerticalAlign;\n\n    // Per-cell style overrides\n    if (cell.style) {\n      if (cell.style.backgroundColor) result['background-color'] = cell.style.backgroundColor;\n      if (cell.style.color) result['color'] = cell.style.color;\n      if (cell.style.fontWeight) result['font-weight'] = cell.style.fontWeight;\n      if (cell.style.fontSize) result['font-size'] = cell.style.fontSize;\n      if (cell.style.textAlign) result['text-align'] = cell.style.textAlign;\n      if (cell.style.padding) result['padding'] = cell.style.padding;\n    }\n\n    return result;\n  }\n}\n"]}
|