kviewer 0.0.1

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 (197) hide show
  1. package/README.md +134 -0
  2. package/dist/module.d.mts +15 -0
  3. package/dist/module.json +9 -0
  4. package/dist/module.mjs +26 -0
  5. package/dist/runtime/annotation/engine/config.d.ts +52 -0
  6. package/dist/runtime/annotation/engine/config.js +283 -0
  7. package/dist/runtime/annotation/engine/const.d.ts +6 -0
  8. package/dist/runtime/annotation/engine/const.js +7 -0
  9. package/dist/runtime/annotation/engine/cursor-preview.d.ts +2 -0
  10. package/dist/runtime/annotation/engine/cursor-preview.js +88 -0
  11. package/dist/runtime/annotation/engine/editor/editor.d.ts +69 -0
  12. package/dist/runtime/annotation/engine/editor/editor.js +233 -0
  13. package/dist/runtime/annotation/engine/editor/selector.d.ts +74 -0
  14. package/dist/runtime/annotation/engine/editor/selector.js +594 -0
  15. package/dist/runtime/annotation/engine/import-normalize.d.ts +5 -0
  16. package/dist/runtime/annotation/engine/import-normalize.js +99 -0
  17. package/dist/runtime/annotation/engine/input-device.d.ts +53 -0
  18. package/dist/runtime/annotation/engine/input-device.js +64 -0
  19. package/dist/runtime/annotation/engine/painter.d.ts +97 -0
  20. package/dist/runtime/annotation/engine/painter.js +591 -0
  21. package/dist/runtime/annotation/engine/store.d.ts +11 -0
  22. package/dist/runtime/annotation/engine/store.js +47 -0
  23. package/dist/runtime/annotation/engine/tools/arrow.d.ts +22 -0
  24. package/dist/runtime/annotation/engine/tools/arrow.js +126 -0
  25. package/dist/runtime/annotation/engine/tools/circle.d.ts +45 -0
  26. package/dist/runtime/annotation/engine/tools/circle.js +148 -0
  27. package/dist/runtime/annotation/engine/tools/cloud.d.ts +50 -0
  28. package/dist/runtime/annotation/engine/tools/cloud.js +244 -0
  29. package/dist/runtime/annotation/engine/tools/free-highlight.d.ts +43 -0
  30. package/dist/runtime/annotation/engine/tools/free-highlight.js +165 -0
  31. package/dist/runtime/annotation/engine/tools/free-text.d.ts +27 -0
  32. package/dist/runtime/annotation/engine/tools/free-text.js +114 -0
  33. package/dist/runtime/annotation/engine/tools/freehand.d.ts +44 -0
  34. package/dist/runtime/annotation/engine/tools/freehand.js +151 -0
  35. package/dist/runtime/annotation/engine/tools/highlight.d.ts +87 -0
  36. package/dist/runtime/annotation/engine/tools/highlight.js +215 -0
  37. package/dist/runtime/annotation/engine/tools/note.d.ts +9 -0
  38. package/dist/runtime/annotation/engine/tools/note.js +34 -0
  39. package/dist/runtime/annotation/engine/tools/rectangle.d.ts +45 -0
  40. package/dist/runtime/annotation/engine/tools/rectangle.js +142 -0
  41. package/dist/runtime/annotation/engine/tools/signature.d.ts +16 -0
  42. package/dist/runtime/annotation/engine/tools/signature.js +74 -0
  43. package/dist/runtime/annotation/engine/tools/stamp.d.ts +18 -0
  44. package/dist/runtime/annotation/engine/tools/stamp.js +94 -0
  45. package/dist/runtime/annotation/engine/types.d.ts +170 -0
  46. package/dist/runtime/annotation/engine/types.js +67 -0
  47. package/dist/runtime/annotation/engine/utils.d.ts +40 -0
  48. package/dist/runtime/annotation/engine/utils.js +257 -0
  49. package/dist/runtime/annotation/parsers/parseFormFields.d.ts +9 -0
  50. package/dist/runtime/annotation/parsers/parseFormFields.js +101 -0
  51. package/dist/runtime/annotation/pdf-export/download.d.ts +1 -0
  52. package/dist/runtime/annotation/pdf-export/download.js +10 -0
  53. package/dist/runtime/annotation/pdf-export/export-form-fields.d.ts +9 -0
  54. package/dist/runtime/annotation/pdf-export/export-form-fields.js +90 -0
  55. package/dist/runtime/annotation/pdf-export/export.d.ts +15 -0
  56. package/dist/runtime/annotation/pdf-export/export.js +145 -0
  57. package/dist/runtime/annotation/pdf-export/parse.d.ts +10 -0
  58. package/dist/runtime/annotation/pdf-export/parse.js +19 -0
  59. package/dist/runtime/annotation/pdf-export/parse_circle.d.ts +4 -0
  60. package/dist/runtime/annotation/pdf-export/parse_circle.js +41 -0
  61. package/dist/runtime/annotation/pdf-export/parse_freetext.d.ts +4 -0
  62. package/dist/runtime/annotation/pdf-export/parse_freetext.js +54 -0
  63. package/dist/runtime/annotation/pdf-export/parse_highlight.d.ts +4 -0
  64. package/dist/runtime/annotation/pdf-export/parse_highlight.js +134 -0
  65. package/dist/runtime/annotation/pdf-export/parse_ink.d.ts +4 -0
  66. package/dist/runtime/annotation/pdf-export/parse_ink.js +124 -0
  67. package/dist/runtime/annotation/pdf-export/parse_line.d.ts +4 -0
  68. package/dist/runtime/annotation/pdf-export/parse_line.js +71 -0
  69. package/dist/runtime/annotation/pdf-export/parse_polyline.d.ts +4 -0
  70. package/dist/runtime/annotation/pdf-export/parse_polyline.js +93 -0
  71. package/dist/runtime/annotation/pdf-export/parse_square.d.ts +4 -0
  72. package/dist/runtime/annotation/pdf-export/parse_square.js +41 -0
  73. package/dist/runtime/annotation/pdf-export/parse_stamp.d.ts +4 -0
  74. package/dist/runtime/annotation/pdf-export/parse_stamp.js +195 -0
  75. package/dist/runtime/annotation/pdf-export/parse_strikeout.d.ts +4 -0
  76. package/dist/runtime/annotation/pdf-export/parse_strikeout.js +59 -0
  77. package/dist/runtime/annotation/pdf-export/parse_text.d.ts +4 -0
  78. package/dist/runtime/annotation/pdf-export/parse_text.js +42 -0
  79. package/dist/runtime/annotation/pdf-export/parse_underline.d.ts +4 -0
  80. package/dist/runtime/annotation/pdf-export/parse_underline.js +59 -0
  81. package/dist/runtime/assets/kviewer.css +1 -0
  82. package/dist/runtime/components/AnnotationToolbar.d.vue.ts +3 -0
  83. package/dist/runtime/components/AnnotationToolbar.vue +125 -0
  84. package/dist/runtime/components/AnnotationToolbar.vue.d.ts +3 -0
  85. package/dist/runtime/components/FloatingPageIndicator.d.vue.ts +6 -0
  86. package/dist/runtime/components/FloatingPageIndicator.vue +93 -0
  87. package/dist/runtime/components/FloatingPageIndicator.vue.d.ts +6 -0
  88. package/dist/runtime/components/FormFieldLayer.d.vue.ts +11 -0
  89. package/dist/runtime/components/FormFieldLayer.vue +40 -0
  90. package/dist/runtime/components/FormFieldLayer.vue.d.ts +11 -0
  91. package/dist/runtime/components/PdfPage.d.vue.ts +9 -0
  92. package/dist/runtime/components/PdfPage.vue +199 -0
  93. package/dist/runtime/components/PdfPage.vue.d.ts +9 -0
  94. package/dist/runtime/components/ToolButton.d.vue.ts +13 -0
  95. package/dist/runtime/components/ToolButton.vue +26 -0
  96. package/dist/runtime/components/ToolButton.vue.d.ts +13 -0
  97. package/dist/runtime/components/Toolbar.d.vue.ts +3 -0
  98. package/dist/runtime/components/Toolbar.vue +11 -0
  99. package/dist/runtime/components/Toolbar.vue.d.ts +3 -0
  100. package/dist/runtime/components/Viewer.d.vue.ts +45 -0
  101. package/dist/runtime/components/Viewer.vue +617 -0
  102. package/dist/runtime/components/Viewer.vue.d.ts +45 -0
  103. package/dist/runtime/components/ViewerBar.d.vue.ts +3 -0
  104. package/dist/runtime/components/ViewerBar.vue +91 -0
  105. package/dist/runtime/components/ViewerBar.vue.d.ts +3 -0
  106. package/dist/runtime/components/ViewerTabs.d.vue.ts +381 -0
  107. package/dist/runtime/components/ViewerTabs.vue +171 -0
  108. package/dist/runtime/components/ViewerTabs.vue.d.ts +381 -0
  109. package/dist/runtime/components/form-fields/FormButton.d.vue.ts +7 -0
  110. package/dist/runtime/components/form-fields/FormButton.vue +39 -0
  111. package/dist/runtime/components/form-fields/FormButton.vue.d.ts +7 -0
  112. package/dist/runtime/components/form-fields/FormCheckbox.d.vue.ts +7 -0
  113. package/dist/runtime/components/form-fields/FormCheckbox.vue +28 -0
  114. package/dist/runtime/components/form-fields/FormCheckbox.vue.d.ts +7 -0
  115. package/dist/runtime/components/form-fields/FormDropdown.d.vue.ts +7 -0
  116. package/dist/runtime/components/form-fields/FormDropdown.vue +112 -0
  117. package/dist/runtime/components/form-fields/FormDropdown.vue.d.ts +7 -0
  118. package/dist/runtime/components/form-fields/FormFieldWrapper.d.vue.ts +8 -0
  119. package/dist/runtime/components/form-fields/FormFieldWrapper.vue +41 -0
  120. package/dist/runtime/components/form-fields/FormFieldWrapper.vue.d.ts +8 -0
  121. package/dist/runtime/components/form-fields/FormRadioButton.d.vue.ts +7 -0
  122. package/dist/runtime/components/form-fields/FormRadioButton.vue +30 -0
  123. package/dist/runtime/components/form-fields/FormRadioButton.vue.d.ts +7 -0
  124. package/dist/runtime/components/form-fields/FormSignatureField.d.vue.ts +7 -0
  125. package/dist/runtime/components/form-fields/FormSignatureField.vue +54 -0
  126. package/dist/runtime/components/form-fields/FormSignatureField.vue.d.ts +7 -0
  127. package/dist/runtime/components/form-fields/FormTextField.d.vue.ts +7 -0
  128. package/dist/runtime/components/form-fields/FormTextField.vue +66 -0
  129. package/dist/runtime/components/form-fields/FormTextField.vue.d.ts +7 -0
  130. package/dist/runtime/components/modals/FreeTextModal.d.vue.ts +25 -0
  131. package/dist/runtime/components/modals/FreeTextModal.vue +89 -0
  132. package/dist/runtime/components/modals/FreeTextModal.vue.d.ts +25 -0
  133. package/dist/runtime/components/modals/SignatureDrawModal.d.vue.ts +14 -0
  134. package/dist/runtime/components/modals/SignatureDrawModal.vue +120 -0
  135. package/dist/runtime/components/modals/SignatureDrawModal.vue.d.ts +14 -0
  136. package/dist/runtime/components/panels/SignaturePicker.d.vue.ts +3 -0
  137. package/dist/runtime/components/panels/SignaturePicker.vue +85 -0
  138. package/dist/runtime/components/panels/SignaturePicker.vue.d.ts +3 -0
  139. package/dist/runtime/components/panels/StampPicker.d.vue.ts +3 -0
  140. package/dist/runtime/components/panels/StampPicker.vue +46 -0
  141. package/dist/runtime/components/panels/StampPicker.vue.d.ts +3 -0
  142. package/dist/runtime/components/tools/ActionTools.d.vue.ts +3 -0
  143. package/dist/runtime/components/tools/ActionTools.vue +32 -0
  144. package/dist/runtime/components/tools/ActionTools.vue.d.ts +3 -0
  145. package/dist/runtime/components/tools/DrawingTools.d.vue.ts +6 -0
  146. package/dist/runtime/components/tools/DrawingTools.vue +57 -0
  147. package/dist/runtime/components/tools/DrawingTools.vue.d.ts +6 -0
  148. package/dist/runtime/components/tools/HandTool.d.vue.ts +3 -0
  149. package/dist/runtime/components/tools/HandTool.vue +14 -0
  150. package/dist/runtime/components/tools/HandTool.vue.d.ts +3 -0
  151. package/dist/runtime/components/tools/MarqueeTool.d.vue.ts +3 -0
  152. package/dist/runtime/components/tools/MarqueeTool.vue +15 -0
  153. package/dist/runtime/components/tools/MarqueeTool.vue.d.ts +3 -0
  154. package/dist/runtime/components/tools/PageInfo.d.vue.ts +3 -0
  155. package/dist/runtime/components/tools/PageInfo.vue +10 -0
  156. package/dist/runtime/components/tools/PageInfo.vue.d.ts +3 -0
  157. package/dist/runtime/components/tools/PageSettings.d.vue.ts +3 -0
  158. package/dist/runtime/components/tools/PageSettings.vue +92 -0
  159. package/dist/runtime/components/tools/PageSettings.vue.d.ts +3 -0
  160. package/dist/runtime/components/tools/SearchTool.d.vue.ts +3 -0
  161. package/dist/runtime/components/tools/SearchTool.vue +149 -0
  162. package/dist/runtime/components/tools/SearchTool.vue.d.ts +3 -0
  163. package/dist/runtime/components/tools/ToolProperties.d.vue.ts +7 -0
  164. package/dist/runtime/components/tools/ToolProperties.vue +174 -0
  165. package/dist/runtime/components/tools/ToolProperties.vue.d.ts +7 -0
  166. package/dist/runtime/components/tools/ZoomControls.d.vue.ts +3 -0
  167. package/dist/runtime/components/tools/ZoomControls.vue +59 -0
  168. package/dist/runtime/components/tools/ZoomControls.vue.d.ts +3 -0
  169. package/dist/runtime/composables/search-utils.d.ts +20 -0
  170. package/dist/runtime/composables/search-utils.js +55 -0
  171. package/dist/runtime/composables/useAnnotationEngine.d.ts +7 -0
  172. package/dist/runtime/composables/useAnnotationEngine.js +70 -0
  173. package/dist/runtime/composables/useAnnotationHistory.d.ts +12 -0
  174. package/dist/runtime/composables/useAnnotationHistory.js +69 -0
  175. package/dist/runtime/composables/useFormFields.d.ts +26 -0
  176. package/dist/runtime/composables/useFormFields.js +112 -0
  177. package/dist/runtime/composables/usePageProxyCache.d.ts +8 -0
  178. package/dist/runtime/composables/usePageProxyCache.js +73 -0
  179. package/dist/runtime/composables/usePageSettings.d.ts +16 -0
  180. package/dist/runtime/composables/usePageSettings.js +66 -0
  181. package/dist/runtime/composables/usePageVirtualization.d.ts +19 -0
  182. package/dist/runtime/composables/usePageVirtualization.js +203 -0
  183. package/dist/runtime/composables/useSearchIndex.d.ts +11 -0
  184. package/dist/runtime/composables/useSearchIndex.js +71 -0
  185. package/dist/runtime/composables/useViewerSearch.d.ts +32 -0
  186. package/dist/runtime/composables/useViewerSearch.js +418 -0
  187. package/dist/runtime/composables/useViewerState.d.ts +62 -0
  188. package/dist/runtime/composables/useViewerState.js +189 -0
  189. package/dist/runtime/composables/viewMode.d.ts +11 -0
  190. package/dist/runtime/composables/viewMode.js +19 -0
  191. package/dist/runtime/plugin.d.ts +2 -0
  192. package/dist/runtime/plugin.js +3 -0
  193. package/dist/runtime/public-types.d.ts +2 -0
  194. package/dist/runtime/public-types.js +0 -0
  195. package/dist/runtime/server/tsconfig.json +3 -0
  196. package/dist/types.d.mts +5 -0
  197. package/package.json +64 -0
@@ -0,0 +1,91 @@
1
+ <template>
2
+ <div class="flex flex-col border-b border-default">
3
+ <!-- Top navigation bar -->
4
+ <div class="flex items-center h-10 px-3 gap-2 bg-elevated/50">
5
+ <UPopover>
6
+ <UButton icon="i-lucide-menu" variant="ghost" color="neutral" size="xs" />
7
+ <template #content>
8
+ <div class="p-1">
9
+ <UButton
10
+ icon="i-lucide-download"
11
+ label="Download"
12
+ variant="ghost"
13
+ color="neutral"
14
+ size="xs"
15
+ class="w-full justify-start"
16
+ @click="onDownload"
17
+ />
18
+ </div>
19
+ </template>
20
+ </UPopover>
21
+ <PageSettings />
22
+
23
+ <USeparator orientation="vertical" class="h-5" color="neutral" />
24
+
25
+ <ZoomControls />
26
+
27
+ <USeparator orientation="vertical" class="h-5" color="neutral" />
28
+
29
+ <HandTool />
30
+ <MarqueeTool />
31
+
32
+ <div class="ml-auto flex items-center gap-2">
33
+ <PageInfo />
34
+ <SearchTool />
35
+ </div>
36
+ </div>
37
+
38
+ <!-- Tool bar -->
39
+ <div class="grid grid-cols-3 items-center h-11 px-3 gap-2 bg-elevated">
40
+ <div class="justify-self-start">
41
+ <ToolProperties @style-updated="bumpStyleVersion" />
42
+ </div>
43
+
44
+ <div class="justify-self-center">
45
+ <DrawingTools :style-version="styleVersion" />
46
+ </div>
47
+
48
+ <div class="justify-self-end flex items-center gap-1">
49
+ <ActionTools />
50
+ </div>
51
+ </div>
52
+ </div>
53
+ </template>
54
+
55
+ <script setup>
56
+ import { ref } from "vue";
57
+ import { useViewerState } from "../composables/useViewerState";
58
+ import { getTimestampString } from "../annotation/engine/utils";
59
+ import { downloadPdfBytes } from "../annotation/pdf-export/download";
60
+ import { exportAnnotationsToPdf } from "../annotation/pdf-export/export";
61
+ import PageSettings from "./tools/PageSettings.vue";
62
+ import ZoomControls from "./tools/ZoomControls.vue";
63
+ import HandTool from "./tools/HandTool.vue";
64
+ import MarqueeTool from "./tools/MarqueeTool.vue";
65
+ import DrawingTools from "./tools/DrawingTools.vue";
66
+ import ActionTools from "./tools/ActionTools.vue";
67
+ import PageInfo from "./tools/PageInfo.vue";
68
+ import SearchTool from "./tools/SearchTool.vue";
69
+ import ToolProperties from "./tools/ToolProperties.vue";
70
+ const state = useViewerState();
71
+ const styleVersion = ref(0);
72
+ function bumpStyleVersion() {
73
+ styleVersion.value += 1;
74
+ }
75
+ async function onDownload() {
76
+ try {
77
+ if (!state.doc.value || !state.painter.value) return;
78
+ const bytes = await exportAnnotationsToPdf({
79
+ pdfData: await state.doc.value.getData(),
80
+ annotations: state.painter.value.getData(),
81
+ options: {
82
+ flatten: false,
83
+ preserveOriginalAnnotations: false
84
+ }
85
+ });
86
+ downloadPdfBytes(bytes, `annotated_${getTimestampString()}.pdf`);
87
+ } catch (error) {
88
+ console.error("Failed to export PDF from menu action", error);
89
+ }
90
+ }
91
+ </script>
@@ -0,0 +1,3 @@
1
+ declare const __VLS_export: import("vue").DefineComponent<{}, {}, {}, {}, {}, import("vue").ComponentOptionsMixin, import("vue").ComponentOptionsMixin, {}, string, import("vue").PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import("vue").ComponentProvideOptions, true, {}, any>;
2
+ declare const _default: typeof __VLS_export;
3
+ export default _default;
@@ -0,0 +1,381 @@
1
+ import { type ComponentPublicInstance } from 'vue';
2
+ import type { ViewMode } from '../composables/useViewerState.js';
3
+ import type { StampDefinition, SignatureHandlers } from '../annotation/engine/types.js';
4
+ export interface ViewerTabItem {
5
+ /** Unique identifier for the tab. */
6
+ id: string;
7
+ /** Display label shown in the tab bar. */
8
+ label: string;
9
+ /** PDF source -- same as KViewer's `source` prop. */
10
+ source: string | Uint8Array | object;
11
+ /** Icon to show in the tab (Lucide icon name). Defaults to 'i-lucide-file-text'. */
12
+ icon?: string;
13
+ /** Whether the tab can be closed. Defaults to true. */
14
+ closable?: boolean;
15
+ /** View mode override for this document. */
16
+ viewMode?: ViewMode;
17
+ /** Zoom override for this document. */
18
+ zoom?: number;
19
+ }
20
+ export interface AddTabOptions {
21
+ /** Insert at specific index. Defaults to end. */
22
+ index?: number;
23
+ /** Switch to the new tab immediately. Defaults to true. */
24
+ activate?: boolean;
25
+ }
26
+ type __VLS_Props = {
27
+ /** Initial set of document tabs. */
28
+ items: ViewerTabItem[];
29
+ /** ID of the initially active tab. Defaults to first item. */
30
+ defaultActiveTab?: string;
31
+ /** Stamps forwarded to all Viewer instances. */
32
+ stamps?: StampDefinition[];
33
+ /** User name forwarded to all Viewer instances. */
34
+ userName?: string;
35
+ /** Signature handlers forwarded to all Viewer instances. */
36
+ signatureHandlers?: SignatureHandlers;
37
+ /** Enable text layer on all Viewer instances. */
38
+ textLayer?: boolean;
39
+ /** Default view mode for tabs without a per-tab override. */
40
+ viewMode?: ViewMode;
41
+ /** Default zoom for tabs without a per-tab override. */
42
+ zoom?: number;
43
+ /** Minimum number of tabs -- prevents closing below this count. */
44
+ minTabs?: number;
45
+ };
46
+ declare function addTab(item: Omit<ViewerTabItem, 'id'> & {
47
+ id?: string;
48
+ }, options?: AddTabOptions): string;
49
+ declare function removeTab(id: string): boolean;
50
+ declare function activateTab(id: string): void;
51
+ declare function getViewer(id: string): ComponentPublicInstance | null;
52
+ declare var __VLS_7: {}, __VLS_21: {}, __VLS_30: {
53
+ tab: {
54
+ id: string;
55
+ label: string;
56
+ source: string | object | {
57
+ [x: number]: number;
58
+ readonly BYTES_PER_ELEMENT: number;
59
+ readonly buffer: {
60
+ readonly byteLength: number;
61
+ slice: (begin?: number, end?: number) => ArrayBuffer;
62
+ readonly maxByteLength: number;
63
+ readonly resizable: boolean;
64
+ resize: (newByteLength?: number) => void;
65
+ readonly detached: boolean;
66
+ transfer: (newByteLength?: number) => ArrayBuffer;
67
+ transferToFixedLength: (newByteLength?: number) => ArrayBuffer;
68
+ readonly [Symbol.toStringTag]: "ArrayBuffer";
69
+ } | {
70
+ readonly byteLength: number;
71
+ slice: (begin?: number, end?: number) => SharedArrayBuffer;
72
+ readonly growable: boolean;
73
+ readonly maxByteLength: number;
74
+ grow: (newByteLength?: number) => void;
75
+ readonly [Symbol.toStringTag]: "SharedArrayBuffer";
76
+ };
77
+ readonly byteLength: number;
78
+ readonly byteOffset: number;
79
+ copyWithin: (target: number, start: number, end?: number) => Uint8Array<ArrayBufferLike>;
80
+ every: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
81
+ fill: (value: number, start?: number, end?: number) => Uint8Array<ArrayBufferLike>;
82
+ filter: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => any, thisArg?: any) => Uint8Array<ArrayBuffer>;
83
+ find: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number | undefined;
84
+ findIndex: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number;
85
+ forEach: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => void, thisArg?: any) => void;
86
+ indexOf: (searchElement: number, fromIndex?: number) => number;
87
+ join: (separator?: string) => string;
88
+ lastIndexOf: (searchElement: number, fromIndex?: number) => number;
89
+ readonly length: number;
90
+ map: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => number, thisArg?: any) => Uint8Array<ArrayBuffer>;
91
+ reduce: {
92
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
93
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
94
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
95
+ };
96
+ reduceRight: {
97
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
98
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
99
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
100
+ };
101
+ reverse: () => Uint8Array<ArrayBufferLike>;
102
+ set: (array: ArrayLike<number>, offset?: number) => void;
103
+ slice: (start?: number, end?: number) => Uint8Array<ArrayBuffer>;
104
+ some: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
105
+ sort: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBufferLike>;
106
+ subarray: (begin?: number, end?: number) => Uint8Array<ArrayBufferLike>;
107
+ toLocaleString: {
108
+ (): string;
109
+ (locales: string | string[], options?: Intl.NumberFormatOptions): string;
110
+ };
111
+ toString: () => string;
112
+ valueOf: () => Uint8Array<ArrayBufferLike>;
113
+ entries: () => ArrayIterator<[number, number]>;
114
+ keys: () => ArrayIterator<number>;
115
+ values: () => ArrayIterator<number>;
116
+ includes: (searchElement: number, fromIndex?: number) => boolean;
117
+ at: (index: number) => number | undefined;
118
+ findLast: {
119
+ <S extends number>(predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => value is S, thisArg?: any): S | undefined;
120
+ (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any): number | undefined;
121
+ };
122
+ findLastIndex: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => number;
123
+ toReversed: () => Uint8Array<ArrayBuffer>;
124
+ toSorted: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBuffer>;
125
+ with: (index: number, value: number) => Uint8Array<ArrayBuffer>;
126
+ toBase64: (options?: {
127
+ alphabet?: "base64" | "base64url" | undefined;
128
+ omitPadding?: boolean | undefined;
129
+ } | undefined) => string;
130
+ setFromBase64: (string: string, options?: {
131
+ alphabet?: "base64" | "base64url" | undefined;
132
+ lastChunkHandling?: "loose" | "strict" | "stop-before-partial" | undefined;
133
+ } | undefined) => {
134
+ read: number;
135
+ written: number;
136
+ };
137
+ toHex: () => string;
138
+ setFromHex: (string: string) => {
139
+ read: number;
140
+ written: number;
141
+ };
142
+ [Symbol.iterator]: () => ArrayIterator<number>;
143
+ readonly [Symbol.toStringTag]: "Uint8Array";
144
+ };
145
+ icon?: string | undefined;
146
+ closable?: boolean | undefined;
147
+ viewMode?: ViewMode | undefined;
148
+ zoom?: number | undefined;
149
+ };
150
+ }, __VLS_33: {
151
+ tab: {
152
+ id: string;
153
+ label: string;
154
+ source: string | object | {
155
+ [x: number]: number;
156
+ readonly BYTES_PER_ELEMENT: number;
157
+ readonly buffer: {
158
+ readonly byteLength: number;
159
+ slice: (begin?: number, end?: number) => ArrayBuffer;
160
+ readonly maxByteLength: number;
161
+ readonly resizable: boolean;
162
+ resize: (newByteLength?: number) => void;
163
+ readonly detached: boolean;
164
+ transfer: (newByteLength?: number) => ArrayBuffer;
165
+ transferToFixedLength: (newByteLength?: number) => ArrayBuffer;
166
+ readonly [Symbol.toStringTag]: "ArrayBuffer";
167
+ } | {
168
+ readonly byteLength: number;
169
+ slice: (begin?: number, end?: number) => SharedArrayBuffer;
170
+ readonly growable: boolean;
171
+ readonly maxByteLength: number;
172
+ grow: (newByteLength?: number) => void;
173
+ readonly [Symbol.toStringTag]: "SharedArrayBuffer";
174
+ };
175
+ readonly byteLength: number;
176
+ readonly byteOffset: number;
177
+ copyWithin: (target: number, start: number, end?: number) => Uint8Array<ArrayBufferLike>;
178
+ every: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
179
+ fill: (value: number, start?: number, end?: number) => Uint8Array<ArrayBufferLike>;
180
+ filter: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => any, thisArg?: any) => Uint8Array<ArrayBuffer>;
181
+ find: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number | undefined;
182
+ findIndex: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number;
183
+ forEach: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => void, thisArg?: any) => void;
184
+ indexOf: (searchElement: number, fromIndex?: number) => number;
185
+ join: (separator?: string) => string;
186
+ lastIndexOf: (searchElement: number, fromIndex?: number) => number;
187
+ readonly length: number;
188
+ map: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => number, thisArg?: any) => Uint8Array<ArrayBuffer>;
189
+ reduce: {
190
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
191
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
192
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
193
+ };
194
+ reduceRight: {
195
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
196
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
197
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
198
+ };
199
+ reverse: () => Uint8Array<ArrayBufferLike>;
200
+ set: (array: ArrayLike<number>, offset?: number) => void;
201
+ slice: (start?: number, end?: number) => Uint8Array<ArrayBuffer>;
202
+ some: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
203
+ sort: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBufferLike>;
204
+ subarray: (begin?: number, end?: number) => Uint8Array<ArrayBufferLike>;
205
+ toLocaleString: {
206
+ (): string;
207
+ (locales: string | string[], options?: Intl.NumberFormatOptions): string;
208
+ };
209
+ toString: () => string;
210
+ valueOf: () => Uint8Array<ArrayBufferLike>;
211
+ entries: () => ArrayIterator<[number, number]>;
212
+ keys: () => ArrayIterator<number>;
213
+ values: () => ArrayIterator<number>;
214
+ includes: (searchElement: number, fromIndex?: number) => boolean;
215
+ at: (index: number) => number | undefined;
216
+ findLast: {
217
+ <S extends number>(predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => value is S, thisArg?: any): S | undefined;
218
+ (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any): number | undefined;
219
+ };
220
+ findLastIndex: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => number;
221
+ toReversed: () => Uint8Array<ArrayBuffer>;
222
+ toSorted: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBuffer>;
223
+ with: (index: number, value: number) => Uint8Array<ArrayBuffer>;
224
+ toBase64: (options?: {
225
+ alphabet?: "base64" | "base64url" | undefined;
226
+ omitPadding?: boolean | undefined;
227
+ } | undefined) => string;
228
+ setFromBase64: (string: string, options?: {
229
+ alphabet?: "base64" | "base64url" | undefined;
230
+ lastChunkHandling?: "loose" | "strict" | "stop-before-partial" | undefined;
231
+ } | undefined) => {
232
+ read: number;
233
+ written: number;
234
+ };
235
+ toHex: () => string;
236
+ setFromHex: (string: string) => {
237
+ read: number;
238
+ written: number;
239
+ };
240
+ [Symbol.iterator]: () => ArrayIterator<number>;
241
+ readonly [Symbol.toStringTag]: "Uint8Array";
242
+ };
243
+ icon?: string | undefined;
244
+ closable?: boolean | undefined;
245
+ viewMode?: ViewMode | undefined;
246
+ zoom?: number | undefined;
247
+ };
248
+ }, __VLS_35: {};
249
+ type __VLS_Slots = {} & {
250
+ 'tabs-leading'?: (props: typeof __VLS_7) => any;
251
+ } & {
252
+ 'tabs-trailing'?: (props: typeof __VLS_21) => any;
253
+ } & {
254
+ header?: (props: typeof __VLS_30) => any;
255
+ } & {
256
+ footer?: (props: typeof __VLS_33) => any;
257
+ } & {
258
+ empty?: (props: typeof __VLS_35) => any;
259
+ };
260
+ declare const __VLS_base: import("vue").DefineComponent<__VLS_Props, {
261
+ addTab: typeof addTab;
262
+ removeTab: typeof removeTab;
263
+ activateTab: typeof activateTab;
264
+ getViewer: typeof getViewer;
265
+ readonly activeTabId: string;
266
+ readonly tabs: {
267
+ id: string;
268
+ label: string;
269
+ source: string | object | {
270
+ [x: number]: number;
271
+ readonly BYTES_PER_ELEMENT: number;
272
+ readonly buffer: {
273
+ readonly byteLength: number;
274
+ slice: (begin?: number, end?: number) => ArrayBuffer;
275
+ readonly maxByteLength: number;
276
+ readonly resizable: boolean;
277
+ resize: (newByteLength?: number) => void;
278
+ readonly detached: boolean;
279
+ transfer: (newByteLength?: number) => ArrayBuffer;
280
+ transferToFixedLength: (newByteLength?: number) => ArrayBuffer;
281
+ readonly [Symbol.toStringTag]: "ArrayBuffer";
282
+ } | {
283
+ readonly byteLength: number;
284
+ slice: (begin?: number, end?: number) => SharedArrayBuffer;
285
+ readonly growable: boolean;
286
+ readonly maxByteLength: number;
287
+ grow: (newByteLength?: number) => void;
288
+ readonly [Symbol.toStringTag]: "SharedArrayBuffer";
289
+ };
290
+ readonly byteLength: number;
291
+ readonly byteOffset: number;
292
+ copyWithin: (target: number, start: number, end?: number) => Uint8Array<ArrayBufferLike>;
293
+ every: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
294
+ fill: (value: number, start?: number, end?: number) => Uint8Array<ArrayBufferLike>;
295
+ filter: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => any, thisArg?: any) => Uint8Array<ArrayBuffer>;
296
+ find: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number | undefined;
297
+ findIndex: (predicate: (value: number, index: number, obj: Uint8Array<ArrayBufferLike>) => boolean, thisArg?: any) => number;
298
+ forEach: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => void, thisArg?: any) => void;
299
+ indexOf: (searchElement: number, fromIndex?: number) => number;
300
+ join: (separator?: string) => string;
301
+ lastIndexOf: (searchElement: number, fromIndex?: number) => number;
302
+ readonly length: number;
303
+ map: (callbackfn: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => number, thisArg?: any) => Uint8Array<ArrayBuffer>;
304
+ reduce: {
305
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
306
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
307
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
308
+ };
309
+ reduceRight: {
310
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number): number;
311
+ (callbackfn: (previousValue: number, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => number, initialValue: number): number;
312
+ <U>(callbackfn: (previousValue: U, currentValue: number, currentIndex: number, array: Uint8Array<ArrayBufferLike>) => U, initialValue: U): U;
313
+ };
314
+ reverse: () => Uint8Array<ArrayBufferLike>;
315
+ set: (array: ArrayLike<number>, offset?: number) => void;
316
+ slice: (start?: number, end?: number) => Uint8Array<ArrayBuffer>;
317
+ some: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => boolean;
318
+ sort: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBufferLike>;
319
+ subarray: (begin?: number, end?: number) => Uint8Array<ArrayBufferLike>;
320
+ toLocaleString: {
321
+ (): string;
322
+ (locales: string | string[], options?: Intl.NumberFormatOptions): string;
323
+ };
324
+ toString: () => string;
325
+ valueOf: () => Uint8Array<ArrayBufferLike>;
326
+ entries: () => ArrayIterator<[number, number]>;
327
+ keys: () => ArrayIterator<number>;
328
+ values: () => ArrayIterator<number>;
329
+ includes: (searchElement: number, fromIndex?: number) => boolean;
330
+ at: (index: number) => number | undefined;
331
+ findLast: {
332
+ <S extends number>(predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => value is S, thisArg?: any): S | undefined;
333
+ (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any): number | undefined;
334
+ };
335
+ findLastIndex: (predicate: (value: number, index: number, array: Uint8Array<ArrayBufferLike>) => unknown, thisArg?: any) => number;
336
+ toReversed: () => Uint8Array<ArrayBuffer>;
337
+ toSorted: (compareFn?: ((a: number, b: number) => number) | undefined) => Uint8Array<ArrayBuffer>;
338
+ with: (index: number, value: number) => Uint8Array<ArrayBuffer>;
339
+ toBase64: (options?: {
340
+ alphabet?: "base64" | "base64url" | undefined;
341
+ omitPadding?: boolean | undefined;
342
+ } | undefined) => string;
343
+ setFromBase64: (string: string, options?: {
344
+ alphabet?: "base64" | "base64url" | undefined;
345
+ lastChunkHandling?: "loose" | "strict" | "stop-before-partial" | undefined;
346
+ } | undefined) => {
347
+ read: number;
348
+ written: number;
349
+ };
350
+ toHex: () => string;
351
+ setFromHex: (string: string) => {
352
+ read: number;
353
+ written: number;
354
+ };
355
+ [Symbol.iterator]: () => ArrayIterator<number>;
356
+ readonly [Symbol.toStringTag]: "Uint8Array";
357
+ };
358
+ icon?: string | undefined;
359
+ closable?: boolean | undefined;
360
+ viewMode?: ViewMode | undefined;
361
+ zoom?: number | undefined;
362
+ }[];
363
+ }, {}, {}, {}, import("vue").ComponentOptionsMixin, import("vue").ComponentOptionsMixin, {
364
+ "update:activeTab": (id: string) => any;
365
+ "tab-added": (tab: ViewerTabItem) => any;
366
+ "tab-close": (id: string) => any;
367
+ "tab-removed": (id: string) => any;
368
+ }, string, import("vue").PublicProps, Readonly<__VLS_Props> & Readonly<{
369
+ "onUpdate:activeTab"?: ((id: string) => any) | undefined;
370
+ "onTab-added"?: ((tab: ViewerTabItem) => any) | undefined;
371
+ "onTab-close"?: ((id: string) => any) | undefined;
372
+ "onTab-removed"?: ((id: string) => any) | undefined;
373
+ }>, {}, {}, {}, {}, string, import("vue").ComponentProvideOptions, false, {}, any>;
374
+ declare const __VLS_export: __VLS_WithSlots<typeof __VLS_base, __VLS_Slots>;
375
+ declare const _default: typeof __VLS_export;
376
+ export default _default;
377
+ type __VLS_WithSlots<T, S> = T & {
378
+ new (): {
379
+ $slots: S;
380
+ };
381
+ };
@@ -0,0 +1,171 @@
1
+ <template>
2
+ <div class="flex flex-col h-full">
3
+ <!-- Tab bar -->
4
+ <ClientOnly>
5
+ <div
6
+ v-if="tabs.length > 0"
7
+ class="flex items-center border-b border-default bg-elevated/50 overflow-x-auto"
8
+ >
9
+ <slot name="tabs-leading" />
10
+
11
+ <button
12
+ v-for="tab in tabs"
13
+ :key="tab.id"
14
+ class="group flex items-center gap-1.5 px-3 h-9 text-sm border-r border-default whitespace-nowrap transition-colors shrink-0"
15
+ :class="[
16
+ tab.id === activeTabId ? 'bg-elevated text-highlighted border-b-2 border-b-primary' : 'text-muted hover:text-highlighted hover:bg-elevated/80'
17
+ ]"
18
+ @click="activateTab(tab.id)"
19
+ >
20
+ <UIcon
21
+ :name="tab.icon ?? 'i-lucide-file-text'"
22
+ class="size-4 shrink-0"
23
+ />
24
+ <span class="truncate max-w-40">{{ tab.label }}</span>
25
+ <UButton
26
+ v-if="tab.closable !== false && tabs.length > minTabs"
27
+ icon="i-lucide-x"
28
+ size="xs"
29
+ variant="ghost"
30
+ color="neutral"
31
+ class="opacity-0 group-hover:opacity-100 transition-opacity -mr-1"
32
+ :class="{ 'opacity-100': tab.id === activeTabId }"
33
+ @click.stop="removeTab(tab.id)"
34
+ />
35
+ </button>
36
+
37
+ <slot name="tabs-trailing" />
38
+ </div>
39
+ </ClientOnly>
40
+
41
+ <!-- Viewer container -->
42
+ <div class="flex-1 relative overflow-hidden">
43
+ <div
44
+ v-for="tab in tabs"
45
+ v-show="tab.id === activeTabId"
46
+ :key="tab.id"
47
+ class="absolute inset-0"
48
+ >
49
+ <Viewer
50
+ :ref="(el) => setViewerRef(tab.id, el)"
51
+ :source="tab.source"
52
+ :active="tab.id === activeTabId"
53
+ :stamps="stamps"
54
+ :text-layer="textLayer"
55
+ :user-name="userName"
56
+ :signature-handlers="signatureHandlers"
57
+ :view-mode="tab.viewMode ?? viewMode"
58
+ :zoom="tab.zoom ?? zoom"
59
+ >
60
+ <template v-if="$slots.header" #header>
61
+ <slot name="header" :tab="tab" />
62
+ </template>
63
+ <template v-if="$slots.footer" #footer>
64
+ <slot name="footer" :tab="tab" />
65
+ </template>
66
+ </Viewer>
67
+ </div>
68
+
69
+ <!-- Empty state when no tabs are open -->
70
+ <div v-if="tabs.length === 0" class="absolute inset-0">
71
+ <slot name="empty" />
72
+ </div>
73
+ </div>
74
+ </div>
75
+ </template>
76
+
77
+ <script setup>
78
+ import { ref, watch } from "vue";
79
+ import Viewer from "./Viewer.vue";
80
+ const props = defineProps({
81
+ items: { type: Array, required: true },
82
+ defaultActiveTab: { type: String, required: false, default: void 0 },
83
+ stamps: { type: Array, required: false, default: void 0 },
84
+ userName: { type: String, required: false, default: void 0 },
85
+ signatureHandlers: { type: Object, required: false, default: void 0 },
86
+ textLayer: { type: Boolean, required: false },
87
+ viewMode: { type: String, required: false, default: "fit-width" },
88
+ zoom: { type: Number, required: false, default: 1 },
89
+ minTabs: { type: Number, required: false, default: 0 }
90
+ });
91
+ const emit = defineEmits(["update:activeTab", "tab-added", "tab-close", "tab-removed"]);
92
+ let idCounter = 0;
93
+ function generateId() {
94
+ return `tab-${Date.now()}-${++idCounter}`;
95
+ }
96
+ const tabs = ref([...props.items]);
97
+ const activeTabId = ref(
98
+ props.defaultActiveTab ?? props.items[0]?.id ?? ""
99
+ );
100
+ const viewerRefs = /* @__PURE__ */ new Map();
101
+ function setViewerRef(id, el) {
102
+ if (el) {
103
+ viewerRefs.set(id, el);
104
+ } else {
105
+ viewerRefs.delete(id);
106
+ }
107
+ }
108
+ watch(
109
+ () => props.items,
110
+ (newItems) => {
111
+ tabs.value = [...newItems];
112
+ if (!tabs.value.some((t) => t.id === activeTabId.value)) {
113
+ activeTabId.value = tabs.value[0]?.id ?? "";
114
+ }
115
+ }
116
+ );
117
+ function addTab(item, options) {
118
+ const id = item.id ?? generateId();
119
+ const tab = {
120
+ ...item,
121
+ id,
122
+ closable: item.closable ?? true
123
+ };
124
+ const index = options?.index ?? tabs.value.length;
125
+ tabs.value.splice(index, 0, tab);
126
+ if (options?.activate !== false) {
127
+ activeTabId.value = id;
128
+ emit("update:activeTab", id);
129
+ }
130
+ emit("tab-added", tab);
131
+ return id;
132
+ }
133
+ function removeTab(id) {
134
+ const index = tabs.value.findIndex((t) => t.id === id);
135
+ if (index === -1) return false;
136
+ if (tabs.value.length <= props.minTabs) return false;
137
+ emit("tab-close", id);
138
+ tabs.value.splice(index, 1);
139
+ viewerRefs.delete(id);
140
+ if (activeTabId.value === id) {
141
+ const newIndex = Math.min(index, tabs.value.length - 1);
142
+ activeTabId.value = tabs.value[newIndex]?.id ?? "";
143
+ if (activeTabId.value) {
144
+ emit("update:activeTab", activeTabId.value);
145
+ }
146
+ }
147
+ emit("tab-removed", id);
148
+ return true;
149
+ }
150
+ function activateTab(id) {
151
+ if (tabs.value.some((t) => t.id === id)) {
152
+ activeTabId.value = id;
153
+ emit("update:activeTab", id);
154
+ }
155
+ }
156
+ function getViewer(id) {
157
+ return viewerRefs.get(id) ?? null;
158
+ }
159
+ defineExpose({
160
+ addTab,
161
+ removeTab,
162
+ activateTab,
163
+ getViewer,
164
+ get activeTabId() {
165
+ return activeTabId.value;
166
+ },
167
+ get tabs() {
168
+ return tabs.value;
169
+ }
170
+ });
171
+ </script>