84 lines
2.2 KiB
TypeScript
84 lines
2.2 KiB
TypeScript
import type { Article } from '@/types/nostr'
|
|
import { ArticleCard } from './ArticleCard'
|
|
import { ErrorState, EmptyState, Skeleton } from './ui'
|
|
import { t } from '@/lib/i18n'
|
|
|
|
interface ArticlesListProps {
|
|
articles: Article[]
|
|
allArticles: Article[]
|
|
loading: boolean
|
|
error: string | null
|
|
onUnlock: (article: Article) => void
|
|
unlockedArticles: Set<string>
|
|
}
|
|
|
|
function ArticleCardSkeleton(): React.ReactElement {
|
|
return (
|
|
<div className="border border-neon-cyan/30 rounded-lg p-6 bg-cyber-dark space-y-4">
|
|
<div className="space-y-2">
|
|
<Skeleton variant="rectangular" height={24} className="w-3/4" />
|
|
<Skeleton variant="rectangular" height={16} className="w-1/2" />
|
|
</div>
|
|
<Skeleton variant="rectangular" height={100} className="w-full" />
|
|
<div className="flex items-center justify-between">
|
|
<Skeleton variant="circular" width={32} height={32} />
|
|
<Skeleton variant="rectangular" height={36} className="w-32" />
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function LoadingState(): React.ReactElement {
|
|
return (
|
|
<div className="space-y-6">
|
|
{Array.from({ length: 3 }).map((_, index) => (
|
|
<ArticleCardSkeleton key={index} />
|
|
))}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function ArticlesEmptyState({ hasAny }: { hasAny: boolean }): React.ReactElement {
|
|
return (
|
|
<EmptyState
|
|
title={hasAny ? t('common.empty.articles.filtered') : t('common.empty.articles')}
|
|
/>
|
|
)
|
|
}
|
|
|
|
export function ArticlesList({
|
|
articles,
|
|
allArticles,
|
|
loading,
|
|
error,
|
|
onUnlock,
|
|
unlockedArticles,
|
|
}: ArticlesListProps): React.ReactElement {
|
|
if (loading) {
|
|
return <LoadingState />
|
|
}
|
|
if (error) {
|
|
return <ErrorState message={error} />
|
|
}
|
|
if (articles.length === 0) {
|
|
return <ArticlesEmptyState hasAny={allArticles.length > 0} />
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className="mb-4 text-sm text-cyber-accent/70">
|
|
Showing {articles.length} of {allArticles.length} article{allArticles.length !== 1 ? 's' : ''}
|
|
</div>
|
|
<div className="space-y-6">
|
|
{articles.map((article) => (
|
|
<ArticleCard
|
|
key={article.id}
|
|
article={{ ...article, paid: unlockedArticles.has(article.id) || article.paid }}
|
|
onUnlock={onUnlock}
|
|
/>
|
|
))}
|
|
</div>
|
|
</>
|
|
)
|
|
}
|