webapp/src/pages/layoutShouts.page.tsx

172 lines
6.1 KiB
TypeScript
Raw Normal View History

2023-02-17 09:21:02 +00:00
import { PageLayout } from '../components/_shared/PageLayout'
import type { PageProps } from './types'
2022-11-13 12:14:04 +00:00
import { createMemo, createSignal, For, onCleanup, onMount, Show } from 'solid-js'
2023-02-17 09:21:02 +00:00
import { loadShouts, resetSortedArticles } from '../stores/zine/articles'
import { useRouter } from '../stores/router'
import { LayoutType, useLayoutsStore } from '../stores/zine/layouts'
import { Loading } from '../components/_shared/Loading'
import { restoreScrollPosition, saveScrollPosition } from '../utils/scroll'
import type { Shout } from '../graphql/types.gen'
import { splitToPages } from '../utils/splitToPages'
2022-11-18 02:23:04 +00:00
import { clsx } from 'clsx'
2023-02-17 09:21:02 +00:00
import { Row3 } from '../components/Feed/Row3'
import { Row2 } from '../components/Feed/Row2'
import { Beside } from '../components/Feed/Beside'
import Slider from '../components/_shared/Slider'
import { Row1 } from '../components/Feed/Row1'
import styles from '../styles/Topic.module.scss'
2023-05-01 18:32:32 +00:00
import { ArticleCard } from '../components/Feed/ArticleCard'
2023-02-17 09:21:02 +00:00
import { useLocalize } from '../context/localize'
2022-11-12 18:59:29 +00:00
2022-11-13 12:14:04 +00:00
export const PRERENDERED_ARTICLES_COUNT = 21
const LOAD_MORE_PAGE_SIZE = 9 // Row3 + Row3 + Row3
2022-11-12 18:59:29 +00:00
export const LayoutShoutsPage = (props: PageProps) => {
2023-02-17 09:21:02 +00:00
const { t } = useLocalize()
const getLayout = createMemo<LayoutType>(() => {
2022-11-12 18:59:29 +00:00
const { page: getPage } = useRouter()
const page = getPage()
2023-05-01 18:32:32 +00:00
if (page.route !== 'expo') {
throw new Error('ts guard')
}
2023-02-17 09:21:02 +00:00
const { layout } = page.params
return layout as LayoutType
2022-11-12 18:59:29 +00:00
})
2022-11-13 12:14:04 +00:00
const [isLoadMoreButtonVisible, setIsLoadMoreButtonVisible] = createSignal(false)
2023-02-17 09:21:02 +00:00
const { sortedLayoutShouts, loadLayoutShoutsBy } = useLayoutsStore(getLayout(), props.layoutShouts)
const sortedArticles = createMemo<Shout[]>(() => sortedLayoutShouts().get(getLayout()) || [])
2023-05-01 18:32:32 +00:00
const loadMoreLayout = async (_kind: LayoutType) => {
2022-11-13 12:14:04 +00:00
saveScrollPosition()
2022-11-16 06:33:58 +00:00
const { hasMore } = await loadLayoutShoutsBy({
2022-11-18 02:23:04 +00:00
// filters: { layout: kind },
2022-11-16 06:33:58 +00:00
limit: LOAD_MORE_PAGE_SIZE,
2022-11-13 12:14:04 +00:00
offset: sortedArticles().length
})
setIsLoadMoreButtonVisible(hasMore)
restoreScrollPosition()
}
2022-11-12 18:59:29 +00:00
onMount(async () => {
2022-11-13 12:14:04 +00:00
if (sortedArticles().length === PRERENDERED_ARTICLES_COUNT) {
2023-02-17 09:21:02 +00:00
loadMoreLayout(getLayout())
2022-11-12 18:59:29 +00:00
}
2022-11-13 12:14:04 +00:00
})
const title = createMemo(() => {
2023-02-17 09:21:02 +00:00
const l = getLayout()
2022-11-13 12:14:04 +00:00
if (l === 'audio') return t('Audio')
if (l === 'video') return t('Video')
if (l === 'image') return t('Artworks')
return t('Literature')
})
2022-11-12 18:59:29 +00:00
2022-11-13 12:14:04 +00:00
const pages = createMemo<Shout[][]>(() =>
splitToPages(sortedArticles(), PRERENDERED_ARTICLES_COUNT, LOAD_MORE_PAGE_SIZE)
)
2022-11-14 08:03:48 +00:00
const isLoaded = createMemo(() => Boolean(sortedArticles()))
2022-11-12 18:59:29 +00:00
2022-11-13 12:14:04 +00:00
onMount(async () => {
if (!isLoaded()) {
2023-02-17 09:21:02 +00:00
await loadShouts({ filters: { layout: getLayout() }, limit: PRERENDERED_ARTICLES_COUNT, offset: 0 })
2022-11-13 12:14:04 +00:00
}
2022-11-12 18:59:29 +00:00
})
onCleanup(() => resetSortedArticles())
2022-11-13 12:14:04 +00:00
const ModeSwitcher = () => (
2022-11-23 20:47:06 +00:00
<div class="wide-container">
2022-11-13 12:14:04 +00:00
<div class={clsx(styles.groupControls, 'row group__controls')}>
2023-03-10 17:42:48 +00:00
<div class="col-md-16">
2022-11-13 12:14:04 +00:00
<ul class="view-switcher">
2023-05-22 22:01:04 +00:00
<li classList={{ 'view-switcher__item--selected': getLayout() === 'audio' }}>
2022-11-13 12:14:04 +00:00
<a href="/expo/audio">{t('Audio')}</a>
</li>
2023-05-22 22:01:04 +00:00
<li classList={{ 'view-switcher__item--selected': getLayout() === 'video' }}>
2022-11-13 12:14:04 +00:00
<a href="/expo/video">{t('Video')}</a>
</li>
2023-05-22 22:01:04 +00:00
<li classList={{ 'view-switcher__item--selected': getLayout() === 'image' }}>
2022-11-13 12:14:04 +00:00
<a href="/expo/image">{t('Artworks')}</a>
</li>
2023-05-22 22:01:04 +00:00
<li classList={{ 'view-switcher__item--selected': getLayout() === 'literature' }}>
2022-11-13 12:14:04 +00:00
<a href="/expo/literature">{t('Literature')}</a>
</li>
</ul>
</div>
2023-03-10 17:42:48 +00:00
<div class="col-md-8">
2022-11-13 12:14:04 +00:00
<div class="mode-switcher">
{`${t('Show')} `}
<span class="mode-switcher__control">{t('All posts')}</span>
</div>
</div>
</div>
</div>
)
2022-11-12 18:59:29 +00:00
return (
2023-02-17 09:21:02 +00:00
<PageLayout>
2022-11-12 18:59:29 +00:00
<Show when={isLoaded()} fallback={<Loading />}>
2022-11-13 12:14:04 +00:00
<div class={styles.topicPage}>
2023-02-17 09:21:02 +00:00
<Show when={getLayout() && Boolean(sortedArticles())}>
2023-01-23 21:31:47 +00:00
<div class="wide-container">
<h1>{title()}</h1>
</div>
2022-11-13 12:14:04 +00:00
<ModeSwitcher />
<Row1 article={sortedArticles()[0]} />
<Row2 articles={sortedArticles().slice(1, 3)} />
2022-11-27 17:02:04 +00:00
<Slider title={title()}>
<For each={sortedArticles().slice(5, 11)}>
{(a: Shout) => (
<ArticleCard
article={a}
settings={{
additionalClass: 'swiper-slide',
isFloorImportant: true,
isWithCover: true,
nodate: true
}}
/>
)}
</For>
</Slider>
2022-11-13 12:14:04 +00:00
<Beside
beside={sortedArticles()[12]}
title={t('Top viewed')}
values={sortedArticles().slice(0, 5)}
wrapper={'top-article'}
/>
<Show when={sortedArticles().length > 5}>
<Row3 articles={sortedArticles().slice(13, 16)} />
<Row2 articles={sortedArticles().slice(16, 18)} />
<Row3 articles={sortedArticles().slice(18, 21)} />
<Row3 articles={sortedArticles().slice(21, 24)} />
<Row3 articles={sortedArticles().slice(24, 27)} />
</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">
2023-02-17 09:21:02 +00:00
<button class="button" onClick={() => loadMoreLayout(getLayout())}>
2022-11-13 12:14:04 +00:00
{t('Load more')}
</button>
</p>
</Show>
</Show>
</div>
2022-11-12 18:59:29 +00:00
</Show>
2023-02-17 09:21:02 +00:00
</PageLayout>
2022-11-12 18:59:29 +00:00
)
}
2023-02-17 09:21:02 +00:00
export const Page = LayoutShoutsPage