Nicolas Cantu 2a191f35f4 Fix all TypeScript errors and warnings
- Fix unused function warnings by renaming to _unusedExtractTags
- Fix type errors in nostrTagSystem.ts for includes() calls
- Fix type errors in reviews.ts for filter kinds array
- Fix ArrayBuffer type errors in articleEncryption.ts
- Remove unused imports (DecryptionKey, decryptArticleContent, extractTagsFromEvent)
- All TypeScript checks now pass without disabling any controls
2025-12-27 22:26:13 +01:00

94 lines
2.6 KiB
TypeScript

import { useEffect, useMemo, useRef, useState } from 'react'
import { nostrService } from '@/lib/nostr'
import type { Article } from '@/types/nostr'
import { applyFiltersAndSort } from '@/lib/articleFiltering'
import type { ArticleFilters } from '@/components/ArticleFilters'
export function useArticles(searchQuery: string = '', filters: ArticleFilters | null = null) {
const [articles, setArticles] = useState<Article[]>([])
const [loading, setLoading] = useState(true)
const [error, setError] = useState<string | null>(null)
const hasArticlesRef = useRef(false)
useEffect(() => {
setLoading(true)
setError(null)
const unsubscribe = nostrService.subscribeToArticles(
(article) => {
setArticles((prev) => {
if (prev.some((a) => a.id === article.id)) {
return prev
}
const next = [article, ...prev].sort((a, b) => b.createdAt - a.createdAt)
hasArticlesRef.current = next.length > 0
return next
})
setLoading(false)
},
50
)
const timeout = setTimeout(() => {
setLoading(false)
if (!hasArticlesRef.current) {
setError('No articles found')
}
}, 10000)
return () => {
unsubscribe()
clearTimeout(timeout)
}
}, [])
const loadArticleContent = async (articleId: string, authorPubkey: string) => {
try {
const article = await nostrService.getArticleById(articleId)
if (article) {
// Try to decrypt article content using decryption key from private messages
const decryptedContent = await nostrService.getDecryptedArticleContent(articleId, authorPubkey)
if (decryptedContent) {
setArticles((prev) =>
prev.map((a) =>
a.id === articleId
? { ...a, content: decryptedContent, paid: true }
: a
)
)
}
return article
}
} catch (e) {
console.error('Error loading article content:', e)
setError(e instanceof Error ? e.message : 'Failed to load article')
}
return null
}
// Apply filters and sorting
const filteredArticles = useMemo(() => {
const effectiveFilters =
filters ??
({
authorPubkey: null,
sortBy: 'newest',
category: 'all',
} as const)
if (!filters && !searchQuery.trim()) {
return articles
}
return applyFiltersAndSort(articles, searchQuery, effectiveFilters)
}, [articles, searchQuery, filters])
return {
articles: filteredArticles,
allArticles: articles, // Return all articles for filters component
loading,
error,
loadArticleContent,
}
}