webapp/src/routes/topic/[slug].tsx

78 lines
2.8 KiB
TypeScript
Raw Normal View History

2024-07-03 17:38:43 +00:00
import { RouteSectionProps, createAsync, useParams } from '@solidjs/router'
2024-07-06 06:41:16 +00:00
import { ErrorBoundary, Suspense, createEffect, createMemo } from 'solid-js'
2024-07-01 13:27:45 +00:00
import { FourOuFourView } from '~/components/Views/FourOuFour'
import { TopicView } from '~/components/Views/Topic'
import { Loading } from '~/components/_shared/Loading'
import { PageLayout } from '~/components/_shared/PageLayout'
import { useLocalize } from '~/context/localize'
import { useTopics } from '~/context/topics'
import { loadShouts } from '~/graphql/api/public'
2024-07-01 13:27:45 +00:00
import { LoadShoutsOptions, Shout, Topic } from '~/graphql/schema/core.gen'
2024-07-09 09:13:13 +00:00
import { getImageUrl } from '~/lib/getImageUrl'
import { getArticleDescription } from '~/utils/meta'
2024-07-01 13:27:45 +00:00
import { SHOUTS_PER_PAGE } from '../(home)'
2024-07-03 17:38:43 +00:00
const fetchTopicShouts = async (slug: string, offset?: number) => {
const opts: LoadShoutsOptions = { filters: { topic: slug }, limit: SHOUTS_PER_PAGE, offset }
2024-07-01 13:27:45 +00:00
const shoutsLoader = loadShouts(opts)
return await shoutsLoader()
}
export const route = {
2024-07-03 17:38:43 +00:00
load: async ({ params, location: { query } }: RouteSectionProps<{ articles: Shout[] }>) => {
const offset: number = Number.parseInt(query.offset, 10)
const result = await fetchTopicShouts(params.slug, offset)
return result
}
2024-07-01 13:27:45 +00:00
}
2024-07-05 17:23:07 +00:00
export default (props: RouteSectionProps<{ articles: Shout[] }>) => {
2024-07-03 17:38:43 +00:00
const params = useParams()
2024-07-01 13:41:22 +00:00
const articles = createAsync(
2024-07-03 17:38:43 +00:00
async () => props.data.articles || (await fetchTopicShouts(params.slug)) || []
2024-07-01 13:41:22 +00:00
)
2024-07-01 13:27:45 +00:00
const { topicEntities } = useTopics()
const { t } = useLocalize()
2024-07-03 17:38:43 +00:00
const topic = createMemo(() => topicEntities?.()[params.slug])
2024-07-05 08:11:57 +00:00
const title = createMemo(() => `${t('Discours')} :: ${topic()?.title || ''}`)
2024-07-06 06:41:16 +00:00
createEffect(() => {
if (topic() && window) {
window?.gtag?.('event', 'page_view', {
2024-07-01 13:27:45 +00:00
page_title: topic()?.title,
2024-07-06 06:41:16 +00:00
page_location: window?.location.href,
page_path: window?.location.pathname
2024-07-01 13:27:45 +00:00
})
}
})
2024-07-09 09:13:13 +00:00
const desc = createMemo(() =>
topic()?.body
? getArticleDescription(topic()?.body || '')
: t('The most interesting publications on the topic', { topicName: title() })
)
const cover = createMemo(() =>
topic()?.pic
? getImageUrl(topic()?.pic || '', { width: 1200 })
: getImageUrl('production/image/logo_image.png')
)
2024-07-01 13:27:45 +00:00
return (
<ErrorBoundary fallback={(_err) => <FourOuFourView />}>
<Suspense fallback={<Loading />}>
<PageLayout
2024-07-09 09:13:13 +00:00
key="topic"
2024-07-01 13:27:45 +00:00
title={title()}
2024-07-09 09:13:13 +00:00
desc={desc()}
2024-07-01 13:27:45 +00:00
headerTitle={topic()?.title || ''}
slug={topic()?.slug}
2024-07-09 09:13:13 +00:00
cover={cover()}
2024-07-01 13:27:45 +00:00
>
2024-07-01 13:41:22 +00:00
<TopicView
topic={topic() as Topic}
topicSlug={props.params.slug}
shouts={articles() as Shout[]}
/>
2024-07-01 13:27:45 +00:00
</PageLayout>
</Suspense>
</ErrorBoundary>
)
}