mirror of
https://github.com/dergigi/boris.git
synced 2026-01-18 14:24:41 +01:00
Makes it easy to see exact timing of saves and identify if debouncing is working correctly.
205 lines
7.6 KiB
TypeScript
205 lines
7.6 KiB
TypeScript
import { useEffect, useRef, useState, useCallback } from 'react'
|
|
|
|
interface UseReadingPositionOptions {
|
|
enabled?: boolean
|
|
onPositionChange?: (position: number) => void
|
|
onReadingComplete?: () => void
|
|
readingCompleteThreshold?: number // Default 0.95 (95%) - matches filter threshold
|
|
syncEnabled?: boolean // Whether to sync positions to Nostr
|
|
onSave?: (position: number) => void // Callback for saving position
|
|
completionHoldMs?: number // How long to hold at 100% before firing complete (default 2000)
|
|
}
|
|
|
|
export const useReadingPosition = ({
|
|
enabled = true,
|
|
onPositionChange,
|
|
onReadingComplete,
|
|
readingCompleteThreshold = 0.95, // Match filter threshold for consistency
|
|
syncEnabled = false,
|
|
onSave,
|
|
completionHoldMs = 2000
|
|
}: UseReadingPositionOptions = {}) => {
|
|
const [position, setPosition] = useState(0)
|
|
const positionRef = useRef(0)
|
|
const [isReadingComplete, setIsReadingComplete] = useState(false)
|
|
const hasTriggeredComplete = useRef(false)
|
|
const lastSavedPosition = useRef(0)
|
|
const saveTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null)
|
|
const hasSavedOnce = useRef(false)
|
|
const completionTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null)
|
|
const lastSavedAtRef = useRef<number>(0)
|
|
const suppressUntilRef = useRef<number>(0)
|
|
|
|
// Suppress auto-saves for a given duration (used after programmatic restore)
|
|
const suppressSavesFor = useCallback((ms: number) => {
|
|
const until = Date.now() + ms
|
|
suppressUntilRef.current = until
|
|
console.log(`[reading-position] [${new Date().toISOString()}] 🛡️ Suppressing saves for ${ms}ms until ${new Date(until).toISOString()}`)
|
|
}, [])
|
|
|
|
// Debounced save function - simple 2s debounce
|
|
const scheduleSave = useCallback((currentPosition: number) => {
|
|
if (!syncEnabled || !onSave) {
|
|
return
|
|
}
|
|
|
|
// Always save instantly when we reach completion (1.0)
|
|
if (currentPosition === 1 && lastSavedPosition.current < 1) {
|
|
if (saveTimerRef.current) {
|
|
clearTimeout(saveTimerRef.current)
|
|
saveTimerRef.current = null
|
|
}
|
|
console.log(`[reading-position] [${new Date().toISOString()}] 💾 Instant save at 100% completion`)
|
|
lastSavedPosition.current = 1
|
|
hasSavedOnce.current = true
|
|
lastSavedAtRef.current = Date.now()
|
|
onSave(1)
|
|
return
|
|
}
|
|
|
|
// Require at least 5% progress change to consider saving
|
|
const MIN_DELTA = 0.05
|
|
const hasSignificantChange = Math.abs(currentPosition - lastSavedPosition.current) >= MIN_DELTA
|
|
|
|
if (!hasSignificantChange) {
|
|
return
|
|
}
|
|
|
|
// Clear any existing timer and schedule new save
|
|
if (saveTimerRef.current) {
|
|
clearTimeout(saveTimerRef.current)
|
|
}
|
|
|
|
const DEBOUNCE_MS = 3000 // Save max every 3 seconds
|
|
console.log(`[reading-position] [${new Date().toISOString()}] ⏱️ Debouncing save for ${DEBOUNCE_MS}ms (pos: ${Math.round(currentPosition * 100)}%)`)
|
|
saveTimerRef.current = setTimeout(() => {
|
|
console.log(`[reading-position] [${new Date().toISOString()}] 💾 Auto-save at ${Math.round(currentPosition * 100)}%`)
|
|
lastSavedPosition.current = currentPosition
|
|
hasSavedOnce.current = true
|
|
lastSavedAtRef.current = Date.now()
|
|
onSave(currentPosition)
|
|
saveTimerRef.current = null
|
|
}, DEBOUNCE_MS)
|
|
}, [syncEnabled, onSave])
|
|
|
|
// Immediate save function
|
|
const saveNow = useCallback(() => {
|
|
if (!syncEnabled || !onSave) return
|
|
if (saveTimerRef.current) {
|
|
clearTimeout(saveTimerRef.current)
|
|
saveTimerRef.current = null
|
|
}
|
|
console.log(`[reading-position] [${new Date().toISOString()}] 💾 saveNow() called at ${Math.round(position * 100)}%`)
|
|
lastSavedPosition.current = position
|
|
hasSavedOnce.current = true
|
|
lastSavedAtRef.current = Date.now()
|
|
onSave(position)
|
|
}, [syncEnabled, onSave, position])
|
|
|
|
useEffect(() => {
|
|
if (!enabled) return
|
|
|
|
const handleScroll = () => {
|
|
// Get the main content area (reader content)
|
|
const readerContent = document.querySelector('.reader-html, .reader-markdown')
|
|
if (!readerContent) return
|
|
|
|
const scrollTop = window.pageYOffset || document.documentElement.scrollTop
|
|
const windowHeight = window.innerHeight
|
|
const documentHeight = document.documentElement.scrollHeight
|
|
|
|
// Calculate position based on how much of the content has been scrolled through
|
|
// Add a small threshold (5px) to account for rounding and make it easier to reach 100%
|
|
const maxScroll = documentHeight - windowHeight
|
|
const scrollProgress = maxScroll > 0 ? scrollTop / maxScroll : 0
|
|
|
|
// If we're within 5px of the bottom, consider it 100%
|
|
const isAtBottom = scrollTop + windowHeight >= documentHeight - 5
|
|
const clampedProgress = isAtBottom ? 1 : Math.max(0, Math.min(1, scrollProgress))
|
|
|
|
setPosition(clampedProgress)
|
|
positionRef.current = clampedProgress
|
|
onPositionChange?.(clampedProgress)
|
|
|
|
// Schedule auto-save if sync is enabled (unless suppressed)
|
|
if (Date.now() >= suppressUntilRef.current) {
|
|
scheduleSave(clampedProgress)
|
|
} else {
|
|
const remainingMs = suppressUntilRef.current - Date.now()
|
|
console.log(`[reading-position] [${new Date().toISOString()}] 🛡️ Save suppressed (${remainingMs}ms remaining) at ${Math.round(clampedProgress * 100)}%`)
|
|
}
|
|
|
|
// Completion detection with 2s hold at 100%
|
|
if (!hasTriggeredComplete.current) {
|
|
// If at exact 100%, start a hold timer; cancel if we scroll up
|
|
if (clampedProgress === 1) {
|
|
if (!completionTimerRef.current) {
|
|
completionTimerRef.current = setTimeout(() => {
|
|
if (!hasTriggeredComplete.current && positionRef.current === 1) {
|
|
setIsReadingComplete(true)
|
|
hasTriggeredComplete.current = true
|
|
onReadingComplete?.()
|
|
}
|
|
completionTimerRef.current = null
|
|
}, completionHoldMs)
|
|
}
|
|
} else {
|
|
// If we moved off 100%, cancel any pending completion hold
|
|
if (completionTimerRef.current) {
|
|
clearTimeout(completionTimerRef.current)
|
|
completionTimerRef.current = null
|
|
// still allow threshold-based completion for near-bottom if configured
|
|
if (clampedProgress >= readingCompleteThreshold) {
|
|
setIsReadingComplete(true)
|
|
hasTriggeredComplete.current = true
|
|
onReadingComplete?.()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Initial calculation
|
|
handleScroll()
|
|
|
|
// Add scroll listener
|
|
window.addEventListener('scroll', handleScroll, { passive: true })
|
|
window.addEventListener('resize', handleScroll, { passive: true })
|
|
|
|
return () => {
|
|
window.removeEventListener('scroll', handleScroll)
|
|
window.removeEventListener('resize', handleScroll)
|
|
|
|
// Clear save timer on unmount
|
|
if (saveTimerRef.current) {
|
|
clearTimeout(saveTimerRef.current)
|
|
}
|
|
if (completionTimerRef.current) {
|
|
clearTimeout(completionTimerRef.current)
|
|
}
|
|
}
|
|
}, [enabled, onPositionChange, onReadingComplete, readingCompleteThreshold, scheduleSave, completionHoldMs])
|
|
|
|
// Reset reading complete state when enabled changes
|
|
useEffect(() => {
|
|
if (!enabled) {
|
|
setIsReadingComplete(false)
|
|
hasTriggeredComplete.current = false
|
|
hasSavedOnce.current = false
|
|
lastSavedPosition.current = 0
|
|
if (completionTimerRef.current) {
|
|
clearTimeout(completionTimerRef.current)
|
|
completionTimerRef.current = null
|
|
}
|
|
}
|
|
}, [enabled])
|
|
|
|
return {
|
|
position,
|
|
isReadingComplete,
|
|
progressPercentage: Math.round(position * 100),
|
|
saveNow,
|
|
suppressSavesFor
|
|
}
|
|
}
|