支持在软件内预览图片

This commit is contained in:
xintaofei
2026-03-24 16:31:33 +08:00
parent 6067d5f9c4
commit b7df63c5f8
13 changed files with 399 additions and 11 deletions

View File

@@ -7,6 +7,7 @@ import type { editor as MonacoEditorNs } from "monaco-editor"
import { useTranslations } from "next-intl"
import { useFolderContext } from "@/contexts/folder-context"
import { useWorkspaceContext } from "@/contexts/workspace-context"
import { ImagePreview } from "@/components/files/image-preview"
import { DiffViewer } from "@/components/diff/diff-viewer"
import { UnifiedDiffPreview } from "@/components/diff/unified-diff-preview"
import {
@@ -1336,6 +1337,11 @@ export function FileWorkspacePanel() {
)
}
// Image preview
if (isFileTab && activeFileTab && activeFileTab.language === "image") {
return <ImagePreview key={activeFileTab.id} tab={activeFileTab} />
}
if (isPreviewMode && activeFileTab) {
const absFilePath =
activeFileTab.path && folderPath

View File

@@ -0,0 +1,290 @@
"use client"
import { useCallback, useEffect, useMemo, useRef, useState } from "react"
import { Minus, Plus, RotateCcw } from "lucide-react"
import { useTranslations } from "next-intl"
import type { FileWorkspaceTab } from "@/contexts/workspace-context"
const ZOOM_STEP = 0.25
const ZOOM_MIN = 0.1
const ZOOM_MAX = 10
const IMAGE_PADDING = 48 // p-6 * 2
function formatFileSize(bytes: number): string {
if (bytes < 1024) return `${bytes} B`
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`
return `${(bytes / (1024 * 1024)).toFixed(1)} MB`
}
/**
* Compute the fitted (object-fit: contain) size of an image within a container.
* Returns the CSS pixel dimensions the image would have at zoom=1.
*/
function fittedSize(
naturalW: number,
naturalH: number,
containerW: number,
containerH: number
): { width: number; height: number } {
if (naturalW === 0 || naturalH === 0 || containerW === 0 || containerH === 0)
return { width: 0, height: 0 }
const availW = containerW - IMAGE_PADDING
const availH = containerH - IMAGE_PADDING
if (availW <= 0 || availH <= 0) return { width: 0, height: 0 }
const scale = Math.min(1, availW / naturalW, availH / naturalH)
return {
width: Math.round(naturalW * scale),
height: Math.round(naturalH * scale),
}
}
export function ImagePreview({ tab }: { tab: FileWorkspaceTab }) {
const t = useTranslations("Folder.fileWorkspacePanel")
const [zoom, setZoom] = useState(1)
const [naturalWidth, setNaturalWidth] = useState(0)
const [naturalHeight, setNaturalHeight] = useState(0)
const fileSize = useMemo(() => {
if (!tab.content) return 0
const base64Part = tab.content.split(",")[1]
if (!base64Part) return 0
const padding = (base64Part.match(/=+$/) ?? [""])[0].length
return Math.floor((base64Part.length * 3) / 4) - padding
}, [tab.content])
const [containerSize, setContainerSize] = useState<{
w: number
h: number
}>({ w: 0, h: 0 })
const handleZoomIn = useCallback(() => {
setZoom((z) => Math.min(ZOOM_MAX, z + ZOOM_STEP))
}, [])
const handleZoomOut = useCallback(() => {
setZoom((z) => Math.max(ZOOM_MIN, z - ZOOM_STEP))
}, [])
const handleZoomReset = useCallback(() => {
setZoom(1)
}, [])
const handleImageLoad = useCallback(
(e: React.SyntheticEvent<HTMLImageElement>) => {
const img = e.currentTarget
setNaturalWidth(img.naturalWidth)
setNaturalHeight(img.naturalHeight)
},
[]
)
// Track container size with ResizeObserver + wheel handler (passive: false).
// Uses a callback ref so setup happens when the DOM node appears
// (it's conditionally rendered behind !tab.loading).
const scrollRef = useRef<HTMLDivElement>(null)
const roRef = useRef<ResizeObserver | null>(null)
const wheelHandler = useRef((e: WheelEvent) => {
if (e.ctrlKey || e.metaKey) {
e.preventDefault()
setZoom((z) => {
const delta = e.deltaY > 0 ? -ZOOM_STEP : ZOOM_STEP
return Math.min(ZOOM_MAX, Math.max(ZOOM_MIN, z + delta))
})
}
})
const scrollCallbackRef = useCallback((el: HTMLDivElement | null) => {
// Tear down previous
const prev = scrollRef.current
if (prev) {
prev.removeEventListener("wheel", wheelHandler.current)
}
if (roRef.current) {
roRef.current.disconnect()
roRef.current = null
}
scrollRef.current = el
if (!el) return
// ResizeObserver
roRef.current = new ResizeObserver((entries) => {
const entry = entries[0]
if (!entry) return
setContainerSize({
w: entry.contentRect.width,
h: entry.contentRect.height,
})
})
roRef.current.observe(el)
// Wheel handler with { passive: false } so preventDefault works
el.addEventListener("wheel", wheelHandler.current, { passive: false })
}, [])
// Right-click drag to pan
const dragRef = useRef<{
active: boolean
startX: number
startY: number
scrollX: number
scrollY: number
} | null>(null)
const handleMouseDown = useCallback((e: React.MouseEvent) => {
if (e.button !== 2) return
const el = scrollRef.current
if (!el) return
e.preventDefault()
dragRef.current = {
active: true,
startX: e.clientX,
startY: e.clientY,
scrollX: el.scrollLeft,
scrollY: el.scrollTop,
}
el.style.cursor = "grabbing"
}, [])
useEffect(() => {
const handleMouseMove = (e: MouseEvent) => {
const drag = dragRef.current
if (!drag?.active) return
const el = scrollRef.current
if (!el) return
el.scrollLeft = drag.scrollX - (e.clientX - drag.startX)
el.scrollTop = drag.scrollY - (e.clientY - drag.startY)
}
const handleMouseUp = () => {
if (!dragRef.current?.active) return
dragRef.current = null
const el = scrollRef.current
if (el) el.style.cursor = ""
}
window.addEventListener("mousemove", handleMouseMove)
window.addEventListener("mouseup", handleMouseUp)
return () => {
window.removeEventListener("mousemove", handleMouseMove)
window.removeEventListener("mouseup", handleMouseUp)
}
}, [])
const handleContextMenu = useCallback((e: React.MouseEvent) => {
e.preventDefault()
}, [])
// Compute display dimensions dynamically from natural size + container size
const ready =
naturalWidth > 0 &&
naturalHeight > 0 &&
containerSize.w > 0 &&
containerSize.h > 0
const base = fittedSize(
naturalWidth,
naturalHeight,
containerSize.w,
containerSize.h
)
const displayWidth = ready ? base.width * zoom : undefined
const displayHeight = ready ? base.height * zoom : undefined
const zoomPercent = Math.round(zoom * 100)
return (
<div className="h-full flex flex-col">
{tab.loading && (
<div className="absolute top-2 right-3 z-10 rounded-md bg-background/70 px-2 py-1 text-[11px] text-muted-foreground backdrop-blur-sm">
{t("loading")}
</div>
)}
{!tab.loading && tab.content && (
<>
{/* Toolbar */}
<div className="flex-none flex items-center gap-1 border-b border-border bg-muted/30 px-3 py-1">
<button
type="button"
onClick={handleZoomOut}
disabled={zoom <= ZOOM_MIN}
className="rounded p-1 hover:bg-muted disabled:opacity-40 transition-colors"
title={t("imageZoomOut")}
>
<Minus className="h-3.5 w-3.5" />
</button>
<button
type="button"
onClick={handleZoomReset}
className="rounded px-1.5 py-0.5 hover:bg-muted transition-colors text-[11px] font-mono text-muted-foreground min-w-[3.5rem] text-center"
title={t("imageZoomReset")}
>
{zoomPercent}%
</button>
<button
type="button"
onClick={handleZoomIn}
disabled={zoom >= ZOOM_MAX}
className="rounded p-1 hover:bg-muted disabled:opacity-40 transition-colors"
title={t("imageZoomIn")}
>
<Plus className="h-3.5 w-3.5" />
</button>
<button
type="button"
onClick={handleZoomReset}
className="rounded p-1 hover:bg-muted transition-colors ml-0.5"
title={t("imageZoomReset")}
>
<RotateCcw className="h-3.5 w-3.5" />
</button>
<div className="ml-auto flex items-center gap-3 text-[11px] text-muted-foreground">
{naturalWidth > 0 && naturalHeight > 0 && (
<span>
{naturalWidth} x {naturalHeight}
</span>
)}
{fileSize > 0 && <span>{formatFileSize(fileSize)}</span>}
</div>
</div>
{/* Image */}
<div
ref={scrollCallbackRef}
className="flex-1 min-h-0 overflow-auto bg-[repeating-conic-gradient(hsl(var(--muted))_0%_25%,transparent_0%_50%)] bg-[length:16px_16px]"
onMouseDown={handleMouseDown}
onContextMenu={handleContextMenu}
>
<div
className="box-border p-6"
style={{
display: "flex",
alignItems: "center",
justifyContent: "center",
minWidth: "100%",
minHeight: "100%",
...(displayWidth != null
? {
width: displayWidth + IMAGE_PADDING,
height: (displayHeight ?? 0) + IMAGE_PADDING,
}
: {}),
}}
>
{/* eslint-disable-next-line @next/next/no-img-element */}
<img
src={tab.content}
alt={tab.title}
onLoad={handleImageLoad}
style={{
display: "block",
flexShrink: 0,
...(displayWidth != null
? { width: displayWidth, height: displayHeight }
: { maxWidth: "100%", maxHeight: "100%" }),
}}
/>
</div>
</div>
</>
)}
</div>
)
}