支持在软件内预览图片
This commit is contained in:
@@ -7,6 +7,7 @@ import type { editor as MonacoEditorNs } from "monaco-editor"
|
|||||||
import { useTranslations } from "next-intl"
|
import { useTranslations } from "next-intl"
|
||||||
import { useFolderContext } from "@/contexts/folder-context"
|
import { useFolderContext } from "@/contexts/folder-context"
|
||||||
import { useWorkspaceContext } from "@/contexts/workspace-context"
|
import { useWorkspaceContext } from "@/contexts/workspace-context"
|
||||||
|
import { ImagePreview } from "@/components/files/image-preview"
|
||||||
import { DiffViewer } from "@/components/diff/diff-viewer"
|
import { DiffViewer } from "@/components/diff/diff-viewer"
|
||||||
import { UnifiedDiffPreview } from "@/components/diff/unified-diff-preview"
|
import { UnifiedDiffPreview } from "@/components/diff/unified-diff-preview"
|
||||||
import {
|
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) {
|
if (isPreviewMode && activeFileTab) {
|
||||||
const absFilePath =
|
const absFilePath =
|
||||||
activeFileTab.path && folderPath
|
activeFileTab.path && folderPath
|
||||||
|
|||||||
290
src/components/files/image-preview.tsx
Normal file
290
src/components/files/image-preview.tsx
Normal 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>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -18,6 +18,7 @@ import {
|
|||||||
gitIsTracked,
|
gitIsTracked,
|
||||||
gitShowDiff,
|
gitShowDiff,
|
||||||
gitShowFile,
|
gitShowFile,
|
||||||
|
readFileBase64,
|
||||||
readFileForEdit,
|
readFileForEdit,
|
||||||
readFilePreview,
|
readFilePreview,
|
||||||
saveFileContent,
|
saveFileContent,
|
||||||
@@ -127,6 +128,33 @@ function isDirtyFileTab(tab: FileWorkspaceTab): boolean {
|
|||||||
return tab.kind === "file" && Boolean(tab.isDirty)
|
return tab.kind === "file" && Boolean(tab.isDirty)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const IMAGE_EXTENSIONS = new Set([
|
||||||
|
"png",
|
||||||
|
"jpg",
|
||||||
|
"jpeg",
|
||||||
|
"gif",
|
||||||
|
"svg",
|
||||||
|
"webp",
|
||||||
|
"bmp",
|
||||||
|
"ico",
|
||||||
|
])
|
||||||
|
|
||||||
|
const IMAGE_MIME: Record<string, string> = {
|
||||||
|
png: "image/png",
|
||||||
|
jpg: "image/jpeg",
|
||||||
|
jpeg: "image/jpeg",
|
||||||
|
gif: "image/gif",
|
||||||
|
svg: "image/svg+xml",
|
||||||
|
webp: "image/webp",
|
||||||
|
bmp: "image/bmp",
|
||||||
|
ico: "image/x-icon",
|
||||||
|
}
|
||||||
|
|
||||||
|
function isImageFile(path: string): boolean {
|
||||||
|
const ext = path.split(".").pop()?.toLowerCase() ?? ""
|
||||||
|
return IMAGE_EXTENSIONS.has(ext)
|
||||||
|
}
|
||||||
|
|
||||||
function loadingTab(
|
function loadingTab(
|
||||||
id: string,
|
id: string,
|
||||||
kind: FileWorkspaceTabKind,
|
kind: FileWorkspaceTabKind,
|
||||||
@@ -350,6 +378,7 @@ export function WorkspaceProvider({ children }: WorkspaceProviderProps) {
|
|||||||
setPendingFileReveal(null)
|
setPendingFileReveal(null)
|
||||||
}
|
}
|
||||||
const tabId = `file:${path}`
|
const tabId = `file:${path}`
|
||||||
|
const image = isImageFile(path)
|
||||||
upsertLoadingTab(
|
upsertLoadingTab(
|
||||||
loadingTab(
|
loadingTab(
|
||||||
tabId,
|
tabId,
|
||||||
@@ -357,10 +386,43 @@ export function WorkspaceProvider({ children }: WorkspaceProviderProps) {
|
|||||||
fileName(path),
|
fileName(path),
|
||||||
path,
|
path,
|
||||||
path,
|
path,
|
||||||
languageFromPath(path)
|
image ? "image" : languageFromPath(path)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if (image) {
|
||||||
|
try {
|
||||||
|
const absPath = `${folderPath}/${path}`
|
||||||
|
const ext = path.split(".").pop()?.toLowerCase() ?? ""
|
||||||
|
const mime = IMAGE_MIME[ext] ?? "image/png"
|
||||||
|
const b64 = await withTimeout(
|
||||||
|
readFileBase64(absPath),
|
||||||
|
15_000,
|
||||||
|
t("previewRequestTimedOut")
|
||||||
|
)
|
||||||
|
setFileTabs((prev) =>
|
||||||
|
prev.map((tab) =>
|
||||||
|
tab.id === tabId
|
||||||
|
? {
|
||||||
|
...tab,
|
||||||
|
content: `data:${mime};base64,${b64}`,
|
||||||
|
readonly: true,
|
||||||
|
loading: false,
|
||||||
|
saveState: "idle",
|
||||||
|
saveError: null,
|
||||||
|
}
|
||||||
|
: tab
|
||||||
|
)
|
||||||
|
)
|
||||||
|
} catch (error) {
|
||||||
|
rejectTab(
|
||||||
|
tabId,
|
||||||
|
error instanceof Error ? error.message : String(error)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const [result, gitBaseContent] = await withTimeout(
|
const [result, gitBaseContent] = await withTimeout(
|
||||||
Promise.all([
|
Promise.all([
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "التالي",
|
"next": "التالي",
|
||||||
"jumpToLine": "الانتقال إلى السطر {line}",
|
"jumpToLine": "الانتقال إلى السطر {line}",
|
||||||
"noParsedDiffSections": "لا توجد أقسام diff محللة",
|
"noParsedDiffSections": "لا توجد أقسام diff محللة",
|
||||||
"loadingEditor": "جارٍ تحميل المحرر..."
|
"loadingEditor": "جارٍ تحميل المحرر...",
|
||||||
|
"imageZoomIn": "تكبير",
|
||||||
|
"imageZoomOut": "تصغير",
|
||||||
|
"imageZoomReset": "إعادة تعيين التكبير"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "Weiter",
|
"next": "Weiter",
|
||||||
"jumpToLine": "Zu Zeile {line} springen",
|
"jumpToLine": "Zu Zeile {line} springen",
|
||||||
"noParsedDiffSections": "Keine geparsten Diff-Abschnitte",
|
"noParsedDiffSections": "Keine geparsten Diff-Abschnitte",
|
||||||
"loadingEditor": "Editor wird geladen..."
|
"loadingEditor": "Editor wird geladen...",
|
||||||
|
"imageZoomIn": "Vergrößern",
|
||||||
|
"imageZoomOut": "Verkleinern",
|
||||||
|
"imageZoomReset": "Zoom zurücksetzen"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "Next",
|
"next": "Next",
|
||||||
"jumpToLine": "Jump to line {line}",
|
"jumpToLine": "Jump to line {line}",
|
||||||
"noParsedDiffSections": "No parsed diff sections",
|
"noParsedDiffSections": "No parsed diff sections",
|
||||||
"loadingEditor": "Loading editor..."
|
"loadingEditor": "Loading editor...",
|
||||||
|
"imageZoomIn": "Zoom in",
|
||||||
|
"imageZoomOut": "Zoom out",
|
||||||
|
"imageZoomReset": "Reset zoom"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "Siguiente",
|
"next": "Siguiente",
|
||||||
"jumpToLine": "Ir a la línea {line}",
|
"jumpToLine": "Ir a la línea {line}",
|
||||||
"noParsedDiffSections": "No hay secciones de diff analizadas",
|
"noParsedDiffSections": "No hay secciones de diff analizadas",
|
||||||
"loadingEditor": "Cargando editor..."
|
"loadingEditor": "Cargando editor...",
|
||||||
|
"imageZoomIn": "Ampliar",
|
||||||
|
"imageZoomOut": "Reducir",
|
||||||
|
"imageZoomReset": "Restablecer zoom"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "Suivant",
|
"next": "Suivant",
|
||||||
"jumpToLine": "Aller à la ligne {line}",
|
"jumpToLine": "Aller à la ligne {line}",
|
||||||
"noParsedDiffSections": "Aucune section de diff analysée",
|
"noParsedDiffSections": "Aucune section de diff analysée",
|
||||||
"loadingEditor": "Chargement de l’éditeur..."
|
"loadingEditor": "Chargement de l’éditeur...",
|
||||||
|
"imageZoomIn": "Agrandir",
|
||||||
|
"imageZoomOut": "Réduire",
|
||||||
|
"imageZoomReset": "Réinitialiser le zoom"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "次",
|
"next": "次",
|
||||||
"jumpToLine": "{line} 行へ移動",
|
"jumpToLine": "{line} 行へ移動",
|
||||||
"noParsedDiffSections": "解析済みの差分セクションがありません",
|
"noParsedDiffSections": "解析済みの差分セクションがありません",
|
||||||
"loadingEditor": "エディターを読み込み中..."
|
"loadingEditor": "エディターを読み込み中...",
|
||||||
|
"imageZoomIn": "拡大",
|
||||||
|
"imageZoomOut": "縮小",
|
||||||
|
"imageZoomReset": "ズームをリセット"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "다음",
|
"next": "다음",
|
||||||
"jumpToLine": "{line}행으로 이동",
|
"jumpToLine": "{line}행으로 이동",
|
||||||
"noParsedDiffSections": "파싱된 diff 섹션이 없습니다",
|
"noParsedDiffSections": "파싱된 diff 섹션이 없습니다",
|
||||||
"loadingEditor": "에디터 로딩 중..."
|
"loadingEditor": "에디터 로딩 중...",
|
||||||
|
"imageZoomIn": "확대",
|
||||||
|
"imageZoomOut": "축소",
|
||||||
|
"imageZoomReset": "확대/축소 초기화"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "Próximo",
|
"next": "Próximo",
|
||||||
"jumpToLine": "Ir para a linha {line}",
|
"jumpToLine": "Ir para a linha {line}",
|
||||||
"noParsedDiffSections": "Sem seções de diff analisadas",
|
"noParsedDiffSections": "Sem seções de diff analisadas",
|
||||||
"loadingEditor": "Carregando editor..."
|
"loadingEditor": "Carregando editor...",
|
||||||
|
"imageZoomIn": "Ampliar",
|
||||||
|
"imageZoomOut": "Reduzir",
|
||||||
|
"imageZoomReset": "Redefinir zoom"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "下一个",
|
"next": "下一个",
|
||||||
"jumpToLine": "跳转到第 {line} 行",
|
"jumpToLine": "跳转到第 {line} 行",
|
||||||
"noParsedDiffSections": "未解析到差异区块",
|
"noParsedDiffSections": "未解析到差异区块",
|
||||||
"loadingEditor": "编辑器加载中..."
|
"loadingEditor": "编辑器加载中...",
|
||||||
|
"imageZoomIn": "放大",
|
||||||
|
"imageZoomOut": "缩小",
|
||||||
|
"imageZoomReset": "重置缩放"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
@@ -872,7 +872,10 @@
|
|||||||
"next": "下一個",
|
"next": "下一個",
|
||||||
"jumpToLine": "跳到第 {line} 行",
|
"jumpToLine": "跳到第 {line} 行",
|
||||||
"noParsedDiffSections": "未解析到差異區塊",
|
"noParsedDiffSections": "未解析到差異區塊",
|
||||||
"loadingEditor": "編輯器載入中..."
|
"loadingEditor": "編輯器載入中...",
|
||||||
|
"imageZoomIn": "放大",
|
||||||
|
"imageZoomOut": "縮小",
|
||||||
|
"imageZoomReset": "重設縮放"
|
||||||
},
|
},
|
||||||
"branchDropdown": {
|
"branchDropdown": {
|
||||||
"toasts": {
|
"toasts": {
|
||||||
|
|||||||
Reference in New Issue
Block a user