add offline part
All checks were successful
Build & Deploy / build-and-deploy (push) Successful in 38s
All checks were successful
Build & Deploy / build-and-deploy (push) Successful in 38s
This commit is contained in:
parent
f2227bc945
commit
5ea5a390ce
21
src/App.tsx
21
src/App.tsx
@ -1,7 +1,8 @@
|
|||||||
import { useEffect } from 'react'
|
import { useEffect, useRef } from 'react'
|
||||||
import { Routes, Route } from 'react-router-dom'
|
import { Routes, Route, useNavigate, useLocation } from 'react-router-dom'
|
||||||
import { supabase } from '@/lib/supabase'
|
import { supabase } from '@/lib/supabase'
|
||||||
import { useAuthStore } from '@/stores/auth'
|
import { useAuthStore } from '@/stores/auth'
|
||||||
|
import { useOnlineStatus } from '@/hooks/useOnlineStatus'
|
||||||
import { Layout } from '@/components/layout/Layout'
|
import { Layout } from '@/components/layout/Layout'
|
||||||
import { Home } from '@/pages/Home'
|
import { Home } from '@/pages/Home'
|
||||||
import { Explore } from '@/pages/Explore'
|
import { Explore } from '@/pages/Explore'
|
||||||
@ -12,9 +13,14 @@ import { Profile } from '@/pages/Profile'
|
|||||||
import { Favorites } from '@/pages/Favorites'
|
import { Favorites } from '@/pages/Favorites'
|
||||||
import { Settings } from '@/pages/Settings'
|
import { Settings } from '@/pages/Settings'
|
||||||
import { MyPodcasts } from '@/pages/MyPodcasts'
|
import { MyPodcasts } from '@/pages/MyPodcasts'
|
||||||
|
import { Offline } from '@/pages/Offline'
|
||||||
|
|
||||||
export default function App() {
|
export default function App() {
|
||||||
const { setUser, fetchProfile } = useAuthStore()
|
const { setUser, fetchProfile } = useAuthStore()
|
||||||
|
const isOnline = useOnlineStatus()
|
||||||
|
const navigate = useNavigate()
|
||||||
|
const location = useLocation()
|
||||||
|
const prevPathRef = useRef(location.pathname)
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
supabase.auth.getSession().then(({ data: { session } }) => {
|
supabase.auth.getSession().then(({ data: { session } }) => {
|
||||||
@ -30,6 +36,16 @@ export default function App() {
|
|||||||
return () => subscription.unsubscribe()
|
return () => subscription.unsubscribe()
|
||||||
}, [setUser, fetchProfile])
|
}, [setUser, fetchProfile])
|
||||||
|
|
||||||
|
// Redirect to /offline when connection drops, restore when back online
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isOnline && location.pathname !== '/offline') {
|
||||||
|
prevPathRef.current = location.pathname
|
||||||
|
navigate('/offline', { replace: true })
|
||||||
|
} else if (isOnline && location.pathname === '/offline') {
|
||||||
|
navigate(prevPathRef.current || '/', { replace: true })
|
||||||
|
}
|
||||||
|
}, [isOnline, location.pathname, navigate])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route element={<Layout />}>
|
<Route element={<Layout />}>
|
||||||
@ -42,6 +58,7 @@ export default function App() {
|
|||||||
<Route path="favorites" element={<Favorites />} />
|
<Route path="favorites" element={<Favorites />} />
|
||||||
<Route path="my-podcasts" element={<MyPodcasts />} />
|
<Route path="my-podcasts" element={<MyPodcasts />} />
|
||||||
<Route path="settings" element={<Settings />} />
|
<Route path="settings" element={<Settings />} />
|
||||||
|
<Route path="offline" element={<Offline />} />
|
||||||
</Route>
|
</Route>
|
||||||
</Routes>
|
</Routes>
|
||||||
)
|
)
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
import { useRef, useState, useCallback } from 'react'
|
import { useRef, useState, useCallback } from 'react'
|
||||||
import { Play, Pause, Volume2, VolumeX, SkipBack, SkipForward, Maximize2, Minimize2 } from 'lucide-react'
|
import { Play, Pause, Volume2, VolumeX, SkipBack, SkipForward, Maximize2, Minimize2, ListMusic, X } from 'lucide-react'
|
||||||
import { usePlayerStore } from '@/stores/player'
|
import { usePlayerStore } from '@/stores/player'
|
||||||
import { formatDuration } from '@/lib/utils'
|
import { formatDuration } from '@/lib/utils'
|
||||||
import { publicUrl } from '@/lib/storage'
|
import { publicUrl } from '@/lib/storage'
|
||||||
@ -7,11 +7,12 @@ import { Avatar } from '@/components/ui/Avatar'
|
|||||||
import { isExternalUrl, getEmbedInfo } from '@/lib/embed'
|
import { isExternalUrl, getEmbedInfo } from '@/lib/embed'
|
||||||
|
|
||||||
export function PlayerBar() {
|
export function PlayerBar() {
|
||||||
const { current, isPlaying, isExternal, progress, duration, volume, toggle, seek, setVolume } = usePlayerStore()
|
const { current, isPlaying, isExternal, progress, duration, volume, toggle, seek, setVolume, playbackRate, cyclePlaybackRate, queue, removeFromQueue, clearQueue, playNext, play } = usePlayerStore()
|
||||||
const seekRef = useRef<HTMLDivElement>(null)
|
const seekRef = useRef<HTMLDivElement>(null)
|
||||||
const [isDragging, setIsDragging] = useState(false)
|
const [isDragging, setIsDragging] = useState(false)
|
||||||
const [dragPct, setDragPct] = useState(0)
|
const [dragPct, setDragPct] = useState(0)
|
||||||
const [ytExpanded, setYtExpanded] = useState(false)
|
const [ytExpanded, setYtExpanded] = useState(false)
|
||||||
|
const [queueOpen, setQueueOpen] = useState(false)
|
||||||
|
|
||||||
const calcPct = useCallback((clientX: number) => {
|
const calcPct = useCallback((clientX: number) => {
|
||||||
if (!seekRef.current) return 0
|
if (!seekRef.current) return 0
|
||||||
@ -24,10 +25,17 @@ export function PlayerBar() {
|
|||||||
const embedInfo = isExternal && current ? getEmbedInfo(current.audio_url) : null
|
const embedInfo = isExternal && current ? getEmbedInfo(current.audio_url) : null
|
||||||
const isYouTube = embedInfo?.platform === 'youtube'
|
const isYouTube = embedInfo?.platform === 'youtube'
|
||||||
const isSpotify = embedInfo?.platform === 'spotify'
|
const isSpotify = embedInfo?.platform === 'spotify'
|
||||||
const pct = duration > 0 ? (progress / duration) * 100 : 0
|
const chapters = current?.chapters || []
|
||||||
|
// Use store duration, fallback to podcast metadata duration
|
||||||
|
const effectiveDuration = duration > 0 ? duration : (current?.duration_seconds || 0)
|
||||||
|
const currentChapter = chapters.length > 0 && effectiveDuration > 0
|
||||||
|
? [...chapters].reverse().find((ch) => progress >= ch.start_time_seconds)
|
||||||
|
: null
|
||||||
|
|
||||||
|
const pct = effectiveDuration > 0 ? (progress / effectiveDuration) * 100 : 0
|
||||||
const displayPct = isDragging ? dragPct : pct
|
const displayPct = isDragging ? dragPct : pct
|
||||||
// Allow seeking for native audio and YouTube (via IFrame API)
|
// Allow seeking for native audio and YouTube (via IFrame API)
|
||||||
const canSeek = duration > 0
|
const canSeek = effectiveDuration > 0
|
||||||
const showBar = true
|
const showBar = true
|
||||||
|
|
||||||
function startDrag(clientX: number) {
|
function startDrag(clientX: number) {
|
||||||
@ -45,7 +53,7 @@ export function PlayerBar() {
|
|||||||
const onUp = (ev: MouseEvent) => {
|
const onUp = (ev: MouseEvent) => {
|
||||||
setIsDragging(false)
|
setIsDragging(false)
|
||||||
const p = calcPct(ev.clientX)
|
const p = calcPct(ev.clientX)
|
||||||
seek((p / 100) * duration)
|
seek((p / 100) * effectiveDuration)
|
||||||
window.removeEventListener('mousemove', onMove)
|
window.removeEventListener('mousemove', onMove)
|
||||||
window.removeEventListener('mouseup', onUp)
|
window.removeEventListener('mouseup', onUp)
|
||||||
}
|
}
|
||||||
@ -61,7 +69,7 @@ export function PlayerBar() {
|
|||||||
const onMove = (ev: TouchEvent) => { ev.preventDefault(); setDragPct(calcPct(ev.touches[0].clientX)) }
|
const onMove = (ev: TouchEvent) => { ev.preventDefault(); setDragPct(calcPct(ev.touches[0].clientX)) }
|
||||||
const onEnd = (ev: TouchEvent) => {
|
const onEnd = (ev: TouchEvent) => {
|
||||||
setIsDragging(false)
|
setIsDragging(false)
|
||||||
seek((calcPct(ev.changedTouches[0].clientX) / 100) * duration)
|
seek((calcPct(ev.changedTouches[0].clientX) / 100) * effectiveDuration)
|
||||||
el.removeEventListener('touchmove', onMove)
|
el.removeEventListener('touchmove', onMove)
|
||||||
el.removeEventListener('touchend', onEnd)
|
el.removeEventListener('touchend', onEnd)
|
||||||
}
|
}
|
||||||
@ -71,10 +79,10 @@ export function PlayerBar() {
|
|||||||
|
|
||||||
function handleClick(e: React.MouseEvent) {
|
function handleClick(e: React.MouseEvent) {
|
||||||
if (!canSeek || isDragging) return
|
if (!canSeek || isDragging) return
|
||||||
seek((calcPct(e.clientX) / 100) * duration)
|
seek((calcPct(e.clientX) / 100) * effectiveDuration)
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentTime = isDragging ? (dragPct / 100) * duration : progress
|
const currentTime = isDragging ? (dragPct / 100) * effectiveDuration : progress
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@ -132,7 +140,9 @@ export function PlayerBar() {
|
|||||||
</div>
|
</div>
|
||||||
<div className="min-w-0">
|
<div className="min-w-0">
|
||||||
<p className="text-[13px] font-semibold truncate text-white">{current.title}</p>
|
<p className="text-[13px] font-semibold truncate text-white">{current.title}</p>
|
||||||
<p className="text-[11px] text-white/50 truncate">{current.creator?.username}</p>
|
<p className="text-[11px] text-white/50 truncate">
|
||||||
|
{currentChapter ? currentChapter.title : current.creator?.username}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -155,20 +165,41 @@ export function PlayerBar() {
|
|||||||
{canSeek && (
|
{canSeek && (
|
||||||
<button
|
<button
|
||||||
className="hidden sm:flex text-white/50 hover:text-white transition-colors cursor-pointer p-1.5 rounded-full hover:bg-white/10"
|
className="hidden sm:flex text-white/50 hover:text-white transition-colors cursor-pointer p-1.5 rounded-full hover:bg-white/10"
|
||||||
onClick={() => seek(Math.min(duration, progress + 15))}
|
onClick={() => seek(Math.min(effectiveDuration, progress + 15))}
|
||||||
>
|
>
|
||||||
<SkipForward size={18} />
|
<SkipForward size={18} />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
|
{/* Speed button — only for native audio + YouTube */}
|
||||||
|
{canSeek && !isSpotify && (
|
||||||
|
<button
|
||||||
|
onClick={cyclePlaybackRate}
|
||||||
|
className="hidden sm:flex items-center justify-center text-[11px] font-bold tabular-nums min-w-[2.5rem] h-7 rounded-full transition-colors cursor-pointer"
|
||||||
|
style={{ color: playbackRate !== 1 ? '#7B6AEF' : 'rgba(255,255,255,0.5)', background: playbackRate !== 1 ? 'rgba(123,106,239,0.15)' : 'transparent' }}
|
||||||
|
title="Vitesse de lecture"
|
||||||
|
>
|
||||||
|
{playbackRate}x
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Time + Volume (desktop) */}
|
{/* Time + Queue + Volume (desktop) */}
|
||||||
<div className="hidden sm:flex items-center gap-3 flex-1 justify-end">
|
<div className="hidden sm:flex items-center gap-3 flex-1 justify-end">
|
||||||
<span className="text-[11px] text-white/40 tabular-nums font-medium min-w-[5rem] text-right">
|
<span className="text-[11px] text-white/40 tabular-nums font-medium min-w-[5rem] text-right">
|
||||||
{duration > 0
|
{effectiveDuration > 0
|
||||||
? `${formatDuration(currentTime)} / ${formatDuration(duration)}`
|
? `${formatDuration(currentTime)} / ${formatDuration(effectiveDuration)}`
|
||||||
: ''}
|
: ''}
|
||||||
</span>
|
</span>
|
||||||
|
<button
|
||||||
|
onClick={() => setQueueOpen(!queueOpen)}
|
||||||
|
className="relative text-white/50 hover:text-white transition-colors cursor-pointer p-1 rounded-full hover:bg-white/10"
|
||||||
|
title="File d'attente"
|
||||||
|
>
|
||||||
|
<ListMusic size={15} />
|
||||||
|
{queue.length > 0 && (
|
||||||
|
<span className="absolute -top-1 -right-1 w-4 h-4 rounded-full bg-[#7B6AEF] text-[9px] font-bold flex items-center justify-center text-white">{queue.length}</span>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
<button
|
<button
|
||||||
onClick={() => setVolume(volume === 0 ? 0.8 : 0)}
|
onClick={() => setVolume(volume === 0 ? 0.8 : 0)}
|
||||||
className="text-white/50 hover:text-white transition-colors cursor-pointer p-1 rounded-full hover:bg-white/10"
|
className="text-white/50 hover:text-white transition-colors cursor-pointer p-1 rounded-full hover:bg-white/10"
|
||||||
@ -209,6 +240,15 @@ export function PlayerBar() {
|
|||||||
background: 'linear-gradient(90deg, #5B4CDB, #7B6AEF)',
|
background: 'linear-gradient(90deg, #5B4CDB, #7B6AEF)',
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
{/* Chapter markers */}
|
||||||
|
{effectiveDuration > 0 && chapters.map((ch) => (
|
||||||
|
<div
|
||||||
|
key={ch.id}
|
||||||
|
className="absolute top-1/2 -translate-y-1/2 w-[3px] h-[11px] rounded-full bg-white/40"
|
||||||
|
style={{ left: `${(ch.start_time_seconds / effectiveDuration) * 100}%` }}
|
||||||
|
title={ch.title}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Thumb — centered on track */}
|
{/* Thumb — centered on track */}
|
||||||
@ -232,6 +272,52 @@ export function PlayerBar() {
|
|||||||
</div>}
|
</div>}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{/* Queue panel */}
|
||||||
|
{queueOpen && (
|
||||||
|
<div className="fixed bottom-[106px] right-4 z-[60] w-80 max-h-[60vh] rounded-xl bg-[#1E1B33] border border-white/10 shadow-[0_4px_30px_rgba(0,0,0,0.4)] overflow-hidden flex flex-col">
|
||||||
|
<div className="flex items-center justify-between px-4 py-3 border-b border-white/10">
|
||||||
|
<span className="text-sm font-semibold text-white">File d'attente ({queue.length})</span>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
{queue.length > 0 && (
|
||||||
|
<button onClick={clearQueue} className="text-[11px] text-white/40 hover:text-white/70 transition-colors cursor-pointer">
|
||||||
|
Vider
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
<button onClick={() => setQueueOpen(false)} className="text-white/40 hover:text-white transition-colors cursor-pointer">
|
||||||
|
<X size={14} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="overflow-y-auto flex-1">
|
||||||
|
{queue.length === 0 ? (
|
||||||
|
<p className="text-sm text-white/30 text-center py-8">La file est vide</p>
|
||||||
|
) : (
|
||||||
|
queue.map((p, i) => (
|
||||||
|
<div key={p.id} className="flex items-center gap-3 px-4 py-2.5 hover:bg-white/5 transition-colors group">
|
||||||
|
<span className="text-[11px] text-white/30 w-4 text-center tabular-nums">{i + 1}</span>
|
||||||
|
{p.cover_url ? (
|
||||||
|
<img src={publicUrl(p.cover_url)} alt="" className="w-9 h-9 rounded-lg object-cover shrink-0" />
|
||||||
|
) : (
|
||||||
|
<div className="w-9 h-9 rounded-lg bg-white/10 flex items-center justify-center shrink-0">
|
||||||
|
<span className="text-[11px] font-bold text-white/30">{p.title[0]?.toUpperCase()}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<p className="text-[12px] font-medium text-white truncate">{p.title}</p>
|
||||||
|
<p className="text-[10px] text-white/40 truncate">{p.external_author || p.creator?.username}</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={() => removeFromQueue(p.id)}
|
||||||
|
className="opacity-0 group-hover:opacity-100 text-white/30 hover:text-white/70 transition-all cursor-pointer p-1"
|
||||||
|
>
|
||||||
|
<X size={12} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,4 +1,4 @@
|
|||||||
import { Play, Pause, Heart, MessageCircle, Clock } from 'lucide-react'
|
import { Play, Pause, Heart, MessageCircle, Clock, ListPlus, Download, Check, Loader2 } from 'lucide-react'
|
||||||
import { Link } from 'react-router-dom'
|
import { Link } from 'react-router-dom'
|
||||||
import type { Podcast } from '@/types'
|
import type { Podcast } from '@/types'
|
||||||
import { formatDuration, timeAgo } from '@/lib/utils'
|
import { formatDuration, timeAgo } from '@/lib/utils'
|
||||||
@ -7,6 +7,8 @@ import { Badge } from '@/components/ui/Badge'
|
|||||||
import { Avatar } from '@/components/ui/Avatar'
|
import { Avatar } from '@/components/ui/Avatar'
|
||||||
import { usePlayerStore } from '@/stores/player'
|
import { usePlayerStore } from '@/stores/player'
|
||||||
import { useListeningProgress } from '@/hooks/useListeningProgress'
|
import { useListeningProgress } from '@/hooks/useListeningProgress'
|
||||||
|
import { useOfflineStore } from '@/stores/offline'
|
||||||
|
import { isExternalUrl } from '@/lib/embed'
|
||||||
|
|
||||||
const coverGradients = [
|
const coverGradients = [
|
||||||
'from-primary/20 via-primary/5 to-accent/10',
|
'from-primary/20 via-primary/5 to-accent/10',
|
||||||
@ -28,11 +30,16 @@ interface PodcastCardProps {
|
|||||||
|
|
||||||
export function PodcastCard({ podcast, progressPercent: propProgress }: PodcastCardProps) {
|
export function PodcastCard({ podcast, progressPercent: propProgress }: PodcastCardProps) {
|
||||||
const play = usePlayerStore((s) => s.play)
|
const play = usePlayerStore((s) => s.play)
|
||||||
|
const addToQueue = usePlayerStore((s) => s.addToQueue)
|
||||||
const currentId = usePlayerStore((s) => s.current?.id)
|
const currentId = usePlayerStore((s) => s.current?.id)
|
||||||
const isPlaying = usePlayerStore((s) => s.isPlaying)
|
const isPlaying = usePlayerStore((s) => s.isPlaying)
|
||||||
const isActive = currentId === podcast.id
|
const isActive = currentId === podcast.id
|
||||||
const progressMap = useListeningProgress()
|
const progressMap = useListeningProgress()
|
||||||
const progressPercent = propProgress ?? progressMap.get(podcast.id)?.percent
|
const progressPercent = propProgress ?? progressMap.get(podcast.id)?.percent
|
||||||
|
const { downloadPodcast, isDownloaded, isDownloading, getDownloadProgress } = useOfflineStore()
|
||||||
|
const downloaded = isDownloaded(podcast.id)
|
||||||
|
const downloading = isDownloading(podcast.id)
|
||||||
|
const isNative = !isExternalUrl(podcast.audio_url)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<article className="group lift bg-surface rounded-3xl shadow-organic-sm hover:shadow-organic overflow-hidden">
|
<article className="group lift bg-surface rounded-3xl shadow-organic-sm hover:shadow-organic overflow-hidden">
|
||||||
@ -56,17 +63,41 @@ export function PodcastCard({ podcast, progressPercent: propProgress }: PodcastC
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Play overlay */}
|
{/* Play overlay */}
|
||||||
<button
|
<div className="absolute inset-0 flex items-end justify-between p-3 bg-gradient-to-t from-black/40 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-all duration-300">
|
||||||
onClick={() => play(podcast)}
|
<button
|
||||||
className="absolute inset-0 flex items-end justify-between p-3 bg-gradient-to-t from-black/40 via-transparent to-transparent opacity-0 group-hover:opacity-100 transition-all duration-300 cursor-pointer"
|
onClick={() => play(podcast)}
|
||||||
>
|
className={`w-10 h-10 rounded-full bg-white/95 shadow-lg flex items-center justify-center transition-all duration-300 cursor-pointer ${isActive && isPlaying ? 'scale-100' : 'scale-90 group-hover:scale-100'}`}
|
||||||
<div className={`w-10 h-10 rounded-full bg-white/95 shadow-lg flex items-center justify-center transition-all duration-300 ${isActive && isPlaying ? 'scale-100' : 'scale-90 group-hover:scale-100'}`}>
|
>
|
||||||
{isActive && isPlaying
|
{isActive && isPlaying
|
||||||
? <Pause size={16} className="text-primary" fill="currentColor" />
|
? <Pause size={16} className="text-primary" fill="currentColor" />
|
||||||
: <Play size={16} className="text-primary ml-0.5" fill="currentColor" />
|
: <Play size={16} className="text-primary ml-0.5" fill="currentColor" />
|
||||||
}
|
}
|
||||||
|
</button>
|
||||||
|
<div className="flex flex-col gap-1.5">
|
||||||
|
<button
|
||||||
|
onClick={(e) => { e.stopPropagation(); addToQueue(podcast) }}
|
||||||
|
className="w-8 h-8 rounded-full bg-white/90 shadow-lg flex items-center justify-center transition-all duration-300 scale-90 group-hover:scale-100 cursor-pointer hover:bg-white"
|
||||||
|
title="Écouter ensuite"
|
||||||
|
>
|
||||||
|
<ListPlus size={14} className="text-primary" />
|
||||||
|
</button>
|
||||||
|
{isNative && !downloaded && (
|
||||||
|
<button
|
||||||
|
onClick={(e) => { e.stopPropagation(); downloadPodcast(podcast) }}
|
||||||
|
disabled={downloading}
|
||||||
|
className="w-8 h-8 rounded-full bg-white/90 shadow-lg flex items-center justify-center transition-all duration-300 scale-90 group-hover:scale-100 cursor-pointer hover:bg-white"
|
||||||
|
title={downloading ? `${getDownloadProgress(podcast.id)}%` : 'Télécharger hors-ligne'}
|
||||||
|
>
|
||||||
|
{downloading ? <Loader2 size={14} className="text-primary animate-spin" /> : <Download size={14} className="text-primary" />}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
{isNative && downloaded && (
|
||||||
|
<div className="w-8 h-8 rounded-full bg-mint/90 shadow-lg flex items-center justify-center scale-90 group-hover:scale-100" title="Disponible hors-ligne">
|
||||||
|
<Check size={14} className="text-white" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</div>
|
||||||
|
|
||||||
{/* Duration pill */}
|
{/* Duration pill */}
|
||||||
<div className="absolute top-2.5 right-2.5 glass rounded-full px-2.5 py-1 flex items-center gap-1 text-[10px] font-bold text-text shadow-sm">
|
<div className="absolute top-2.5 right-2.5 glass rounded-full px-2.5 py-1 flex items-center gap-1 text-[10px] font-bold text-text shadow-sm">
|
||||||
|
|||||||
18
src/hooks/useOnlineStatus.ts
Normal file
18
src/hooks/useOnlineStatus.ts
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import { useSyncExternalStore } from 'react'
|
||||||
|
|
||||||
|
function subscribe(callback: () => void) {
|
||||||
|
window.addEventListener('online', callback)
|
||||||
|
window.addEventListener('offline', callback)
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('online', callback)
|
||||||
|
window.removeEventListener('offline', callback)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSnapshot() {
|
||||||
|
return navigator.onLine
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useOnlineStatus() {
|
||||||
|
return useSyncExternalStore(subscribe, getSnapshot)
|
||||||
|
}
|
||||||
93
src/lib/offline.ts
Normal file
93
src/lib/offline.ts
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
const CACHE_NAME = 'wetalk-audio-cache'
|
||||||
|
|
||||||
|
export async function cacheAudio(
|
||||||
|
url: string,
|
||||||
|
onProgress?: (percent: number) => void,
|
||||||
|
): Promise<boolean> {
|
||||||
|
try {
|
||||||
|
const cache = await caches.open(CACHE_NAME)
|
||||||
|
|
||||||
|
// Check if already cached
|
||||||
|
const existing = await cache.match(url)
|
||||||
|
if (existing) return true
|
||||||
|
|
||||||
|
const response = await fetch(url)
|
||||||
|
if (!response.ok) return false
|
||||||
|
|
||||||
|
if (onProgress && response.body) {
|
||||||
|
const contentLength = Number(response.headers.get('content-length') || 0)
|
||||||
|
const reader = response.body.getReader()
|
||||||
|
const chunks: Uint8Array[] = []
|
||||||
|
let received = 0
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read()
|
||||||
|
if (done) break
|
||||||
|
chunks.push(value)
|
||||||
|
received += value.length
|
||||||
|
if (contentLength > 0) onProgress(Math.round((received / contentLength) * 100))
|
||||||
|
}
|
||||||
|
|
||||||
|
const blob = new Blob(chunks, { type: response.headers.get('content-type') || 'audio/mpeg' })
|
||||||
|
const cachedResponse = new Response(blob, {
|
||||||
|
status: 200,
|
||||||
|
headers: { 'Content-Type': blob.type, 'Content-Length': String(blob.size) },
|
||||||
|
})
|
||||||
|
await cache.put(url, cachedResponse)
|
||||||
|
} else {
|
||||||
|
await cache.put(url, response)
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
} catch (e) {
|
||||||
|
console.error('[offline] cache failed:', e)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getCachedAudioUrl(url: string): Promise<string | null> {
|
||||||
|
try {
|
||||||
|
const cache = await caches.open(CACHE_NAME)
|
||||||
|
const response = await cache.match(url)
|
||||||
|
if (!response) return null
|
||||||
|
const blob = await response.blob()
|
||||||
|
return URL.createObjectURL(blob)
|
||||||
|
} catch {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function isAudioCached(url: string): Promise<boolean> {
|
||||||
|
try {
|
||||||
|
const cache = await caches.open(CACHE_NAME)
|
||||||
|
const response = await cache.match(url)
|
||||||
|
return !!response
|
||||||
|
} catch {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function removeCachedAudio(url: string): Promise<void> {
|
||||||
|
try {
|
||||||
|
const cache = await caches.open(CACHE_NAME)
|
||||||
|
await cache.delete(url)
|
||||||
|
} catch { /* ignore */ }
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getCacheSize(): Promise<number> {
|
||||||
|
try {
|
||||||
|
const cache = await caches.open(CACHE_NAME)
|
||||||
|
const keys = await cache.keys()
|
||||||
|
let total = 0
|
||||||
|
for (const req of keys) {
|
||||||
|
const res = await cache.match(req)
|
||||||
|
if (res) {
|
||||||
|
const blob = await res.blob()
|
||||||
|
total += blob.size
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return total
|
||||||
|
} catch {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,4 +1,5 @@
|
|||||||
export function formatDuration(seconds: number): string {
|
export function formatDuration(seconds: number): string {
|
||||||
|
if (!seconds || !isFinite(seconds) || isNaN(seconds)) return '0:00'
|
||||||
const h = Math.floor(seconds / 3600)
|
const h = Math.floor(seconds / 3600)
|
||||||
const m = Math.floor((seconds % 3600) / 60)
|
const m = Math.floor((seconds % 3600) / 60)
|
||||||
const s = Math.floor(seconds % 60)
|
const s = Math.floor(seconds % 60)
|
||||||
|
|||||||
142
src/pages/Offline.tsx
Normal file
142
src/pages/Offline.tsx
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
import { useMemo } from 'react'
|
||||||
|
import { WifiOff, Play, Pause, Trash2 } from 'lucide-react'
|
||||||
|
import { usePlayerStore } from '@/stores/player'
|
||||||
|
import { useOfflineStore } from '@/stores/offline'
|
||||||
|
import { useOnlineStatus } from '@/hooks/useOnlineStatus'
|
||||||
|
import { publicUrl } from '@/lib/storage'
|
||||||
|
import { formatDuration } from '@/lib/utils'
|
||||||
|
import { Avatar } from '@/components/ui/Avatar'
|
||||||
|
import { Button } from '@/components/ui/Button'
|
||||||
|
import type { Podcast } from '@/types'
|
||||||
|
|
||||||
|
function getOfflinePodcasts(): Podcast[] {
|
||||||
|
try {
|
||||||
|
const meta = JSON.parse(localStorage.getItem('wetalk-offline-meta') || '{}')
|
||||||
|
return Object.values(meta)
|
||||||
|
} catch {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Offline() {
|
||||||
|
const isOnline = useOnlineStatus()
|
||||||
|
const { play, toggle, current, isPlaying } = usePlayerStore()
|
||||||
|
const { downloaded, removePodcast } = useOfflineStore()
|
||||||
|
|
||||||
|
const podcasts = useMemo(() => {
|
||||||
|
const all = getOfflinePodcasts()
|
||||||
|
return all.filter((p) => downloaded.has(p.id))
|
||||||
|
}, [downloaded])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="max-w-3xl mx-auto">
|
||||||
|
{/* Header */}
|
||||||
|
<div className="flex items-center gap-3 mb-2">
|
||||||
|
<div className="w-10 h-10 rounded-xl bg-gradient-to-br from-accent/20 to-accent/5 flex items-center justify-center">
|
||||||
|
<WifiOff size={20} className="text-accent" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h1 className="text-2xl font-heading font-bold">Mode hors-ligne</h1>
|
||||||
|
<p className="text-sm text-text-secondary">
|
||||||
|
{isOnline
|
||||||
|
? 'Vos épisodes téléchargés, prêts à écouter sans connexion.'
|
||||||
|
: 'Pas de connexion — voici vos épisodes disponibles.'}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{!isOnline && (
|
||||||
|
<div className="mt-4 mb-6 px-4 py-3 rounded-xl bg-accent/5 border border-accent/20 text-sm text-accent font-medium flex items-center gap-2">
|
||||||
|
<WifiOff size={16} />
|
||||||
|
Vous êtes hors-ligne. Seuls les épisodes téléchargés sont disponibles.
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{podcasts.length === 0 ? (
|
||||||
|
<div className="text-center py-16 space-y-3">
|
||||||
|
<div className="w-16 h-16 rounded-2xl bg-border-light flex items-center justify-center mx-auto">
|
||||||
|
<WifiOff size={28} className="text-text-secondary" />
|
||||||
|
</div>
|
||||||
|
<p className="text-text-secondary font-medium">Aucun épisode téléchargé</p>
|
||||||
|
<p className="text-sm text-text-secondary/60">
|
||||||
|
{isOnline
|
||||||
|
? 'Téléchargez des épisodes depuis la page d\'un podcast pour les écouter hors-ligne.'
|
||||||
|
: 'Reconnectez-vous pour télécharger des épisodes.'}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="mt-6 space-y-2">
|
||||||
|
{podcasts.map((podcast) => {
|
||||||
|
const isActive = current?.id === podcast.id
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={podcast.id}
|
||||||
|
className="flex items-center gap-4 p-3 rounded-2xl bg-surface hover:bg-surface-warm/70 transition-colors group"
|
||||||
|
>
|
||||||
|
{/* Cover */}
|
||||||
|
<div className="relative shrink-0">
|
||||||
|
{podcast.cover_url ? (
|
||||||
|
<img
|
||||||
|
src={publicUrl(podcast.cover_url)}
|
||||||
|
alt=""
|
||||||
|
className="w-14 h-14 rounded-xl object-cover"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Avatar name={podcast.title} size="md" className="!rounded-xl !w-14 !h-14" />
|
||||||
|
)}
|
||||||
|
{isActive && isPlaying && (
|
||||||
|
<div className="absolute inset-0 flex items-center justify-center bg-black/30 rounded-xl">
|
||||||
|
<div className="flex items-end gap-[2px] h-4">
|
||||||
|
<div className="wave-bar" />
|
||||||
|
<div className="wave-bar" />
|
||||||
|
<div className="wave-bar" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Info */}
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<p className="font-heading font-bold text-sm truncate">{podcast.title}</p>
|
||||||
|
<p className="text-xs text-text-secondary truncate">
|
||||||
|
{podcast.external_author || (podcast.creator as any)?.username || 'Inconnu'}
|
||||||
|
</p>
|
||||||
|
{podcast.duration_seconds > 0 && (
|
||||||
|
<p className="text-[11px] text-text-secondary/60 mt-0.5">
|
||||||
|
{formatDuration(podcast.duration_seconds)}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Actions */}
|
||||||
|
<div className="flex items-center gap-2 shrink-0">
|
||||||
|
<button
|
||||||
|
onClick={() => (isActive ? toggle() : play(podcast))}
|
||||||
|
className="w-10 h-10 rounded-full bg-primary text-white flex items-center justify-center hover:scale-105 transition-transform cursor-pointer shadow-md"
|
||||||
|
>
|
||||||
|
{isActive && isPlaying ? (
|
||||||
|
<Pause size={16} />
|
||||||
|
) : (
|
||||||
|
<Play size={16} className="ml-0.5" />
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => removePodcast(podcast)}
|
||||||
|
className="w-8 h-8 rounded-full flex items-center justify-center text-text-secondary hover:text-accent hover:bg-accent/10 transition-colors cursor-pointer opacity-0 group-hover:opacity-100"
|
||||||
|
title="Supprimer le téléchargement"
|
||||||
|
>
|
||||||
|
<Trash2 size={14} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
|
||||||
|
<p className="text-center text-[11px] text-text-secondary/50 pt-4">
|
||||||
|
{podcasts.length} épisode{podcasts.length > 1 ? 's' : ''} téléchargé{podcasts.length > 1 ? 's' : ''}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@ -1,22 +1,25 @@
|
|||||||
import { useEffect, useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
import { useParams, Link } from 'react-router-dom'
|
import { useParams, Link } from 'react-router-dom'
|
||||||
import { Play, Pause, Heart, MessageCircle, Clock, Share2 } from 'lucide-react'
|
import { Play, Pause, Heart, MessageCircle, Clock, Share2, ListPlus, Download, Check, Loader2 } from 'lucide-react'
|
||||||
import { supabase } from '@/lib/supabase'
|
import { supabase } from '@/lib/supabase'
|
||||||
import { useAuthStore } from '@/stores/auth'
|
import { useAuthStore } from '@/stores/auth'
|
||||||
import { usePlayerStore } from '@/stores/player'
|
import { usePlayerStore } from '@/stores/player'
|
||||||
import type { Podcast, Comment } from '@/types'
|
import type { Podcast, Comment, Chapter } from '@/types'
|
||||||
import { formatDuration, timeAgo } from '@/lib/utils'
|
import { formatDuration, timeAgo } from '@/lib/utils'
|
||||||
import { Avatar } from '@/components/ui/Avatar'
|
import { Avatar } from '@/components/ui/Avatar'
|
||||||
import { Badge } from '@/components/ui/Badge'
|
import { Badge } from '@/components/ui/Badge'
|
||||||
import { publicUrl } from '@/lib/storage'
|
import { publicUrl } from '@/lib/storage'
|
||||||
import { Button } from '@/components/ui/Button'
|
import { Button } from '@/components/ui/Button'
|
||||||
import { useListeningProgress } from '@/hooks/useListeningProgress'
|
import { useListeningProgress } from '@/hooks/useListeningProgress'
|
||||||
|
import { useOfflineStore } from '@/stores/offline'
|
||||||
|
import { isExternalUrl } from '@/lib/embed'
|
||||||
|
|
||||||
export function PodcastDetail() {
|
export function PodcastDetail() {
|
||||||
const { id } = useParams<{ id: string }>()
|
const { id } = useParams<{ id: string }>()
|
||||||
const { user } = useAuthStore()
|
const { user } = useAuthStore()
|
||||||
const { play, toggle, current, isPlaying } = usePlayerStore()
|
const { play, toggle, current, isPlaying, addToQueue, seek: playerSeek } = usePlayerStore()
|
||||||
const [podcast, setPodcast] = useState<Podcast | null>(null)
|
const [podcast, setPodcast] = useState<Podcast | null>(null)
|
||||||
|
const [chapters, setChapters] = useState<Chapter[]>([])
|
||||||
const [comments, setComments] = useState<Comment[]>([])
|
const [comments, setComments] = useState<Comment[]>([])
|
||||||
const [newComment, setNewComment] = useState('')
|
const [newComment, setNewComment] = useState('')
|
||||||
const [isLiked, setIsLiked] = useState(false)
|
const [isLiked, setIsLiked] = useState(false)
|
||||||
@ -26,6 +29,7 @@ export function PodcastDetail() {
|
|||||||
const isActive = current?.id === id
|
const isActive = current?.id === id
|
||||||
const progressMap = useListeningProgress()
|
const progressMap = useListeningProgress()
|
||||||
const progressInfo = id ? progressMap.get(id) : undefined
|
const progressInfo = id ? progressMap.get(id) : undefined
|
||||||
|
const { downloadPodcast, removePodcast, isDownloaded, isDownloading, getDownloadProgress } = useOfflineStore()
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!id) return
|
if (!id) return
|
||||||
@ -50,6 +54,13 @@ export function PodcastDetail() {
|
|||||||
|
|
||||||
if (commentsData) setComments(commentsData)
|
if (commentsData) setComments(commentsData)
|
||||||
|
|
||||||
|
const { data: chaptersData } = await supabase
|
||||||
|
.from('chapters')
|
||||||
|
.select('*')
|
||||||
|
.eq('podcast_id', id)
|
||||||
|
.order('sort_order', { ascending: true })
|
||||||
|
if (chaptersData) setChapters(chaptersData)
|
||||||
|
|
||||||
if (user) {
|
if (user) {
|
||||||
const { data: like } = await supabase
|
const { data: like } = await supabase
|
||||||
.from('likes')
|
.from('likes')
|
||||||
@ -176,6 +187,25 @@ export function PodcastDetail() {
|
|||||||
<Heart size={16} fill={isLiked ? 'currentColor' : 'none'} />
|
<Heart size={16} fill={isLiked ? 'currentColor' : 'none'} />
|
||||||
{likesCount}
|
{likesCount}
|
||||||
</Button>
|
</Button>
|
||||||
|
<Button variant="ghost" onClick={() => podcast && addToQueue(podcast)} title="Écouter ensuite">
|
||||||
|
<ListPlus size={16} />
|
||||||
|
</Button>
|
||||||
|
{podcast && !isExternalUrl(podcast.audio_url) && (
|
||||||
|
isDownloaded(podcast.id) ? (
|
||||||
|
<Button variant="ghost" onClick={() => removePodcast(podcast)} title="Supprimer le téléchargement">
|
||||||
|
<Check size={16} className="text-mint" />
|
||||||
|
</Button>
|
||||||
|
) : isDownloading(podcast.id) ? (
|
||||||
|
<Button variant="ghost" disabled>
|
||||||
|
<Loader2 size={16} className="animate-spin" />
|
||||||
|
<span className="text-xs">{getDownloadProgress(podcast.id)}%</span>
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<Button variant="ghost" onClick={() => downloadPodcast(podcast)} title="Télécharger hors-ligne">
|
||||||
|
<Download size={16} />
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
)}
|
||||||
<Button variant="ghost">
|
<Button variant="ghost">
|
||||||
<Share2 size={16} />
|
<Share2 size={16} />
|
||||||
</Button>
|
</Button>
|
||||||
@ -190,6 +220,32 @@ export function PodcastDetail() {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{chapters.length > 0 && (
|
||||||
|
<div>
|
||||||
|
<h2 className="text-lg font-heading font-bold mb-3 flex items-center gap-2">
|
||||||
|
<Clock size={18} />
|
||||||
|
Chapitres ({chapters.length})
|
||||||
|
</h2>
|
||||||
|
<div className="space-y-1">
|
||||||
|
{chapters.map((ch) => (
|
||||||
|
<button
|
||||||
|
key={ch.id}
|
||||||
|
onClick={() => {
|
||||||
|
if (!isActive) play(podcast)
|
||||||
|
setTimeout(() => playerSeek(ch.start_time_seconds), 300)
|
||||||
|
}}
|
||||||
|
className="w-full flex items-center gap-3 px-4 py-2.5 rounded-xl hover:bg-primary/5 transition-colors cursor-pointer text-left group"
|
||||||
|
>
|
||||||
|
<span className="text-xs font-mono text-primary font-semibold min-w-[3rem]">
|
||||||
|
{formatDuration(ch.start_time_seconds)}
|
||||||
|
</span>
|
||||||
|
<span className="text-sm font-medium group-hover:text-primary transition-colors">{ch.title}</span>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<h2 className="text-lg font-heading font-bold mb-4 flex items-center gap-2">
|
<h2 className="text-lg font-heading font-bold mb-4 flex items-center gap-2">
|
||||||
<MessageCircle size={18} />
|
<MessageCircle size={18} />
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { useState, useRef, useCallback, useEffect } from 'react'
|
import { useState, useRef, useCallback, useEffect } from 'react'
|
||||||
import { useNavigate } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
import { Upload as UploadIcon, Music, X, Image, Mic, Link2, Loader2, Circle, Square, MicOff, ShieldAlert } from 'lucide-react'
|
import { Upload as UploadIcon, Music, X, Image, Mic, Link2, Loader2, Circle, Square, MicOff, ShieldAlert, Plus, Trash2 } from 'lucide-react'
|
||||||
import { supabase } from '@/lib/supabase'
|
import { supabase } from '@/lib/supabase'
|
||||||
import { useAuthStore } from '@/stores/auth'
|
import { useAuthStore } from '@/stores/auth'
|
||||||
import { Button } from '@/components/ui/Button'
|
import { Button } from '@/components/ui/Button'
|
||||||
@ -57,6 +57,10 @@ export function Upload() {
|
|||||||
const mediaRecorderRef = useRef<MediaRecorder | null>(null)
|
const mediaRecorderRef = useRef<MediaRecorder | null>(null)
|
||||||
const chunksRef = useRef<Blob[]>([])
|
const chunksRef = useRef<Blob[]>([])
|
||||||
const timerRef = useRef<ReturnType<typeof setInterval> | null>(null)
|
const timerRef = useRef<ReturnType<typeof setInterval> | null>(null)
|
||||||
|
const recordingTimeRef = useRef(0)
|
||||||
|
|
||||||
|
// Chapters
|
||||||
|
const [chapters, setChapters] = useState<{ title: string; minutes: string; seconds: string }[]>([])
|
||||||
|
|
||||||
// External mode
|
// External mode
|
||||||
const [externalUrl, setExternalUrl] = useState('')
|
const [externalUrl, setExternalUrl] = useState('')
|
||||||
@ -114,7 +118,7 @@ export function Upload() {
|
|||||||
const blob = new Blob(chunksRef.current, { type: 'audio/webm' })
|
const blob = new Blob(chunksRef.current, { type: 'audio/webm' })
|
||||||
const file = new File([blob], `enregistrement-${Date.now()}.webm`, { type: 'audio/webm' })
|
const file = new File([blob], `enregistrement-${Date.now()}.webm`, { type: 'audio/webm' })
|
||||||
setAudioFile(file)
|
setAudioFile(file)
|
||||||
setDuration(recordingTime)
|
setDuration(recordingTimeRef.current)
|
||||||
if (!title) setTitle('Mon enregistrement')
|
if (!title) setTitle('Mon enregistrement')
|
||||||
setIsRecording(false)
|
setIsRecording(false)
|
||||||
}
|
}
|
||||||
@ -122,7 +126,11 @@ export function Upload() {
|
|||||||
mediaRecorder.start(1000)
|
mediaRecorder.start(1000)
|
||||||
setIsRecording(true)
|
setIsRecording(true)
|
||||||
setRecordingTime(0)
|
setRecordingTime(0)
|
||||||
timerRef.current = setInterval(() => setRecordingTime(t => t + 1), 1000)
|
recordingTimeRef.current = 0
|
||||||
|
timerRef.current = setInterval(() => {
|
||||||
|
recordingTimeRef.current += 1
|
||||||
|
setRecordingTime(recordingTimeRef.current)
|
||||||
|
}, 1000)
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
const e = err as DOMException
|
const e = err as DOMException
|
||||||
if (e.name === 'NotAllowedError') {
|
if (e.name === 'NotAllowedError') {
|
||||||
@ -255,10 +263,13 @@ export function Upload() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// External content → wetalk system account, original → logged-in user
|
||||||
|
const WETALK_SYSTEM_ID = 'a1000000-0000-0000-0000-000000000005'
|
||||||
|
|
||||||
const { data: podcast, error: insertErr } = await supabase
|
const { data: podcast, error: insertErr } = await supabase
|
||||||
.from('podcasts')
|
.from('podcasts')
|
||||||
.insert({
|
.insert({
|
||||||
creator_id: user.id,
|
creator_id: mode === 'external' ? WETALK_SYSTEM_ID : user.id,
|
||||||
title,
|
title,
|
||||||
description,
|
description,
|
||||||
audio_url,
|
audio_url,
|
||||||
@ -275,6 +286,21 @@ export function Upload() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Save chapters if any
|
||||||
|
if (chapters.length > 0 && podcast) {
|
||||||
|
const validChapters = chapters
|
||||||
|
.filter((ch) => ch.title.trim())
|
||||||
|
.map((ch, i) => ({
|
||||||
|
podcast_id: podcast.id,
|
||||||
|
title: ch.title.trim(),
|
||||||
|
start_time_seconds: parseInt(ch.minutes || '0') * 60 + parseInt(ch.seconds || '0'),
|
||||||
|
sort_order: i,
|
||||||
|
}))
|
||||||
|
if (validChapters.length > 0) {
|
||||||
|
await supabase.from('chapters').insert(validChapters)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
navigate(`/podcast/${podcast.id}`)
|
navigate(`/podcast/${podcast.id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -571,6 +597,72 @@ export function Upload() {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{/* Chapter editor — original mode only */}
|
||||||
|
{mode === 'original' && audioFile && (
|
||||||
|
<div className="space-y-3">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<label className="text-[13px] font-semibold text-text tracking-wide">Chapitres (optionnel)</label>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => setChapters([...chapters, { title: '', minutes: '0', seconds: '00' }])}
|
||||||
|
className="text-[12px] text-primary hover:text-primary-hover font-medium flex items-center gap-1 transition-colors cursor-pointer"
|
||||||
|
>
|
||||||
|
<Plus size={14} /> Ajouter
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{chapters.map((ch, i) => (
|
||||||
|
<div key={i} className="flex items-center gap-2">
|
||||||
|
<div className="flex items-center gap-1 shrink-0">
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
min={0}
|
||||||
|
value={ch.minutes}
|
||||||
|
onChange={(e) => {
|
||||||
|
const next = [...chapters]
|
||||||
|
next[i] = { ...ch, minutes: e.target.value }
|
||||||
|
setChapters(next)
|
||||||
|
}}
|
||||||
|
className="w-12 rounded-lg border border-border bg-surface px-2 py-1.5 text-sm text-center outline-none focus:border-primary"
|
||||||
|
placeholder="0"
|
||||||
|
/>
|
||||||
|
<span className="text-text-secondary text-sm">:</span>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
min={0}
|
||||||
|
max={59}
|
||||||
|
value={ch.seconds}
|
||||||
|
onChange={(e) => {
|
||||||
|
const next = [...chapters]
|
||||||
|
next[i] = { ...ch, seconds: e.target.value.padStart(2, '0').slice(-2) }
|
||||||
|
setChapters(next)
|
||||||
|
}}
|
||||||
|
className="w-12 rounded-lg border border-border bg-surface px-2 py-1.5 text-sm text-center outline-none focus:border-primary"
|
||||||
|
placeholder="00"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={ch.title}
|
||||||
|
onChange={(e) => {
|
||||||
|
const next = [...chapters]
|
||||||
|
next[i] = { ...ch, title: e.target.value }
|
||||||
|
setChapters(next)
|
||||||
|
}}
|
||||||
|
className="flex-1 rounded-lg border border-border bg-surface px-3 py-1.5 text-sm outline-none focus:border-primary"
|
||||||
|
placeholder="Titre du chapitre"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => setChapters(chapters.filter((_, j) => j !== i))}
|
||||||
|
className="text-text-secondary hover:text-accent transition-colors cursor-pointer p-1"
|
||||||
|
>
|
||||||
|
<Trash2 size={14} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{error && <p className="text-sm text-accent">{error}</p>}
|
{error && <p className="text-sm text-accent">{error}</p>}
|
||||||
|
|
||||||
{(mode === 'original' || platform) && (
|
{(mode === 'original' || platform) && (
|
||||||
|
|||||||
112
src/stores/offline.ts
Normal file
112
src/stores/offline.ts
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
import { create } from 'zustand'
|
||||||
|
import { cacheAudio, removeCachedAudio, isAudioCached } from '@/lib/offline'
|
||||||
|
import { publicUrl } from '@/lib/storage'
|
||||||
|
import { isExternalUrl } from '@/lib/embed'
|
||||||
|
import type { Podcast } from '@/types'
|
||||||
|
|
||||||
|
interface DownloadInfo {
|
||||||
|
podcastId: string
|
||||||
|
progress: number // 0-100
|
||||||
|
}
|
||||||
|
|
||||||
|
interface OfflineState {
|
||||||
|
downloaded: Set<string>
|
||||||
|
downloading: Map<string, number>
|
||||||
|
downloadPodcast: (podcast: Podcast) => Promise<void>
|
||||||
|
removePodcast: (podcast: Podcast) => Promise<void>
|
||||||
|
isDownloaded: (podcastId: string) => boolean
|
||||||
|
isDownloading: (podcastId: string) => boolean
|
||||||
|
getDownloadProgress: (podcastId: string) => number
|
||||||
|
initFromStorage: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
function loadDownloaded(): Set<string> {
|
||||||
|
try {
|
||||||
|
const saved = localStorage.getItem('wetalk-offline-podcasts')
|
||||||
|
return saved ? new Set(JSON.parse(saved)) : new Set()
|
||||||
|
} catch { return new Set() }
|
||||||
|
}
|
||||||
|
|
||||||
|
function saveDownloaded(ids: Set<string>) {
|
||||||
|
localStorage.setItem('wetalk-offline-podcasts', JSON.stringify([...ids]))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store podcast metadata for offline access
|
||||||
|
function savePodcastMeta(podcast: Podcast) {
|
||||||
|
try {
|
||||||
|
const stored = JSON.parse(localStorage.getItem('wetalk-offline-meta') || '{}')
|
||||||
|
stored[podcast.id] = {
|
||||||
|
id: podcast.id,
|
||||||
|
title: podcast.title,
|
||||||
|
audio_url: podcast.audio_url,
|
||||||
|
cover_url: podcast.cover_url,
|
||||||
|
duration_seconds: podcast.duration_seconds,
|
||||||
|
creator: podcast.creator,
|
||||||
|
external_author: podcast.external_author,
|
||||||
|
}
|
||||||
|
localStorage.setItem('wetalk-offline-meta', JSON.stringify(stored))
|
||||||
|
} catch { /* ignore */ }
|
||||||
|
}
|
||||||
|
|
||||||
|
function removePodcastMeta(podcastId: string) {
|
||||||
|
try {
|
||||||
|
const stored = JSON.parse(localStorage.getItem('wetalk-offline-meta') || '{}')
|
||||||
|
delete stored[podcastId]
|
||||||
|
localStorage.setItem('wetalk-offline-meta', JSON.stringify(stored))
|
||||||
|
} catch { /* ignore */ }
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useOfflineStore = create<OfflineState>((set, get) => ({
|
||||||
|
downloaded: loadDownloaded(),
|
||||||
|
downloading: new Map(),
|
||||||
|
|
||||||
|
downloadPodcast: async (podcast) => {
|
||||||
|
if (isExternalUrl(podcast.audio_url)) return
|
||||||
|
if (get().downloaded.has(podcast.id)) return
|
||||||
|
if (get().downloading.has(podcast.id)) return
|
||||||
|
|
||||||
|
const url = publicUrl(podcast.audio_url)
|
||||||
|
|
||||||
|
set((s) => ({ downloading: new Map(s.downloading).set(podcast.id, 0) }))
|
||||||
|
|
||||||
|
const success = await cacheAudio(url, (pct) => {
|
||||||
|
set((s) => ({ downloading: new Map(s.downloading).set(podcast.id, pct) }))
|
||||||
|
})
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
savePodcastMeta(podcast)
|
||||||
|
const next = new Set(get().downloaded)
|
||||||
|
next.add(podcast.id)
|
||||||
|
saveDownloaded(next)
|
||||||
|
set((s) => {
|
||||||
|
const dl = new Map(s.downloading)
|
||||||
|
dl.delete(podcast.id)
|
||||||
|
return { downloaded: next, downloading: dl }
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
set((s) => {
|
||||||
|
const dl = new Map(s.downloading)
|
||||||
|
dl.delete(podcast.id)
|
||||||
|
return { downloading: dl }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
removePodcast: async (podcast) => {
|
||||||
|
const url = publicUrl(podcast.audio_url)
|
||||||
|
await removeCachedAudio(url)
|
||||||
|
removePodcastMeta(podcast.id)
|
||||||
|
const next = new Set(get().downloaded)
|
||||||
|
next.delete(podcast.id)
|
||||||
|
saveDownloaded(next)
|
||||||
|
set({ downloaded: next })
|
||||||
|
},
|
||||||
|
|
||||||
|
isDownloaded: (podcastId) => get().downloaded.has(podcastId),
|
||||||
|
isDownloading: (podcastId) => get().downloading.has(podcastId),
|
||||||
|
getDownloadProgress: (podcastId) => get().downloading.get(podcastId) ?? 0,
|
||||||
|
|
||||||
|
initFromStorage: () => {
|
||||||
|
set({ downloaded: loadDownloaded() })
|
||||||
|
},
|
||||||
|
}))
|
||||||
@ -5,6 +5,7 @@ import { publicUrl } from '@/lib/storage'
|
|||||||
import { supabase } from '@/lib/supabase'
|
import { supabase } from '@/lib/supabase'
|
||||||
import { useAuthStore } from '@/stores/auth'
|
import { useAuthStore } from '@/stores/auth'
|
||||||
import { invalidateProgressCache } from '@/hooks/useListeningProgress'
|
import { invalidateProgressCache } from '@/hooks/useListeningProgress'
|
||||||
|
import { getCachedAudioUrl } from '@/lib/offline'
|
||||||
|
|
||||||
// ──── YouTube IFrame API ────
|
// ──── YouTube IFrame API ────
|
||||||
declare global {
|
declare global {
|
||||||
@ -80,14 +81,39 @@ function createHiddenIframe(embedUrl: string) {
|
|||||||
hiddenIframe = iframe
|
hiddenIframe = iframe
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ──── Offline progress persistence (localStorage fallback) ────
|
||||||
|
const OFFLINE_PROGRESS_KEY = 'wetalk-offline-progress'
|
||||||
|
|
||||||
|
function saveProgressLocal(podcastId: string, seconds: number, duration: number, completed = false) {
|
||||||
|
try {
|
||||||
|
const stored = JSON.parse(localStorage.getItem(OFFLINE_PROGRESS_KEY) || '{}')
|
||||||
|
stored[podcastId] = { seconds: Math.floor(seconds), duration, completed, at: Date.now() }
|
||||||
|
localStorage.setItem(OFFLINE_PROGRESS_KEY, JSON.stringify(stored))
|
||||||
|
} catch { /* ignore */ }
|
||||||
|
}
|
||||||
|
|
||||||
|
function fetchProgressLocal(podcastId: string): number {
|
||||||
|
try {
|
||||||
|
const stored = JSON.parse(localStorage.getItem(OFFLINE_PROGRESS_KEY) || '{}')
|
||||||
|
const entry = stored[podcastId]
|
||||||
|
return entry?.seconds > 5 ? entry.seconds : 0
|
||||||
|
} catch { return 0 }
|
||||||
|
}
|
||||||
|
|
||||||
// ──── Progress persistence ────
|
// ──── Progress persistence ────
|
||||||
let saveTimeout: ReturnType<typeof setTimeout> | null = null
|
let saveTimeout: ReturnType<typeof setTimeout> | null = null
|
||||||
|
|
||||||
async function saveProgressNow(podcastId: string, seconds: number, duration: number, markCompleted = false): Promise<void> {
|
async function saveProgressNow(podcastId: string, seconds: number, duration: number, markCompleted = false): Promise<void> {
|
||||||
if (saveTimeout) { clearTimeout(saveTimeout); saveTimeout = null }
|
if (saveTimeout) { clearTimeout(saveTimeout); saveTimeout = null }
|
||||||
const userId = useAuthStore.getState().user?.id
|
if (seconds < 1) return
|
||||||
if (!userId || seconds < 1) return
|
|
||||||
|
// Always save to localStorage (works offline)
|
||||||
const completed = markCompleted || (duration > 0 && seconds / duration > 0.95)
|
const completed = markCompleted || (duration > 0 && seconds / duration > 0.95)
|
||||||
|
saveProgressLocal(podcastId, seconds, duration, completed)
|
||||||
|
|
||||||
|
// Try Supabase if online + authenticated
|
||||||
|
const userId = useAuthStore.getState().user?.id
|
||||||
|
if (!userId || !navigator.onLine) return
|
||||||
const { error } = await supabase.from('listen_history').upsert(
|
const { error } = await supabase.from('listen_history').upsert(
|
||||||
{ user_id: userId, podcast_id: podcastId, progress_seconds: Math.floor(seconds), completed, listened_at: new Date().toISOString() },
|
{ user_id: userId, podcast_id: podcastId, progress_seconds: Math.floor(seconds), completed, listened_at: new Date().toISOString() },
|
||||||
{ onConflict: 'user_id,podcast_id' }
|
{ onConflict: 'user_id,podcast_id' }
|
||||||
@ -105,20 +131,35 @@ function debouncedSaveProgress(podcastId: string, seconds: number, duration: num
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function fetchSavedProgress(podcastId: string): Promise<number> {
|
async function fetchSavedProgress(podcastId: string): Promise<number> {
|
||||||
|
// Try Supabase first if online
|
||||||
const userId = useAuthStore.getState().user?.id
|
const userId = useAuthStore.getState().user?.id
|
||||||
if (!userId) return 0
|
if (userId && navigator.onLine) {
|
||||||
const { data } = await supabase
|
const { data } = await supabase
|
||||||
.from('listen_history')
|
.from('listen_history')
|
||||||
.select('progress_seconds')
|
.select('progress_seconds')
|
||||||
.eq('user_id', userId)
|
.eq('user_id', userId)
|
||||||
.eq('podcast_id', podcastId)
|
.eq('podcast_id', podcastId)
|
||||||
.maybeSingle()
|
.maybeSingle()
|
||||||
const saved = data?.progress_seconds || 0
|
const saved = data?.progress_seconds || 0
|
||||||
console.log('[player] fetched progress:', podcastId, saved, 'seconds, data:', data)
|
if (saved > 5) return saved
|
||||||
return saved > 5 ? saved : 0
|
}
|
||||||
|
// Fallback to localStorage
|
||||||
|
return fetchProgressLocal(podcastId)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ──── Store ────
|
// ──── Store ────
|
||||||
|
const PLAYBACK_RATES = [0.5, 0.75, 1, 1.25, 1.5, 2] as const
|
||||||
|
type PlaybackRate = (typeof PLAYBACK_RATES)[number]
|
||||||
|
|
||||||
|
function getSavedPlaybackRate(): PlaybackRate {
|
||||||
|
const saved = localStorage.getItem('wetalk-playback-rate')
|
||||||
|
if (saved) {
|
||||||
|
const n = parseFloat(saved)
|
||||||
|
if (PLAYBACK_RATES.includes(n as PlaybackRate)) return n as PlaybackRate
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
interface PlayerState {
|
interface PlayerState {
|
||||||
current: Podcast | null
|
current: Podcast | null
|
||||||
isPlaying: boolean
|
isPlaying: boolean
|
||||||
@ -127,6 +168,8 @@ interface PlayerState {
|
|||||||
duration: number
|
duration: number
|
||||||
volume: number
|
volume: number
|
||||||
audio: HTMLAudioElement | null
|
audio: HTMLAudioElement | null
|
||||||
|
playbackRate: PlaybackRate
|
||||||
|
queue: Podcast[]
|
||||||
|
|
||||||
play: (podcast: Podcast) => void
|
play: (podcast: Podcast) => void
|
||||||
toggle: () => void
|
toggle: () => void
|
||||||
@ -135,6 +178,23 @@ interface PlayerState {
|
|||||||
setVolume: (vol: number) => void
|
setVolume: (vol: number) => void
|
||||||
setProgress: (progress: number) => void
|
setProgress: (progress: number) => void
|
||||||
setDuration: (duration: number) => void
|
setDuration: (duration: number) => void
|
||||||
|
cyclePlaybackRate: () => void
|
||||||
|
setPlaybackRate: (rate: PlaybackRate) => void
|
||||||
|
addToQueue: (podcast: Podcast) => void
|
||||||
|
removeFromQueue: (podcastId: string) => void
|
||||||
|
clearQueue: () => void
|
||||||
|
playNext: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSavedQueue(): Podcast[] {
|
||||||
|
try {
|
||||||
|
const saved = localStorage.getItem('wetalk-queue')
|
||||||
|
return saved ? JSON.parse(saved) : []
|
||||||
|
} catch { return [] }
|
||||||
|
}
|
||||||
|
|
||||||
|
function persistQueue(queue: Podcast[]) {
|
||||||
|
localStorage.setItem('wetalk-queue', JSON.stringify(queue))
|
||||||
}
|
}
|
||||||
|
|
||||||
export const usePlayerStore = create<PlayerState>((set, get) => ({
|
export const usePlayerStore = create<PlayerState>((set, get) => ({
|
||||||
@ -145,6 +205,8 @@ export const usePlayerStore = create<PlayerState>((set, get) => ({
|
|||||||
duration: 0,
|
duration: 0,
|
||||||
volume: 0.8,
|
volume: 0.8,
|
||||||
audio: null,
|
audio: null,
|
||||||
|
playbackRate: getSavedPlaybackRate(),
|
||||||
|
queue: getSavedQueue(),
|
||||||
|
|
||||||
play: (podcast) => {
|
play: (podcast) => {
|
||||||
const { audio, current, volume } = get()
|
const { audio, current, volume } = get()
|
||||||
@ -213,6 +275,7 @@ export const usePlayerStore = create<PlayerState>((set, get) => ({
|
|||||||
events: {
|
events: {
|
||||||
onReady: (e: any) => {
|
onReady: (e: any) => {
|
||||||
e.target.setVolume(volume * 100)
|
e.target.setVolume(volume * 100)
|
||||||
|
e.target.setPlaybackRate(get().playbackRate)
|
||||||
e.target.playVideo()
|
e.target.playVideo()
|
||||||
const dur = e.target.getDuration()
|
const dur = e.target.getDuration()
|
||||||
if (dur > 0) set({ duration: dur })
|
if (dur > 0) set({ duration: dur })
|
||||||
@ -230,6 +293,8 @@ export const usePlayerStore = create<PlayerState>((set, get) => ({
|
|||||||
const podId = usePlayerStore.getState().current?.id
|
const podId = usePlayerStore.getState().current?.id
|
||||||
if (podId) saveProgressNow(podId, e.target.getDuration(), e.target.getDuration(), true)
|
if (podId) saveProgressNow(podId, e.target.getDuration(), e.target.getDuration(), true)
|
||||||
set({ isPlaying: false, progress: 0 })
|
set({ isPlaying: false, progress: 0 })
|
||||||
|
// Auto-play next from queue
|
||||||
|
setTimeout(() => usePlayerStore.getState().playNext(), 100)
|
||||||
} else if (e.data === 1) {
|
} else if (e.data === 1) {
|
||||||
const dur = e.target.getDuration()
|
const dur = e.target.getDuration()
|
||||||
if (dur > 0) set({ duration: dur })
|
if (dur > 0) set({ duration: dur })
|
||||||
@ -264,30 +329,45 @@ export const usePlayerStore = create<PlayerState>((set, get) => ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ── Native audio ──
|
// ── Native audio ──
|
||||||
const newAudio = new Audio(publicUrl(podcast.audio_url))
|
const audioSrc = publicUrl(podcast.audio_url)
|
||||||
newAudio.volume = volume
|
|
||||||
|
|
||||||
newAudio.addEventListener('timeupdate', () => {
|
// Try to use cached version first
|
||||||
set({ progress: newAudio.currentTime })
|
const setupAudio = (src: string) => {
|
||||||
debouncedSaveProgress(podcast.id, newAudio.currentTime, newAudio.duration || 0)
|
const newAudio = new Audio(src)
|
||||||
})
|
newAudio.volume = volume
|
||||||
newAudio.addEventListener('loadedmetadata', () => {
|
newAudio.playbackRate = get().playbackRate
|
||||||
set({ duration: newAudio.duration })
|
|
||||||
// Wait for previous save, then restore saved position
|
newAudio.addEventListener('timeupdate', () => {
|
||||||
savePrevPromise.then(() => fetchSavedProgress(podcast.id)).then((savedTime) => {
|
const currentDur = isFinite(newAudio.duration) ? newAudio.duration : (get().duration || 0)
|
||||||
if (savedTime > 0 && savedTime < newAudio.duration - 5) {
|
set({ progress: newAudio.currentTime })
|
||||||
newAudio.currentTime = savedTime
|
debouncedSaveProgress(podcast.id, newAudio.currentTime, currentDur)
|
||||||
set({ progress: savedTime })
|
})
|
||||||
}
|
newAudio.addEventListener('loadedmetadata', () => {
|
||||||
|
const dur = isFinite(newAudio.duration) ? newAudio.duration : podcast.duration_seconds || 0
|
||||||
|
set({ duration: dur })
|
||||||
|
// Wait for previous save, then restore saved position
|
||||||
|
savePrevPromise.then(() => fetchSavedProgress(podcast.id)).then((savedTime) => {
|
||||||
|
if (savedTime > 0 && savedTime < newAudio.duration - 5) {
|
||||||
|
newAudio.currentTime = savedTime
|
||||||
|
set({ progress: savedTime })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
newAudio.addEventListener('ended', () => {
|
||||||
|
saveProgressNow(podcast.id, newAudio.duration, newAudio.duration, true)
|
||||||
|
set({ isPlaying: false, progress: 0 })
|
||||||
|
// Auto-play next from queue
|
||||||
|
setTimeout(() => get().playNext(), 100)
|
||||||
})
|
})
|
||||||
})
|
|
||||||
newAudio.addEventListener('ended', () => {
|
|
||||||
saveProgressNow(podcast.id, newAudio.duration, newAudio.duration, true)
|
|
||||||
set({ isPlaying: false, progress: 0 })
|
|
||||||
})
|
|
||||||
|
|
||||||
newAudio.play()
|
newAudio.play()
|
||||||
set({ audio: newAudio, current: podcast, isPlaying: true, isExternal: false, progress: 0 })
|
set({ audio: newAudio, current: podcast, isPlaying: true, isExternal: false, progress: 0 })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check cache, fall back to network
|
||||||
|
getCachedAudioUrl(audioSrc).then((cachedUrl) => {
|
||||||
|
setupAudio(cachedUrl || audioSrc)
|
||||||
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
toggle: () => {
|
toggle: () => {
|
||||||
@ -374,6 +454,49 @@ export const usePlayerStore = create<PlayerState>((set, get) => ({
|
|||||||
|
|
||||||
setProgress: (progress) => set({ progress }),
|
setProgress: (progress) => set({ progress }),
|
||||||
setDuration: (duration) => set({ duration }),
|
setDuration: (duration) => set({ duration }),
|
||||||
|
|
||||||
|
cyclePlaybackRate: () => {
|
||||||
|
const { playbackRate } = get()
|
||||||
|
const idx = PLAYBACK_RATES.indexOf(playbackRate)
|
||||||
|
const next = PLAYBACK_RATES[(idx + 1) % PLAYBACK_RATES.length]
|
||||||
|
get().setPlaybackRate(next)
|
||||||
|
},
|
||||||
|
|
||||||
|
setPlaybackRate: (rate) => {
|
||||||
|
const { audio } = get()
|
||||||
|
if (audio) audio.playbackRate = rate
|
||||||
|
if (ytPlayer) try { ytPlayer.setPlaybackRate(rate) } catch { /* ignore */ }
|
||||||
|
localStorage.setItem('wetalk-playback-rate', String(rate))
|
||||||
|
set({ playbackRate: rate })
|
||||||
|
},
|
||||||
|
|
||||||
|
addToQueue: (podcast) => {
|
||||||
|
const { queue } = get()
|
||||||
|
if (queue.some((p) => p.id === podcast.id)) return
|
||||||
|
const next = [...queue, podcast]
|
||||||
|
persistQueue(next)
|
||||||
|
set({ queue: next })
|
||||||
|
},
|
||||||
|
|
||||||
|
removeFromQueue: (podcastId) => {
|
||||||
|
const next = get().queue.filter((p) => p.id !== podcastId)
|
||||||
|
persistQueue(next)
|
||||||
|
set({ queue: next })
|
||||||
|
},
|
||||||
|
|
||||||
|
clearQueue: () => {
|
||||||
|
persistQueue([])
|
||||||
|
set({ queue: [] })
|
||||||
|
},
|
||||||
|
|
||||||
|
playNext: () => {
|
||||||
|
const { queue, play } = get()
|
||||||
|
if (queue.length === 0) return
|
||||||
|
const [next, ...rest] = queue
|
||||||
|
persistQueue(rest)
|
||||||
|
set({ queue: rest })
|
||||||
|
play(next)
|
||||||
|
},
|
||||||
}))
|
}))
|
||||||
|
|
||||||
// Track YouTube progress via polling
|
// Track YouTube progress via polling
|
||||||
|
|||||||
@ -26,6 +26,7 @@ export interface Podcast {
|
|||||||
likes_count?: number
|
likes_count?: number
|
||||||
comments_count?: number
|
comments_count?: number
|
||||||
is_liked?: boolean
|
is_liked?: boolean
|
||||||
|
chapters?: Chapter[]
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Tag {
|
export interface Tag {
|
||||||
@ -34,6 +35,14 @@ export interface Tag {
|
|||||||
slug: string
|
slug: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface Chapter {
|
||||||
|
id: string
|
||||||
|
podcast_id: string
|
||||||
|
title: string
|
||||||
|
start_time_seconds: number
|
||||||
|
sort_order: number
|
||||||
|
}
|
||||||
|
|
||||||
export interface Comment {
|
export interface Comment {
|
||||||
id: string
|
id: string
|
||||||
user_id: string
|
user_id: string
|
||||||
|
|||||||
33
supabase/migrations/005_chapters.sql
Normal file
33
supabase/migrations/005_chapters.sql
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
-- Chapters: timestamped sections within a podcast episode
|
||||||
|
CREATE TABLE public.chapters (
|
||||||
|
id uuid DEFAULT gen_random_uuid() PRIMARY KEY,
|
||||||
|
podcast_id uuid REFERENCES public.podcasts(id) ON DELETE CASCADE NOT NULL,
|
||||||
|
title text NOT NULL,
|
||||||
|
start_time_seconds integer NOT NULL DEFAULT 0,
|
||||||
|
sort_order integer NOT NULL DEFAULT 0,
|
||||||
|
created_at timestamptz DEFAULT now()
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX idx_chapters_podcast ON public.chapters(podcast_id);
|
||||||
|
|
||||||
|
ALTER TABLE public.chapters ENABLE ROW LEVEL SECURITY;
|
||||||
|
|
||||||
|
-- Everyone can read chapters
|
||||||
|
CREATE POLICY "chapters_read" ON public.chapters
|
||||||
|
FOR SELECT USING (true);
|
||||||
|
|
||||||
|
-- Only the podcast creator can manage chapters
|
||||||
|
CREATE POLICY "chapters_insert" ON public.chapters
|
||||||
|
FOR INSERT WITH CHECK (
|
||||||
|
podcast_id IN (SELECT id FROM public.podcasts WHERE creator_id = auth.uid())
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE POLICY "chapters_update" ON public.chapters
|
||||||
|
FOR UPDATE USING (
|
||||||
|
podcast_id IN (SELECT id FROM public.podcasts WHERE creator_id = auth.uid())
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE POLICY "chapters_delete" ON public.chapters
|
||||||
|
FOR DELETE USING (
|
||||||
|
podcast_id IN (SELECT id FROM public.podcasts WHERE creator_id = auth.uid())
|
||||||
|
);
|
||||||
@ -12,6 +12,17 @@ export default defineConfig({
|
|||||||
workbox: {
|
workbox: {
|
||||||
skipWaiting: true,
|
skipWaiting: true,
|
||||||
clientsClaim: true,
|
clientsClaim: true,
|
||||||
|
runtimeCaching: [
|
||||||
|
{
|
||||||
|
urlPattern: /\/storage\/v1\/object\/public\//,
|
||||||
|
handler: 'CacheFirst',
|
||||||
|
options: {
|
||||||
|
cacheName: 'wetalk-supabase-assets',
|
||||||
|
expiration: { maxEntries: 100, maxAgeSeconds: 30 * 24 * 60 * 60 },
|
||||||
|
cacheableResponse: { statuses: [0, 200] },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
includeAssets: ['favicon.svg', 'icons/*.png'],
|
includeAssets: ['favicon.svg', 'icons/*.png'],
|
||||||
manifest: {
|
manifest: {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user