mirror of
https://github.com/dergigi/boris.git
synced 2025-12-27 11:34:50 +01:00
feat(ui): resolve nprofile strings to human-readable names
- Add extractNprofilePubkeys utility to parse nprofile strings from content - Create ContentWithResolvedProfiles component using applesauce ProfileModel - Replace nprofile strings with @displayName in bookmark content - Update BookmarkItem to use resolved content rendering - Improves readability by showing names instead of long nprofile strings
This commit is contained in:
@@ -1,10 +1,77 @@
|
||||
import React from 'react'
|
||||
import { ParsedContent, ParsedNode } from '../types/bookmarks'
|
||||
import { decode } from 'nostr-tools/nip19'
|
||||
import { getPubkeyFromDecodeResult } from 'applesauce-core/helpers'
|
||||
import { useEventModel } from 'applesauce-react/hooks'
|
||||
import { Models } from 'applesauce-core'
|
||||
|
||||
export const formatDate = (timestamp: number) => {
|
||||
return new Date(timestamp * 1000).toLocaleDateString()
|
||||
}
|
||||
|
||||
// Extract pubkeys from nprofile strings in content
|
||||
export const extractNprofilePubkeys = (content: string): string[] => {
|
||||
const nprofileRegex = /nprofile1[a-z0-9]+/gi
|
||||
const matches = content.match(nprofileRegex) || []
|
||||
const pubkeys: string[] = []
|
||||
|
||||
for (const match of matches) {
|
||||
try {
|
||||
const decoded = decode(match)
|
||||
const pubkey = getPubkeyFromDecodeResult(decoded)
|
||||
if (pubkey && !pubkeys.includes(pubkey)) {
|
||||
pubkeys.push(pubkey)
|
||||
}
|
||||
} catch (error) {
|
||||
// Invalid nprofile string, skip
|
||||
console.warn('Failed to decode nprofile:', match, error)
|
||||
}
|
||||
}
|
||||
|
||||
return pubkeys
|
||||
}
|
||||
|
||||
// Component to render content with resolved nprofile names
|
||||
export const ContentWithResolvedProfiles: React.FC<{ content: string }> = ({ content }) => {
|
||||
const nprofilePubkeys = extractNprofilePubkeys(content)
|
||||
|
||||
// Create individual profile hooks for each pubkey
|
||||
const profiles = nprofilePubkeys.map(pubkey => {
|
||||
// eslint-disable-next-line react-hooks/rules-of-hooks
|
||||
const profile = useEventModel(Models.ProfileModel, [pubkey])
|
||||
return { pubkey, profile }
|
||||
})
|
||||
|
||||
// Replace nprofile strings with resolved names
|
||||
const renderContent = () => {
|
||||
let renderedContent = content
|
||||
|
||||
profiles.forEach(({ pubkey, profile }) => {
|
||||
const displayName = profile?.name || profile?.display_name || profile?.nip05 || `${pubkey.slice(0, 8)}...`
|
||||
|
||||
// Replace all instances of this nprofile with the display name
|
||||
const nprofileRegex = new RegExp(`nprofile1[a-z0-9]+`, 'gi')
|
||||
const matches = content.match(nprofileRegex) || []
|
||||
|
||||
matches.forEach(match => {
|
||||
try {
|
||||
const decoded = decode(match)
|
||||
const matchPubkey = getPubkeyFromDecodeResult(decoded)
|
||||
if (matchPubkey === pubkey) {
|
||||
renderedContent = renderedContent.replace(match, `@${displayName}`)
|
||||
}
|
||||
} catch (error) {
|
||||
// Skip invalid nprofile
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
return renderedContent
|
||||
}
|
||||
|
||||
return <div className="bookmark-content">{renderContent()}</div>
|
||||
}
|
||||
|
||||
// Component to render parsed content using applesauce-content
|
||||
export const renderParsedContent = (parsedContent: ParsedContent) => {
|
||||
if (!parsedContent || !parsedContent.children) {
|
||||
|
||||
Reference in New Issue
Block a user