273 lines
10 KiB
JavaScript
Raw Normal View History

import React, { useEffect, useState } from 'react';
2024-08-08 16:29:16 -05:00
import axios from 'axios';
import { useRouter } from 'next/router';
2024-08-08 16:29:16 -05:00
import { parseEvent, findKind0Fields } from '@/utils/nostr';
import { Button } from 'primereact/button';
import { nip19, nip04 } from 'nostr-tools';
import { useSession } from 'next-auth/react';
import dynamic from 'next/dynamic';
import ZapThreadsWrapper from '@/components/ZapThreadsWrapper';
2024-08-08 16:29:16 -05:00
import { useToast } from '@/hooks/useToast';
import { useNDKContext } from '@/context/NDKContext';
import ResourceDetails from '@/components/content/resources/ResourceDetails';
import {ProgressSpinner} from 'primereact/progressspinner';
import 'primeicons/primeicons.css';
2024-08-09 14:28:57 -05:00
2024-07-21 19:56:55 -05:00
const MDDisplay = dynamic(
() => import("@uiw/react-markdown-preview"),
{
ssr: false,
}
);
const privkey = process.env.NEXT_PUBLIC_APP_PRIV_KEY;
2024-08-09 14:28:57 -05:00
const pubkey = process.env.NEXT_PUBLIC_APP_PUBLIC_KEY;
export default function Details() {
const [event, setEvent] = useState(null);
const [processedEvent, setProcessedEvent] = useState({});
2024-03-16 16:37:47 -05:00
const [author, setAuthor] = useState(null);
const [nAddress, setNAddress] = useState(null);
const [paidResource, setPaidResource] = useState(false);
const [decryptedContent, setDecryptedContent] = useState(null);
2024-08-08 16:29:16 -05:00
const [authorView, setAuthorView] = useState(false);
const [loading, setLoading] = useState(true);
const [error, setError] = useState(null);
const {ndk, addSigner} = useNDKContext();
2024-08-12 17:27:47 -05:00
const { data: session, update } = useSession();
const [user, setUser] = useState(null);
2024-08-08 16:29:16 -05:00
const { showToast } = useToast();
const router = useRouter();
useEffect(() => {
if (session) {
setUser(session.user);
}
}, [session]);
useEffect(() => {
if (processedEvent.price) {
setPaidResource(true);
}
}, [processedEvent]);
useEffect(() => {
const decryptContent = async () => {
if (user && paidResource) {
2024-08-12 17:27:47 -05:00
if (user?.purchased?.length > 0) {
const purchasedResource = user?.purchased.find(purchase => purchase.resourceId === processedEvent.d);
if (purchasedResource) {
console.log("purchasedResource", purchasedResource)
const decryptedContent = await nip04.decrypt(privkey, pubkey, processedEvent.content);
setDecryptedContent(decryptedContent);
}
} else if (user?.role && user?.role.subscribed) {
2024-08-09 14:28:57 -05:00
// decrypt the content
const decryptedContent = await nip04.decrypt(privkey, pubkey, processedEvent.content);
setDecryptedContent(decryptedContent);
}
}
2024-08-12 17:27:47 -05:00
}
decryptContent();
}, [user, paidResource, processedEvent]);
useEffect(() => {
if (router.isReady) {
const { slug } = router.query;
const fetchEvent = async (slug, retryCount = 0) => {
setLoading(true);
setError(null);
try {
await ndk.connect();
const filter = {
ids: [slug]
}
const event = await ndk.fetchEvent(filter);
if (event) {
setEvent(event);
2024-08-08 16:29:16 -05:00
if (user && user.pubkey === event.pubkey) {
const decryptedContent = await nip04.decrypt(privkey, pubkey, event.content);
setDecryptedContent(decryptedContent);
2024-08-08 16:29:16 -05:00
setAuthorView(true);
}
} else {
if (retryCount < 1) {
// Wait for 2 seconds before retrying
await new Promise(resolve => setTimeout(resolve, 3000));
return fetchEvent(slug, retryCount + 1);
} else {
setError("Event not found");
}
}
} catch (error) {
console.error('Error fetching event:', error);
if (retryCount < 1) {
// Wait for 2 seconds before retrying
await new Promise(resolve => setTimeout(resolve, 3000));
return fetchEvent(slug, retryCount + 1);
} else {
setError("Failed to fetch event. Please try again.");
}
} finally {
setLoading(false);
}
};
if (ndk) {
fetchEvent(slug);
}
}
2024-08-08 16:29:16 -05:00
}, [router.isReady, router.query, ndk, user]);
2024-03-16 16:37:47 -05:00
useEffect(() => {
const fetchAuthor = async (pubkey) => {
try {
await ndk.connect();
const filter = {
kinds: [0],
authors: [pubkey]
}
const author = await ndk.fetchEvent(filter);
if (author) {
const fields = await findKind0Fields(JSON.parse(author.content));
console.log("fields", fields);
setAuthor(fields);
}
} catch (error) {
console.error('Error fetching author:', error);
2024-03-16 16:37:47 -05:00
}
}
if (event && ndk) {
2024-03-16 16:37:47 -05:00
fetchAuthor(event.pubkey);
}
}, [ndk, event]);
2024-03-16 16:37:47 -05:00
useEffect(() => {
if (event) {
const parsedEvent = parseEvent(event);
setProcessedEvent(parsedEvent);
}
}, [event]);
useEffect(() => {
if (processedEvent?.d) {
const naddr = nip19.naddrEncode({
pubkey: processedEvent.pubkey,
kind: processedEvent.kind,
identifier: processedEvent.d,
});
setNAddress(naddr);
}
}, [processedEvent]);
2024-08-08 16:29:16 -05:00
const handleDelete = async () => {
try {
2024-08-09 14:28:57 -05:00
const response = await axios.delete(`/api/resources/${processedEvent.d}`);
2024-08-08 16:29:16 -05:00
if (response.status === 204) {
showToast('success', 'Success', 'Resource deleted successfully.');
router.push('/');
}
} catch (error) {
if (error.response && error.response.data && error.response.data.error.includes("Invalid `prisma.resource.delete()`")) {
showToast('error', 'Error', 'Resource cannot be deleted because it is part of a course, delete the course first.');
}
else if (error.response && error.response.data && error.response.data.error) {
showToast('error', 'Error', error.response.data.error);
} else {
showToast('error', 'Error', 'Failed to delete resource. Please try again.');
}
}
}
const handlePaymentSuccess = async (response, newResource) => {
if (response && response?.preimage) {
console.log("newResource", newResource);
const updated = await update();
console.log("session after update", updated);
} 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}`);
}
const renderContent = () => {
if (decryptedContent) {
2024-09-03 17:02:24 -05:00
return <MDDisplay className='p-4 rounded-lg' source={decryptedContent} />;
}
if (paidResource && !decryptedContent) {
return <p className="text-center text-xl text-red-500">This content is paid and needs to be purchased before viewing.</p>;
}
if (processedEvent?.content) {
2024-09-03 17:02:24 -05:00
return <MDDisplay className='p-4 rounded-lg' source={processedEvent.content} />;
}
return null;
}
if (loading) {
return <div className="mx-auto">
<ProgressSpinner />
</div>;
}
if (error) {
return <div className="w-full mx-auto h-screen">
<div className="text-red-500 text-xl">{error}</div>
</div>;
}
return (
2024-03-19 13:04:46 -05:00
<div className='w-full px-24 pt-12 mx-auto mt-4 max-tab:px-0 max-mob:px-0 max-tab:pt-2 max-mob:pt-2'>
{processedEvent && (
<ResourceDetails
processedEvent={processedEvent}
topics={processedEvent.topics}
title={processedEvent.title}
summary={processedEvent.summary}
image={processedEvent.image}
price={processedEvent.price}
author={author}
paidResource={paidResource}
decryptedContent={decryptedContent}
handlePaymentSuccess={handlePaymentSuccess}
handlePaymentError={handlePaymentError}
/>
)}
2024-08-08 16:29:16 -05:00
{authorView && (
<div className='w-[75vw] mx-auto flex flex-row justify-end mt-12'>
<div className='w-fit flex flex-row justify-between'>
<Button onClick={() => router.push(`/details/${processedEvent.id}/edit`)} label="Edit" severity='warning' outlined className="w-auto m-2" />
<Button onClick={handleDelete} label="Delete" severity='danger' outlined className="w-auto m-2 mr-0" />
2024-08-08 16:29:16 -05:00
</div>
</div>
)}
{typeof window !== 'undefined' && nAddress !== null && (
<div className='px-24'>
<ZapThreadsWrapper
anchor={nAddress}
user={user?.pubkey || null}
relays="wss://nos.lol/, wss://relay.damus.io/, wss://relay.snort.social/, wss://relay.nostr.band/, wss://nostr.mutinywallet.com/, wss://relay.mutinywallet.com/, wss://relay.primal.net/"
disable=""
/>
</div>
)}
<div className='w-[75vw] mx-auto mt-12 p-12 border-t-2 border-gray-300 max-tab:p-0 max-mob:p-0 max-tab:max-w-[100vw] max-mob:max-w-[100vw]'>
{
processedEvent && processedEvent.content && renderContent()
}
</div>
</div>
);
2024-08-09 14:28:57 -05:00
}