mirror of
https://github.com/r-freeman/portfolio.git
synced 2024-11-14 12:05:40 +00:00
64 lines
1.8 KiB
TypeScript
64 lines
1.8 KiB
TypeScript
'use client'
|
|
|
|
import {createPagesBrowserClient} from '@supabase/auth-helpers-nextjs'
|
|
import {ElementType, useEffect} from 'react'
|
|
import useSWR, {useSWRConfig} from 'swr'
|
|
import fetcher from '@/lib/fetcher'
|
|
import {numberFormat} from '@/lib/numberFormat'
|
|
|
|
type ViewsProps = {
|
|
as?: ElementType
|
|
slug: string
|
|
className?: string
|
|
shouldUpdateViews?: boolean
|
|
shouldRender?: boolean
|
|
}
|
|
|
|
export function Views({as: Component = 'span', slug, className, shouldUpdateViews = true, shouldRender = true}: ViewsProps) {
|
|
const supabase = createPagesBrowserClient()
|
|
const {data} = useSWR(`/api/views/${slug}`, fetcher) as { data: { views: number } }
|
|
const {mutate} = useSWRConfig()
|
|
|
|
useEffect(() => {
|
|
if (shouldUpdateViews) {
|
|
// subscribe to analytics table and react to updates at row level
|
|
const sub = supabase
|
|
.channel('any')
|
|
.on('postgres_changes', {
|
|
event: 'UPDATE',
|
|
schema: 'public',
|
|
table: 'analytics',
|
|
filter: `slug=eq.${slug}`
|
|
}, () => {
|
|
mutate(`/api/views/${slug}`)
|
|
})
|
|
.subscribe();
|
|
|
|
return () => {
|
|
sub.unsubscribe()
|
|
}
|
|
}
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
if (shouldUpdateViews) {
|
|
const registerView = async () => {
|
|
await fetcher(`/api/views/${slug}`,
|
|
{
|
|
method: 'POST'
|
|
}
|
|
)
|
|
}
|
|
|
|
registerView().then(() => mutate(`/api/views/${slug}`))
|
|
}
|
|
}, [])
|
|
|
|
if (!shouldRender) return null
|
|
|
|
return (
|
|
<Component className={className}>
|
|
{` · ${data?.views > 0 ? numberFormat(data.views) : '—'} views`}
|
|
</Component>
|
|
)
|
|
} |