@turinhub/atomix-common-ui 0.4.0 → 0.6.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 (104) hide show
  1. package/README.md +20 -2
  2. package/dist/AuthPanel-CTKx618F.cjs +2 -0
  3. package/dist/AuthPanel-CTKx618F.cjs.map +1 -0
  4. package/dist/AuthPanel-Cn_WwmjX.js +703 -0
  5. package/dist/AuthPanel-Cn_WwmjX.js.map +1 -0
  6. package/dist/PDFSidebar-4DtXqqzN.cjs +2 -0
  7. package/dist/PDFSidebar-4DtXqqzN.cjs.map +1 -0
  8. package/dist/PDFSidebar-ClnrF4Br.js +239 -0
  9. package/dist/PDFSidebar-ClnrF4Br.js.map +1 -0
  10. package/dist/auth.cjs +2 -0
  11. package/dist/auth.cjs.map +1 -0
  12. package/dist/auth.d.ts +11 -0
  13. package/dist/auth.d.ts.map +1 -0
  14. package/dist/auth.js +9 -0
  15. package/dist/auth.js.map +1 -0
  16. package/dist/components/AuthLoginPanel.d.ts +55 -0
  17. package/dist/components/AuthLoginPanel.d.ts.map +1 -0
  18. package/dist/components/AuthPageShell.d.ts +11 -0
  19. package/dist/components/AuthPageShell.d.ts.map +1 -0
  20. package/dist/components/AuthPanel.d.ts +20 -0
  21. package/dist/components/AuthPanel.d.ts.map +1 -0
  22. package/dist/components/AuthRegisterPanel.d.ts +34 -0
  23. package/dist/components/AuthRegisterPanel.d.ts.map +1 -0
  24. package/dist/components/AuthVisualCarousel.d.ts +22 -0
  25. package/dist/components/AuthVisualCarousel.d.ts.map +1 -0
  26. package/dist/components/DataTable.d.ts +2 -2
  27. package/dist/components/DataTable.d.ts.map +1 -1
  28. package/dist/components/ImageReader.d.ts +44 -0
  29. package/dist/components/ImageReader.d.ts.map +1 -0
  30. package/dist/components/MarkdownReader.d.ts.map +1 -1
  31. package/dist/components/PDFReader.d.ts.map +1 -1
  32. package/dist/components/PDFSidebar.d.ts.map +1 -1
  33. package/dist/components/SimplePDFReader.d.ts.map +1 -1
  34. package/dist/components/TableHeader.d.ts.map +1 -1
  35. package/dist/components/TablePagination.d.ts +2 -1
  36. package/dist/components/TablePagination.d.ts.map +1 -1
  37. package/dist/components/VideoReader.d.ts +39 -0
  38. package/dist/components/VideoReader.d.ts.map +1 -0
  39. package/dist/components/media-utils.d.ts +9 -0
  40. package/dist/components/media-utils.d.ts.map +1 -0
  41. package/dist/components/ui/switch.d.ts +5 -0
  42. package/dist/components/ui/switch.d.ts.map +1 -0
  43. package/dist/data-table.cjs +1 -1
  44. package/dist/data-table.cjs.map +1 -1
  45. package/dist/data-table.js +83 -73
  46. package/dist/data-table.js.map +1 -1
  47. package/dist/file-upload.cjs +1 -1
  48. package/dist/file-upload.cjs.map +1 -1
  49. package/dist/file-upload.js +36 -36
  50. package/dist/file-upload.js.map +1 -1
  51. package/dist/image-reader.cjs +2 -0
  52. package/dist/image-reader.cjs.map +1 -0
  53. package/dist/image-reader.d.ts +3 -0
  54. package/dist/image-reader.d.ts.map +1 -0
  55. package/dist/image-reader.js +215 -0
  56. package/dist/image-reader.js.map +1 -0
  57. package/dist/index.cjs +1 -1
  58. package/dist/index.d.ts +10 -0
  59. package/dist/index.d.ts.map +1 -1
  60. package/dist/index.js +8 -2
  61. package/dist/index.js.map +1 -1
  62. package/dist/markdown-reader.cjs +1 -1
  63. package/dist/markdown-reader.cjs.map +1 -1
  64. package/dist/markdown-reader.js +28 -24
  65. package/dist/markdown-reader.js.map +1 -1
  66. package/dist/media-utils-5UPuocc1.js +23 -0
  67. package/dist/media-utils-5UPuocc1.js.map +1 -0
  68. package/dist/media-utils-X1dDYP9W.cjs +2 -0
  69. package/dist/media-utils-X1dDYP9W.cjs.map +1 -0
  70. package/dist/pdf-reader.cjs +1 -1
  71. package/dist/pdf-reader.cjs.map +1 -1
  72. package/dist/pdf-reader.js +170 -121
  73. package/dist/pdf-reader.js.map +1 -1
  74. package/dist/pdf-sidebar.cjs +1 -1
  75. package/dist/pdf-sidebar.js +1 -1
  76. package/dist/simple-pdf-reader.cjs +1 -1
  77. package/dist/simple-pdf-reader.cjs.map +1 -1
  78. package/dist/simple-pdf-reader.js +138 -105
  79. package/dist/simple-pdf-reader.js.map +1 -1
  80. package/dist/table-header.cjs +1 -1
  81. package/dist/table-header.cjs.map +1 -1
  82. package/dist/table-header.js +42 -34
  83. package/dist/table-header.js.map +1 -1
  84. package/dist/table-pagination.cjs +1 -1
  85. package/dist/table-pagination.cjs.map +1 -1
  86. package/dist/table-pagination.js +49 -43
  87. package/dist/table-pagination.js.map +1 -1
  88. package/dist/types/component-types.d.ts +2 -0
  89. package/dist/types/component-types.d.ts.map +1 -1
  90. package/dist/video-reader.cjs +2 -0
  91. package/dist/video-reader.cjs.map +1 -0
  92. package/dist/video-reader.d.ts +3 -0
  93. package/dist/video-reader.d.ts.map +1 -0
  94. package/dist/video-reader.js +158 -0
  95. package/dist/video-reader.js.map +1 -0
  96. package/package.json +32 -1
  97. package/dist/PDFSidebar-BBtucLK6.js +0 -232
  98. package/dist/PDFSidebar-BBtucLK6.js.map +0 -1
  99. package/dist/PDFSidebar-Di0D-yPS.cjs +0 -2
  100. package/dist/PDFSidebar-Di0D-yPS.cjs.map +0 -1
  101. package/dist/index-BiA_tnaq.cjs +0 -13
  102. package/dist/index-BiA_tnaq.cjs.map +0 -1
  103. package/dist/index-BypbGNpR.js +0 -18821
  104. package/dist/index-BypbGNpR.js.map +0 -1
@@ -0,0 +1,158 @@
1
+ import { j as e } from "./jsx-runtime-B4hRZ52C.js";
2
+ import { useState as y, useMemo as K, useEffect as Q, createElement as T } from "react";
3
+ import { ExternalLink as W } from "lucide-react";
4
+ import { c as x } from "./utils-B6yFEsav.js";
5
+ import { i as Z, g as ee } from "./media-utils-5UPuocc1.js";
6
+ const te = [
7
+ "mp4",
8
+ "webm",
9
+ "ogg",
10
+ "ogv",
11
+ "mov",
12
+ "m4v"
13
+ ], se = [
14
+ "video/mp4",
15
+ "video/webm",
16
+ "video/ogg",
17
+ "video/quicktime",
18
+ "video/x-m4v"
19
+ ], ne = (s) => s ? s instanceof Error ? s : new Error(s) : null;
20
+ function de({
21
+ src: s,
22
+ fileName: o,
23
+ mimeType: r,
24
+ title: f,
25
+ components: V,
26
+ tracks: i,
27
+ loading: _ = !1,
28
+ error: I,
29
+ className: p,
30
+ containerClassName: M,
31
+ videoClassName: O,
32
+ toolbarClassName: P,
33
+ loadingText: C = "正在加载视频...",
34
+ errorText: h = "视频加载失败",
35
+ unsupportedText: R = "暂不支持该视频格式",
36
+ showToolbar: k = !0,
37
+ showOpenInNewTab: L = !0,
38
+ allowUnsupportedFormat: U = !1,
39
+ supportedExtensions: z = te,
40
+ supportedMimeTypes: B = se,
41
+ controls: F = !0,
42
+ preload: $ = "metadata",
43
+ playsInline: q = !0,
44
+ onLoadedData: l,
45
+ onError: a,
46
+ ...D
47
+ }) {
48
+ const { Card: g, CardContent: v, Button: j, Skeleton: d } = V || {}, [X, u] = y(!0), [Y, c] = y(null), n = ne(I) || Y, b = U || Z({
49
+ src: s,
50
+ fileName: o,
51
+ mimeType: r,
52
+ supportedExtensions: z,
53
+ supportedMimeTypes: B
54
+ }), w = _ || X && !n, A = K(() => {
55
+ const t = ee(s, o);
56
+ return t ? t.toUpperCase() : r || "视频";
57
+ }, [o, r, s]);
58
+ Q(() => {
59
+ u(!0), c(null);
60
+ }, [s]);
61
+ const G = (t, N, S) => {
62
+ const J = "inline-flex h-8 w-8 items-center justify-center rounded-md border border-input bg-background text-foreground shadow-sm transition-colors hover:bg-accent hover:text-accent-foreground";
63
+ return j ? /* @__PURE__ */ e.jsx(
64
+ j,
65
+ {
66
+ "aria-label": t,
67
+ className: "h-8 w-8",
68
+ onClick: S,
69
+ size: "icon",
70
+ title: t,
71
+ type: "button",
72
+ variant: "outline",
73
+ children: N
74
+ }
75
+ ) : /* @__PURE__ */ e.jsx(
76
+ "button",
77
+ {
78
+ "aria-label": t,
79
+ className: J,
80
+ onClick: S,
81
+ title: t,
82
+ type: "button",
83
+ children: N
84
+ }
85
+ );
86
+ }, H = () => /* @__PURE__ */ e.jsx("div", { className: "w-full space-y-3 p-4", role: "status", "aria-live": "polite", children: d ? /* @__PURE__ */ e.jsxs(e.Fragment, { children: [
87
+ /* @__PURE__ */ e.jsx(d, { className: "h-4 w-32" }),
88
+ /* @__PURE__ */ e.jsx(d, { className: "aspect-video w-full" })
89
+ ] }) : /* @__PURE__ */ e.jsx("p", { className: "text-sm text-muted-foreground", children: C }) }), E = (t) => /* @__PURE__ */ e.jsxs("div", { className: "p-4 text-center text-sm text-destructive", role: "alert", children: [
90
+ /* @__PURE__ */ e.jsx("p", { className: "font-medium", children: t || h }),
91
+ n != null && n.message ? /* @__PURE__ */ e.jsx("p", { className: "mt-1 opacity-80", children: n.message }) : null
92
+ ] }), m = /* @__PURE__ */ e.jsxs(
93
+ "div",
94
+ {
95
+ className: x(
96
+ "flex h-full min-h-[360px] flex-col overflow-hidden rounded-md border bg-background",
97
+ M
98
+ ),
99
+ children: [
100
+ k ? /* @__PURE__ */ e.jsxs(
101
+ "div",
102
+ {
103
+ className: x(
104
+ "flex flex-wrap items-center justify-between gap-2 border-b px-3 py-2",
105
+ P
106
+ ),
107
+ children: [
108
+ /* @__PURE__ */ e.jsx("span", { className: "text-xs font-medium text-muted-foreground", children: f || A }),
109
+ L ? G(
110
+ "新窗口打开",
111
+ /* @__PURE__ */ e.jsx(W, { className: "h-4 w-4" }),
112
+ () => window.open(s, "_blank", "noreferrer")
113
+ ) : null
114
+ ]
115
+ }
116
+ ) : null,
117
+ /* @__PURE__ */ e.jsxs("div", { className: "relative flex min-h-0 flex-1 items-center justify-center bg-muted/20 p-4", children: [
118
+ b ? n ? E() : null : E(R),
119
+ b && !n ? /* @__PURE__ */ e.jsxs(e.Fragment, { children: [
120
+ w ? H() : null,
121
+ /* @__PURE__ */ T(
122
+ "video",
123
+ {
124
+ ...D,
125
+ key: s,
126
+ className: x(
127
+ "max-h-full w-full max-w-full bg-black",
128
+ w ? "invisible absolute" : "visible",
129
+ O
130
+ ),
131
+ controls: F,
132
+ onError: () => {
133
+ const t = new Error(h);
134
+ u(!1), c(t), a == null || a(t);
135
+ },
136
+ onLoadedData: () => {
137
+ u(!1), c(null), l == null || l();
138
+ },
139
+ playsInline: q,
140
+ preload: $,
141
+ title: f
142
+ },
143
+ /* @__PURE__ */ e.jsx("source", { src: s, type: r }),
144
+ i == null ? void 0 : i.map((t) => /* @__PURE__ */ e.jsx("track", { ...t }, `${t.src}-${t.kind || ""}`))
145
+ )
146
+ ] }) : null
147
+ ] })
148
+ ]
149
+ }
150
+ );
151
+ return g ? /* @__PURE__ */ e.jsx(g, { className: p, children: v ? /* @__PURE__ */ e.jsx(v, { children: m }) : m }) : /* @__PURE__ */ e.jsx("div", { className: p, children: m });
152
+ }
153
+ export {
154
+ te as SUPPORTED_VIDEO_EXTENSIONS,
155
+ se as SUPPORTED_VIDEO_MIME_TYPES,
156
+ de as VideoReader
157
+ };
158
+ //# sourceMappingURL=video-reader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"video-reader.js","sources":["../src/components/VideoReader.tsx"],"sourcesContent":["import { ExternalLink as ExternalLinkIcon } from 'lucide-react';\nimport { useEffect, useMemo, useState } from 'react';\nimport type {\n HTMLAttributes,\n ReactNode,\n TrackHTMLAttributes,\n VideoHTMLAttributes,\n} from 'react';\n\nimport { cn } from '../lib/utils';\nimport type {\n ButtonComponent,\n CardComponent,\n SkeletonComponent,\n UIComponent,\n} from '../types/component-types';\n\nimport { getMediaExtension, isSupportedMediaSource } from './media-utils';\n\nexport const SUPPORTED_VIDEO_EXTENSIONS = [\n 'mp4',\n 'webm',\n 'ogg',\n 'ogv',\n 'mov',\n 'm4v',\n] as const;\n\nexport const SUPPORTED_VIDEO_MIME_TYPES = [\n 'video/mp4',\n 'video/webm',\n 'video/ogg',\n 'video/quicktime',\n 'video/x-m4v',\n] as const;\n\nexport interface VideoReaderUIComponents {\n Card?: CardComponent;\n CardContent?: UIComponent<HTMLAttributes<HTMLDivElement>>;\n Button?: ButtonComponent;\n Skeleton?: SkeletonComponent;\n}\n\nexport interface VideoReaderTrack extends Omit<\n TrackHTMLAttributes<HTMLTrackElement>,\n 'children'\n> {\n src: string;\n}\n\nexport interface VideoReaderProps extends Omit<\n VideoHTMLAttributes<HTMLVideoElement>,\n 'children' | 'className' | 'onError' | 'onLoadedData' | 'src'\n> {\n src: string;\n fileName?: string;\n mimeType?: string;\n title?: string;\n components?: VideoReaderUIComponents;\n tracks?: VideoReaderTrack[];\n loading?: boolean;\n error?: Error | string | null;\n className?: string;\n containerClassName?: string;\n videoClassName?: string;\n toolbarClassName?: string;\n loadingText?: string;\n errorText?: string;\n unsupportedText?: string;\n showToolbar?: boolean;\n showOpenInNewTab?: boolean;\n allowUnsupportedFormat?: boolean;\n supportedExtensions?: readonly string[];\n supportedMimeTypes?: readonly string[];\n onLoadedData?: () => void;\n onError?: (error: Error) => void;\n}\n\nconst normalizeError = (error: Error | string | null | undefined) => {\n if (!error) return null;\n return error instanceof Error ? error : new Error(error);\n};\n\nexport function VideoReader({\n src,\n fileName,\n mimeType,\n title,\n components,\n tracks,\n loading = false,\n error,\n className,\n containerClassName,\n videoClassName,\n toolbarClassName,\n loadingText = '正在加载视频...',\n errorText = '视频加载失败',\n unsupportedText = '暂不支持该视频格式',\n showToolbar = true,\n showOpenInNewTab = true,\n allowUnsupportedFormat = false,\n supportedExtensions = SUPPORTED_VIDEO_EXTENSIONS,\n supportedMimeTypes = SUPPORTED_VIDEO_MIME_TYPES,\n controls = true,\n preload = 'metadata',\n playsInline = true,\n onLoadedData,\n onError,\n ...videoProps\n}: VideoReaderProps) {\n const { Card, CardContent, Button, Skeleton } = components || {};\n const [isVideoLoading, setIsVideoLoading] = useState(true);\n const [videoError, setVideoError] = useState<Error | null>(null);\n\n const displayedError = normalizeError(error) || videoError;\n const isSupported =\n allowUnsupportedFormat ||\n isSupportedMediaSource({\n src,\n fileName,\n mimeType,\n supportedExtensions,\n supportedMimeTypes,\n });\n const isLoading = loading || (isVideoLoading && !displayedError);\n\n const formatLabel = useMemo(() => {\n const extension = getMediaExtension(src, fileName);\n return extension ? extension.toUpperCase() : mimeType || '视频';\n }, [fileName, mimeType, src]);\n\n useEffect(() => {\n setIsVideoLoading(true);\n setVideoError(null);\n }, [src]);\n\n const renderButton = (\n label: string,\n icon: ReactNode,\n onClick: () => void\n ) => {\n const buttonClassName =\n 'inline-flex h-8 w-8 items-center justify-center rounded-md border border-input bg-background text-foreground shadow-sm transition-colors hover:bg-accent hover:text-accent-foreground';\n\n if (Button) {\n return (\n <Button\n aria-label={label}\n className=\"h-8 w-8\"\n onClick={onClick}\n size=\"icon\"\n title={label}\n type=\"button\"\n variant=\"outline\"\n >\n {icon}\n </Button>\n );\n }\n\n return (\n <button\n aria-label={label}\n className={buttonClassName}\n onClick={onClick}\n title={label}\n type=\"button\"\n >\n {icon}\n </button>\n );\n };\n\n const renderLoading = () => (\n <div className=\"w-full space-y-3 p-4\" role=\"status\" aria-live=\"polite\">\n {Skeleton ? (\n <>\n <Skeleton className=\"h-4 w-32\" />\n <Skeleton className=\"aspect-video w-full\" />\n </>\n ) : (\n <p className=\"text-sm text-muted-foreground\">{loadingText}</p>\n )}\n </div>\n );\n\n const renderError = (message?: string) => (\n <div className=\"p-4 text-center text-sm text-destructive\" role=\"alert\">\n <p className=\"font-medium\">{message || errorText}</p>\n {displayedError?.message ? (\n <p className=\"mt-1 opacity-80\">{displayedError.message}</p>\n ) : null}\n </div>\n );\n\n const body = (\n <div\n className={cn(\n 'flex h-full min-h-[360px] flex-col overflow-hidden rounded-md border bg-background',\n containerClassName\n )}\n >\n {showToolbar ? (\n <div\n className={cn(\n 'flex flex-wrap items-center justify-between gap-2 border-b px-3 py-2',\n toolbarClassName\n )}\n >\n <span className=\"text-xs font-medium text-muted-foreground\">\n {title || formatLabel}\n </span>\n {showOpenInNewTab\n ? renderButton(\n '新窗口打开',\n <ExternalLinkIcon className=\"h-4 w-4\" />,\n () => window.open(src, '_blank', 'noreferrer')\n )\n : null}\n </div>\n ) : null}\n <div className=\"relative flex min-h-0 flex-1 items-center justify-center bg-muted/20 p-4\">\n {!isSupported\n ? renderError(unsupportedText)\n : displayedError\n ? renderError()\n : null}\n {isSupported && !displayedError ? (\n <>\n {isLoading ? renderLoading() : null}\n <video\n {...videoProps}\n key={src}\n className={cn(\n 'max-h-full w-full max-w-full bg-black',\n isLoading ? 'invisible absolute' : 'visible',\n videoClassName\n )}\n controls={controls}\n onError={() => {\n const nextError = new Error(errorText);\n setIsVideoLoading(false);\n setVideoError(nextError);\n onError?.(nextError);\n }}\n onLoadedData={() => {\n setIsVideoLoading(false);\n setVideoError(null);\n onLoadedData?.();\n }}\n playsInline={playsInline}\n preload={preload}\n title={title}\n >\n <source src={src} type={mimeType} />\n {tracks?.map((track) => (\n <track key={`${track.src}-${track.kind || ''}`} {...track} />\n ))}\n </video>\n </>\n ) : null}\n </div>\n </div>\n );\n\n if (Card) {\n return (\n <Card className={className}>\n {CardContent ? <CardContent>{body}</CardContent> : body}\n </Card>\n );\n }\n\n return <div className={className}>{body}</div>;\n}\n"],"names":["SUPPORTED_VIDEO_EXTENSIONS","SUPPORTED_VIDEO_MIME_TYPES","normalizeError","error","VideoReader","src","fileName","mimeType","title","components","tracks","loading","className","containerClassName","videoClassName","toolbarClassName","loadingText","errorText","unsupportedText","showToolbar","showOpenInNewTab","allowUnsupportedFormat","supportedExtensions","supportedMimeTypes","controls","preload","playsInline","onLoadedData","onError","videoProps","Card","CardContent","Button","Skeleton","isVideoLoading","setIsVideoLoading","useState","videoError","setVideoError","displayedError","isSupported","isSupportedMediaSource","isLoading","formatLabel","useMemo","extension","getMediaExtension","useEffect","renderButton","label","icon","onClick","buttonClassName","jsx","renderLoading","jsxs","Fragment","renderError","message","body","cn","ExternalLinkIcon","createElement","nextError","track"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@turinhub/atomix-common-ui",
3
- "version": "0.4.0",
3
+ "version": "0.6.0",
4
4
  "description": "Common UI components for TurinHub projects",
5
5
  "type": "module",
6
6
  "main": "./dist/index.cjs",
@@ -17,6 +17,16 @@
17
17
  "default": "./dist/index.cjs"
18
18
  }
19
19
  },
20
+ "./auth": {
21
+ "import": {
22
+ "types": "./dist/auth.d.ts",
23
+ "default": "./dist/auth.js"
24
+ },
25
+ "require": {
26
+ "types": "./dist/auth.d.ts",
27
+ "default": "./dist/auth.cjs"
28
+ }
29
+ },
20
30
  "./data-table": {
21
31
  "import": {
22
32
  "types": "./dist/data-table.d.ts",
@@ -87,6 +97,26 @@
87
97
  "default": "./dist/file-upload.cjs"
88
98
  }
89
99
  },
100
+ "./image-reader": {
101
+ "import": {
102
+ "types": "./dist/image-reader.d.ts",
103
+ "default": "./dist/image-reader.js"
104
+ },
105
+ "require": {
106
+ "types": "./dist/image-reader.d.ts",
107
+ "default": "./dist/image-reader.cjs"
108
+ }
109
+ },
110
+ "./video-reader": {
111
+ "import": {
112
+ "types": "./dist/video-reader.d.ts",
113
+ "default": "./dist/video-reader.js"
114
+ },
115
+ "require": {
116
+ "types": "./dist/video-reader.d.ts",
117
+ "default": "./dist/video-reader.cjs"
118
+ }
119
+ },
90
120
  "./simple-pdf-reader": {
91
121
  "import": {
92
122
  "types": "./dist/simple-pdf-reader.d.ts",
@@ -211,6 +241,7 @@
211
241
  }
212
242
  },
213
243
  "devDependencies": {
244
+ "@radix-ui/react-switch": "^1.3.1",
214
245
  "@testing-library/jest-dom": "^6.4.0",
215
246
  "@testing-library/react": "^16.0.0",
216
247
  "@testing-library/user-event": "^14.5.0",
@@ -1,232 +0,0 @@
1
- import { j as a } from "./jsx-runtime-B4hRZ52C.js";
2
- import { memo as H, useState as N, useRef as $, useMemo as C, useEffect as g, useCallback as k } from "react";
3
- const B = 2, K = (t, i) => {
4
- if (t <= 0) return [];
5
- const l = Math.max(1, Math.min(i, t)), o = Math.max(1, l - B), c = Math.min(t, l + B);
6
- return Array.from({ length: c - o + 1 }, (f, d) => o + d);
7
- }, L = H(
8
- ({
9
- thumbnail: t,
10
- isCurrentPage: i,
11
- onClick: l
12
- }) => /* @__PURE__ */ a.jsxs(
13
- "div",
14
- {
15
- className: `flex w-full flex-col items-center rounded p-1 ${i ? "bg-primary/10" : ""}`,
16
- children: [
17
- /* @__PURE__ */ a.jsx(
18
- "img",
19
- {
20
- src: t.url,
21
- alt: `Page ${t.pageNumber}`,
22
- className: "w-48 cursor-pointer border transition-opacity hover:opacity-80",
23
- onClick: l,
24
- onKeyDown: (o) => {
25
- (o.key === "Enter" || o.key === " ") && (o.preventDefault(), l());
26
- },
27
- role: "button",
28
- tabIndex: 0
29
- }
30
- ),
31
- /* @__PURE__ */ a.jsxs("span", { className: "mt-1 text-sm", children: [
32
- "第 ",
33
- t.pageNumber,
34
- " 页"
35
- ] })
36
- ]
37
- }
38
- )
39
- );
40
- L.displayName = "PDFThumbnail";
41
- const R = ({
42
- bookmark: t,
43
- depth: i,
44
- onClick: l
45
- }) => /* @__PURE__ */ a.jsxs("div", { style: { paddingLeft: `${i * 16}px` }, children: [
46
- /* @__PURE__ */ a.jsx(
47
- "button",
48
- {
49
- onClick: () => l(t),
50
- className: "w-full rounded px-2 py-1 text-left text-sm hover:bg-primary/10 hover:text-primary",
51
- children: t.title
52
- }
53
- ),
54
- t.items && t.items.map((o, c) => /* @__PURE__ */ a.jsx(
55
- R,
56
- {
57
- bookmark: o,
58
- depth: i + 1,
59
- onClick: l
60
- },
61
- `${t.title}-${c}`
62
- ))
63
- ] });
64
- function Q({
65
- pdfDocument: t,
66
- currentPage: i,
67
- onPageClick: l,
68
- components: o
69
- }) {
70
- const [c, f] = N([]), [d, w] = N([]), [T, b] = N(
71
- []
72
- ), m = $(null), x = $(/* @__PURE__ */ new Map()), { Tabs: F, TabsList: O, TabsTrigger: P, TabsContent: E, ScrollArea: A, Skeleton: S } = o, v = C(
73
- () => c.reduce((r, e) => (r[e.pageNumber] = e, r), {}),
74
- [c]
75
- ), p = C(() => {
76
- if (!t) return [];
77
- const r = /* @__PURE__ */ new Set([
78
- ...K(t.numPages, i),
79
- ...T
80
- ]);
81
- return Array.from(r).filter(
82
- (e) => e >= 1 && e <= t.numPages && !v[e]
83
- ).sort((e, s) => e - s);
84
- }, [t, i, v, T]);
85
- g(() => {
86
- f([]), b([]);
87
- }, [t]);
88
- const I = k((r) => {
89
- b(
90
- (e) => e.includes(r) ? e : [...e, r]
91
- );
92
- }, []);
93
- g(() => {
94
- if (typeof IntersectionObserver > "u") {
95
- t && b(
96
- Array.from({ length: t.numPages }, (e, s) => s + 1)
97
- );
98
- return;
99
- }
100
- const r = new IntersectionObserver(
101
- (e) => {
102
- e.forEach((s) => {
103
- if (!s.isIntersecting) return;
104
- const n = Number(
105
- s.target.dataset.pageNumber
106
- );
107
- Number.isFinite(n) && I(n);
108
- });
109
- },
110
- { rootMargin: "600px 0px" }
111
- );
112
- return m.current = r, x.current.forEach((e) => {
113
- r.observe(e);
114
- }), () => {
115
- r.disconnect(), m.current = null;
116
- };
117
- }, [t, I]);
118
- const _ = k(
119
- (r, e) => {
120
- var n, u;
121
- const s = x.current.get(r);
122
- s && ((n = m.current) == null || n.unobserve(s), x.current.delete(r)), e && (x.current.set(r, e), (u = m.current) == null || u.observe(e));
123
- },
124
- []
125
- );
126
- g(() => {
127
- if (!t || p.length === 0) return;
128
- let r = !1;
129
- return (async () => {
130
- for (const s of p) {
131
- if (r) return;
132
- try {
133
- const n = await t.getPage(s), u = n.getViewport({ scale: 0.2 }), h = document.createElement("canvas"), M = h.getContext("2d");
134
- if (!M) continue;
135
- if (h.width = u.width, h.height = u.height, await n.render({
136
- canvasContext: M,
137
- viewport: u,
138
- canvas: h
139
- }).promise, r) return;
140
- f((y) => y.some(
141
- (j) => j.pageNumber === s
142
- ) ? y : [
143
- ...y,
144
- {
145
- pageNumber: s,
146
- url: h.toDataURL()
147
- }
148
- ].sort((j, G) => j.pageNumber - G.pageNumber));
149
- } catch (n) {
150
- console.error(
151
- `Error loading thumbnail for page ${s}:`,
152
- n
153
- );
154
- }
155
- }
156
- })(), () => {
157
- r = !0;
158
- };
159
- }, [t, p]), g(() => {
160
- (async () => {
161
- if (t)
162
- try {
163
- const e = await t.getOutline();
164
- w(e || []);
165
- } catch (e) {
166
- console.error("Error loading bookmarks:", e), w([]);
167
- }
168
- })();
169
- }, [t]);
170
- const V = async (r) => {
171
- if (t)
172
- try {
173
- let e;
174
- if (r.dest)
175
- if (typeof r.dest == "string") {
176
- const s = await t.getDestination(r.dest);
177
- s && s[0] && (e = await t.getPageIndex(s[0]));
178
- } else Array.isArray(r.dest) && r.dest[0] && (e = await t.getPageIndex(r.dest[0]));
179
- else r.pageNumber && (e = r.pageNumber - 1);
180
- typeof e == "number" && l(e + 1);
181
- } catch (e) {
182
- console.error("Error navigating to bookmark:", e);
183
- }
184
- }, U = (r) => /* @__PURE__ */ a.jsxs(
185
- "div",
186
- {
187
- ref: (e) => _(r, e),
188
- "data-page-number": r,
189
- className: `flex w-full flex-col items-center rounded p-1 ${i === r ? "bg-primary/10" : ""}`,
190
- children: [
191
- /* @__PURE__ */ a.jsx(S, { className: "h-32 w-48" }),
192
- /* @__PURE__ */ a.jsxs("span", { className: "mt-1 text-sm", children: [
193
- "第 ",
194
- r,
195
- " 页"
196
- ] })
197
- ]
198
- },
199
- r
200
- ), W = () => !t || t.numPages <= 0 ? null : Array.from({ length: t.numPages }, (r, e) => {
201
- const s = e + 1, n = v[s];
202
- return n ? /* @__PURE__ */ a.jsx(
203
- L,
204
- {
205
- thumbnail: n,
206
- isCurrentPage: i === s,
207
- onClick: () => l(s)
208
- },
209
- s
210
- ) : U(s);
211
- });
212
- return /* @__PURE__ */ a.jsx("div", { className: "w-64 border-r bg-muted", children: /* @__PURE__ */ a.jsxs(F, { defaultValue: "thumbnails", children: [
213
- /* @__PURE__ */ a.jsxs(O, { className: "w-full p-2", children: [
214
- /* @__PURE__ */ a.jsx(P, { value: "thumbnails", className: "flex-1 bg-transparent", children: "缩略图" }),
215
- /* @__PURE__ */ a.jsx(P, { value: "bookmarks", className: "flex-1 bg-transparent", children: "书签" })
216
- ] }),
217
- /* @__PURE__ */ a.jsx(E, { value: "thumbnails", children: /* @__PURE__ */ a.jsx(A, { className: "h-[calc(100vh-8rem)]", children: /* @__PURE__ */ a.jsx("div", { className: "space-y-2 p-4", children: W() }) }) }),
218
- /* @__PURE__ */ a.jsx(E, { value: "bookmarks", children: /* @__PURE__ */ a.jsx(A, { className: "h-[calc(100vh-8rem)]", children: /* @__PURE__ */ a.jsx("div", { className: "p-4", children: d.length === 0 ? /* @__PURE__ */ a.jsx("div", { className: "py-4 text-center text-sm text-muted-foreground", children: "没有可用的书签" }) : /* @__PURE__ */ a.jsx("div", { className: "space-y-2", children: d.map((r, e) => /* @__PURE__ */ a.jsx(
219
- R,
220
- {
221
- bookmark: r,
222
- depth: 0,
223
- onClick: V
224
- },
225
- `${r.title}-${e}`
226
- )) }) }) }) })
227
- ] }) });
228
- }
229
- export {
230
- Q as P
231
- };
232
- //# sourceMappingURL=PDFSidebar-BBtucLK6.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"PDFSidebar-BBtucLK6.js","sources":["../src/components/PDFSidebar.tsx"],"sourcesContent":["import { useState, useEffect, memo, useCallback, useMemo, useRef } from 'react';\n\nimport type {\n TabsComponent,\n TabsListComponent,\n TabsTriggerComponent,\n TabsContentComponent,\n ScrollAreaComponent,\n SkeletonComponent,\n} from '../types/component-types';\n\n/**\n * react-pdf 类型定义\n */\nexport interface PDFDocumentProxy {\n numPages: number;\n getPage(pageNumber: number): Promise<PDFPageProxy>;\n getOutline(): Promise<PDFOutline[] | null>;\n getDestination(dest: string): Promise<unknown[] | null>;\n getPageIndex(ref: unknown): Promise<number>;\n}\n\nexport interface PDFPageProxy {\n getViewport(options: { scale: number }): PDFViewport;\n render(renderContext: {\n canvasContext: CanvasRenderingContext2D;\n viewport: PDFViewport;\n canvas: HTMLCanvasElement;\n }): { promise: Promise<void> };\n}\n\nexport interface PDFViewport {\n width: number;\n height: number;\n}\n\nexport interface PDFOutline {\n title: string;\n bold?: boolean;\n italic?: boolean;\n color?: Uint8ClampedArray;\n dest?: string | unknown[] | null;\n url?: string | null;\n unsafeUrl?: string;\n newWindow?: boolean;\n count?: number;\n items?: PDFOutline[];\n pageNumber?: number;\n}\n\n/**\n * PDF 缩略图\n */\ninterface PDFThumbnail {\n pageNumber: number;\n url: string;\n}\n\nconst THUMBNAIL_PAGE_WINDOW = 2;\n\nconst getThumbnailPages = (numPages: number, currentPage: number) => {\n if (numPages <= 0) return [];\n\n const safeCurrentPage = Math.max(1, Math.min(currentPage, numPages));\n const start = Math.max(1, safeCurrentPage - THUMBNAIL_PAGE_WINDOW);\n const end = Math.min(numPages, safeCurrentPage + THUMBNAIL_PAGE_WINDOW);\n\n return Array.from({ length: end - start + 1 }, (_, index) => start + index);\n};\n\n/**\n * PDFSidebar 组件 Props\n */\nexport interface PDFSidebarProps {\n /** PDF 文档对象 */\n pdfDocument: PDFDocumentProxy | null;\n /** 当前页码 */\n currentPage: number;\n /** 页面点击回调 */\n onPageClick: (pageNumber: number) => void;\n /** UI 组件注入 */\n components: {\n Tabs: TabsComponent;\n TabsList: TabsListComponent;\n TabsTrigger: TabsTriggerComponent;\n TabsContent: TabsContentComponent;\n ScrollArea: ScrollAreaComponent;\n Skeleton: SkeletonComponent;\n };\n}\n\n/**\n * PDF 缩略图组件 (使用 React.memo 优化性能)\n */\nconst PDFThumbnail = memo(\n ({\n thumbnail,\n isCurrentPage,\n onClick,\n }: {\n thumbnail: PDFThumbnail;\n isCurrentPage: boolean;\n onClick: () => void;\n }) => (\n <div\n className={`flex w-full flex-col items-center rounded p-1 ${\n isCurrentPage ? 'bg-primary/10' : ''\n }`}\n >\n <img\n src={thumbnail.url}\n alt={`Page ${thumbnail.pageNumber}`}\n className=\"w-48 cursor-pointer border transition-opacity hover:opacity-80\"\n onClick={onClick}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault();\n onClick();\n }\n }}\n role=\"button\"\n tabIndex={0}\n />\n <span className=\"mt-1 text-sm\">第 {thumbnail.pageNumber} 页</span>\n </div>\n )\n);\nPDFThumbnail.displayName = 'PDFThumbnail';\n\n/**\n * PDF 书签组件\n */\nconst PDFBookmark = ({\n bookmark,\n depth,\n onClick,\n}: {\n bookmark: PDFOutline;\n depth: number;\n onClick: (bookmark: PDFOutline) => void;\n}) => (\n <div style={{ paddingLeft: `${depth * 16}px` }}>\n <button\n onClick={() => onClick(bookmark)}\n className=\"w-full rounded px-2 py-1 text-left text-sm hover:bg-primary/10 hover:text-primary\"\n >\n {bookmark.title}\n </button>\n {bookmark.items &&\n bookmark.items.map((item, index) => (\n <PDFBookmark\n key={`${bookmark.title}-${index}`}\n bookmark={item}\n depth={depth + 1}\n onClick={onClick}\n />\n ))}\n </div>\n);\n\n/**\n * PDFSidebar 组件\n *\n * 提供缩略图和书签导航功能。\n *\n * @example\n * ```tsx\n * import { PDFSidebar } from '@turinhub/atomix-common-ui/pdf-sidebar';\n * import { Tabs, ScrollArea, Skeleton } from '@/components/ui';\n *\n * <PDFSidebar\n * pdfDocument={pdfDocument}\n * currentPage={currentPage}\n * onPageClick={handlePageClick}\n * components={{\n * Tabs,\n * TabsList: Tabs.List,\n * TabsTrigger: Tabs.Trigger,\n * TabsContent: Tabs.Content,\n * ScrollArea,\n * Skeleton,\n * }}\n * />\n * ```\n */\nexport function PDFSidebar({\n pdfDocument,\n currentPage,\n onPageClick,\n components,\n}: PDFSidebarProps) {\n const [thumbnails, setThumbnails] = useState<PDFThumbnail[]>([]);\n const [bookmarks, setBookmarks] = useState<PDFOutline[]>([]);\n const [visibleThumbnailPages, setVisibleThumbnailPages] = useState<number[]>(\n []\n );\n const intersectionObserverRef = useRef<IntersectionObserver | null>(null);\n const placeholderElementsRef = useRef<Map<number, HTMLDivElement>>(new Map());\n\n const { Tabs, TabsList, TabsTrigger, TabsContent, ScrollArea, Skeleton } =\n components;\n\n const thumbnailMap = useMemo(\n () =>\n thumbnails.reduce<Record<number, PDFThumbnail>>((map, thumbnail) => {\n map[thumbnail.pageNumber] = thumbnail;\n return map;\n }, {}),\n [thumbnails]\n );\n\n const pagesToLoad = useMemo(() => {\n if (!pdfDocument) return [];\n\n const pageNumbers = new Set([\n ...getThumbnailPages(pdfDocument.numPages, currentPage),\n ...visibleThumbnailPages,\n ]);\n\n return Array.from(pageNumbers)\n .filter(\n (pageNumber) =>\n pageNumber >= 1 &&\n pageNumber <= pdfDocument.numPages &&\n !thumbnailMap[pageNumber]\n )\n .sort((a, b) => a - b);\n }, [pdfDocument, currentPage, thumbnailMap, visibleThumbnailPages]);\n\n useEffect(() => {\n setThumbnails([]);\n setVisibleThumbnailPages([]);\n }, [pdfDocument]);\n\n const trackVisibleThumbnailPage = useCallback((pageNumber: number) => {\n setVisibleThumbnailPages((pages) =>\n pages.includes(pageNumber) ? pages : [...pages, pageNumber]\n );\n }, []);\n\n useEffect(() => {\n if (typeof IntersectionObserver === 'undefined') {\n if (pdfDocument) {\n setVisibleThumbnailPages(\n Array.from({ length: pdfDocument.numPages }, (_, index) => index + 1)\n );\n }\n return;\n }\n\n const observer = new IntersectionObserver(\n (entries) => {\n entries.forEach((entry) => {\n if (!entry.isIntersecting) return;\n\n const pageNumber = Number(\n (entry.target as HTMLElement).dataset.pageNumber\n );\n\n if (Number.isFinite(pageNumber)) {\n trackVisibleThumbnailPage(pageNumber);\n }\n });\n },\n { rootMargin: '600px 0px' }\n );\n\n intersectionObserverRef.current = observer;\n placeholderElementsRef.current.forEach((element) => {\n observer.observe(element);\n });\n\n return () => {\n observer.disconnect();\n intersectionObserverRef.current = null;\n };\n }, [pdfDocument, trackVisibleThumbnailPage]);\n\n const setThumbnailPlaceholderRef = useCallback(\n (pageNumber: number, element: HTMLDivElement | null) => {\n const previousElement = placeholderElementsRef.current.get(pageNumber);\n\n if (previousElement) {\n intersectionObserverRef.current?.unobserve(previousElement);\n placeholderElementsRef.current.delete(pageNumber);\n }\n\n if (!element) return;\n\n placeholderElementsRef.current.set(pageNumber, element);\n intersectionObserverRef.current?.observe(element);\n },\n []\n );\n\n // 按当前页附近的窗口懒加载缩略图,避免大 PDF 一次性生成全部页面。\n useEffect(() => {\n if (!pdfDocument || pagesToLoad.length === 0) return;\n\n let cancelled = false;\n\n const loadThumbnails = async () => {\n for (const pageNumber of pagesToLoad) {\n if (cancelled) return;\n\n try {\n const page = await pdfDocument.getPage(pageNumber);\n const viewport = page.getViewport({ scale: 0.2 });\n const canvas = document.createElement('canvas');\n const context = canvas.getContext('2d');\n\n if (!context) continue;\n\n canvas.width = viewport.width;\n canvas.height = viewport.height;\n\n const renderTask = page.render({\n canvasContext: context,\n viewport,\n canvas,\n });\n\n await renderTask.promise;\n\n if (cancelled) return;\n\n setThumbnails((currentThumbnails) => {\n if (\n currentThumbnails.some(\n (thumbnail) => thumbnail.pageNumber === pageNumber\n )\n ) {\n return currentThumbnails;\n }\n\n return [\n ...currentThumbnails,\n {\n pageNumber,\n url: canvas.toDataURL(),\n },\n ].sort((a, b) => a.pageNumber - b.pageNumber);\n });\n } catch (error) {\n console.error(\n `Error loading thumbnail for page ${pageNumber}:`,\n error\n );\n }\n }\n };\n\n loadThumbnails();\n\n return () => {\n cancelled = true;\n };\n }, [pdfDocument, pagesToLoad]);\n\n // 加载书签\n useEffect(() => {\n const loadBookmarks = async () => {\n if (!pdfDocument) return;\n try {\n const outline = await pdfDocument.getOutline();\n setBookmarks((outline as PDFOutline[]) || []);\n } catch (error) {\n console.error('Error loading bookmarks:', error);\n setBookmarks([]);\n }\n };\n\n loadBookmarks();\n }, [pdfDocument]);\n\n // 处理书签点击\n const handleBookmarkClick = async (bookmark: PDFOutline) => {\n if (!pdfDocument) return;\n\n try {\n let pageIndex: number | undefined;\n\n // 处理不同类型的书签目标\n if (bookmark.dest) {\n if (typeof bookmark.dest === 'string') {\n // 命名目标\n const destination = await pdfDocument.getDestination(bookmark.dest);\n if (destination && destination[0]) {\n pageIndex = await pdfDocument.getPageIndex(destination[0]);\n }\n } else if (Array.isArray(bookmark.dest) && bookmark.dest[0]) {\n // 显式目标\n pageIndex = await pdfDocument.getPageIndex(bookmark.dest[0]);\n }\n } else if (bookmark.pageNumber) {\n // 直接页码\n pageIndex = bookmark.pageNumber - 1;\n }\n\n if (typeof pageIndex === 'number') {\n onPageClick(pageIndex + 1);\n }\n } catch (error) {\n console.error('Error navigating to bookmark:', error);\n }\n };\n\n const renderThumbnailPlaceholder = (pageNumber: number) => (\n <div\n ref={(element) => setThumbnailPlaceholderRef(pageNumber, element)}\n key={pageNumber}\n data-page-number={pageNumber}\n className={`flex w-full flex-col items-center rounded p-1 ${\n currentPage === pageNumber ? 'bg-primary/10' : ''\n }`}\n >\n <Skeleton className=\"h-32 w-48\" />\n <span className=\"mt-1 text-sm\">第 {pageNumber} 页</span>\n </div>\n );\n\n const renderThumbnailList = () => {\n if (!pdfDocument || pdfDocument.numPages <= 0) return null;\n\n return Array.from({ length: pdfDocument.numPages }, (_, index) => {\n const pageNumber = index + 1;\n const thumbnail = thumbnailMap[pageNumber];\n\n return thumbnail ? (\n <PDFThumbnail\n key={pageNumber}\n thumbnail={thumbnail}\n isCurrentPage={currentPage === pageNumber}\n onClick={() => onPageClick(pageNumber)}\n />\n ) : (\n renderThumbnailPlaceholder(pageNumber)\n );\n });\n };\n\n return (\n <div className=\"w-64 border-r bg-muted\">\n <Tabs defaultValue=\"thumbnails\">\n <TabsList className=\"w-full p-2\">\n <TabsTrigger value=\"thumbnails\" className=\"flex-1 bg-transparent\">\n 缩略图\n </TabsTrigger>\n <TabsTrigger value=\"bookmarks\" className=\"flex-1 bg-transparent\">\n 书签\n </TabsTrigger>\n </TabsList>\n\n <TabsContent value=\"thumbnails\">\n <ScrollArea className=\"h-[calc(100vh-8rem)]\">\n <div className=\"space-y-2 p-4\">{renderThumbnailList()}</div>\n </ScrollArea>\n </TabsContent>\n\n <TabsContent value=\"bookmarks\">\n <ScrollArea className=\"h-[calc(100vh-8rem)]\">\n <div className=\"p-4\">\n {bookmarks.length === 0 ? (\n <div className=\"py-4 text-center text-sm text-muted-foreground\">\n 没有可用的书签\n </div>\n ) : (\n <div className=\"space-y-2\">\n {bookmarks.map((bookmark, index) => (\n <PDFBookmark\n key={`${bookmark.title}-${index}`}\n bookmark={bookmark}\n depth={0}\n onClick={handleBookmarkClick}\n />\n ))}\n </div>\n )}\n </div>\n </ScrollArea>\n </TabsContent>\n </Tabs>\n </div>\n );\n}\n"],"names":["THUMBNAIL_PAGE_WINDOW","getThumbnailPages","numPages","currentPage","safeCurrentPage","start","end","_","index","PDFThumbnail","memo","thumbnail","isCurrentPage","onClick","jsxs","jsx","e","PDFBookmark","bookmark","depth","item","PDFSidebar","pdfDocument","onPageClick","components","thumbnails","setThumbnails","useState","bookmarks","setBookmarks","visibleThumbnailPages","setVisibleThumbnailPages","intersectionObserverRef","useRef","placeholderElementsRef","Tabs","TabsList","TabsTrigger","TabsContent","ScrollArea","Skeleton","thumbnailMap","useMemo","map","pagesToLoad","pageNumbers","pageNumber","a","b","useEffect","trackVisibleThumbnailPage","useCallback","pages","observer","entries","entry","element","setThumbnailPlaceholderRef","previousElement","_a","_b","cancelled","page","viewport","canvas","context","currentThumbnails","error","outline","handleBookmarkClick","pageIndex","destination","renderThumbnailPlaceholder","renderThumbnailList"],"mappings":"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"}
@@ -1,2 +0,0 @@
1
- "use strict";const n=require("./jsx-runtime-BB_1_6y_.cjs"),a=require("react"),M=2,V=(t,l)=>{if(t<=0)return[];const o=Math.max(1,Math.min(l,t)),c=Math.max(1,o-M),u=Math.min(t,o+M);return Array.from({length:u-c+1},(p,d)=>c+d)},A=a.memo(({thumbnail:t,isCurrentPage:l,onClick:o})=>n.jsxRuntimeExports.jsxs("div",{className:`flex w-full flex-col items-center rounded p-1 ${l?"bg-primary/10":""}`,children:[n.jsxRuntimeExports.jsx("img",{src:t.url,alt:`Page ${t.pageNumber}`,className:"w-48 cursor-pointer border transition-opacity hover:opacity-80",onClick:o,onKeyDown:c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),o())},role:"button",tabIndex:0}),n.jsxRuntimeExports.jsxs("span",{className:"mt-1 text-sm",children:["第 ",t.pageNumber," 页"]})]}));A.displayName="PDFThumbnail";const I=({bookmark:t,depth:l,onClick:o})=>n.jsxRuntimeExports.jsxs("div",{style:{paddingLeft:`${l*16}px`},children:[n.jsxRuntimeExports.jsx("button",{onClick:()=>o(t),className:"w-full rounded px-2 py-1 text-left text-sm hover:bg-primary/10 hover:text-primary",children:t.title}),t.items&&t.items.map((c,u)=>n.jsxRuntimeExports.jsx(I,{bookmark:c,depth:l+1,onClick:o},`${t.title}-${u}`))]});function q({pdfDocument:t,currentPage:l,onPageClick:o,components:c}){const[u,p]=a.useState([]),[d,R]=a.useState([]),[y,f]=a.useState([]),h=a.useRef(null),g=a.useRef(new Map),{Tabs:C,TabsList:S,TabsTrigger:N,TabsContent:w,ScrollArea:T,Skeleton:k}=c,j=a.useMemo(()=>u.reduce((s,e)=>(s[e.pageNumber]=e,s),{}),[u]),b=a.useMemo(()=>{if(!t)return[];const s=new Set([...V(t.numPages,l),...y]);return Array.from(s).filter(e=>e>=1&&e<=t.numPages&&!j[e]).sort((e,r)=>e-r)},[t,l,j,y]);a.useEffect(()=>{p([]),f([])},[t]);const P=a.useCallback(s=>{f(e=>e.includes(s)?e:[...e,s])},[]);a.useEffect(()=>{if(typeof IntersectionObserver>"u"){t&&f(Array.from({length:t.numPages},(e,r)=>r+1));return}const s=new IntersectionObserver(e=>{e.forEach(r=>{if(!r.isIntersecting)return;const i=Number(r.target.dataset.pageNumber);Number.isFinite(i)&&P(i)})},{rootMargin:"600px 0px"});return h.current=s,g.current.forEach(e=>{s.observe(e)}),()=>{s.disconnect(),h.current=null}},[t,P]);const B=a.useCallback((s,e)=>{var i,x;const r=g.current.get(s);r&&((i=h.current)==null||i.unobserve(r),g.current.delete(s)),e&&(g.current.set(s,e),(x=h.current)==null||x.observe(e))},[]);a.useEffect(()=>{if(!t||b.length===0)return;let s=!1;return(async()=>{for(const r of b){if(s)return;try{const i=await t.getPage(r),x=i.getViewport({scale:.2}),m=document.createElement("canvas"),$=m.getContext("2d");if(!$)continue;if(m.width=x.width,m.height=x.height,await i.render({canvasContext:$,viewport:x,canvas:m}).promise,s)return;p(E=>E.some(v=>v.pageNumber===r)?E:[...E,{pageNumber:r,url:m.toDataURL()}].sort((v,_)=>v.pageNumber-_.pageNumber))}catch(i){console.error(`Error loading thumbnail for page ${r}:`,i)}}})(),()=>{s=!0}},[t,b]),a.useEffect(()=>{(async()=>{if(t)try{const e=await t.getOutline();R(e||[])}catch(e){console.error("Error loading bookmarks:",e),R([])}})()},[t]);const F=async s=>{if(t)try{let e;if(s.dest)if(typeof s.dest=="string"){const r=await t.getDestination(s.dest);r&&r[0]&&(e=await t.getPageIndex(r[0]))}else Array.isArray(s.dest)&&s.dest[0]&&(e=await t.getPageIndex(s.dest[0]));else s.pageNumber&&(e=s.pageNumber-1);typeof e=="number"&&o(e+1)}catch(e){console.error("Error navigating to bookmark:",e)}},L=s=>n.jsxRuntimeExports.jsxs("div",{ref:e=>B(s,e),"data-page-number":s,className:`flex w-full flex-col items-center rounded p-1 ${l===s?"bg-primary/10":""}`,children:[n.jsxRuntimeExports.jsx(k,{className:"h-32 w-48"}),n.jsxRuntimeExports.jsxs("span",{className:"mt-1 text-sm",children:["第 ",s," 页"]})]},s),O=()=>!t||t.numPages<=0?null:Array.from({length:t.numPages},(s,e)=>{const r=e+1,i=j[r];return i?n.jsxRuntimeExports.jsx(A,{thumbnail:i,isCurrentPage:l===r,onClick:()=>o(r)},r):L(r)});return n.jsxRuntimeExports.jsx("div",{className:"w-64 border-r bg-muted",children:n.jsxRuntimeExports.jsxs(C,{defaultValue:"thumbnails",children:[n.jsxRuntimeExports.jsxs(S,{className:"w-full p-2",children:[n.jsxRuntimeExports.jsx(N,{value:"thumbnails",className:"flex-1 bg-transparent",children:"缩略图"}),n.jsxRuntimeExports.jsx(N,{value:"bookmarks",className:"flex-1 bg-transparent",children:"书签"})]}),n.jsxRuntimeExports.jsx(w,{value:"thumbnails",children:n.jsxRuntimeExports.jsx(T,{className:"h-[calc(100vh-8rem)]",children:n.jsxRuntimeExports.jsx("div",{className:"space-y-2 p-4",children:O()})})}),n.jsxRuntimeExports.jsx(w,{value:"bookmarks",children:n.jsxRuntimeExports.jsx(T,{className:"h-[calc(100vh-8rem)]",children:n.jsxRuntimeExports.jsx("div",{className:"p-4",children:d.length===0?n.jsxRuntimeExports.jsx("div",{className:"py-4 text-center text-sm text-muted-foreground",children:"没有可用的书签"}):n.jsxRuntimeExports.jsx("div",{className:"space-y-2",children:d.map((s,e)=>n.jsxRuntimeExports.jsx(I,{bookmark:s,depth:0,onClick:F},`${s.title}-${e}`))})})})})]})})}exports.PDFSidebar=q;
2
- //# sourceMappingURL=PDFSidebar-Di0D-yPS.cjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"PDFSidebar-Di0D-yPS.cjs","sources":["../src/components/PDFSidebar.tsx"],"sourcesContent":["import { useState, useEffect, memo, useCallback, useMemo, useRef } from 'react';\n\nimport type {\n TabsComponent,\n TabsListComponent,\n TabsTriggerComponent,\n TabsContentComponent,\n ScrollAreaComponent,\n SkeletonComponent,\n} from '../types/component-types';\n\n/**\n * react-pdf 类型定义\n */\nexport interface PDFDocumentProxy {\n numPages: number;\n getPage(pageNumber: number): Promise<PDFPageProxy>;\n getOutline(): Promise<PDFOutline[] | null>;\n getDestination(dest: string): Promise<unknown[] | null>;\n getPageIndex(ref: unknown): Promise<number>;\n}\n\nexport interface PDFPageProxy {\n getViewport(options: { scale: number }): PDFViewport;\n render(renderContext: {\n canvasContext: CanvasRenderingContext2D;\n viewport: PDFViewport;\n canvas: HTMLCanvasElement;\n }): { promise: Promise<void> };\n}\n\nexport interface PDFViewport {\n width: number;\n height: number;\n}\n\nexport interface PDFOutline {\n title: string;\n bold?: boolean;\n italic?: boolean;\n color?: Uint8ClampedArray;\n dest?: string | unknown[] | null;\n url?: string | null;\n unsafeUrl?: string;\n newWindow?: boolean;\n count?: number;\n items?: PDFOutline[];\n pageNumber?: number;\n}\n\n/**\n * PDF 缩略图\n */\ninterface PDFThumbnail {\n pageNumber: number;\n url: string;\n}\n\nconst THUMBNAIL_PAGE_WINDOW = 2;\n\nconst getThumbnailPages = (numPages: number, currentPage: number) => {\n if (numPages <= 0) return [];\n\n const safeCurrentPage = Math.max(1, Math.min(currentPage, numPages));\n const start = Math.max(1, safeCurrentPage - THUMBNAIL_PAGE_WINDOW);\n const end = Math.min(numPages, safeCurrentPage + THUMBNAIL_PAGE_WINDOW);\n\n return Array.from({ length: end - start + 1 }, (_, index) => start + index);\n};\n\n/**\n * PDFSidebar 组件 Props\n */\nexport interface PDFSidebarProps {\n /** PDF 文档对象 */\n pdfDocument: PDFDocumentProxy | null;\n /** 当前页码 */\n currentPage: number;\n /** 页面点击回调 */\n onPageClick: (pageNumber: number) => void;\n /** UI 组件注入 */\n components: {\n Tabs: TabsComponent;\n TabsList: TabsListComponent;\n TabsTrigger: TabsTriggerComponent;\n TabsContent: TabsContentComponent;\n ScrollArea: ScrollAreaComponent;\n Skeleton: SkeletonComponent;\n };\n}\n\n/**\n * PDF 缩略图组件 (使用 React.memo 优化性能)\n */\nconst PDFThumbnail = memo(\n ({\n thumbnail,\n isCurrentPage,\n onClick,\n }: {\n thumbnail: PDFThumbnail;\n isCurrentPage: boolean;\n onClick: () => void;\n }) => (\n <div\n className={`flex w-full flex-col items-center rounded p-1 ${\n isCurrentPage ? 'bg-primary/10' : ''\n }`}\n >\n <img\n src={thumbnail.url}\n alt={`Page ${thumbnail.pageNumber}`}\n className=\"w-48 cursor-pointer border transition-opacity hover:opacity-80\"\n onClick={onClick}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault();\n onClick();\n }\n }}\n role=\"button\"\n tabIndex={0}\n />\n <span className=\"mt-1 text-sm\">第 {thumbnail.pageNumber} 页</span>\n </div>\n )\n);\nPDFThumbnail.displayName = 'PDFThumbnail';\n\n/**\n * PDF 书签组件\n */\nconst PDFBookmark = ({\n bookmark,\n depth,\n onClick,\n}: {\n bookmark: PDFOutline;\n depth: number;\n onClick: (bookmark: PDFOutline) => void;\n}) => (\n <div style={{ paddingLeft: `${depth * 16}px` }}>\n <button\n onClick={() => onClick(bookmark)}\n className=\"w-full rounded px-2 py-1 text-left text-sm hover:bg-primary/10 hover:text-primary\"\n >\n {bookmark.title}\n </button>\n {bookmark.items &&\n bookmark.items.map((item, index) => (\n <PDFBookmark\n key={`${bookmark.title}-${index}`}\n bookmark={item}\n depth={depth + 1}\n onClick={onClick}\n />\n ))}\n </div>\n);\n\n/**\n * PDFSidebar 组件\n *\n * 提供缩略图和书签导航功能。\n *\n * @example\n * ```tsx\n * import { PDFSidebar } from '@turinhub/atomix-common-ui/pdf-sidebar';\n * import { Tabs, ScrollArea, Skeleton } from '@/components/ui';\n *\n * <PDFSidebar\n * pdfDocument={pdfDocument}\n * currentPage={currentPage}\n * onPageClick={handlePageClick}\n * components={{\n * Tabs,\n * TabsList: Tabs.List,\n * TabsTrigger: Tabs.Trigger,\n * TabsContent: Tabs.Content,\n * ScrollArea,\n * Skeleton,\n * }}\n * />\n * ```\n */\nexport function PDFSidebar({\n pdfDocument,\n currentPage,\n onPageClick,\n components,\n}: PDFSidebarProps) {\n const [thumbnails, setThumbnails] = useState<PDFThumbnail[]>([]);\n const [bookmarks, setBookmarks] = useState<PDFOutline[]>([]);\n const [visibleThumbnailPages, setVisibleThumbnailPages] = useState<number[]>(\n []\n );\n const intersectionObserverRef = useRef<IntersectionObserver | null>(null);\n const placeholderElementsRef = useRef<Map<number, HTMLDivElement>>(new Map());\n\n const { Tabs, TabsList, TabsTrigger, TabsContent, ScrollArea, Skeleton } =\n components;\n\n const thumbnailMap = useMemo(\n () =>\n thumbnails.reduce<Record<number, PDFThumbnail>>((map, thumbnail) => {\n map[thumbnail.pageNumber] = thumbnail;\n return map;\n }, {}),\n [thumbnails]\n );\n\n const pagesToLoad = useMemo(() => {\n if (!pdfDocument) return [];\n\n const pageNumbers = new Set([\n ...getThumbnailPages(pdfDocument.numPages, currentPage),\n ...visibleThumbnailPages,\n ]);\n\n return Array.from(pageNumbers)\n .filter(\n (pageNumber) =>\n pageNumber >= 1 &&\n pageNumber <= pdfDocument.numPages &&\n !thumbnailMap[pageNumber]\n )\n .sort((a, b) => a - b);\n }, [pdfDocument, currentPage, thumbnailMap, visibleThumbnailPages]);\n\n useEffect(() => {\n setThumbnails([]);\n setVisibleThumbnailPages([]);\n }, [pdfDocument]);\n\n const trackVisibleThumbnailPage = useCallback((pageNumber: number) => {\n setVisibleThumbnailPages((pages) =>\n pages.includes(pageNumber) ? pages : [...pages, pageNumber]\n );\n }, []);\n\n useEffect(() => {\n if (typeof IntersectionObserver === 'undefined') {\n if (pdfDocument) {\n setVisibleThumbnailPages(\n Array.from({ length: pdfDocument.numPages }, (_, index) => index + 1)\n );\n }\n return;\n }\n\n const observer = new IntersectionObserver(\n (entries) => {\n entries.forEach((entry) => {\n if (!entry.isIntersecting) return;\n\n const pageNumber = Number(\n (entry.target as HTMLElement).dataset.pageNumber\n );\n\n if (Number.isFinite(pageNumber)) {\n trackVisibleThumbnailPage(pageNumber);\n }\n });\n },\n { rootMargin: '600px 0px' }\n );\n\n intersectionObserverRef.current = observer;\n placeholderElementsRef.current.forEach((element) => {\n observer.observe(element);\n });\n\n return () => {\n observer.disconnect();\n intersectionObserverRef.current = null;\n };\n }, [pdfDocument, trackVisibleThumbnailPage]);\n\n const setThumbnailPlaceholderRef = useCallback(\n (pageNumber: number, element: HTMLDivElement | null) => {\n const previousElement = placeholderElementsRef.current.get(pageNumber);\n\n if (previousElement) {\n intersectionObserverRef.current?.unobserve(previousElement);\n placeholderElementsRef.current.delete(pageNumber);\n }\n\n if (!element) return;\n\n placeholderElementsRef.current.set(pageNumber, element);\n intersectionObserverRef.current?.observe(element);\n },\n []\n );\n\n // 按当前页附近的窗口懒加载缩略图,避免大 PDF 一次性生成全部页面。\n useEffect(() => {\n if (!pdfDocument || pagesToLoad.length === 0) return;\n\n let cancelled = false;\n\n const loadThumbnails = async () => {\n for (const pageNumber of pagesToLoad) {\n if (cancelled) return;\n\n try {\n const page = await pdfDocument.getPage(pageNumber);\n const viewport = page.getViewport({ scale: 0.2 });\n const canvas = document.createElement('canvas');\n const context = canvas.getContext('2d');\n\n if (!context) continue;\n\n canvas.width = viewport.width;\n canvas.height = viewport.height;\n\n const renderTask = page.render({\n canvasContext: context,\n viewport,\n canvas,\n });\n\n await renderTask.promise;\n\n if (cancelled) return;\n\n setThumbnails((currentThumbnails) => {\n if (\n currentThumbnails.some(\n (thumbnail) => thumbnail.pageNumber === pageNumber\n )\n ) {\n return currentThumbnails;\n }\n\n return [\n ...currentThumbnails,\n {\n pageNumber,\n url: canvas.toDataURL(),\n },\n ].sort((a, b) => a.pageNumber - b.pageNumber);\n });\n } catch (error) {\n console.error(\n `Error loading thumbnail for page ${pageNumber}:`,\n error\n );\n }\n }\n };\n\n loadThumbnails();\n\n return () => {\n cancelled = true;\n };\n }, [pdfDocument, pagesToLoad]);\n\n // 加载书签\n useEffect(() => {\n const loadBookmarks = async () => {\n if (!pdfDocument) return;\n try {\n const outline = await pdfDocument.getOutline();\n setBookmarks((outline as PDFOutline[]) || []);\n } catch (error) {\n console.error('Error loading bookmarks:', error);\n setBookmarks([]);\n }\n };\n\n loadBookmarks();\n }, [pdfDocument]);\n\n // 处理书签点击\n const handleBookmarkClick = async (bookmark: PDFOutline) => {\n if (!pdfDocument) return;\n\n try {\n let pageIndex: number | undefined;\n\n // 处理不同类型的书签目标\n if (bookmark.dest) {\n if (typeof bookmark.dest === 'string') {\n // 命名目标\n const destination = await pdfDocument.getDestination(bookmark.dest);\n if (destination && destination[0]) {\n pageIndex = await pdfDocument.getPageIndex(destination[0]);\n }\n } else if (Array.isArray(bookmark.dest) && bookmark.dest[0]) {\n // 显式目标\n pageIndex = await pdfDocument.getPageIndex(bookmark.dest[0]);\n }\n } else if (bookmark.pageNumber) {\n // 直接页码\n pageIndex = bookmark.pageNumber - 1;\n }\n\n if (typeof pageIndex === 'number') {\n onPageClick(pageIndex + 1);\n }\n } catch (error) {\n console.error('Error navigating to bookmark:', error);\n }\n };\n\n const renderThumbnailPlaceholder = (pageNumber: number) => (\n <div\n ref={(element) => setThumbnailPlaceholderRef(pageNumber, element)}\n key={pageNumber}\n data-page-number={pageNumber}\n className={`flex w-full flex-col items-center rounded p-1 ${\n currentPage === pageNumber ? 'bg-primary/10' : ''\n }`}\n >\n <Skeleton className=\"h-32 w-48\" />\n <span className=\"mt-1 text-sm\">第 {pageNumber} 页</span>\n </div>\n );\n\n const renderThumbnailList = () => {\n if (!pdfDocument || pdfDocument.numPages <= 0) return null;\n\n return Array.from({ length: pdfDocument.numPages }, (_, index) => {\n const pageNumber = index + 1;\n const thumbnail = thumbnailMap[pageNumber];\n\n return thumbnail ? (\n <PDFThumbnail\n key={pageNumber}\n thumbnail={thumbnail}\n isCurrentPage={currentPage === pageNumber}\n onClick={() => onPageClick(pageNumber)}\n />\n ) : (\n renderThumbnailPlaceholder(pageNumber)\n );\n });\n };\n\n return (\n <div className=\"w-64 border-r bg-muted\">\n <Tabs defaultValue=\"thumbnails\">\n <TabsList className=\"w-full p-2\">\n <TabsTrigger value=\"thumbnails\" className=\"flex-1 bg-transparent\">\n 缩略图\n </TabsTrigger>\n <TabsTrigger value=\"bookmarks\" className=\"flex-1 bg-transparent\">\n 书签\n </TabsTrigger>\n </TabsList>\n\n <TabsContent value=\"thumbnails\">\n <ScrollArea className=\"h-[calc(100vh-8rem)]\">\n <div className=\"space-y-2 p-4\">{renderThumbnailList()}</div>\n </ScrollArea>\n </TabsContent>\n\n <TabsContent value=\"bookmarks\">\n <ScrollArea className=\"h-[calc(100vh-8rem)]\">\n <div className=\"p-4\">\n {bookmarks.length === 0 ? (\n <div className=\"py-4 text-center text-sm text-muted-foreground\">\n 没有可用的书签\n </div>\n ) : (\n <div className=\"space-y-2\">\n {bookmarks.map((bookmark, index) => (\n <PDFBookmark\n key={`${bookmark.title}-${index}`}\n bookmark={bookmark}\n depth={0}\n onClick={handleBookmarkClick}\n />\n ))}\n </div>\n )}\n </div>\n </ScrollArea>\n </TabsContent>\n </Tabs>\n </div>\n );\n}\n"],"names":["THUMBNAIL_PAGE_WINDOW","getThumbnailPages","numPages","currentPage","safeCurrentPage","start","end","_","index","PDFThumbnail","memo","thumbnail","isCurrentPage","onClick","jsxs","jsx","e","PDFBookmark","bookmark","depth","item","PDFSidebar","pdfDocument","onPageClick","components","thumbnails","setThumbnails","useState","bookmarks","setBookmarks","visibleThumbnailPages","setVisibleThumbnailPages","intersectionObserverRef","useRef","placeholderElementsRef","Tabs","TabsList","TabsTrigger","TabsContent","ScrollArea","Skeleton","thumbnailMap","useMemo","map","pagesToLoad","pageNumbers","pageNumber","a","b","useEffect","trackVisibleThumbnailPage","useCallback","pages","observer","entries","entry","element","setThumbnailPlaceholderRef","previousElement","_a","_b","cancelled","page","viewport","canvas","context","currentThumbnails","error","outline","handleBookmarkClick","pageIndex","destination","renderThumbnailPlaceholder","renderThumbnailList"],"mappings":"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"}