2023-11-28 13:18:25 +00:00
|
|
|
import type { Shout, Topic } from '../../graphql/schema/core.gen'
|
2023-02-17 09:21:02 +00:00
|
|
|
|
2023-12-13 10:39:31 +00:00
|
|
|
import { Meta } from '@solidjs/meta'
|
2023-11-14 15:10:00 +00:00
|
|
|
import { clsx } from 'clsx'
|
2024-02-04 11:25:21 +00:00
|
|
|
import { For, Show, createEffect, createMemo, createSignal, onMount } from 'solid-js'
|
2023-11-14 15:10:00 +00:00
|
|
|
|
|
|
|
import { useLocalize } from '../../context/localize'
|
2022-09-22 09:37:49 +00:00
|
|
|
import { useRouter } from '../../stores/router'
|
2022-11-18 02:23:04 +00:00
|
|
|
import { loadShouts, useArticlesStore } from '../../stores/zine/articles'
|
2022-09-13 09:59:04 +00:00
|
|
|
import { useAuthorsStore } from '../../stores/zine/authors'
|
2023-11-14 15:10:00 +00:00
|
|
|
import { useTopicsStore } from '../../stores/zine/topics'
|
2023-12-13 10:39:31 +00:00
|
|
|
import { capitalize } from '../../utils/capitalize'
|
|
|
|
import { getImageUrl } from '../../utils/getImageUrl'
|
|
|
|
import { getDescription } from '../../utils/meta'
|
2022-10-28 21:21:47 +00:00
|
|
|
import { restoreScrollPosition, saveScrollPosition } from '../../utils/scroll'
|
|
|
|
import { splitToPages } from '../../utils/splitToPages'
|
2023-11-14 15:10:00 +00:00
|
|
|
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'
|
2024-02-04 11:25:21 +00:00
|
|
|
import { ArticleCardSwiper } from '../_shared/SolidSwiper/ArticleCardSwiper'
|
2023-11-14 15:10:00 +00:00
|
|
|
|
|
|
|
import styles from '../../styles/Topic.module.scss'
|
2022-09-22 09:37:49 +00:00
|
|
|
|
|
|
|
type TopicsPageSearchParams = {
|
|
|
|
by: 'comments' | '' | 'recent' | 'viewed' | 'rating' | 'commented'
|
|
|
|
}
|
2022-09-09 11:53:35 +00:00
|
|
|
|
2023-12-13 10:39:31 +00:00
|
|
|
interface Props {
|
2022-09-09 11:53:35 +00:00
|
|
|
topic: Topic
|
2022-11-15 14:24:50 +00:00
|
|
|
shouts: Shout[]
|
2022-10-05 15:11:14 +00:00
|
|
|
topicSlug: string
|
2022-09-09 11:53:35 +00:00
|
|
|
}
|
|
|
|
|
2022-11-13 19:35:57 +00:00
|
|
|
export const PRERENDERED_ARTICLES_COUNT = 28
|
2022-10-28 21:21:47 +00:00
|
|
|
const LOAD_MORE_PAGE_SIZE = 9 // Row3 + Row3 + Row3
|
|
|
|
|
2023-12-13 10:39:31 +00:00
|
|
|
export const TopicView = (props: Props) => {
|
2023-12-20 07:45:29 +00:00
|
|
|
const { t, lang } = useLocalize()
|
2023-12-24 12:56:30 +00:00
|
|
|
const { searchParams, changeSearchParams } = useRouter<TopicsPageSearchParams>()
|
2022-10-28 21:21:47 +00:00
|
|
|
const [isLoadMoreButtonVisible, setIsLoadMoreButtonVisible] = createSignal(false)
|
2022-11-15 14:24:50 +00:00
|
|
|
const { sortedArticles } = useArticlesStore({ shouts: props.shouts })
|
2022-09-28 20:16:44 +00:00
|
|
|
const { topicEntities } = useTopicsStore({ topics: [props.topic] })
|
|
|
|
const { authorsByTopic } = useAuthorsStore()
|
2022-09-13 09:59:04 +00:00
|
|
|
|
2024-01-31 12:34:15 +00:00
|
|
|
const [topic, setTopic] = createSignal<Topic>()
|
|
|
|
createEffect(() => {
|
|
|
|
const topics = topicEntities()
|
|
|
|
if (props.topicSlug && !topic() && topics) {
|
|
|
|
setTopic(topics[props.topicSlug])
|
|
|
|
}
|
|
|
|
})
|
|
|
|
const title = createMemo(
|
|
|
|
() =>
|
|
|
|
`#${capitalize(
|
|
|
|
lang() === 'en'
|
|
|
|
? topic()?.slug.replace(/-/, ' ')
|
|
|
|
: topic()?.title || topic()?.slug.replace(/-/, ' '),
|
|
|
|
true,
|
|
|
|
)}`,
|
2023-12-20 07:45:29 +00:00
|
|
|
)
|
2023-11-14 10:45:44 +00:00
|
|
|
|
2022-10-28 21:21:47 +00:00
|
|
|
const loadMore = async () => {
|
|
|
|
saveScrollPosition()
|
|
|
|
|
2022-11-18 02:23:04 +00:00
|
|
|
const { hasMore } = await loadShouts({
|
2023-12-20 07:45:29 +00:00
|
|
|
filters: { topic: topic()?.slug },
|
2022-10-28 21:21:47 +00:00
|
|
|
limit: LOAD_MORE_PAGE_SIZE,
|
2023-11-14 15:10:00 +00:00
|
|
|
offset: sortedArticles().length,
|
2022-10-28 21:21:47 +00:00
|
|
|
})
|
|
|
|
setIsLoadMoreButtonVisible(hasMore)
|
|
|
|
|
|
|
|
restoreScrollPosition()
|
|
|
|
}
|
|
|
|
|
2024-02-05 15:04:23 +00:00
|
|
|
onMount(() => {
|
2022-10-28 21:21:47 +00:00
|
|
|
if (sortedArticles().length === PRERENDERED_ARTICLES_COUNT) {
|
|
|
|
loadMore()
|
|
|
|
}
|
|
|
|
})
|
2023-12-24 12:56:30 +00:00
|
|
|
/*
|
2023-12-20 07:45:29 +00:00
|
|
|
const selectionTitle = createMemo(() => {
|
2022-10-25 16:25:42 +00:00
|
|
|
const m = searchParams().by
|
2022-10-05 15:11:14 +00:00
|
|
|
if (m === 'viewed') return t('Top viewed')
|
|
|
|
if (m === 'rating') return t('Top rated')
|
|
|
|
if (m === 'commented') return t('Top discussed')
|
2022-09-09 11:53:35 +00:00
|
|
|
return t('Top recent')
|
|
|
|
})
|
2023-12-24 12:56:30 +00:00
|
|
|
*/
|
2022-10-28 21:21:47 +00:00
|
|
|
const pages = createMemo<Shout[][]>(() =>
|
2023-11-14 15:10:00 +00:00
|
|
|
splitToPages(sortedArticles(), PRERENDERED_ARTICLES_COUNT, LOAD_MORE_PAGE_SIZE),
|
2022-10-28 21:21:47 +00:00
|
|
|
)
|
|
|
|
|
2023-12-20 07:45:29 +00:00
|
|
|
const ogImage = () =>
|
|
|
|
topic()?.pic
|
|
|
|
? getImageUrl(topic().pic, { width: 1200 })
|
|
|
|
: getImageUrl('production/image/logo_image.png')
|
|
|
|
const description = () =>
|
|
|
|
topic()?.body
|
|
|
|
? getDescription(topic().body)
|
|
|
|
: t('The most interesting publications on the topic', { topicName: title() })
|
2023-12-13 10:39:31 +00:00
|
|
|
|
2022-09-09 11:53:35 +00:00
|
|
|
return (
|
2022-11-07 21:07:42 +00:00
|
|
|
<div class={styles.topicPage}>
|
2023-12-20 07:45:29 +00:00
|
|
|
<Meta name="descprition" content={description()} />
|
|
|
|
<Meta name="keywords" content={t('topicKeywords', { topic: title() })} />
|
2023-12-13 10:39:31 +00:00
|
|
|
<Meta name="og:type" content="article" />
|
2023-12-20 07:45:29 +00:00
|
|
|
<Meta name="og:title" content={title()} />
|
|
|
|
<Meta name="og:image" content={ogImage()} />
|
|
|
|
<Meta name="twitter:image" content={ogImage()} />
|
|
|
|
<Meta name="og:description" content={description()} />
|
2023-12-13 10:39:31 +00:00
|
|
|
<Meta name="twitter:card" content="summary_large_image" />
|
2023-12-20 07:45:29 +00:00
|
|
|
<Meta name="twitter:title" content={title()} />
|
|
|
|
<Meta name="twitter:description" content={description()} />
|
2023-12-21 12:48:13 +00:00
|
|
|
<FullTopic topic={topic()} />
|
|
|
|
<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>
|
2022-09-09 11:53:35 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2023-12-21 12:48:13 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<Row1 article={sortedArticles()[0]} />
|
|
|
|
<Row2 articles={sortedArticles().slice(1, 3)} isEqual={true} />
|
|
|
|
|
|
|
|
<Beside
|
|
|
|
title={t('Topic is supported by')}
|
2023-12-27 22:50:42 +00:00
|
|
|
values={authorsByTopic()[topic()?.slug]?.slice(0, 6)}
|
2023-12-21 12:48:13 +00:00
|
|
|
beside={sortedArticles()[4]}
|
|
|
|
wrapper={'author'}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<ArticleCardSwiper title={title()} slides={sortedArticles().slice(5, 11)} />
|
|
|
|
|
|
|
|
<Beside
|
|
|
|
beside={sortedArticles()[12]}
|
|
|
|
title={t('Top viewed')}
|
|
|
|
values={sortedArticles().slice(0, 5)}
|
|
|
|
wrapper={'top-article'}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<Row2 articles={sortedArticles().slice(13, 15)} isEqual={true} />
|
|
|
|
<Row1 article={sortedArticles()[15]} />
|
|
|
|
|
|
|
|
<Show when={sortedArticles().length > 15}>
|
|
|
|
<ArticleCardSwiper slides={sortedArticles().slice(16, 22)} />
|
|
|
|
<Row3 articles={sortedArticles().slice(23, 26)} />
|
|
|
|
<Row2 articles={sortedArticles().slice(26, 28)} />
|
|
|
|
</Show>
|
2022-11-07 21:07:42 +00:00
|
|
|
|
2023-12-21 12:48:13 +00:00
|
|
|
<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>
|
2022-09-09 11:53:35 +00:00
|
|
|
</Show>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|