mirror of
https://github.com/hicccc77/WeFlow.git
synced 2026-03-25 07:16:51 +00:00
图片解密策略更加激进
This commit is contained in:
@@ -1501,6 +1501,10 @@ function MessageBubble({ message, session, showTime, myAvatarUrl, isGroupChat, o
|
|||||||
const imageClickTimerRef = useRef<number | null>(null)
|
const imageClickTimerRef = useRef<number | null>(null)
|
||||||
const imageContainerRef = useRef<HTMLDivElement>(null)
|
const imageContainerRef = useRef<HTMLDivElement>(null)
|
||||||
const imageAutoDecryptTriggered = useRef(false)
|
const imageAutoDecryptTriggered = useRef(false)
|
||||||
|
const imageAutoHdTriggered = useRef<string | null>(null)
|
||||||
|
const [imageInView, setImageInView] = useState(false)
|
||||||
|
const imageForceHdAttempted = useRef<string | null>(null)
|
||||||
|
const imageForceHdPending = useRef(false)
|
||||||
const [voiceError, setVoiceError] = useState(false)
|
const [voiceError, setVoiceError] = useState(false)
|
||||||
const [voiceLoading, setVoiceLoading] = useState(false)
|
const [voiceLoading, setVoiceLoading] = useState(false)
|
||||||
const [isVoicePlaying, setIsVoicePlaying] = useState(false)
|
const [isVoicePlaying, setIsVoicePlaying] = useState(false)
|
||||||
@@ -1697,10 +1701,13 @@ function MessageBubble({ message, session, showTime, myAvatarUrl, isGroupChat, o
|
|||||||
}
|
}
|
||||||
}, [isEmoji, message.emojiCdnUrl, emojiLocalPath, emojiLoading, emojiError])
|
}, [isEmoji, message.emojiCdnUrl, emojiLocalPath, emojiLoading, emojiError])
|
||||||
|
|
||||||
const requestImageDecrypt = useCallback(async (forceUpdate = false) => {
|
const requestImageDecrypt = useCallback(async (forceUpdate = false, silent = false) => {
|
||||||
if (!isImage || imageLoading) return
|
if (!isImage) return
|
||||||
setImageLoading(true)
|
if (imageLoading) return
|
||||||
setImageError(false)
|
if (!silent) {
|
||||||
|
setImageLoading(true)
|
||||||
|
setImageError(false)
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
if (message.imageMd5 || message.imageDatName) {
|
if (message.imageMd5 || message.imageDatName) {
|
||||||
const result = await window.electronAPI.image.decrypt({
|
const result = await window.electronAPI.image.decrypt({
|
||||||
@@ -1726,14 +1733,25 @@ function MessageBubble({ message, session, showTime, myAvatarUrl, isGroupChat, o
|
|||||||
setImageHasUpdate(false)
|
setImageHasUpdate(false)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
setImageError(true)
|
if (!silent) setImageError(true)
|
||||||
} catch {
|
} catch {
|
||||||
setImageError(true)
|
if (!silent) setImageError(true)
|
||||||
} finally {
|
} finally {
|
||||||
setImageLoading(false)
|
if (!silent) setImageLoading(false)
|
||||||
}
|
}
|
||||||
}, [isImage, imageLoading, message.imageMd5, message.imageDatName, message.localId, session.username, imageCacheKey, detectImageMimeFromBase64])
|
}, [isImage, imageLoading, message.imageMd5, message.imageDatName, message.localId, session.username, imageCacheKey, detectImageMimeFromBase64])
|
||||||
|
|
||||||
|
const triggerForceHd = useCallback(() => {
|
||||||
|
if (!message.imageMd5 && !message.imageDatName) return
|
||||||
|
if (imageForceHdAttempted.current === imageCacheKey) return
|
||||||
|
if (imageForceHdPending.current) return
|
||||||
|
imageForceHdAttempted.current = imageCacheKey
|
||||||
|
imageForceHdPending.current = true
|
||||||
|
requestImageDecrypt(true, true).finally(() => {
|
||||||
|
imageForceHdPending.current = false
|
||||||
|
})
|
||||||
|
}, [imageCacheKey, message.imageDatName, message.imageMd5, requestImageDecrypt])
|
||||||
|
|
||||||
const handleImageClick = useCallback(() => {
|
const handleImageClick = useCallback(() => {
|
||||||
if (imageClickTimerRef.current) {
|
if (imageClickTimerRef.current) {
|
||||||
window.clearTimeout(imageClickTimerRef.current)
|
window.clearTimeout(imageClickTimerRef.current)
|
||||||
@@ -1846,6 +1864,47 @@ function MessageBubble({ message, session, showTime, myAvatarUrl, isGroupChat, o
|
|||||||
return () => observer.disconnect()
|
return () => observer.disconnect()
|
||||||
}, [isImage, imageLocalPath, message.imageMd5, message.imageDatName, requestImageDecrypt])
|
}, [isImage, imageLocalPath, message.imageMd5, message.imageDatName, requestImageDecrypt])
|
||||||
|
|
||||||
|
// 进入视野时自动尝试切换高清图
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isImage) return
|
||||||
|
const container = imageContainerRef.current
|
||||||
|
if (!container) return
|
||||||
|
const observer = new IntersectionObserver(
|
||||||
|
(entries) => {
|
||||||
|
const entry = entries[0]
|
||||||
|
setImageInView(entry.isIntersecting)
|
||||||
|
},
|
||||||
|
{ rootMargin: '120px', threshold: 0 }
|
||||||
|
)
|
||||||
|
observer.observe(container)
|
||||||
|
return () => observer.disconnect()
|
||||||
|
}, [isImage])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isImage || !imageHasUpdate || !imageInView) return
|
||||||
|
if (imageAutoHdTriggered.current === imageCacheKey) return
|
||||||
|
imageAutoHdTriggered.current = imageCacheKey
|
||||||
|
triggerForceHd()
|
||||||
|
}, [isImage, imageHasUpdate, imageInView, imageCacheKey, triggerForceHd])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isImage || !showImagePreview || !imageHasUpdate) return
|
||||||
|
if (imageAutoHdTriggered.current === imageCacheKey) return
|
||||||
|
imageAutoHdTriggered.current = imageCacheKey
|
||||||
|
triggerForceHd()
|
||||||
|
}, [isImage, showImagePreview, imageHasUpdate, imageCacheKey, triggerForceHd])
|
||||||
|
|
||||||
|
// 更激进:进入视野/打开预览时,无论 hasUpdate 与否都尝试强制高清
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isImage || !imageInView) return
|
||||||
|
triggerForceHd()
|
||||||
|
}, [isImage, imageInView, triggerForceHd])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isImage || !showImagePreview) return
|
||||||
|
triggerForceHd()
|
||||||
|
}, [isImage, showImagePreview, triggerForceHd])
|
||||||
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isVoice) return
|
if (!isVoice) return
|
||||||
@@ -2196,23 +2255,15 @@ function MessageBubble({ message, session, showTime, myAvatarUrl, isGroupChat, o
|
|||||||
src={imageLocalPath}
|
src={imageLocalPath}
|
||||||
alt="图片"
|
alt="图片"
|
||||||
className="image-message"
|
className="image-message"
|
||||||
onClick={() => setShowImagePreview(true)}
|
onClick={() => {
|
||||||
|
if (imageHasUpdate) {
|
||||||
|
void requestImageDecrypt(true, true)
|
||||||
|
}
|
||||||
|
setShowImagePreview(true)
|
||||||
|
}}
|
||||||
onLoad={() => setImageError(false)}
|
onLoad={() => setImageError(false)}
|
||||||
onError={() => setImageError(true)}
|
onError={() => setImageError(true)}
|
||||||
/>
|
/>
|
||||||
{imageHasUpdate && (
|
|
||||||
<button
|
|
||||||
className="image-update-button"
|
|
||||||
type="button"
|
|
||||||
title="发现更高清图片,点击更新"
|
|
||||||
onClick={(event) => {
|
|
||||||
event.stopPropagation()
|
|
||||||
void requestImageDecrypt(true)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<RefreshCw size={14} />
|
|
||||||
</button>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
{showImagePreview && (
|
{showImagePreview && (
|
||||||
<ImagePreview src={imageLocalPath} onClose={() => setShowImagePreview(false)} />
|
<ImagePreview src={imageLocalPath} onClose={() => setShowImagePreview(false)} />
|
||||||
|
|||||||
Reference in New Issue
Block a user