|
import { memo, useCallback, useEffect, useRef, useState } from 'react'; |
|
import { toast } from 'react-toastify'; |
|
|
|
interface ScreenshotSelectorProps { |
|
isSelectionMode: boolean; |
|
setIsSelectionMode: (mode: boolean) => void; |
|
containerRef: React.RefObject<HTMLElement>; |
|
} |
|
|
|
export const ScreenshotSelector = memo( |
|
({ isSelectionMode, setIsSelectionMode, containerRef }: ScreenshotSelectorProps) => { |
|
const [isCapturing, setIsCapturing] = useState(false); |
|
const [selectionStart, setSelectionStart] = useState<{ x: number; y: number } | null>(null); |
|
const [selectionEnd, setSelectionEnd] = useState<{ x: number; y: number } | null>(null); |
|
const mediaStreamRef = useRef<MediaStream | null>(null); |
|
const videoRef = useRef<HTMLVideoElement | null>(null); |
|
|
|
useEffect(() => { |
|
|
|
return () => { |
|
if (videoRef.current) { |
|
videoRef.current.pause(); |
|
videoRef.current.srcObject = null; |
|
videoRef.current.remove(); |
|
videoRef.current = null; |
|
} |
|
|
|
if (mediaStreamRef.current) { |
|
mediaStreamRef.current.getTracks().forEach((track) => track.stop()); |
|
mediaStreamRef.current = null; |
|
} |
|
}; |
|
}, []); |
|
|
|
const initializeStream = async () => { |
|
if (!mediaStreamRef.current) { |
|
try { |
|
const stream = await navigator.mediaDevices.getDisplayMedia({ |
|
audio: false, |
|
video: { |
|
displaySurface: 'window', |
|
preferCurrentTab: true, |
|
surfaceSwitching: 'include', |
|
systemAudio: 'exclude', |
|
}, |
|
} as MediaStreamConstraints); |
|
|
|
|
|
stream.addEventListener('inactive', () => { |
|
if (videoRef.current) { |
|
videoRef.current.pause(); |
|
videoRef.current.srcObject = null; |
|
videoRef.current.remove(); |
|
videoRef.current = null; |
|
} |
|
|
|
if (mediaStreamRef.current) { |
|
mediaStreamRef.current.getTracks().forEach((track) => track.stop()); |
|
mediaStreamRef.current = null; |
|
} |
|
|
|
setIsSelectionMode(false); |
|
setSelectionStart(null); |
|
setSelectionEnd(null); |
|
setIsCapturing(false); |
|
}); |
|
|
|
mediaStreamRef.current = stream; |
|
|
|
|
|
if (!videoRef.current) { |
|
const video = document.createElement('video'); |
|
video.style.opacity = '0'; |
|
video.style.position = 'fixed'; |
|
video.style.pointerEvents = 'none'; |
|
video.style.zIndex = '-1'; |
|
document.body.appendChild(video); |
|
videoRef.current = video; |
|
} |
|
|
|
|
|
videoRef.current.srcObject = stream; |
|
await videoRef.current.play(); |
|
} catch (error) { |
|
console.error('Failed to initialize stream:', error); |
|
setIsSelectionMode(false); |
|
toast.error('Failed to initialize screen capture'); |
|
} |
|
} |
|
|
|
return mediaStreamRef.current; |
|
}; |
|
|
|
const handleCopySelection = useCallback(async () => { |
|
if (!isSelectionMode || !selectionStart || !selectionEnd || !containerRef.current) { |
|
return; |
|
} |
|
|
|
setIsCapturing(true); |
|
|
|
try { |
|
const stream = await initializeStream(); |
|
|
|
if (!stream || !videoRef.current) { |
|
return; |
|
} |
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 300)); |
|
|
|
|
|
const tempCanvas = document.createElement('canvas'); |
|
tempCanvas.width = videoRef.current.videoWidth; |
|
tempCanvas.height = videoRef.current.videoHeight; |
|
|
|
const tempCtx = tempCanvas.getContext('2d'); |
|
|
|
if (!tempCtx) { |
|
throw new Error('Failed to get temporary canvas context'); |
|
} |
|
|
|
|
|
tempCtx.drawImage(videoRef.current, 0, 0); |
|
|
|
|
|
const scaleX = videoRef.current.videoWidth / window.innerWidth; |
|
const scaleY = videoRef.current.videoHeight / window.innerHeight; |
|
|
|
|
|
const scrollX = window.scrollX; |
|
const scrollY = window.scrollY + 40; |
|
|
|
|
|
const containerRect = containerRef.current.getBoundingClientRect(); |
|
|
|
|
|
const leftOffset = -9; |
|
const bottomOffset = -14; |
|
|
|
|
|
const scaledX = Math.round( |
|
(containerRect.left + Math.min(selectionStart.x, selectionEnd.x) + scrollX + leftOffset) * scaleX, |
|
); |
|
const scaledY = Math.round( |
|
(containerRect.top + Math.min(selectionStart.y, selectionEnd.y) + scrollY + bottomOffset) * scaleY, |
|
); |
|
const scaledWidth = Math.round(Math.abs(selectionEnd.x - selectionStart.x) * scaleX); |
|
const scaledHeight = Math.round(Math.abs(selectionEnd.y - selectionStart.y) * scaleY); |
|
|
|
|
|
const canvas = document.createElement('canvas'); |
|
canvas.width = Math.round(Math.abs(selectionEnd.x - selectionStart.x)); |
|
canvas.height = Math.round(Math.abs(selectionEnd.y - selectionStart.y)); |
|
|
|
const ctx = canvas.getContext('2d'); |
|
|
|
if (!ctx) { |
|
throw new Error('Failed to get canvas context'); |
|
} |
|
|
|
|
|
ctx.drawImage(tempCanvas, scaledX, scaledY, scaledWidth, scaledHeight, 0, 0, canvas.width, canvas.height); |
|
|
|
|
|
const blob = await new Promise<Blob>((resolve, reject) => { |
|
canvas.toBlob((blob) => { |
|
if (blob) { |
|
resolve(blob); |
|
} else { |
|
reject(new Error('Failed to create blob')); |
|
} |
|
}, 'image/png'); |
|
}); |
|
|
|
|
|
const reader = new FileReader(); |
|
|
|
reader.onload = (e) => { |
|
const base64Image = e.target?.result as string; |
|
|
|
|
|
const textarea = document.querySelector('textarea'); |
|
|
|
if (textarea) { |
|
|
|
const setUploadedFiles = (window as any).__BOLT_SET_UPLOADED_FILES__; |
|
const setImageDataList = (window as any).__BOLT_SET_IMAGE_DATA_LIST__; |
|
const uploadedFiles = (window as any).__BOLT_UPLOADED_FILES__ || []; |
|
const imageDataList = (window as any).__BOLT_IMAGE_DATA_LIST__ || []; |
|
|
|
if (setUploadedFiles && setImageDataList) { |
|
|
|
const file = new File([blob], 'screenshot.png', { type: 'image/png' }); |
|
setUploadedFiles([...uploadedFiles, file]); |
|
setImageDataList([...imageDataList, base64Image]); |
|
toast.success('Screenshot captured and added to chat'); |
|
} else { |
|
toast.error('Could not add screenshot to chat'); |
|
} |
|
} |
|
}; |
|
reader.readAsDataURL(blob); |
|
} catch (error) { |
|
console.error('Failed to capture screenshot:', error); |
|
toast.error('Failed to capture screenshot'); |
|
|
|
if (mediaStreamRef.current) { |
|
mediaStreamRef.current.getTracks().forEach((track) => track.stop()); |
|
mediaStreamRef.current = null; |
|
} |
|
} finally { |
|
setIsCapturing(false); |
|
setSelectionStart(null); |
|
setSelectionEnd(null); |
|
setIsSelectionMode(false); |
|
} |
|
}, [isSelectionMode, selectionStart, selectionEnd, containerRef, setIsSelectionMode]); |
|
|
|
const handleSelectionStart = useCallback( |
|
(e: React.MouseEvent) => { |
|
e.preventDefault(); |
|
e.stopPropagation(); |
|
|
|
if (!isSelectionMode) { |
|
return; |
|
} |
|
|
|
const rect = e.currentTarget.getBoundingClientRect(); |
|
const x = e.clientX - rect.left; |
|
const y = e.clientY - rect.top; |
|
setSelectionStart({ x, y }); |
|
setSelectionEnd({ x, y }); |
|
}, |
|
[isSelectionMode], |
|
); |
|
|
|
const handleSelectionMove = useCallback( |
|
(e: React.MouseEvent) => { |
|
e.preventDefault(); |
|
e.stopPropagation(); |
|
|
|
if (!isSelectionMode || !selectionStart) { |
|
return; |
|
} |
|
|
|
const rect = e.currentTarget.getBoundingClientRect(); |
|
const x = e.clientX - rect.left; |
|
const y = e.clientY - rect.top; |
|
setSelectionEnd({ x, y }); |
|
}, |
|
[isSelectionMode, selectionStart], |
|
); |
|
|
|
if (!isSelectionMode) { |
|
return null; |
|
} |
|
|
|
return ( |
|
<div |
|
className="absolute inset-0 cursor-crosshair" |
|
onMouseDown={handleSelectionStart} |
|
onMouseMove={handleSelectionMove} |
|
onMouseUp={handleCopySelection} |
|
onMouseLeave={() => { |
|
if (selectionStart) { |
|
setSelectionStart(null); |
|
} |
|
}} |
|
style={{ |
|
backgroundColor: isCapturing ? 'transparent' : 'rgba(0, 0, 0, 0.1)', |
|
userSelect: 'none', |
|
WebkitUserSelect: 'none', |
|
pointerEvents: 'all', |
|
opacity: isCapturing ? 0 : 1, |
|
zIndex: 50, |
|
transition: 'opacity 0.1s ease-in-out', |
|
}} |
|
> |
|
{selectionStart && selectionEnd && !isCapturing && ( |
|
<div |
|
className="absolute border-2 border-blue-500 bg-blue-200 bg-opacity-20" |
|
style={{ |
|
left: Math.min(selectionStart.x, selectionEnd.x), |
|
top: Math.min(selectionStart.y, selectionEnd.y), |
|
width: Math.abs(selectionEnd.x - selectionStart.x), |
|
height: Math.abs(selectionEnd.y - selectionStart.y), |
|
}} |
|
/> |
|
)} |
|
</div> |
|
); |
|
}, |
|
); |
|
|