stacker.news/components/pull-to-refresh.js

82 lines
2.8 KiB
JavaScript
Raw Permalink Normal View History

import { useRouter } from 'next/router'
2024-11-18 19:05:51 +00:00
import { useState, useRef, useEffect, useCallback, useMemo } from 'react'
import styles from './pull-to-refresh.module.css'
2024-12-24 01:50:51 +00:00
const REFRESH_THRESHOLD = 150
2024-11-17 20:58:40 +00:00
export default function PullToRefresh ({ children, className }) {
const router = useRouter()
const [pullDistance, setPullDistance] = useState(0)
const [isPWA, setIsPWA] = useState(false)
const touchStartY = useRef(0)
const touchEndY = useRef(0)
2024-11-17 20:58:40 +00:00
const checkPWA = () => {
const androidPWA = window.matchMedia('(display-mode: standalone)').matches
const iosPWA = window.navigator.standalone === true
setIsPWA(androidPWA || iosPWA)
}
useEffect(checkPWA, [])
const handleTouchStart = useCallback((e) => {
2024-11-17 20:58:40 +00:00
// don't handle if the user is not scrolling from the top of the page, is not on a PWA or if we want Android's native PTR
if (!isPWA || window.scrollY > 0) return
touchStartY.current = e.touches[0].clientY
}, [isPWA])
const handleTouchMove = useCallback((e) => {
2024-11-18 19:05:51 +00:00
if (touchStartY.current === 0) return
if (!isPWA) return
touchEndY.current = e.touches[0].clientY
2024-11-18 19:05:51 +00:00
const distance = touchEndY.current - touchStartY.current
setPullDistance(distance)
document.body.style.marginTop = `${Math.max(0, Math.min(distance / 2, 25))}px`
}, [isPWA])
const handleTouchEnd = useCallback(() => {
2024-11-18 19:05:51 +00:00
if (touchStartY.current === 0 || touchEndY.current === 0) return
if (touchEndY.current - touchStartY.current > REFRESH_THRESHOLD) {
2024-11-18 19:05:51 +00:00
router.push(router.asPath)
}
2024-11-18 19:05:51 +00:00
setPullDistance(0)
document.body.style.marginTop = '0px'
touchStartY.current = 0
touchEndY.current = 0
}, [router])
useEffect(() => {
if (!isPWA) return
2024-11-18 19:05:51 +00:00
document.body.style.overscrollBehaviorY = 'contain'
document.addEventListener('touchstart', handleTouchStart)
document.addEventListener('touchmove', handleTouchMove)
document.addEventListener('touchend', handleTouchEnd)
return () => {
2024-11-18 19:05:51 +00:00
document.body.style.overscrollBehaviorY = ''
document.body.style.marginTop = '0px'
document.removeEventListener('touchstart', handleTouchStart)
document.removeEventListener('touchmove', handleTouchMove)
document.removeEventListener('touchend', handleTouchEnd)
}
}, [isPWA, handleTouchStart, handleTouchMove, handleTouchEnd])
2024-11-18 19:05:51 +00:00
const pullMessage = useMemo(() => {
if (pullDistance > REFRESH_THRESHOLD) return 'release to refresh'
2024-11-18 19:05:51 +00:00
return 'pull down to refresh'
}, [pullDistance])
return (
<main
className={className}
onTouchStart={handleTouchStart}
onTouchMove={handleTouchMove}
onTouchEnd={handleTouchEnd}
>
<p className={`${styles.pullMessage}`} style={{ top: `${Math.max(-20, Math.min(-20 + pullDistance / 2, 5))}px` }}>
{pullMessage}
</p>
{children}
</main>
)
}