webapp/src/components/Views/Topic.tsx

249 lines
8.9 KiB
TypeScript

import { useSearchParams } from '@solidjs/router'
import { clsx } from 'clsx'
import { For, Show, Suspense, createEffect, createMemo, createSignal, on, onMount } from 'solid-js'
import { useAuthors } from '~/context/authors'
import { useFeed } from '~/context/feed'
import { useLocalize } from '~/context/localize'
import { useTopics } from '~/context/topics'
import { loadAuthors, loadFollowersByTopic, loadShouts } from '~/graphql/api/public'
import { Author, AuthorsBy, LoadShoutsOptions, Shout, Topic } from '~/graphql/schema/core.gen'
import { SHOUTS_PER_PAGE } from '~/routes/(main)'
import { getUnixtime } from '~/utils/getServerDate'
import { restoreScrollPosition, saveScrollPosition } from '~/utils/scroll'
import { splitToPages } from '~/utils/splitToPages'
import styles from '../../styles/Topic.module.scss'
import { Beside } from '../Feed/Beside'
import { Row1 } from '../Feed/Row1'
import { Row2 } from '../Feed/Row2'
import { Row3 } from '../Feed/Row3'
import { FullTopic } from '../Topic/Full'
import { Loading } from '../_shared/Loading'
import { ArticleCardSwiper } from '../_shared/SolidSwiper/ArticleCardSwiper'
type TopicsPageSearchParams = {
by: 'comments' | '' | 'recent' | 'viewed' | 'rating' | 'commented'
}
interface Props {
topic: Topic
shouts: Shout[]
topicSlug: string
followers?: Author[]
}
export const PRERENDERED_ARTICLES_COUNT = 28
const LOAD_MORE_PAGE_SIZE = 9 // Row3 + Row3 + Row3
export const TopicView = (props: Props) => {
const { t } = useLocalize()
const { feedByTopic, addFeed } = useFeed()
const { topicEntities } = useTopics()
const { authorsByTopic } = useAuthors()
const [searchParams, changeSearchParams] = useSearchParams<TopicsPageSearchParams>()
const [isLoadMoreButtonVisible, setIsLoadMoreButtonVisible] = createSignal(false)
const [favoriteTopArticles, setFavoriteTopArticles] = createSignal<Shout[]>([])
const [reactedTopMonthArticles, setReactedTopMonthArticles] = createSignal<Shout[]>([])
const [topic, setTopic] = createSignal<Topic>()
const [followers, setFollowers] = createSignal<Author[]>(props.followers || [])
const sortedFeed = createMemo(() => feedByTopic()[topic()?.slug || ''] || []) // TODO: filter + sort
const loadTopicFollowers = async () => {
const topicFollowersFetcher = loadFollowersByTopic(props.topicSlug)
const topicFollowers = await topicFollowersFetcher()
topicFollowers && setFollowers(topicFollowers)
}
const [topicAuthors, setTopicAuthors] = createSignal<Author[]>([])
const loadTopicAuthors = async () => {
const by: AuthorsBy = { topic: props.topicSlug }
const topicAuthorsFetcher = await loadAuthors({ by, limit: 10, offset: 0 })
const result = await topicAuthorsFetcher()
result && setTopicAuthors(result)
}
const loadFavoriteTopArticles = async () => {
const options: LoadShoutsOptions = {
filters: { featured: true, topic: props.topicSlug },
limit: 10,
random_limit: 100
}
const topicRandomShoutsFetcher = loadShouts(options)
const result = await topicRandomShoutsFetcher()
result && setFavoriteTopArticles(result)
}
const loadReactedTopMonthArticles = async () => {
const now = new Date()
const after = getUnixtime(new Date(now.setMonth(now.getMonth() - 1)))
const options: LoadShoutsOptions = {
filters: { after: after, featured: true, topic: props.topicSlug },
limit: 10,
random_limit: 10
}
const reactedTopMonthShoutsFetcher = loadShouts(options)
const result = await reactedTopMonthShoutsFetcher()
result && setReactedTopMonthArticles(result)
}
// второй этап начальной загрузки данных
createEffect(
on(
topicEntities,
(ttt: Record<string, Topic>) => {
if (props.topicSlug in ttt) {
Promise.all([
loadFavoriteTopArticles(),
loadReactedTopMonthArticles(),
loadTopicAuthors(),
loadTopicFollowers()
]).finally(() => {
setTopic(ttt[props.topicSlug])
})
}
},
{ defer: true }
)
)
// дозагрузка
const loadMore = async () => {
saveScrollPosition()
const amountBefore = feedByTopic()?.[props.topicSlug]?.length || 0
const topicShoutsFetcher = loadShouts({
filters: { topic: props.topicSlug },
limit: SHOUTS_PER_PAGE,
offset: amountBefore
})
const result = await topicShoutsFetcher()
if (result) {
addFeed(result)
const amountAfter = feedByTopic()[props.topicSlug].length
setIsLoadMoreButtonVisible(amountBefore !== amountAfter)
}
restoreScrollPosition()
}
onMount(() => {
if (sortedFeed() || [].length === PRERENDERED_ARTICLES_COUNT) {
loadMore()
}
})
/*
const selectionTitle = createMemo(() => {
const m = searchParams?.by
if (m === 'viewed') return t('Top viewed')
if (m === 'rating') return t('Top rated')
if (m === 'commented') return t('Top discussed')
return t('Top recent')
})
*/
const pages = createMemo<Shout[][]>(() =>
splitToPages(sortedFeed(), PRERENDERED_ARTICLES_COUNT, LOAD_MORE_PAGE_SIZE)
)
return (
<div class={styles.topicPage}>
<Suspense fallback={<Loading />}>
<Show when={topic()}>
<FullTopic topic={topic() as Topic} followers={followers()} authors={topicAuthors()} />
</Show>
<div class="wide-container">
<div class={clsx(styles.groupControls, 'row group__controls')}>
<div class="col-md-16">
<ul class="view-switcher">
<li
classList={{
'view-switcher__item--selected': searchParams?.by === 'recent' || !searchParams?.by
}}
>
<button
type="button"
onClick={() =>
changeSearchParams({
by: 'recent'
})
}
>
{t('Recent')}
</button>
</li>
{/*TODO: server sort*/}
{/*<li classList={{ 'view-switcher__item--selected': getSearchParams().by === 'rating' }}>*/}
{/* <button type="button" onClick={() => changeSearchParams('by', 'rating')}>*/}
{/* {t('Popular')}*/}
{/* </button>*/}
{/*</li>*/}
{/*<li classList={{ 'view-switcher__item--selected': getSearchParams().by === 'viewed' }}>*/}
{/* <button type="button" onClick={() => changeSearchParams('by', 'viewed')}>*/}
{/* {t('Views')}*/}
{/* </button>*/}
{/*</li>*/}
{/*<li classList={{ 'view-switcher__item--selected': getSearchParams().by === 'commented' }}>*/}
{/* <button type="button" onClick={() => changeSearchParams('by', 'commented')}>*/}
{/* {t('Discussing')}*/}
{/* </button>*/}
{/*</li>*/}
</ul>
</div>
<div class="col-md-8">
<div class="mode-switcher">
{`${t('Show')} `}
<span class="mode-switcher__control">{t('All posts')}</span>
</div>
</div>
</div>
</div>
<Row1 article={sortedFeed()[0]} />
<Row2 articles={sortedFeed().slice(1, 3)} isEqual={true} />
<Beside
title={t('Topic is supported by')}
values={authorsByTopic?.()?.[topic()?.slug || '']?.slice(0, 6)}
beside={sortedFeed()[4]}
wrapper={'author'}
/>
<Show when={reactedTopMonthArticles()?.length > 0} keyed={true}>
<ArticleCardSwiper title={t('Top month')} slides={reactedTopMonthArticles()} />
</Show>
<Beside
beside={sortedFeed()[12]}
title={t('Top viewed')}
values={sortedFeed().slice(0, 5)}
wrapper={'top-article'}
/>
<Row2 articles={sortedFeed().slice(13, 15)} isEqual={true} />
<Row1 article={sortedFeed()[15]} />
<Show when={favoriteTopArticles()?.length > 0} keyed={true}>
<ArticleCardSwiper title={t('Favorite')} slides={favoriteTopArticles()} />
</Show>
<Show when={sortedFeed().length > 15}>
<Row3 articles={sortedFeed().slice(23, 26)} />
<Row2 articles={sortedFeed().slice(26, 28)} />
</Show>
<For each={pages()}>
{(page) => (
<>
<Row3 articles={page.slice(0, 3)} />
<Row3 articles={page.slice(3, 6)} />
<Row3 articles={page.slice(6, 9)} />
</>
)}
</For>
<Show when={isLoadMoreButtonVisible()}>
<p class="load-more-container">
<button class="button" onClick={loadMore}>
{t('Load more')}
</button>
</p>
</Show>
</Suspense>
</div>
)
}