mirror of
https://github.com/dergigi/boris.git
synced 2026-02-16 20:45:01 +01:00
116 lines
3.7 KiB
TypeScript
116 lines
3.7 KiB
TypeScript
import React, { useState, useEffect, useRef } from 'react'
|
|
import { RelayPool } from 'applesauce-relay'
|
|
import { extractNaddrUris, replaceNostrUrisInMarkdownWithProfileLabels } from '../utils/nostrUriResolver'
|
|
import { fetchArticleTitles } from '../services/articleTitleResolver'
|
|
import { useProfileLabels } from './useProfileLabels'
|
|
|
|
/**
|
|
* Hook to convert markdown to HTML using a hidden ReactMarkdown component
|
|
* Also processes nostr: URIs in the markdown and resolves article titles
|
|
*/
|
|
export const useMarkdownToHTML = (
|
|
markdown?: string,
|
|
relayPool?: RelayPool | null
|
|
): {
|
|
renderedHtml: string
|
|
previewRef: React.RefObject<HTMLDivElement>
|
|
processedMarkdown: string
|
|
} => {
|
|
const previewRef = useRef<HTMLDivElement>(null)
|
|
const [renderedHtml, setRenderedHtml] = useState<string>('')
|
|
const [processedMarkdown, setProcessedMarkdown] = useState<string>('')
|
|
const [articleTitles, setArticleTitles] = useState<Map<string, string>>(new Map())
|
|
|
|
// Resolve profile labels progressively as profiles load
|
|
const { labels: profileLabels, loading: profileLoading } = useProfileLabels(markdown || '', relayPool)
|
|
|
|
// Fetch article titles
|
|
useEffect(() => {
|
|
if (!markdown || !relayPool) {
|
|
setArticleTitles(new Map())
|
|
return
|
|
}
|
|
|
|
let isCancelled = false
|
|
|
|
const fetchTitles = async () => {
|
|
const naddrs = extractNaddrUris(markdown)
|
|
if (naddrs.length === 0) {
|
|
setArticleTitles(new Map())
|
|
return
|
|
}
|
|
|
|
try {
|
|
const titlesMap = await fetchArticleTitles(relayPool!, naddrs)
|
|
if (!isCancelled) {
|
|
setArticleTitles(titlesMap)
|
|
}
|
|
} catch {
|
|
if (!isCancelled) setArticleTitles(new Map())
|
|
}
|
|
}
|
|
|
|
fetchTitles()
|
|
return () => { isCancelled = true }
|
|
}, [markdown, relayPool])
|
|
|
|
// Process markdown with progressive profile labels and article titles
|
|
useEffect(() => {
|
|
console.log(`[markdown-to-html] Processing markdown, profileLabels=${profileLabels.size}, profileLoading=${profileLoading.size}, articleTitles=${articleTitles.size}`)
|
|
console.log(`[markdown-to-html] Clearing rendered HTML and processed markdown`)
|
|
// Always clear previous render immediately to avoid showing stale content while processing
|
|
setRenderedHtml('')
|
|
setProcessedMarkdown('')
|
|
|
|
if (!markdown) {
|
|
return
|
|
}
|
|
|
|
let isCancelled = false
|
|
|
|
const processMarkdown = () => {
|
|
try {
|
|
// Replace nostr URIs with profile labels (progressive) and article titles
|
|
const processed = replaceNostrUrisInMarkdownWithProfileLabels(
|
|
markdown,
|
|
profileLabels,
|
|
articleTitles,
|
|
profileLoading
|
|
)
|
|
|
|
if (isCancelled) return
|
|
|
|
console.log(`[markdown-to-html] Processed markdown, loading states:`, Array.from(profileLoading.entries()).filter(([, l]) => l).map(([e]) => e.slice(0, 16) + '...'))
|
|
setProcessedMarkdown(processed)
|
|
} catch (error) {
|
|
console.error(`[markdown-to-html] Error processing markdown:`, error)
|
|
if (!isCancelled) {
|
|
setProcessedMarkdown(markdown) // Fallback to original
|
|
}
|
|
}
|
|
|
|
const rafId = requestAnimationFrame(() => {
|
|
if (previewRef.current && !isCancelled) {
|
|
const html = previewRef.current.innerHTML
|
|
setRenderedHtml(html)
|
|
} else if (!isCancelled) {
|
|
console.warn('⚠️ markdownPreviewRef.current is null')
|
|
}
|
|
})
|
|
|
|
return () => cancelAnimationFrame(rafId)
|
|
}
|
|
|
|
processMarkdown()
|
|
|
|
return () => {
|
|
isCancelled = true
|
|
}
|
|
}, [markdown, profileLabels, profileLoading, articleTitles])
|
|
|
|
return { renderedHtml, previewRef, processedMarkdown }
|
|
}
|
|
|
|
// Removed separate useMarkdownPreviewRef; use useMarkdownToHTML to obtain previewRef
|
|
|