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.
Files changed (76) hide show
  1. package/README.md +24 -0
  2. package/esm2022/lib/components/chart-renderer/chart-renderer.component.mjs +75 -0
  3. package/esm2022/lib/components/complex-table-renderer/complex-table-renderer.component.mjs +155 -0
  4. package/esm2022/lib/components/component-renderer/component-renderer.component.mjs +158 -0
  5. package/esm2022/lib/components/container-renderer/container-renderer.component.mjs +53 -0
  6. package/esm2022/lib/components/image-renderer/image-renderer.component.mjs +67 -0
  7. package/esm2022/lib/components/line-renderer/line-renderer.component.mjs +94 -0
  8. package/esm2022/lib/components/list-renderer/list-renderer.component.mjs +96 -0
  9. package/esm2022/lib/components/qrcode-renderer/qrcode-renderer.component.mjs +67 -0
  10. package/esm2022/lib/components/shape-renderer/shape-renderer.component.mjs +133 -0
  11. package/esm2022/lib/components/table-renderer/table-renderer.component.mjs +148 -0
  12. package/esm2022/lib/components/text-renderer/text-renderer.component.mjs +33 -0
  13. package/esm2022/lib/constants/page-sizes.mjs +3 -0
  14. package/esm2022/lib/constants/report-tokens.mjs +21 -0
  15. package/esm2022/lib/layout/report-band.component.mjs +33 -0
  16. package/esm2022/lib/layout/report-page.component.mjs +159 -0
  17. package/esm2022/lib/models/component.model.mjs +3 -0
  18. package/esm2022/lib/models/report-template.model.mjs +19 -0
  19. package/esm2022/lib/page-container/page-container.component.mjs +248 -0
  20. package/esm2022/lib/report-viewer.component.mjs +393 -0
  21. package/esm2022/lib/services/chart-options-builder.service.mjs +749 -0
  22. package/esm2022/lib/services/data-resolver.service.mjs +385 -0
  23. package/esm2022/lib/services/export.service.mjs +82 -0
  24. package/esm2022/lib/services/formatting.service.mjs +59 -0
  25. package/esm2022/lib/services/print.service.mjs +133 -0
  26. package/esm2022/lib/services/search.service.mjs +117 -0
  27. package/esm2022/lib/services/style-mapper.service.mjs +247 -0
  28. package/esm2022/lib/services/template-normalizer.service.mjs +213 -0
  29. package/esm2022/lib/services/template-validator.service.mjs +293 -0
  30. package/esm2022/lib/services/token-resolver.service.mjs +14 -0
  31. package/esm2022/lib/services/viewer-state.service.mjs +155 -0
  32. package/esm2022/lib/sidebar/sidebar-container.component.mjs +86 -0
  33. package/esm2022/lib/sidebar/thumbnail-sidebar.component.mjs +110 -0
  34. package/esm2022/lib/sidebar/toc-sidebar.component.mjs +155 -0
  35. package/esm2022/lib/toolbar/viewer-toolbar.component.mjs +486 -0
  36. package/esm2022/public-api.mjs +43 -0
  37. package/esm2022/snap-report-viewer.mjs +5 -0
  38. package/fesm2022/snap-report-viewer.mjs +5110 -0
  39. package/fesm2022/snap-report-viewer.mjs.map +1 -0
  40. package/index.d.ts +5 -0
  41. package/lib/components/chart-renderer/chart-renderer.component.d.ts +14 -0
  42. package/lib/components/complex-table-renderer/complex-table-renderer.component.d.ts +17 -0
  43. package/lib/components/component-renderer/component-renderer.component.d.ts +14 -0
  44. package/lib/components/container-renderer/container-renderer.component.d.ts +11 -0
  45. package/lib/components/image-renderer/image-renderer.component.d.ts +14 -0
  46. package/lib/components/line-renderer/line-renderer.component.d.ts +10 -0
  47. package/lib/components/list-renderer/list-renderer.component.d.ts +16 -0
  48. package/lib/components/qrcode-renderer/qrcode-renderer.component.d.ts +16 -0
  49. package/lib/components/shape-renderer/shape-renderer.component.d.ts +11 -0
  50. package/lib/components/table-renderer/table-renderer.component.d.ts +20 -0
  51. package/lib/components/text-renderer/text-renderer.component.d.ts +13 -0
  52. package/lib/constants/page-sizes.d.ts +2 -0
  53. package/lib/constants/report-tokens.d.ts +8 -0
  54. package/lib/layout/report-band.component.d.ts +10 -0
  55. package/lib/layout/report-page.component.d.ts +21 -0
  56. package/lib/models/component.model.d.ts +315 -0
  57. package/lib/models/report-template.model.d.ts +122 -0
  58. package/lib/page-container/page-container.component.d.ts +29 -0
  59. package/lib/report-viewer.component.d.ts +51 -0
  60. package/lib/services/chart-options-builder.service.d.ts +31 -0
  61. package/lib/services/data-resolver.service.d.ts +27 -0
  62. package/lib/services/export.service.d.ts +11 -0
  63. package/lib/services/formatting.service.d.ts +10 -0
  64. package/lib/services/print.service.d.ts +12 -0
  65. package/lib/services/search.service.d.ts +13 -0
  66. package/lib/services/style-mapper.service.d.ts +14 -0
  67. package/lib/services/template-normalizer.service.d.ts +24 -0
  68. package/lib/services/template-validator.service.d.ts +19 -0
  69. package/lib/services/token-resolver.service.d.ts +6 -0
  70. package/lib/services/viewer-state.service.d.ts +49 -0
  71. package/lib/sidebar/sidebar-container.component.d.ts +8 -0
  72. package/lib/sidebar/thumbnail-sidebar.component.d.ts +15 -0
  73. package/lib/sidebar/toc-sidebar.component.d.ts +17 -0
  74. package/lib/toolbar/viewer-toolbar.component.d.ts +17 -0
  75. package/package.json +43 -0
  76. package/public-api.d.ts +35 -0
@@ -0,0 +1,155 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class ViewerStateService {
4
+ constructor() {
5
+ // Pagination
6
+ this.currentPage = signal(1);
7
+ this.totalPages = signal(1);
8
+ // Zoom
9
+ this.zoom = signal(100);
10
+ this.fitMode = signal('none');
11
+ // View modes
12
+ this.viewMode = signal('single');
13
+ this.isFullScreen = signal(false);
14
+ // Sidebars
15
+ this.tocSidebarOpen = signal(false);
16
+ this.thumbnailSidebarOpen = signal(false);
17
+ // Search
18
+ this.searchQuery = signal('');
19
+ this.searchMatches = signal([]);
20
+ this.currentMatchIndex = signal(-1);
21
+ this.searchOpen = signal(false);
22
+ // Theme
23
+ this.ribbonTheme = signal('light');
24
+ // Computed
25
+ this.pageDisplay = computed(() => `${this.currentPage()} of ${this.totalPages()}`);
26
+ this.zoomDisplay = computed(() => `${this.zoom()}%`);
27
+ this.matchDisplay = computed(() => {
28
+ const matches = this.searchMatches();
29
+ const idx = this.currentMatchIndex();
30
+ if (matches.length === 0)
31
+ return 'No matches';
32
+ return `${idx + 1} of ${matches.length}`;
33
+ });
34
+ this.isFirstPage = computed(() => this.currentPage() <= 1);
35
+ this.isLastPage = computed(() => this.currentPage() >= this.totalPages());
36
+ this.hasSidebarOpen = computed(() => this.tocSidebarOpen() || this.thumbnailSidebarOpen());
37
+ this.isDarkTheme = computed(() => this.ribbonTheme() === 'dark');
38
+ }
39
+ // Navigation
40
+ goToPage(page) {
41
+ const clamped = Math.max(1, Math.min(page, this.totalPages()));
42
+ this.currentPage.set(clamped);
43
+ }
44
+ nextPage() {
45
+ if (!this.isLastPage()) {
46
+ this.currentPage.update(p => p + 1);
47
+ }
48
+ }
49
+ previousPage() {
50
+ if (!this.isFirstPage()) {
51
+ this.currentPage.update(p => p - 1);
52
+ }
53
+ }
54
+ firstPage() {
55
+ this.currentPage.set(1);
56
+ }
57
+ lastPage() {
58
+ this.currentPage.set(this.totalPages());
59
+ }
60
+ // Zoom
61
+ setZoom(level) {
62
+ this.zoom.set(Math.max(10, Math.min(400, level)));
63
+ this.fitMode.set('none');
64
+ }
65
+ zoomIn() {
66
+ const levels = [25, 50, 75, 100, 125, 150, 200, 300, 400];
67
+ const current = this.zoom();
68
+ const next = levels.find(l => l > current) || 400;
69
+ this.setZoom(next);
70
+ }
71
+ zoomOut() {
72
+ const levels = [25, 50, 75, 100, 125, 150, 200, 300, 400];
73
+ const current = this.zoom();
74
+ const prev = [...levels].reverse().find(l => l < current) || 25;
75
+ this.setZoom(prev);
76
+ }
77
+ fitToWidth() {
78
+ this.fitMode.set('width');
79
+ }
80
+ fitToPage() {
81
+ this.fitMode.set('page');
82
+ }
83
+ // View modes
84
+ toggleViewMode() {
85
+ this.viewMode.update(m => m === 'single' ? 'continuous' : 'single');
86
+ }
87
+ setViewMode(mode) {
88
+ this.viewMode.set(mode);
89
+ }
90
+ toggleFullScreen() {
91
+ this.isFullScreen.update(v => !v);
92
+ }
93
+ // Sidebars
94
+ toggleTocSidebar() {
95
+ const opening = !this.tocSidebarOpen();
96
+ this.tocSidebarOpen.set(opening);
97
+ if (opening) {
98
+ this.thumbnailSidebarOpen.set(false);
99
+ }
100
+ }
101
+ toggleThumbnailSidebar() {
102
+ const opening = !this.thumbnailSidebarOpen();
103
+ this.thumbnailSidebarOpen.set(opening);
104
+ if (opening) {
105
+ this.tocSidebarOpen.set(false);
106
+ }
107
+ }
108
+ closeSidebars() {
109
+ this.tocSidebarOpen.set(false);
110
+ this.thumbnailSidebarOpen.set(false);
111
+ }
112
+ // Search
113
+ setSearchQuery(query) {
114
+ this.searchQuery.set(query);
115
+ this.currentMatchIndex.set(query ? 0 : -1);
116
+ }
117
+ nextMatch() {
118
+ const total = this.searchMatches().length;
119
+ if (total > 0) {
120
+ this.currentMatchIndex.update(i => (i + 1) % total);
121
+ }
122
+ }
123
+ previousMatch() {
124
+ const total = this.searchMatches().length;
125
+ if (total > 0) {
126
+ this.currentMatchIndex.update(i => (i - 1 + total) % total);
127
+ }
128
+ }
129
+ clearSearch() {
130
+ this.searchQuery.set('');
131
+ this.searchMatches.set([]);
132
+ this.currentMatchIndex.set(-1);
133
+ this.searchOpen.set(false);
134
+ }
135
+ toggleSearch() {
136
+ const opening = !this.searchOpen();
137
+ this.searchOpen.set(opening);
138
+ if (!opening) {
139
+ this.clearSearch();
140
+ }
141
+ }
142
+ // Theme
143
+ toggleTheme() {
144
+ this.ribbonTheme.update(t => t === 'light' ? 'dark' : 'light');
145
+ }
146
+ setTheme(theme) {
147
+ this.ribbonTheme.set(theme);
148
+ }
149
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ViewerStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
150
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ViewerStateService }); }
151
+ }
152
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ViewerStateService, decorators: [{
153
+ type: Injectable
154
+ }] });
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewer-state.service.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/services/viewer-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed } from '@angular/core';\r\nimport { SearchMatch } from '../models/report-template.model';\r\n\r\n@Injectable()\r\nexport class ViewerStateService {\r\n  // Pagination\r\n  readonly currentPage = signal(1);\r\n  readonly totalPages = signal(1);\r\n\r\n  // Zoom\r\n  readonly zoom = signal(100);\r\n  readonly fitMode = signal<'none' | 'width' | 'page'>('none');\r\n\r\n  // View modes\r\n  readonly viewMode = signal<'single' | 'continuous'>('single');\r\n  readonly isFullScreen = signal(false);\r\n\r\n  // Sidebars\r\n  readonly tocSidebarOpen = signal(false);\r\n  readonly thumbnailSidebarOpen = signal(false);\r\n\r\n  // Search\r\n  readonly searchQuery = signal('');\r\n  readonly searchMatches = signal<SearchMatch[]>([]);\r\n  readonly currentMatchIndex = signal(-1);\r\n  readonly searchOpen = signal(false);\r\n\r\n  // Theme\r\n  readonly ribbonTheme = signal<'light' | 'dark'>('light');\r\n\r\n  // Computed\r\n  readonly pageDisplay = computed(() => `${this.currentPage()} of ${this.totalPages()}`);\r\n  readonly zoomDisplay = computed(() => `${this.zoom()}%`);\r\n  readonly matchDisplay = computed(() => {\r\n    const matches = this.searchMatches();\r\n    const idx = this.currentMatchIndex();\r\n    if (matches.length === 0) return 'No matches';\r\n    return `${idx + 1} of ${matches.length}`;\r\n  });\r\n  readonly isFirstPage = computed(() => this.currentPage() <= 1);\r\n  readonly isLastPage = computed(() => this.currentPage() >= this.totalPages());\r\n  readonly hasSidebarOpen = computed(() => this.tocSidebarOpen() || this.thumbnailSidebarOpen());\r\n  readonly isDarkTheme = computed(() => this.ribbonTheme() === 'dark');\r\n\r\n  // Navigation\r\n  goToPage(page: number): void {\r\n    const clamped = Math.max(1, Math.min(page, this.totalPages()));\r\n    this.currentPage.set(clamped);\r\n  }\r\n\r\n  nextPage(): void {\r\n    if (!this.isLastPage()) {\r\n      this.currentPage.update(p => p + 1);\r\n    }\r\n  }\r\n\r\n  previousPage(): void {\r\n    if (!this.isFirstPage()) {\r\n      this.currentPage.update(p => p - 1);\r\n    }\r\n  }\r\n\r\n  firstPage(): void {\r\n    this.currentPage.set(1);\r\n  }\r\n\r\n  lastPage(): void {\r\n    this.currentPage.set(this.totalPages());\r\n  }\r\n\r\n  // Zoom\r\n  setZoom(level: number): void {\r\n    this.zoom.set(Math.max(10, Math.min(400, level)));\r\n    this.fitMode.set('none');\r\n  }\r\n\r\n  zoomIn(): void {\r\n    const levels = [25, 50, 75, 100, 125, 150, 200, 300, 400];\r\n    const current = this.zoom();\r\n    const next = levels.find(l => l > current) || 400;\r\n    this.setZoom(next);\r\n  }\r\n\r\n  zoomOut(): void {\r\n    const levels = [25, 50, 75, 100, 125, 150, 200, 300, 400];\r\n    const current = this.zoom();\r\n    const prev = [...levels].reverse().find(l => l < current) || 25;\r\n    this.setZoom(prev);\r\n  }\r\n\r\n  fitToWidth(): void {\r\n    this.fitMode.set('width');\r\n  }\r\n\r\n  fitToPage(): void {\r\n    this.fitMode.set('page');\r\n  }\r\n\r\n  // View modes\r\n  toggleViewMode(): void {\r\n    this.viewMode.update(m => m === 'single' ? 'continuous' : 'single');\r\n  }\r\n\r\n  setViewMode(mode: 'single' | 'continuous'): void {\r\n    this.viewMode.set(mode);\r\n  }\r\n\r\n  toggleFullScreen(): void {\r\n    this.isFullScreen.update(v => !v);\r\n  }\r\n\r\n  // Sidebars\r\n  toggleTocSidebar(): void {\r\n    const opening = !this.tocSidebarOpen();\r\n    this.tocSidebarOpen.set(opening);\r\n    if (opening) {\r\n      this.thumbnailSidebarOpen.set(false);\r\n    }\r\n  }\r\n\r\n  toggleThumbnailSidebar(): void {\r\n    const opening = !this.thumbnailSidebarOpen();\r\n    this.thumbnailSidebarOpen.set(opening);\r\n    if (opening) {\r\n      this.tocSidebarOpen.set(false);\r\n    }\r\n  }\r\n\r\n  closeSidebars(): void {\r\n    this.tocSidebarOpen.set(false);\r\n    this.thumbnailSidebarOpen.set(false);\r\n  }\r\n\r\n  // Search\r\n  setSearchQuery(query: string): void {\r\n    this.searchQuery.set(query);\r\n    this.currentMatchIndex.set(query ? 0 : -1);\r\n  }\r\n\r\n  nextMatch(): void {\r\n    const total = this.searchMatches().length;\r\n    if (total > 0) {\r\n      this.currentMatchIndex.update(i => (i + 1) % total);\r\n    }\r\n  }\r\n\r\n  previousMatch(): void {\r\n    const total = this.searchMatches().length;\r\n    if (total > 0) {\r\n      this.currentMatchIndex.update(i => (i - 1 + total) % total);\r\n    }\r\n  }\r\n\r\n  clearSearch(): void {\r\n    this.searchQuery.set('');\r\n    this.searchMatches.set([]);\r\n    this.currentMatchIndex.set(-1);\r\n    this.searchOpen.set(false);\r\n  }\r\n\r\n  toggleSearch(): void {\r\n    const opening = !this.searchOpen();\r\n    this.searchOpen.set(opening);\r\n    if (!opening) {\r\n      this.clearSearch();\r\n    }\r\n  }\r\n\r\n  // Theme\r\n  toggleTheme(): void {\r\n    this.ribbonTheme.update(t => t === 'light' ? 'dark' : 'light');\r\n  }\r\n\r\n  setTheme(theme: 'light' | 'dark'): void {\r\n    this.ribbonTheme.set(theme);\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,86 @@
1
+ import { Component, ChangeDetectionStrategy, Input, output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { MatIconModule } from '@angular/material/icon';
4
+ import { MatButtonModule } from '@angular/material/button';
5
+ import { animate, state, style, transition, trigger } from '@angular/animations';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/material/icon";
8
+ import * as i2 from "@angular/material/button";
9
+ export class SidebarContainerComponent {
10
+ constructor() {
11
+ this.isOpen = false;
12
+ this.title = '';
13
+ this.closed = output();
14
+ }
15
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SidebarContainerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
16
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SidebarContainerComponent, isStandalone: true, selector: "rv-sidebar-container", inputs: { isOpen: "isOpen", title: "title" }, outputs: { closed: "closed" }, ngImport: i0, template: `
17
+ <div class="rv-sidebar-wrapper"
18
+ [class.rv-sidebar-hidden]="!isOpen"
19
+ [@sidebarSlide]="isOpen ? 'open' : 'closed'">
20
+ <div class="rv-sidebar-panel">
21
+ <div class="rv-sidebar-header">
22
+ <span class="rv-sidebar-title">{{ title }}</span>
23
+ <button mat-icon-button class="rv-sidebar-close" (click)="closed.emit()">
24
+ <mat-icon>close</mat-icon>
25
+ </button>
26
+ </div>
27
+ <div class="rv-sidebar-body">
28
+ <ng-content></ng-content>
29
+ </div>
30
+ </div>
31
+ </div>
32
+ `, isInline: true, styles: [":host{display:block;position:absolute;left:0;top:0;bottom:0;z-index:50;pointer-events:none}.rv-sidebar-wrapper{width:240px;height:100%;pointer-events:auto}.rv-sidebar-hidden{pointer-events:none}.rv-sidebar-panel{width:100%;height:100%;display:flex;flex-direction:column;background:var(--rv-sidebar-bg, #ffffff);border-right:1px solid var(--rv-sidebar-border, #e0e0e0);box-shadow:2px 0 8px #0000001a}:host-context(.rv-theme-dark) .rv-sidebar-panel{--rv-sidebar-bg: #1e1e2d;--rv-sidebar-border: #3a3a4a;box-shadow:2px 0 12px #0000004d}.rv-sidebar-header{display:flex;align-items:center;justify-content:space-between;height:40px;min-height:40px;padding:0 8px 0 16px;border-bottom:1px solid var(--rv-sidebar-border, #e0e0e0);background:var(--rv-sidebar-bg, #ffffff)}:host-context(.rv-theme-dark) .rv-sidebar-header{background:#252538}.rv-sidebar-title{font-size:12px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--rv-sidebar-text, #555555)}:host-context(.rv-theme-dark) .rv-sidebar-title{color:#b0b0c0}.rv-sidebar-close{width:28px!important;height:28px!important}.rv-sidebar-close .mat-icon{font-size:18px;width:18px;height:18px;color:var(--rv-sidebar-text, #555555)}:host-context(.rv-theme-dark) .rv-sidebar-close .mat-icon{color:#b0b0c0}.rv-sidebar-body{flex:1;overflow-y:auto;overflow-x:hidden}.rv-sidebar-body::-webkit-scrollbar{width:6px}.rv-sidebar-body::-webkit-scrollbar-track{background:transparent}.rv-sidebar-body::-webkit-scrollbar-thumb{background:#ccc;border-radius:3px}:host-context(.rv-theme-dark) .rv-sidebar-body::-webkit-scrollbar-thumb{background:#444}.rv-sidebar-body::-webkit-scrollbar-thumb:hover{background:#aaa}:host-context(.rv-theme-dark) .rv-sidebar-body::-webkit-scrollbar-thumb:hover{background:#555}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }], animations: [
33
+ trigger('sidebarSlide', [
34
+ state('open', style({
35
+ transform: 'translateX(0)',
36
+ opacity: 1,
37
+ })),
38
+ state('closed', style({
39
+ transform: 'translateX(-100%)',
40
+ opacity: 0,
41
+ })),
42
+ transition('open <=> closed', [
43
+ animate('200ms ease-in-out')
44
+ ]),
45
+ ]),
46
+ ], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
47
+ }
48
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SidebarContainerComponent, decorators: [{
49
+ type: Component,
50
+ args: [{ selector: 'rv-sidebar-container', standalone: true, imports: [CommonModule, MatIconModule, MatButtonModule], changeDetection: ChangeDetectionStrategy.OnPush, animations: [
51
+ trigger('sidebarSlide', [
52
+ state('open', style({
53
+ transform: 'translateX(0)',
54
+ opacity: 1,
55
+ })),
56
+ state('closed', style({
57
+ transform: 'translateX(-100%)',
58
+ opacity: 0,
59
+ })),
60
+ transition('open <=> closed', [
61
+ animate('200ms ease-in-out')
62
+ ]),
63
+ ]),
64
+ ], template: `
65
+ <div class="rv-sidebar-wrapper"
66
+ [class.rv-sidebar-hidden]="!isOpen"
67
+ [@sidebarSlide]="isOpen ? 'open' : 'closed'">
68
+ <div class="rv-sidebar-panel">
69
+ <div class="rv-sidebar-header">
70
+ <span class="rv-sidebar-title">{{ title }}</span>
71
+ <button mat-icon-button class="rv-sidebar-close" (click)="closed.emit()">
72
+ <mat-icon>close</mat-icon>
73
+ </button>
74
+ </div>
75
+ <div class="rv-sidebar-body">
76
+ <ng-content></ng-content>
77
+ </div>
78
+ </div>
79
+ </div>
80
+ `, styles: [":host{display:block;position:absolute;left:0;top:0;bottom:0;z-index:50;pointer-events:none}.rv-sidebar-wrapper{width:240px;height:100%;pointer-events:auto}.rv-sidebar-hidden{pointer-events:none}.rv-sidebar-panel{width:100%;height:100%;display:flex;flex-direction:column;background:var(--rv-sidebar-bg, #ffffff);border-right:1px solid var(--rv-sidebar-border, #e0e0e0);box-shadow:2px 0 8px #0000001a}:host-context(.rv-theme-dark) .rv-sidebar-panel{--rv-sidebar-bg: #1e1e2d;--rv-sidebar-border: #3a3a4a;box-shadow:2px 0 12px #0000004d}.rv-sidebar-header{display:flex;align-items:center;justify-content:space-between;height:40px;min-height:40px;padding:0 8px 0 16px;border-bottom:1px solid var(--rv-sidebar-border, #e0e0e0);background:var(--rv-sidebar-bg, #ffffff)}:host-context(.rv-theme-dark) .rv-sidebar-header{background:#252538}.rv-sidebar-title{font-size:12px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--rv-sidebar-text, #555555)}:host-context(.rv-theme-dark) .rv-sidebar-title{color:#b0b0c0}.rv-sidebar-close{width:28px!important;height:28px!important}.rv-sidebar-close .mat-icon{font-size:18px;width:18px;height:18px;color:var(--rv-sidebar-text, #555555)}:host-context(.rv-theme-dark) .rv-sidebar-close .mat-icon{color:#b0b0c0}.rv-sidebar-body{flex:1;overflow-y:auto;overflow-x:hidden}.rv-sidebar-body::-webkit-scrollbar{width:6px}.rv-sidebar-body::-webkit-scrollbar-track{background:transparent}.rv-sidebar-body::-webkit-scrollbar-thumb{background:#ccc;border-radius:3px}:host-context(.rv-theme-dark) .rv-sidebar-body::-webkit-scrollbar-thumb{background:#444}.rv-sidebar-body::-webkit-scrollbar-thumb:hover{background:#aaa}:host-context(.rv-theme-dark) .rv-sidebar-body::-webkit-scrollbar-thumb:hover{background:#555}\n"] }]
81
+ }], propDecorators: { isOpen: [{
82
+ type: Input
83
+ }], title: [{
84
+ type: Input
85
+ }] } });
86
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2lkZWJhci1jb250YWluZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcmVwb3J0LXZpZXdlci9zcmMvbGliL3NpZGViYXIvc2lkZWJhci1jb250YWluZXIuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCxTQUFTLEVBQUUsdUJBQXVCLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFDbEQsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN2RCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDM0QsT0FBTyxFQUFFLE9BQU8sRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLFVBQVUsRUFBRSxPQUFPLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQzs7OztBQXdKakYsTUFBTSxPQUFPLHlCQUF5QjtJQXRKdEM7UUF1SlcsV0FBTSxHQUFHLEtBQUssQ0FBQztRQUNmLFVBQUssR0FBRyxFQUFFLENBQUM7UUFFcEIsV0FBTSxHQUFHLE1BQU0sRUFBUSxDQUFDO0tBQ3pCOytHQUxZLHlCQUF5QjttR0FBekIseUJBQXlCLDZKQWxJMUI7Ozs7Ozs7Ozs7Ozs7Ozs7R0FnQlQsbXlEQWpDUyxZQUFZLDhCQUFFLGFBQWEsbUxBQUUsZUFBZSw4SEFFMUM7WUFDVixPQUFPLENBQUMsY0FBYyxFQUFFO2dCQUN0QixLQUFLLENBQUMsTUFBTSxFQUFFLEtBQUssQ0FBQztvQkFDbEIsU0FBUyxFQUFFLGVBQWU7b0JBQzFCLE9BQU8sRUFBRSxDQUFDO2lCQUNYLENBQUMsQ0FBQztnQkFDSCxLQUFLLENBQUMsUUFBUSxFQUFFLEtBQUssQ0FBQztvQkFDcEIsU0FBUyxFQUFFLG1CQUFtQjtvQkFDOUIsT0FBTyxFQUFFLENBQUM7aUJBQ1gsQ0FBQyxDQUFDO2dCQUNILFVBQVUsQ0FBQyxpQkFBaUIsRUFBRTtvQkFDNUIsT0FBTyxDQUFDLG1CQUFtQixDQUFDO2lCQUM3QixDQUFDO2FBQ0gsQ0FBQztTQUNIOzs0RkFtSVUseUJBQXlCO2tCQXRKckMsU0FBUzsrQkFDRSxzQkFBc0IsY0FDcEIsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLGFBQWEsRUFBRSxlQUFlLENBQUMsbUJBQ3RDLHVCQUF1QixDQUFDLE1BQU0sY0FDbkM7d0JBQ1YsT0FBTyxDQUFDLGNBQWMsRUFBRTs0QkFDdEIsS0FBSyxDQUFDLE1BQU0sRUFBRSxLQUFLLENBQUM7Z0NBQ2xCLFNBQVMsRUFBRSxlQUFlO2dDQUMxQixPQUFPLEVBQUUsQ0FBQzs2QkFDWCxDQUFDLENBQUM7NEJBQ0gsS0FBSyxDQUFDLFFBQVEsRUFBRSxLQUFLLENBQUM7Z0NBQ3BCLFNBQVMsRUFBRSxtQkFBbUI7Z0NBQzlCLE9BQU8sRUFBRSxDQUFDOzZCQUNYLENBQUMsQ0FBQzs0QkFDSCxVQUFVLENBQUMsaUJBQWlCLEVBQUU7Z0NBQzVCLE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQzs2QkFDN0IsQ0FBQzt5QkFDSCxDQUFDO3FCQUNILFlBQ1M7Ozs7Ozs7Ozs7Ozs7Ozs7R0FnQlQ7OEJBbUhRLE1BQU07c0JBQWQsS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBDb21wb25lbnQsIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBJbnB1dCwgb3V0cHV0LCBzaWduYWxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgTWF0SWNvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2ljb24nO1xuaW1wb3J0IHsgTWF0QnV0dG9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvYnV0dG9uJztcbmltcG9ydCB7IGFuaW1hdGUsIHN0YXRlLCBzdHlsZSwgdHJhbnNpdGlvbiwgdHJpZ2dlciB9IGZyb20gJ0Bhbmd1bGFyL2FuaW1hdGlvbnMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdydi1zaWRlYmFyLWNvbnRhaW5lcicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIE1hdEljb25Nb2R1bGUsIE1hdEJ1dHRvbk1vZHVsZV0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBhbmltYXRpb25zOiBbXG4gICAgdHJpZ2dlcignc2lkZWJhclNsaWRlJywgW1xuICAgICAgc3RhdGUoJ29wZW4nLCBzdHlsZSh7XG4gICAgICAgIHRyYW5zZm9ybTogJ3RyYW5zbGF0ZVgoMCknLFxuICAgICAgICBvcGFjaXR5OiAxLFxuICAgICAgfSkpLFxuICAgICAgc3RhdGUoJ2Nsb3NlZCcsIHN0eWxlKHtcbiAgICAgICAgdHJhbnNmb3JtOiAndHJhbnNsYXRlWCgtMTAwJSknLFxuICAgICAgICBvcGFjaXR5OiAwLFxuICAgICAgfSkpLFxuICAgICAgdHJhbnNpdGlvbignb3BlbiA8PT4gY2xvc2VkJywgW1xuICAgICAgICBhbmltYXRlKCcyMDBtcyBlYXNlLWluLW91dCcpXG4gICAgICBdKSxcbiAgICBdKSxcbiAgXSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGl2IGNsYXNzPVwicnYtc2lkZWJhci13cmFwcGVyXCJcbiAgICAgICAgIFtjbGFzcy5ydi1zaWRlYmFyLWhpZGRlbl09XCIhaXNPcGVuXCJcbiAgICAgICAgIFtAc2lkZWJhclNsaWRlXT1cImlzT3BlbiA/ICdvcGVuJyA6ICdjbG9zZWQnXCI+XG4gICAgICA8ZGl2IGNsYXNzPVwicnYtc2lkZWJhci1wYW5lbFwiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwicnYtc2lkZWJhci1oZWFkZXJcIj5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInJ2LXNpZGViYXItdGl0bGVcIj57eyB0aXRsZSB9fTwvc3Bhbj5cbiAgICAgICAgICA8YnV0dG9uIG1hdC1pY29uLWJ1dHRvbiBjbGFzcz1cInJ2LXNpZGViYXItY2xvc2VcIiAoY2xpY2spPVwiY2xvc2VkLmVtaXQoKVwiPlxuICAgICAgICAgICAgPG1hdC1pY29uPmNsb3NlPC9tYXQtaWNvbj5cbiAgICAgICAgICA8L2J1dHRvbj5cbiAgICAgICAgPC9kaXY+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJydi1zaWRlYmFyLWJvZHlcIj5cbiAgICAgICAgICA8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIGAsXG4gIHN0eWxlczogW2BcbiAgICA6aG9zdCB7XG4gICAgICBkaXNwbGF5OiBibG9jaztcbiAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgIGxlZnQ6IDA7XG4gICAgICB0b3A6IDA7XG4gICAgICBib3R0b206IDA7XG4gICAgICB6LWluZGV4OiA1MDtcbiAgICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLXdyYXBwZXIge1xuICAgICAgd2lkdGg6IDI0MHB4O1xuICAgICAgaGVpZ2h0OiAxMDAlO1xuICAgICAgcG9pbnRlci1ldmVudHM6IGF1dG87XG4gICAgfVxuXG4gICAgLnJ2LXNpZGViYXItaGlkZGVuIHtcbiAgICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLXBhbmVsIHtcbiAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgaGVpZ2h0OiAxMDAlO1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47XG4gICAgICBiYWNrZ3JvdW5kOiB2YXIoLS1ydi1zaWRlYmFyLWJnLCAjZmZmZmZmKTtcbiAgICAgIGJvcmRlci1yaWdodDogMXB4IHNvbGlkIHZhcigtLXJ2LXNpZGViYXItYm9yZGVyLCAjZTBlMGUwKTtcbiAgICAgIGJveC1zaGFkb3c6IDJweCAwIDhweCByZ2JhKDAsIDAsIDAsIDAuMSk7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXNpZGViYXItcGFuZWwge1xuICAgICAgLS1ydi1zaWRlYmFyLWJnOiAjMWUxZTJkO1xuICAgICAgLS1ydi1zaWRlYmFyLWJvcmRlcjogIzNhM2E0YTtcbiAgICAgIGJveC1zaGFkb3c6IDJweCAwIDEycHggcmdiYSgwLCAwLCAwLCAwLjMpO1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLWhlYWRlciB7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIGp1c3RpZnktY29udGVudDogc3BhY2UtYmV0d2VlbjtcbiAgICAgIGhlaWdodDogNDBweDtcbiAgICAgIG1pbi1oZWlnaHQ6IDQwcHg7XG4gICAgICBwYWRkaW5nOiAwIDhweCAwIDE2cHg7XG4gICAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgdmFyKC0tcnYtc2lkZWJhci1ib3JkZXIsICNlMGUwZTApO1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tcnYtc2lkZWJhci1iZywgI2ZmZmZmZik7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXNpZGViYXItaGVhZGVyIHtcbiAgICAgIGJhY2tncm91bmQ6ICMyNTI1Mzg7XG4gICAgfVxuXG4gICAgLnJ2LXNpZGViYXItdGl0bGUge1xuICAgICAgZm9udC1zaXplOiAxMnB4O1xuICAgICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICAgIHRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7XG4gICAgICBsZXR0ZXItc3BhY2luZzogMC41cHg7XG4gICAgICBjb2xvcjogdmFyKC0tcnYtc2lkZWJhci10ZXh0LCAjNTU1NTU1KTtcbiAgICB9XG5cbiAgICA6aG9zdC1jb250ZXh0KC5ydi10aGVtZS1kYXJrKSAucnYtc2lkZWJhci10aXRsZSB7XG4gICAgICBjb2xvcjogI2IwYjBjMDtcbiAgICB9XG5cbiAgICAucnYtc2lkZWJhci1jbG9zZSB7XG4gICAgICB3aWR0aDogMjhweCAhaW1wb3J0YW50O1xuICAgICAgaGVpZ2h0OiAyOHB4ICFpbXBvcnRhbnQ7XG4gICAgfVxuXG4gICAgLnJ2LXNpZGViYXItY2xvc2UgLm1hdC1pY29uIHtcbiAgICAgIGZvbnQtc2l6ZTogMThweDtcbiAgICAgIHdpZHRoOiAxOHB4O1xuICAgICAgaGVpZ2h0OiAxOHB4O1xuICAgICAgY29sb3I6IHZhcigtLXJ2LXNpZGViYXItdGV4dCwgIzU1NTU1NSk7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXNpZGViYXItY2xvc2UgLm1hdC1pY29uIHtcbiAgICAgIGNvbG9yOiAjYjBiMGMwO1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLWJvZHkge1xuICAgICAgZmxleDogMTtcbiAgICAgIG92ZXJmbG93LXk6IGF1dG87XG4gICAgICBvdmVyZmxvdy14OiBoaWRkZW47XG4gICAgfVxuXG4gICAgLyogU2Nyb2xsYmFyIHN0eWxpbmcgKi9cbiAgICAucnYtc2lkZWJhci1ib2R5Ojotd2Via2l0LXNjcm9sbGJhciB7XG4gICAgICB3aWR0aDogNnB4O1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLWJvZHk6Oi13ZWJraXQtc2Nyb2xsYmFyLXRyYWNrIHtcbiAgICAgIGJhY2tncm91bmQ6IHRyYW5zcGFyZW50O1xuICAgIH1cblxuICAgIC5ydi1zaWRlYmFyLWJvZHk6Oi13ZWJraXQtc2Nyb2xsYmFyLXRodW1iIHtcbiAgICAgIGJhY2tncm91bmQ6ICNjY2M7XG4gICAgICBib3JkZXItcmFkaXVzOiAzcHg7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXNpZGViYXItYm9keTo6LXdlYmtpdC1zY3JvbGxiYXItdGh1bWIge1xuICAgICAgYmFja2dyb3VuZDogIzQ0NDtcbiAgICB9XG5cbiAgICAucnYtc2lkZWJhci1ib2R5Ojotd2Via2l0LXNjcm9sbGJhci10aHVtYjpob3ZlciB7XG4gICAgICBiYWNrZ3JvdW5kOiAjYWFhO1xuICAgIH1cblxuICAgIDpob3N0LWNvbnRleHQoLnJ2LXRoZW1lLWRhcmspIC5ydi1zaWRlYmFyLWJvZHk6Oi13ZWJraXQtc2Nyb2xsYmFyLXRodW1iOmhvdmVyIHtcbiAgICAgIGJhY2tncm91bmQ6ICM1NTU7XG4gICAgfVxuICBgXVxufSlcbmV4cG9ydCBjbGFzcyBTaWRlYmFyQ29udGFpbmVyQ29tcG9uZW50IHtcbiAgQElucHV0KCkgaXNPcGVuID0gZmFsc2U7XG4gIEBJbnB1dCgpIHRpdGxlID0gJyc7XG5cbiAgY2xvc2VkID0gb3V0cHV0PHZvaWQ+KCk7XG59XG4iXX0=
@@ -0,0 +1,110 @@
1
+ import { Component, ChangeDetectionStrategy, Input, inject, output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ViewerStateService } from '../services/viewer-state.service';
4
+ import { SidebarContainerComponent } from './sidebar-container.component';
5
+ import { ReportPageComponent } from '../layout/report-page.component';
6
+ import * as i0 from "@angular/core";
7
+ export class ThumbnailSidebarComponent {
8
+ constructor() {
9
+ this.pages = [];
10
+ this.pageWidth = 210;
11
+ this.pageHeight = 297;
12
+ this.globalStyles = { fontFamily: 'Arial', fontSize: '11px', color: '#000', backgroundColor: '#fff' };
13
+ this.data = null;
14
+ this.pageSelected = output();
15
+ this.state = inject(ViewerStateService);
16
+ }
17
+ onPageClick(page) {
18
+ this.state.goToPage(page);
19
+ this.pageSelected.emit(page);
20
+ }
21
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ThumbnailSidebarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
22
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ThumbnailSidebarComponent, isStandalone: true, selector: "rv-thumbnail-sidebar", inputs: { pages: "pages", pageWidth: "pageWidth", pageHeight: "pageHeight", globalStyles: "globalStyles", data: "data" }, outputs: { pageSelected: "pageSelected" }, ngImport: i0, template: `
23
+ <rv-sidebar-container
24
+ [isOpen]="state.thumbnailSidebarOpen()"
25
+ title="Pages"
26
+ (closed)="state.toggleThumbnailSidebar()">
27
+
28
+ <div class="rv-thumbnail-list">
29
+ @for (page of pages; track page.index) {
30
+ <div class="rv-thumbnail-item"
31
+ [class.rv-thumb-active]="page.index === state.currentPage() - 1"
32
+ (click)="onPageClick(page.index + 1)">
33
+ <div class="rv-thumbnail-preview">
34
+ <div class="rv-thumbnail-scale">
35
+ <rv-report-page
36
+ [pageWidth]="pageWidth"
37
+ [pageHeight]="pageHeight"
38
+ [headerSection]="page.header"
39
+ [footerSection]="page.footer"
40
+ [bodyComponents]="page.components"
41
+ [globalStyles]="globalStyles"
42
+ [data]="data"
43
+ [pageNumber]="page.index + 1"
44
+ [totalPages]="pages.length">
45
+ </rv-report-page>
46
+ </div>
47
+ </div>
48
+ <span class="rv-thumb-label">{{ page.index + 1 }}</span>
49
+ </div>
50
+ }
51
+
52
+ @if (!pages || pages.length === 0) {
53
+ <div class="rv-thumb-empty">No pages</div>
54
+ }
55
+ </div>
56
+
57
+ </rv-sidebar-container>
58
+ `, isInline: true, styles: [".rv-thumbnail-list{padding:12px 8px;display:flex;flex-direction:column;align-items:center;gap:12px}.rv-thumbnail-item{cursor:pointer;display:flex;flex-direction:column;align-items:center;gap:4px;padding:6px;border-radius:6px;border:2px solid transparent;transition:all .15s ease}.rv-thumbnail-item:hover:not(.rv-thumb-active){background:#00000008;border-color:#00000014}:host-context(.rv-theme-dark) .rv-thumbnail-item:hover:not(.rv-thumb-active){background:#ffffff0a;border-color:#ffffff14}.rv-thumb-active{border-color:#1976d2;background:#1976d20f}:host-context(.rv-theme-dark) .rv-thumb-active{border-color:#64b5f6;background:#64b5f61a}.rv-thumbnail-preview{width:140px;height:198px;overflow:hidden;border-radius:3px;box-shadow:0 1px 4px #00000026;background:#fff}.rv-thumbnail-scale{transform:scale(.167);transform-origin:top left;width:600%;height:600%;pointer-events:none}.rv-thumb-label{font-size:11px;font-weight:500;color:#666}:host-context(.rv-theme-dark) .rv-thumb-label{color:#a0a0b0}.rv-thumb-active .rv-thumb-label{color:#1976d2;font-weight:600}:host-context(.rv-theme-dark) .rv-thumb-active .rv-thumb-label{color:#64b5f6}.rv-thumb-empty{padding:24px;color:#999;font-size:12px;text-align:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: SidebarContainerComponent, selector: "rv-sidebar-container", inputs: ["isOpen", "title"], outputs: ["closed"] }, { kind: "component", type: ReportPageComponent, selector: "rv-report-page", inputs: ["pageWidth", "pageHeight", "headerSection", "footerSection", "bodyComponents", "globalStyles", "data", "pageNumber", "totalPages", "reportTitle"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
59
+ }
60
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ThumbnailSidebarComponent, decorators: [{
61
+ type: Component,
62
+ args: [{ selector: 'rv-thumbnail-sidebar', standalone: true, imports: [CommonModule, SidebarContainerComponent, ReportPageComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
63
+ <rv-sidebar-container
64
+ [isOpen]="state.thumbnailSidebarOpen()"
65
+ title="Pages"
66
+ (closed)="state.toggleThumbnailSidebar()">
67
+
68
+ <div class="rv-thumbnail-list">
69
+ @for (page of pages; track page.index) {
70
+ <div class="rv-thumbnail-item"
71
+ [class.rv-thumb-active]="page.index === state.currentPage() - 1"
72
+ (click)="onPageClick(page.index + 1)">
73
+ <div class="rv-thumbnail-preview">
74
+ <div class="rv-thumbnail-scale">
75
+ <rv-report-page
76
+ [pageWidth]="pageWidth"
77
+ [pageHeight]="pageHeight"
78
+ [headerSection]="page.header"
79
+ [footerSection]="page.footer"
80
+ [bodyComponents]="page.components"
81
+ [globalStyles]="globalStyles"
82
+ [data]="data"
83
+ [pageNumber]="page.index + 1"
84
+ [totalPages]="pages.length">
85
+ </rv-report-page>
86
+ </div>
87
+ </div>
88
+ <span class="rv-thumb-label">{{ page.index + 1 }}</span>
89
+ </div>
90
+ }
91
+
92
+ @if (!pages || pages.length === 0) {
93
+ <div class="rv-thumb-empty">No pages</div>
94
+ }
95
+ </div>
96
+
97
+ </rv-sidebar-container>
98
+ `, styles: [".rv-thumbnail-list{padding:12px 8px;display:flex;flex-direction:column;align-items:center;gap:12px}.rv-thumbnail-item{cursor:pointer;display:flex;flex-direction:column;align-items:center;gap:4px;padding:6px;border-radius:6px;border:2px solid transparent;transition:all .15s ease}.rv-thumbnail-item:hover:not(.rv-thumb-active){background:#00000008;border-color:#00000014}:host-context(.rv-theme-dark) .rv-thumbnail-item:hover:not(.rv-thumb-active){background:#ffffff0a;border-color:#ffffff14}.rv-thumb-active{border-color:#1976d2;background:#1976d20f}:host-context(.rv-theme-dark) .rv-thumb-active{border-color:#64b5f6;background:#64b5f61a}.rv-thumbnail-preview{width:140px;height:198px;overflow:hidden;border-radius:3px;box-shadow:0 1px 4px #00000026;background:#fff}.rv-thumbnail-scale{transform:scale(.167);transform-origin:top left;width:600%;height:600%;pointer-events:none}.rv-thumb-label{font-size:11px;font-weight:500;color:#666}:host-context(.rv-theme-dark) .rv-thumb-label{color:#a0a0b0}.rv-thumb-active .rv-thumb-label{color:#1976d2;font-weight:600}:host-context(.rv-theme-dark) .rv-thumb-active .rv-thumb-label{color:#64b5f6}.rv-thumb-empty{padding:24px;color:#999;font-size:12px;text-align:center}\n"] }]
99
+ }], propDecorators: { pages: [{
100
+ type: Input
101
+ }], pageWidth: [{
102
+ type: Input
103
+ }], pageHeight: [{
104
+ type: Input
105
+ }], globalStyles: [{
106
+ type: Input
107
+ }], data: [{
108
+ type: Input
109
+ }] } });
110
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGh1bWJuYWlsLXNpZGViYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcmVwb3J0LXZpZXdlci9zcmMvbGliL3NpZGViYXIvdGh1bWJuYWlsLXNpZGViYXIuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCxTQUFTLEVBQUUsdUJBQXVCLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQzFELE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUUvQyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0RSxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUMxRSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQzs7QUFpSXRFLE1BQU0sT0FBTyx5QkFBeUI7SUEvSHRDO1FBZ0lXLFVBQUssR0FBZSxFQUFFLENBQUM7UUFDdkIsY0FBUyxHQUFHLEdBQUcsQ0FBQztRQUNoQixlQUFVLEdBQUcsR0FBRyxDQUFDO1FBQ2pCLGlCQUFZLEdBQWlCLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsZUFBZSxFQUFFLE1BQU0sRUFBRSxDQUFDO1FBQy9HLFNBQUksR0FBUSxJQUFJLENBQUM7UUFFMUIsaUJBQVksR0FBRyxNQUFNLEVBQVUsQ0FBQztRQUVoQyxVQUFLLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUM7S0FNcEM7SUFKQyxXQUFXLENBQUMsSUFBWTtRQUN0QixJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMxQixJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixDQUFDOytHQWRVLHlCQUF5QjttR0FBekIseUJBQXlCLHFQQTFIMUI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9DVCxvd0NBdENTLFlBQVksK0JBQUUseUJBQXlCLG1IQUFFLG1CQUFtQjs7NEZBNEgzRCx5QkFBeUI7a0JBL0hyQyxTQUFTOytCQUNFLHNCQUFzQixjQUNwQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUseUJBQXlCLEVBQUUsbUJBQW1CLENBQUMsbUJBQ3RELHVCQUF1QixDQUFDLE1BQU0sWUFDckM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9DVDs4QkF1RlEsS0FBSztzQkFBYixLQUFLO2dCQUNHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBQ0csVUFBVTtzQkFBbEIsS0FBSztnQkFDRyxZQUFZO3NCQUFwQixLQUFLO2dCQUNHLElBQUk7c0JBQVosS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENvbXBvbmVudCwgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIElucHV0LCBpbmplY3QsIG91dHB1dFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBQYWdlRGF0YSwgR2xvYmFsU3R5bGVzIH0gZnJvbSAnLi4vbW9kZWxzL3JlcG9ydC10ZW1wbGF0ZS5tb2RlbCc7XG5pbXBvcnQgeyBWaWV3ZXJTdGF0ZVNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy92aWV3ZXItc3RhdGUuc2VydmljZSc7XG5pbXBvcnQgeyBTaWRlYmFyQ29udGFpbmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zaWRlYmFyLWNvbnRhaW5lci5jb21wb25lbnQnO1xuaW1wb3J0IHsgUmVwb3J0UGFnZUNvbXBvbmVudCB9IGZyb20gJy4uL2xheW91dC9yZXBvcnQtcGFnZS5jb21wb25lbnQnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdydi10aHVtYm5haWwtc2lkZWJhcicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFNpZGViYXJDb250YWluZXJDb21wb25lbnQsIFJlcG9ydFBhZ2VDb21wb25lbnRdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgdGVtcGxhdGU6IGBcbiAgICA8cnYtc2lkZWJhci1jb250YWluZXJcbiAgICAgIFtpc09wZW5dPVwic3RhdGUudGh1bWJuYWlsU2lkZWJhck9wZW4oKVwiXG4gICAgICB0aXRsZT1cIlBhZ2VzXCJcbiAgICAgIChjbG9zZWQpPVwic3RhdGUudG9nZ2xlVGh1bWJuYWlsU2lkZWJhcigpXCI+XG5cbiAgICAgIDxkaXYgY2xhc3M9XCJydi10aHVtYm5haWwtbGlzdFwiPlxuICAgICAgICBAZm9yIChwYWdlIG9mIHBhZ2VzOyB0cmFjayBwYWdlLmluZGV4KSB7XG4gICAgICAgICAgPGRpdiBjbGFzcz1cInJ2LXRodW1ibmFpbC1pdGVtXCJcbiAgICAgICAgICAgICAgIFtjbGFzcy5ydi10aHVtYi1hY3RpdmVdPVwicGFnZS5pbmRleCA9PT0gc3RhdGUuY3VycmVudFBhZ2UoKSAtIDFcIlxuICAgICAgICAgICAgICAgKGNsaWNrKT1cIm9uUGFnZUNsaWNrKHBhZ2UuaW5kZXggKyAxKVwiPlxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cInJ2LXRodW1ibmFpbC1wcmV2aWV3XCI+XG4gICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJydi10aHVtYm5haWwtc2NhbGVcIj5cbiAgICAgICAgICAgICAgICA8cnYtcmVwb3J0LXBhZ2VcbiAgICAgICAgICAgICAgICAgIFtwYWdlV2lkdGhdPVwicGFnZVdpZHRoXCJcbiAgICAgICAgICAgICAgICAgIFtwYWdlSGVpZ2h0XT1cInBhZ2VIZWlnaHRcIlxuICAgICAgICAgICAgICAgICAgW2hlYWRlclNlY3Rpb25dPVwicGFnZS5oZWFkZXJcIlxuICAgICAgICAgICAgICAgICAgW2Zvb3RlclNlY3Rpb25dPVwicGFnZS5mb290ZXJcIlxuICAgICAgICAgICAgICAgICAgW2JvZHlDb21wb25lbnRzXT1cInBhZ2UuY29tcG9uZW50c1wiXG4gICAgICAgICAgICAgICAgICBbZ2xvYmFsU3R5bGVzXT1cImdsb2JhbFN0eWxlc1wiXG4gICAgICAgICAgICAgICAgICBbZGF0YV09XCJkYXRhXCJcbiAgICAgICAgICAgICAgICAgIFtwYWdlTnVtYmVyXT1cInBhZ2UuaW5kZXggKyAxXCJcbiAgICAgICAgICAgICAgICAgIFt0b3RhbFBhZ2VzXT1cInBhZ2VzLmxlbmd0aFwiPlxuICAgICAgICAgICAgICAgIDwvcnYtcmVwb3J0LXBhZ2U+XG4gICAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgICA8c3BhbiBjbGFzcz1cInJ2LXRodW1iLWxhYmVsXCI+e3sgcGFnZS5pbmRleCArIDEgfX08L3NwYW4+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIH1cblxuICAgICAgICBAaWYgKCFwYWdlcyB8fCBwYWdlcy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwicnYtdGh1bWItZW1wdHlcIj5ObyBwYWdlczwvZGl2PlxuICAgICAgICB9XG4gICAgICA8L2Rpdj5cblxuICAgIDwvcnYtc2lkZWJhci1jb250YWluZXI+XG4gIGAsXG4gIHN0eWxlczogW2BcbiAgICAucnYtdGh1bWJuYWlsLWxpc3Qge1xuICAgICAgcGFkZGluZzogMTJweCA4cHg7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICBnYXA6IDEycHg7XG4gICAgfVxuXG4gICAgLnJ2LXRodW1ibmFpbC1pdGVtIHtcbiAgICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIGdhcDogNHB4O1xuICAgICAgcGFkZGluZzogNnB4O1xuICAgICAgYm9yZGVyLXJhZGl1czogNnB4O1xuICAgICAgYm9yZGVyOiAycHggc29saWQgdHJhbnNwYXJlbnQ7XG4gICAgICB0cmFuc2l0aW9uOiBhbGwgMC4xNXMgZWFzZTtcbiAgICB9XG5cbiAgICAucnYtdGh1bWJuYWlsLWl0ZW06aG92ZXI6bm90KC5ydi10aHVtYi1hY3RpdmUpIHtcbiAgICAgIGJhY2tncm91bmQ6IHJnYmEoMCwgMCwgMCwgMC4wMyk7XG4gICAgICBib3JkZXItY29sb3I6IHJnYmEoMCwgMCwgMCwgMC4wOCk7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXRodW1ibmFpbC1pdGVtOmhvdmVyOm5vdCgucnYtdGh1bWItYWN0aXZlKSB7XG4gICAgICBiYWNrZ3JvdW5kOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMDQpO1xuICAgICAgYm9yZGVyLWNvbG9yOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuMDgpO1xuICAgIH1cblxuICAgIC5ydi10aHVtYi1hY3RpdmUge1xuICAgICAgYm9yZGVyLWNvbG9yOiAjMTk3NmQyO1xuICAgICAgYmFja2dyb3VuZDogcmdiYSgyNSwgMTE4LCAyMTAsIDAuMDYpO1xuICAgIH1cblxuICAgIDpob3N0LWNvbnRleHQoLnJ2LXRoZW1lLWRhcmspIC5ydi10aHVtYi1hY3RpdmUge1xuICAgICAgYm9yZGVyLWNvbG9yOiAjNjRiNWY2O1xuICAgICAgYmFja2dyb3VuZDogcmdiYSgxMDAsIDE4MSwgMjQ2LCAwLjEpO1xuICAgIH1cblxuICAgIC5ydi10aHVtYm5haWwtcHJldmlldyB7XG4gICAgICB3aWR0aDogMTQwcHg7XG4gICAgICBoZWlnaHQ6IDE5OHB4OyAvKiBBNCByYXRpbyAoMjEwLzI5NyAqIDE0MCkgKi9cbiAgICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgICBib3JkZXItcmFkaXVzOiAzcHg7XG4gICAgICBib3gtc2hhZG93OiAwIDFweCA0cHggcmdiYSgwLCAwLCAwLCAwLjE1KTtcbiAgICAgIGJhY2tncm91bmQ6IHdoaXRlO1xuICAgIH1cblxuICAgIC5ydi10aHVtYm5haWwtc2NhbGUge1xuICAgICAgdHJhbnNmb3JtOiBzY2FsZSgwLjE2Nyk7IC8qIDE0MHB4IC8gMjEwbW0gKiB+MC4yNSByb3VnaCBzY2FsZSAqL1xuICAgICAgdHJhbnNmb3JtLW9yaWdpbjogdG9wIGxlZnQ7XG4gICAgICB3aWR0aDogNjAwJTtcbiAgICAgIGhlaWdodDogNjAwJTtcbiAgICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xuICAgIH1cblxuICAgIC5ydi10aHVtYi1sYWJlbCB7XG4gICAgICBmb250LXNpemU6IDExcHg7XG4gICAgICBmb250LXdlaWdodDogNTAwO1xuICAgICAgY29sb3I6ICM2NjY7XG4gICAgfVxuXG4gICAgOmhvc3QtY29udGV4dCgucnYtdGhlbWUtZGFyaykgLnJ2LXRodW1iLWxhYmVsIHtcbiAgICAgIGNvbG9yOiAjYTBhMGIwO1xuICAgIH1cblxuICAgIC5ydi10aHVtYi1hY3RpdmUgLnJ2LXRodW1iLWxhYmVsIHtcbiAgICAgIGNvbG9yOiAjMTk3NmQyO1xuICAgICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICB9XG5cbiAgICA6aG9zdC1jb250ZXh0KC5ydi10aGVtZS1kYXJrKSAucnYtdGh1bWItYWN0aXZlIC5ydi10aHVtYi1sYWJlbCB7XG4gICAgICBjb2xvcjogIzY0YjVmNjtcbiAgICB9XG5cbiAgICAucnYtdGh1bWItZW1wdHkge1xuICAgICAgcGFkZGluZzogMjRweDtcbiAgICAgIGNvbG9yOiAjOTk5O1xuICAgICAgZm9udC1zaXplOiAxMnB4O1xuICAgICAgdGV4dC1hbGlnbjogY2VudGVyO1xuICAgIH1cbiAgYF1cbn0pXG5leHBvcnQgY2xhc3MgVGh1bWJuYWlsU2lkZWJhckNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIHBhZ2VzOiBQYWdlRGF0YVtdID0gW107XG4gIEBJbnB1dCgpIHBhZ2VXaWR0aCA9IDIxMDtcbiAgQElucHV0KCkgcGFnZUhlaWdodCA9IDI5NztcbiAgQElucHV0KCkgZ2xvYmFsU3R5bGVzOiBHbG9iYWxTdHlsZXMgPSB7IGZvbnRGYW1pbHk6ICdBcmlhbCcsIGZvbnRTaXplOiAnMTFweCcsIGNvbG9yOiAnIzAwMCcsIGJhY2tncm91bmRDb2xvcjogJyNmZmYnIH07XG4gIEBJbnB1dCgpIGRhdGE6IGFueSA9IG51bGw7XG5cbiAgcGFnZVNlbGVjdGVkID0gb3V0cHV0PG51bWJlcj4oKTtcblxuICBzdGF0ZSA9IGluamVjdChWaWV3ZXJTdGF0ZVNlcnZpY2UpO1xuXG4gIG9uUGFnZUNsaWNrKHBhZ2U6IG51bWJlcik6IHZvaWQge1xuICAgIHRoaXMuc3RhdGUuZ29Ub1BhZ2UocGFnZSk7XG4gICAgdGhpcy5wYWdlU2VsZWN0ZWQuZW1pdChwYWdlKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,155 @@
1
+ import { Component, ChangeDetectionStrategy, Input, inject, output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { MatIconModule } from '@angular/material/icon';
4
+ import { ViewerStateService } from '../services/viewer-state.service';
5
+ import { SidebarContainerComponent } from './sidebar-container.component';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/material/icon";
8
+ export class TocSidebarComponent {
9
+ constructor() {
10
+ this.pages = [];
11
+ this.pageSelected = output();
12
+ this.state = inject(ViewerStateService);
13
+ this.tocItems = [];
14
+ }
15
+ ngOnChanges(changes) {
16
+ if (changes['pages']) {
17
+ this.buildToc();
18
+ }
19
+ }
20
+ onItemClick(item) {
21
+ this.state.goToPage(item.pageIndex + 1);
22
+ this.pageSelected.emit(item.pageIndex + 1);
23
+ }
24
+ buildToc() {
25
+ this.tocItems = [];
26
+ if (!this.pages || this.pages.length === 0)
27
+ return;
28
+ for (const page of this.pages) {
29
+ const components = [
30
+ ...(page.header?.components || []),
31
+ ...page.components,
32
+ ...(page.footer?.components || []),
33
+ ];
34
+ for (const comp of components) {
35
+ const tocItem = this.componentToTocItem(comp, page.index);
36
+ if (tocItem) {
37
+ this.tocItems.push(tocItem);
38
+ }
39
+ }
40
+ }
41
+ }
42
+ componentToTocItem(comp, pageIndex) {
43
+ // Only create TOC items for text/field components with certain characteristics
44
+ if (comp.type !== 'text' && comp.type !== 'field')
45
+ return null;
46
+ const content = comp.content || '';
47
+ if (!content.trim())
48
+ return null;
49
+ // Detect headings by font size or bold style
50
+ const fontSize = this.parseFontSize(comp.style?.fontSize);
51
+ const isBold = comp.style?.fontWeight === 'bold' || comp.style?.fontWeight === '700';
52
+ const isLarge = fontSize >= 16;
53
+ const isMedium = fontSize >= 13;
54
+ // Only include text that looks like a heading
55
+ if (!isLarge && !isBold)
56
+ return null;
57
+ let level = 2;
58
+ let icon = 'text_fields';
59
+ if (isLarge && isBold) {
60
+ level = 0;
61
+ icon = 'title';
62
+ }
63
+ else if (isLarge) {
64
+ level = 1;
65
+ icon = 'text_fields';
66
+ }
67
+ else if (isBold) {
68
+ level = 2;
69
+ icon = 'short_text';
70
+ }
71
+ // Clean content - remove HTML tags and template tokens
72
+ const label = content
73
+ .replace(/<[^>]*>/g, '')
74
+ .replace(/\{\{[^}]*\}\}/g, '')
75
+ .trim();
76
+ if (!label || label.length < 2)
77
+ return null;
78
+ return {
79
+ id: comp.id,
80
+ label: label.substring(0, 60) + (label.length > 60 ? '...' : ''),
81
+ pageIndex,
82
+ level,
83
+ icon,
84
+ };
85
+ }
86
+ parseFontSize(fontSize) {
87
+ if (!fontSize)
88
+ return 11;
89
+ const match = fontSize.match(/(\d+(?:\.\d+)?)/);
90
+ return match ? parseFloat(match[1]) : 11;
91
+ }
92
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TocSidebarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
93
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: TocSidebarComponent, isStandalone: true, selector: "rv-toc-sidebar", inputs: { pages: "pages" }, outputs: { pageSelected: "pageSelected" }, usesOnChanges: true, ngImport: i0, template: `
94
+ <rv-sidebar-container
95
+ [isOpen]="state.tocSidebarOpen()"
96
+ title="Document Map"
97
+ (closed)="state.toggleTocSidebar()">
98
+
99
+ <div class="rv-toc-tree">
100
+ @for (item of tocItems; track item.id) {
101
+ <div class="rv-toc-item"
102
+ [class.rv-toc-active]="item.pageIndex === state.currentPage() - 1"
103
+ [style.padding-left.px]="12 + item.level * 16"
104
+ (click)="onItemClick(item)">
105
+ <mat-icon class="rv-toc-icon">{{ item.icon }}</mat-icon>
106
+ <span class="rv-toc-label">{{ item.label }}</span>
107
+ <span class="rv-toc-page">{{ item.pageIndex + 1 }}</span>
108
+ </div>
109
+ }
110
+
111
+ @if (tocItems.length === 0) {
112
+ <div class="rv-toc-empty">
113
+ <mat-icon>toc</mat-icon>
114
+ <p>No document structure found</p>
115
+ </div>
116
+ }
117
+ </div>
118
+
119
+ </rv-sidebar-container>
120
+ `, isInline: true, styles: [".rv-toc-tree{padding:8px 0}.rv-toc-item{display:flex;align-items:center;gap:8px;padding:8px 12px;cursor:pointer;font-size:12px;transition:background .12s ease;border-left:3px solid transparent}.rv-toc-item:hover{background:#0000000a}:host-context(.rv-theme-dark) .rv-toc-item:hover{background:#ffffff0a}.rv-toc-active{background:#1976d20f!important;border-left-color:#1976d2}:host-context(.rv-theme-dark) .rv-toc-active{background:#64b5f61a!important;border-left-color:#64b5f6}.rv-toc-icon{font-size:16px!important;width:16px!important;height:16px!important;color:#888;flex-shrink:0}:host-context(.rv-theme-dark) .rv-toc-icon{color:#889}.rv-toc-active .rv-toc-icon{color:#1976d2}:host-context(.rv-theme-dark) .rv-toc-active .rv-toc-icon{color:#64b5f6}.rv-toc-label{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;color:#333}:host-context(.rv-theme-dark) .rv-toc-label{color:#d0d0d8}.rv-toc-active .rv-toc-label{color:#1976d2;font-weight:600}:host-context(.rv-theme-dark) .rv-toc-active .rv-toc-label{color:#64b5f6}.rv-toc-page{font-size:10px;color:#999;flex-shrink:0;min-width:18px;text-align:right}:host-context(.rv-theme-dark) .rv-toc-page{color:#777}.rv-toc-empty{display:flex;flex-direction:column;align-items:center;padding:32px 16px;color:#999;gap:8px}.rv-toc-empty .mat-icon{font-size:32px;width:32px;height:32px;opacity:.4}.rv-toc-empty p{margin:0;font-size:12px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: SidebarContainerComponent, selector: "rv-sidebar-container", inputs: ["isOpen", "title"], outputs: ["closed"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
121
+ }
122
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TocSidebarComponent, decorators: [{
123
+ type: Component,
124
+ args: [{ selector: 'rv-toc-sidebar', standalone: true, imports: [CommonModule, MatIconModule, SidebarContainerComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
125
+ <rv-sidebar-container
126
+ [isOpen]="state.tocSidebarOpen()"
127
+ title="Document Map"
128
+ (closed)="state.toggleTocSidebar()">
129
+
130
+ <div class="rv-toc-tree">
131
+ @for (item of tocItems; track item.id) {
132
+ <div class="rv-toc-item"
133
+ [class.rv-toc-active]="item.pageIndex === state.currentPage() - 1"
134
+ [style.padding-left.px]="12 + item.level * 16"
135
+ (click)="onItemClick(item)">
136
+ <mat-icon class="rv-toc-icon">{{ item.icon }}</mat-icon>
137
+ <span class="rv-toc-label">{{ item.label }}</span>
138
+ <span class="rv-toc-page">{{ item.pageIndex + 1 }}</span>
139
+ </div>
140
+ }
141
+
142
+ @if (tocItems.length === 0) {
143
+ <div class="rv-toc-empty">
144
+ <mat-icon>toc</mat-icon>
145
+ <p>No document structure found</p>
146
+ </div>
147
+ }
148
+ </div>
149
+
150
+ </rv-sidebar-container>
151
+ `, styles: [".rv-toc-tree{padding:8px 0}.rv-toc-item{display:flex;align-items:center;gap:8px;padding:8px 12px;cursor:pointer;font-size:12px;transition:background .12s ease;border-left:3px solid transparent}.rv-toc-item:hover{background:#0000000a}:host-context(.rv-theme-dark) .rv-toc-item:hover{background:#ffffff0a}.rv-toc-active{background:#1976d20f!important;border-left-color:#1976d2}:host-context(.rv-theme-dark) .rv-toc-active{background:#64b5f61a!important;border-left-color:#64b5f6}.rv-toc-icon{font-size:16px!important;width:16px!important;height:16px!important;color:#888;flex-shrink:0}:host-context(.rv-theme-dark) .rv-toc-icon{color:#889}.rv-toc-active .rv-toc-icon{color:#1976d2}:host-context(.rv-theme-dark) .rv-toc-active .rv-toc-icon{color:#64b5f6}.rv-toc-label{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;color:#333}:host-context(.rv-theme-dark) .rv-toc-label{color:#d0d0d8}.rv-toc-active .rv-toc-label{color:#1976d2;font-weight:600}:host-context(.rv-theme-dark) .rv-toc-active .rv-toc-label{color:#64b5f6}.rv-toc-page{font-size:10px;color:#999;flex-shrink:0;min-width:18px;text-align:right}:host-context(.rv-theme-dark) .rv-toc-page{color:#777}.rv-toc-empty{display:flex;flex-direction:column;align-items:center;padding:32px 16px;color:#999;gap:8px}.rv-toc-empty .mat-icon{font-size:32px;width:32px;height:32px;opacity:.4}.rv-toc-empty p{margin:0;font-size:12px}\n"] }]
152
+ }], propDecorators: { pages: [{
153
+ type: Input
154
+ }] } });
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toc-sidebar.component.js","sourceRoot":"","sources":["../../../../../projects/report-viewer/src/lib/sidebar/toc-sidebar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, ChangeDetectionStrategy, Input, inject, output, OnChanges, SimpleChanges\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { MatIconModule } from '@angular/material/icon';\nimport { ComponentModel } from '../models/component.model';\nimport { PageData, TocItem } from '../models/report-template.model';\nimport { ViewerStateService } from '../services/viewer-state.service';\nimport { SidebarContainerComponent } from './sidebar-container.component';\n\n@Component({\n  selector: 'rv-toc-sidebar',\n  standalone: true,\n  imports: [CommonModule, MatIconModule, SidebarContainerComponent],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `\n    <rv-sidebar-container\n      [isOpen]=\"state.tocSidebarOpen()\"\n      title=\"Document Map\"\n      (closed)=\"state.toggleTocSidebar()\">\n\n      <div class=\"rv-toc-tree\">\n        @for (item of tocItems; track item.id) {\n          <div class=\"rv-toc-item\"\n               [class.rv-toc-active]=\"item.pageIndex === state.currentPage() - 1\"\n               [style.padding-left.px]=\"12 + item.level * 16\"\n               (click)=\"onItemClick(item)\">\n            <mat-icon class=\"rv-toc-icon\">{{ item.icon }}</mat-icon>\n            <span class=\"rv-toc-label\">{{ item.label }}</span>\n            <span class=\"rv-toc-page\">{{ item.pageIndex + 1 }}</span>\n          </div>\n        }\n\n        @if (tocItems.length === 0) {\n          <div class=\"rv-toc-empty\">\n            <mat-icon>toc</mat-icon>\n            <p>No document structure found</p>\n          </div>\n        }\n      </div>\n\n    </rv-sidebar-container>\n  `,\n  styles: [`\n    .rv-toc-tree {\n      padding: 8px 0;\n    }\n\n    .rv-toc-item {\n      display: flex;\n      align-items: center;\n      gap: 8px;\n      padding: 8px 12px;\n      cursor: pointer;\n      font-size: 12px;\n      transition: background 0.12s ease;\n      border-left: 3px solid transparent;\n    }\n\n    .rv-toc-item:hover {\n      background: rgba(0, 0, 0, 0.04);\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-item:hover {\n      background: rgba(255, 255, 255, 0.04);\n    }\n\n    .rv-toc-active {\n      background: rgba(25, 118, 210, 0.06) !important;\n      border-left-color: #1976d2;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-active {\n      background: rgba(100, 181, 246, 0.1) !important;\n      border-left-color: #64b5f6;\n    }\n\n    .rv-toc-icon {\n      font-size: 16px !important;\n      width: 16px !important;\n      height: 16px !important;\n      color: #888;\n      flex-shrink: 0;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-icon {\n      color: #888899;\n    }\n\n    .rv-toc-active .rv-toc-icon {\n      color: #1976d2;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-active .rv-toc-icon {\n      color: #64b5f6;\n    }\n\n    .rv-toc-label {\n      flex: 1;\n      overflow: hidden;\n      text-overflow: ellipsis;\n      white-space: nowrap;\n      color: #333;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-label {\n      color: #d0d0d8;\n    }\n\n    .rv-toc-active .rv-toc-label {\n      color: #1976d2;\n      font-weight: 600;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-active .rv-toc-label {\n      color: #64b5f6;\n    }\n\n    .rv-toc-page {\n      font-size: 10px;\n      color: #999;\n      flex-shrink: 0;\n      min-width: 18px;\n      text-align: right;\n    }\n\n    :host-context(.rv-theme-dark) .rv-toc-page {\n      color: #777;\n    }\n\n    .rv-toc-empty {\n      display: flex;\n      flex-direction: column;\n      align-items: center;\n      padding: 32px 16px;\n      color: #999;\n      gap: 8px;\n    }\n\n    .rv-toc-empty .mat-icon {\n      font-size: 32px;\n      width: 32px;\n      height: 32px;\n      opacity: 0.4;\n    }\n\n    .rv-toc-empty p {\n      margin: 0;\n      font-size: 12px;\n    }\n  `]\n})\nexport class TocSidebarComponent implements OnChanges {\n  @Input() pages: PageData[] = [];\n\n  pageSelected = output<number>();\n\n  state = inject(ViewerStateService);\n\n  tocItems: TocItem[] = [];\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['pages']) {\n      this.buildToc();\n    }\n  }\n\n  onItemClick(item: TocItem): void {\n    this.state.goToPage(item.pageIndex + 1);\n    this.pageSelected.emit(item.pageIndex + 1);\n  }\n\n  private buildToc(): void {\n    this.tocItems = [];\n\n    if (!this.pages || this.pages.length === 0) return;\n\n    for (const page of this.pages) {\n      const components = [\n        ...(page.header?.components || []),\n        ...page.components,\n        ...(page.footer?.components || []),\n      ];\n\n      for (const comp of components) {\n        const tocItem = this.componentToTocItem(comp, page.index);\n        if (tocItem) {\n          this.tocItems.push(tocItem);\n        }\n      }\n    }\n  }\n\n  private componentToTocItem(comp: ComponentModel, pageIndex: number): TocItem | null {\n    // Only create TOC items for text/field components with certain characteristics\n    if (comp.type !== 'text' && comp.type !== 'field') return null;\n\n    const content = comp.content || '';\n    if (!content.trim()) return null;\n\n    // Detect headings by font size or bold style\n    const fontSize = this.parseFontSize(comp.style?.fontSize);\n    const isBold = comp.style?.fontWeight === 'bold' || comp.style?.fontWeight === '700';\n    const isLarge = fontSize >= 16;\n    const isMedium = fontSize >= 13;\n\n    // Only include text that looks like a heading\n    if (!isLarge && !isBold) return null;\n\n    let level = 2;\n    let icon = 'text_fields';\n\n    if (isLarge && isBold) {\n      level = 0;\n      icon = 'title';\n    } else if (isLarge) {\n      level = 1;\n      icon = 'text_fields';\n    } else if (isBold) {\n      level = 2;\n      icon = 'short_text';\n    }\n\n    // Clean content - remove HTML tags and template tokens\n    const label = content\n      .replace(/<[^>]*>/g, '')\n      .replace(/\\{\\{[^}]*\\}\\}/g, '')\n      .trim();\n\n    if (!label || label.length < 2) return null;\n\n    return {\n      id: comp.id,\n      label: label.substring(0, 60) + (label.length > 60 ? '...' : ''),\n      pageIndex,\n      level,\n      icon,\n    };\n  }\n\n  private parseFontSize(fontSize: string | undefined): number {\n    if (!fontSize) return 11;\n    const match = fontSize.match(/(\\d+(?:\\.\\d+)?)/);\n    return match ? parseFloat(match[1]) : 11;\n  }\n}\n"]}