179 lines
6.6 KiB
JavaScript
Raw Normal View History

import React, { useState, useCallback, useEffect } from "react";
import DocumentDetails from "@/components/content/documents/DocumentDetails";
import VideoDetails from "@/components/content/videos/VideoDetails";
2024-08-08 16:29:16 -05:00
import { parseEvent, findKind0Fields } from '@/utils/nostr';
import { nip19 } from 'nostr-tools';
import { useSession } from 'next-auth/react';
import { useNDKContext } from "@/context/NDKContext";
import { useDecryptContent } from "@/hooks/encryption/useDecryptContent";
import { useToast } from "@/hooks/useToast";
import { useRouter } from "next/router";
2024-09-12 12:07:38 -05:00
import { ProgressSpinner } from 'primereact/progressspinner';
import axios from 'axios';
import ZapThreadsWrapper from '@/components/ZapThreadsWrapper';
import { appConfig } from "@/config/appConfig";
2024-08-09 14:28:57 -05:00
const Details = () => {
const [event, setEvent] = useState(null);
2024-03-16 16:37:47 -05:00
const [author, setAuthor] = useState(null);
const [nAddress, setNAddress] = useState(null);
const [decryptedContent, setDecryptedContent] = useState(null);
2024-08-08 16:29:16 -05:00
const [authorView, setAuthorView] = useState(false);
const [loading, setLoading] = useState(true);
const [lessons, setLessons] = useState([]);
2024-11-05 15:13:38 -06:00
const { data: session, update } = useSession();
const { ndk } = useNDKContext();
const { decryptContent } = useDecryptContent();
const router = useRouter();
const { showToast } = useToast();
useEffect(() => {
axios.get('/api/lessons').then(res => {
if (res.data) {
console.log("res.data", res.data);
res.data.forEach(lesson => {
setLessons(prev => [...prev, { resourceId: lesson?.resourceId, courseId: lesson?.courseId || null }]);
});
}
}).catch(err => {
console.log('err', err);
});
}, []);
const fetchAuthor = useCallback(async (pubkey) => {
if (!pubkey) return;
const author = await ndk.getUser({ pubkey });
const profile = await author.fetchProfile();
const fields = await findKind0Fields(profile);
if (fields) {
setAuthor(fields);
}
}, [ndk]);
useEffect(() => {
if (event?.d && !nAddress) {
const naddr = nip19.naddrEncode({
pubkey: event.pubkey,
kind: event.kind,
identifier: event.d,
relayUrls: appConfig.defaultRelayUrls
});
setNAddress(naddr);
}
}, [event, nAddress]);
2024-10-12 17:37:26 -05:00
useEffect(() => {
if (!author && event?.pubkey) {
fetchAuthor(event?.pubkey);
}
}, [author, event, fetchAuthor]);
useEffect(() => {
const fetchAndProcessEvent = async () => {
if (!router.isReady || !router.query.slug) return;
2024-08-12 17:27:47 -05:00
const { slug } = router.query;
let id;
if (slug.includes("naddr")) {
const { data } = nip19.decode(slug);
if (!data) {
showToast('error', 'Error', 'Resource not found');
setLoading(false);
return;
}
id = data?.identifier;
setNAddress(slug);
} else {
id = slug;
}
try {
await ndk.connect();
const event = await ndk.fetchEvent({ ids: [id] });
if (event) {
const parsedEvent = parseEvent(event);
setEvent(parsedEvent);
await fetchAuthor(event.pubkey);
const isAuthor = session?.user?.pubkey === event.pubkey;
setAuthorView(isAuthor);
if (parsedEvent.price || (isAuthor && event.kind === 30402)) {
const shouldDecrypt = isAuthor ||
session?.user?.role?.subscribed ||
session?.user?.purchased?.some(purchase => purchase.resourceId === parsedEvent.d) ||
lessons.some(lesson => lesson.resourceId === parsedEvent.d && session?.user?.purchased?.some(purchase => purchase.courseId === lesson.courseId));
if (shouldDecrypt) {
const decrypted = await decryptContent(event.content);
setDecryptedContent(decrypted);
}
}
}
} catch (error) {
console.error('Error fetching event:', error);
showToast('error', 'Error', 'Failed to fetch event. Please try again.');
} finally {
setLoading(false);
2024-03-16 16:37:47 -05:00
}
};
fetchAndProcessEvent();
}, [router.isReady, router.query, ndk, session, decryptContent, fetchAuthor, showToast]);
const handlePaymentSuccess = (response) => {
if (response && response?.preimage) {
update();
} else {
showToast('error', 'Error', 'Failed to purchase resource. Please try again.');
}
}
const handlePaymentError = (error) => {
showToast('error', 'Payment Error', `Failed to purchase resource. Please try again. Error: ${error}`);
}
if (loading) {
return <div className='w-full h-full flex items-center justify-center mt-24'><ProgressSpinner /></div>
}
if (!author || !event) return null;
const DetailComponent = event.type === "document" ? DocumentDetails : VideoDetails;
return (
<>
<DetailComponent
processedEvent={event}
topics={event.topics}
title={event.title}
summary={event.summary}
image={event.image}
price={event.price}
author={author}
paidResource={!!event.price}
isLesson={lessons.some(lesson => lesson.resourceId === event.d)}
nAddress={nAddress}
decryptedContent={decryptedContent}
handlePaymentSuccess={handlePaymentSuccess}
handlePaymentError={handlePaymentError}
authorView={authorView}
/>
{typeof window !== 'undefined' && nAddress !== null && session?.user?.pubkey && (
<div className='px-4'>
<ZapThreadsWrapper
anchor={nAddress}
user={session?.user?.pubkey || null}
2024-09-12 12:07:38 -05:00
relays="wss://nos.lol/, wss://relay.damus.io/, wss://relay.snort.social/, wss://relay.nostr.band/, wss://relay.mutinywallet.com/, wss://relay.primal.net/"
disable="zaps"
/>
</div>
)}
</>
);
};
export default Details;