2024-10-04 11:15:54 +02:00
|
|
|
import React, { useRef, useState } from "react";
|
|
|
|
|
|
|
|
const LensViewer = ({
|
|
|
|
zoomAmount,
|
|
|
|
lensSize,
|
|
|
|
sanitizedImagePath,
|
|
|
|
sanitizedUpscaledImagePath,
|
|
|
|
}: {
|
|
|
|
zoomAmount: string;
|
|
|
|
lensSize: number;
|
|
|
|
sanitizedImagePath: string;
|
|
|
|
sanitizedUpscaledImagePath: string;
|
|
|
|
}) => {
|
2024-10-30 15:36:14 +01:00
|
|
|
const [hoverPosition, setHoverPosition] = useState({ x: 0, y: 0 });
|
2024-10-30 15:48:32 +01:00
|
|
|
const zoomLevel = 4; // Adjust zoom level as needed
|
2024-10-04 11:15:54 +02:00
|
|
|
|
2024-10-30 15:48:32 +01:00
|
|
|
const handleMouseMove = (e: React.MouseEvent) => {
|
2024-10-30 15:36:14 +01:00
|
|
|
const { left, top, width, height } =
|
|
|
|
e.currentTarget.getBoundingClientRect();
|
|
|
|
const x = ((e.clientX - left) / width) * 100;
|
|
|
|
const y = ((e.clientY - top) / height) * 100;
|
2024-10-04 11:15:54 +02:00
|
|
|
|
2024-10-30 15:36:14 +01:00
|
|
|
setHoverPosition({ x, y });
|
2024-10-04 11:15:54 +02:00
|
|
|
};
|
|
|
|
|
2024-10-30 15:36:14 +01:00
|
|
|
const originalImage = "file:///" + sanitizedImagePath;
|
|
|
|
const upscaledImage = "file:///" + sanitizedUpscaledImagePath;
|
|
|
|
|
2024-10-04 11:15:54 +02:00
|
|
|
return (
|
2024-10-30 15:48:32 +01:00
|
|
|
<div className="group relative flex h-screen flex-col items-center">
|
2024-10-30 15:36:14 +01:00
|
|
|
{/* Main image container */}
|
2024-10-04 11:15:54 +02:00
|
|
|
<div
|
2024-10-30 15:36:14 +01:00
|
|
|
className="relative h-full w-full cursor-crosshair bg-cover bg-no-repeat"
|
|
|
|
onMouseMove={handleMouseMove}
|
2024-10-04 11:15:54 +02:00
|
|
|
>
|
2024-10-30 15:36:14 +01:00
|
|
|
<img
|
|
|
|
src={originalImage}
|
|
|
|
alt="Original"
|
|
|
|
className="h-full w-full object-contain"
|
|
|
|
/>
|
2024-10-30 15:48:32 +01:00
|
|
|
<div
|
|
|
|
className="pointer-events-none absolute hidden h-12 w-12 border-2 border-white group-hover:block"
|
|
|
|
style={{
|
|
|
|
left: `${hoverPosition.x}%`,
|
|
|
|
top: `${hoverPosition.y}%`,
|
|
|
|
transform: "translate(-50%, -50%)",
|
|
|
|
}}
|
|
|
|
/>
|
2024-10-04 11:15:54 +02:00
|
|
|
</div>
|
2024-10-30 15:36:14 +01:00
|
|
|
|
|
|
|
{/* Enlarged views for original and upscaled images */}
|
2024-10-30 15:48:32 +01:00
|
|
|
<div
|
|
|
|
className="pointer-events-none absolute hidden gap-4 group-hover:flex "
|
|
|
|
style={{
|
|
|
|
left: `${hoverPosition.x}%`,
|
|
|
|
top: `${hoverPosition.y}%`, // Position below the cursor
|
|
|
|
transform: "translate(-50%, 0)",
|
|
|
|
}}
|
|
|
|
>
|
2024-10-30 15:36:14 +01:00
|
|
|
<div
|
2024-10-30 15:48:32 +01:00
|
|
|
className="relative h-48 w-48 border border-gray-300 bg-cover bg-no-repeat"
|
2024-10-30 15:36:14 +01:00
|
|
|
style={{
|
2024-10-30 15:48:32 +01:00
|
|
|
backgroundImage: `url(${originalImage})`,
|
|
|
|
backgroundPosition: `${hoverPosition.x}% ${hoverPosition.y}%`,
|
|
|
|
backgroundSize: `${100 * zoomLevel}%`, // Increase zoom level to match the white box
|
2024-10-30 15:36:14 +01:00
|
|
|
}}
|
|
|
|
>
|
2024-10-30 15:48:32 +01:00
|
|
|
<span className="absolute bottom-1 left-1 rounded bg-black bg-opacity-60 px-2 py-1 text-sm text-white">
|
|
|
|
Original
|
|
|
|
</span>
|
2024-10-30 15:36:14 +01:00
|
|
|
</div>
|
2024-10-30 15:48:32 +01:00
|
|
|
<div
|
|
|
|
className="relative h-48 w-48 border border-gray-300 bg-cover bg-no-repeat"
|
|
|
|
style={{
|
|
|
|
backgroundImage: `url(${upscaledImage})`,
|
|
|
|
backgroundPosition: `${hoverPosition.x}% ${hoverPosition.y}%`,
|
|
|
|
backgroundSize: `${100 * zoomLevel}%`, // Increase zoom level to match the white box
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<span className="absolute bottom-1 left-1 rounded bg-black bg-opacity-60 px-2 py-1 text-sm text-white">
|
|
|
|
AI Upscaled
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
2024-10-04 11:15:54 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default LensViewer;
|