source: chapterx-frontend/src/pages/story/StoryDetailPage.tsx@ 73b69b2

main
Last change on this file since 73b69b2 was 73b69b2, checked in by kikisrbinoska <srbinoskakristina07@…>, 3 months ago

Fixed reading lists,comments and likes

  • Property mode set to 100644
File size: 12.5 KB
Line 
1import React, { useState } from 'react'
2import { useParams, useNavigate } from 'react-router-dom'
3import { ArrowLeft, BookOpen, Eye, Users, Calendar, Plus, BookmarkPlus } from 'lucide-react'
4import { useStoryStore } from '../../store/storyStore'
5import { useAuthStore } from '../../store/authStore'
6import { useUIStore } from '../../store/uiStore'
7import { Button } from '../../components/ui/Button'
8import { GenreBadge } from '../../components/ui/Badge'
9import { Avatar } from '../../components/ui/Avatar'
10import { ChapterList } from '../../components/story/ChapterList'
11import { CommentSection } from '../../components/story/CommentSection'
12import { LikeButton } from '../../components/story/LikeButton'
13import { Modal } from '../../components/ui/Modal'
14import { getGenreGradient } from '../../components/story/GenreBadge'
15import { ReadingListItem } from '../../types'
16
17export const StoryDetailPage: React.FC = () => {
18 const { id } = useParams<{ id: string }>()
19 const navigate = useNavigate()
20 const { stories, chapters, collaborations, readingLists, addStoryToList, createReadingList } = useStoryStore()
21 const { currentUser } = useAuthStore()
22 const { addToast } = useUIStore()
23 const [listModalOpen, setListModalOpen] = useState(false)
24 const [newListName, setNewListName] = useState('')
25 const [liveLikes, setLiveLikes] = useState<number | null>(null)
26 const [liveComments, setLiveComments] = useState<number | null>(null)
27
28 const story = stories.find(s => s.story_id === Number(id))
29 if (!story) {
30 return (
31 <div className="max-w-4xl mx-auto px-4 py-20 text-center">
32 <h2 className="text-2xl text-white mb-4">Story not found</h2>
33 <Button onClick={() => navigate('/browse')}>Browse Stories</Button>
34 </div>
35 )
36 }
37
38 const storyChapters = chapters.filter(c => c.story_id === story.story_id)
39 const storyCollabs = collaborations.filter(c => c.story_id === story.story_id)
40 const gradient = getGenreGradient(story.genres[0])
41 const myLists = currentUser ? readingLists.filter(l => l.user_id === currentUser.user_id) : []
42
43 const handleAddToList = async (listId: number) => {
44 const list = readingLists.find(l => l.list_id === listId)
45 if (!list) return
46 if (list.stories.some(s => s.story_id === story.story_id)) {
47 addToast('Already in this list', 'info')
48 return
49 }
50 const item: ReadingListItem = {
51 item_id: Date.now(),
52 list_id: listId,
53 story_id: story.story_id,
54 story_title: story.title,
55 author_username: story.author_username,
56 added_at: new Date().toISOString(),
57 genres: story.genres,
58 }
59 try {
60 await addStoryToList(listId, item)
61 addToast(`Added to "${list.name}"!`)
62 } catch (err: any) {
63 const msg = err?.response?.data?.message || ''
64 if (msg.includes('already') || msg.includes('duplicate') || err?.response?.status === 400) {
65 addToast('Already in this list', 'info')
66 } else {
67 addToast('Failed to add to list.', 'error')
68 }
69 }
70 setListModalOpen(false)
71 }
72
73 const handleCreateList = async () => {
74 if (!newListName.trim() || !currentUser) return
75 const newList = {
76 list_id: Date.now(),
77 user_id: currentUser.user_id,
78 username: currentUser.username,
79 name: newListName.trim(),
80 is_public: false,
81 created_at: new Date().toISOString(),
82 stories: [],
83 }
84 try {
85 const realListId = await createReadingList(newList)
86 await addStoryToList(realListId, {
87 item_id: Date.now() + 1,
88 list_id: realListId,
89 story_id: story.story_id,
90 story_title: story.title,
91 author_username: story.author_username,
92 added_at: new Date().toISOString(),
93 genres: story.genres,
94 })
95 addToast(`Created "${newListName}" and added story!`)
96 } catch {
97 addToast('Failed to create list.', 'error')
98 }
99 setNewListName('')
100 setListModalOpen(false)
101 }
102
103 const isOwner = currentUser?.user_id === story.user_id
104
105 return (
106 <div className="max-w-5xl mx-auto px-4 py-8">
107 {/* Back */}
108 <button
109 onClick={() => navigate(-1)}
110 className="flex items-center gap-2 text-slate-400 hover:text-white mb-6 text-sm transition-colors"
111 >
112 <ArrowLeft size={16} />
113 Back
114 </button>
115
116 {/* Hero */}
117 <div className={`relative rounded-2xl overflow-hidden mb-8 bg-gradient-to-br ${gradient}`}>
118 <div className="absolute inset-0 bg-gradient-to-t from-slate-950/90 via-slate-950/30 to-transparent" />
119 <div className="relative p-8 sm:p-12">
120 <div className="flex flex-wrap gap-2 mb-4">
121 {story.genres.map(g => <GenreBadge key={g} genre={g} />)}
122 {story.mature_content && (
123 <span className="px-2 py-0.5 text-xs font-medium rounded-full border bg-rose-500/20 text-rose-400 border-rose-500/30">
124 18+
125 </span>
126 )}
127 </div>
128 <h1 className="font-serif text-3xl sm:text-4xl font-bold text-white mb-3">{story.title}</h1>
129 <p className="text-slate-300 text-lg mb-6 leading-relaxed max-w-2xl">{story.short_description}</p>
130
131 <div className="flex items-center gap-4 flex-wrap">
132 <div className="flex items-center gap-2">
133 <Avatar name={story.author_username} size="sm" />
134 <span className="text-white text-sm font-medium">{story.author_username}</span>
135 </div>
136 <div className="flex items-center gap-1 text-slate-400 text-sm">
137 <Eye size={14} />
138 {story.total_views.toLocaleString()} views
139 </div>
140 <div className="flex items-center gap-1 text-slate-400 text-sm">
141 <BookOpen size={14} />
142 {story.total_chapters} chapters
143 </div>
144 <div className="flex items-center gap-1 text-slate-400 text-sm">
145 <Calendar size={14} />
146 {new Date(story.created_at).toLocaleDateString('en-US', { month: 'long', year: 'numeric' })}
147 </div>
148 </div>
149 </div>
150 </div>
151
152 <div className="grid grid-cols-1 lg:grid-cols-3 gap-8">
153 {/* Main content */}
154 <div className="lg:col-span-2 space-y-8">
155 {/* Action bar */}
156 <div className="flex items-center gap-3 flex-wrap">
157 <LikeButton storyId={story.story_id} authorUserId={story.user_id} totalLikes={story.total_likes} onCountChange={setLiveLikes} />
158 {currentUser && (
159 <Button variant="secondary" size="sm" onClick={() => setListModalOpen(true)}>
160 <BookmarkPlus size={14} />
161 Save to List
162 </Button>
163 )}
164 {isOwner && (
165 <Button variant="ghost" size="sm" onClick={() => navigate(`/writer/edit-story/${story.story_id}`)}>
166 Edit Story
167 </Button>
168 )}
169 </div>
170
171 {/* Chapters */}
172 <div>
173 <div className="flex items-center justify-between mb-4">
174 <h2 className="font-serif text-xl font-bold text-white">Chapters</h2>
175 {isOwner && (
176 <Button size="sm" onClick={() => navigate(`/writer/create-chapter/${story.story_id}`)}>
177 <Plus size={14} />
178 Add Chapter
179 </Button>
180 )}
181 </div>
182 <ChapterList chapters={storyChapters} storyId={story.story_id} />
183 </div>
184
185 {/* Comments */}
186 <div className="border-t border-slate-700 pt-8">
187 <CommentSection storyId={story.story_id} authorUserId={story.user_id} onCountChange={setLiveComments} />
188 </div>
189 </div>
190
191 {/* Sidebar */}
192 <div className="space-y-6">
193 {/* Story info */}
194 <div className="bg-slate-800 border border-slate-700 rounded-2xl p-5">
195 <h3 className="text-white font-semibold mb-4">Story Info</h3>
196 <div className="space-y-3 text-sm">
197 <div className="flex justify-between">
198 <span className="text-slate-400">Status</span>
199 <span className="text-white capitalize">{story.status}</span>
200 </div>
201 <div className="flex justify-between">
202 <span className="text-slate-400">Chapters</span>
203 <span className="text-white">{story.total_chapters}</span>
204 </div>
205 <div className="flex justify-between">
206 <span className="text-slate-400">Likes</span>
207 <span className="text-white">{(liveLikes ?? story.total_likes).toLocaleString()}</span>
208 </div>
209 <div className="flex justify-between">
210 <span className="text-slate-400">Views</span>
211 <span className="text-white">{story.total_views.toLocaleString()}</span>
212 </div>
213 <div className="flex justify-between">
214 <span className="text-slate-400">Comments</span>
215 <span className="text-white">{liveComments ?? story.total_comments}</span>
216 </div>
217 <div className="flex justify-between">
218 <span className="text-slate-400">Published</span>
219 <span className="text-white">{new Date(story.created_at).toLocaleDateString()}</span>
220 </div>
221 </div>
222 </div>
223
224 {/* Collaborators */}
225 {storyCollabs.length > 0 && (
226 <div className="bg-slate-800 border border-slate-700 rounded-2xl p-5">
227 <div className="flex items-center gap-2 mb-4">
228 <Users size={16} className="text-violet-400" />
229 <h3 className="text-white font-semibold">Collaborators</h3>
230 </div>
231 <div className="space-y-3">
232 {storyCollabs.map(c => (
233 <div key={c.collab_id} className="flex items-center gap-2">
234 <Avatar name={c.name} size="sm" />
235 <div>
236 <p className="text-white text-sm">@{c.username}</p>
237 <p className="text-slate-500 text-xs">{c.role}</p>
238 </div>
239 </div>
240 ))}
241 </div>
242 </div>
243 )}
244 </div>
245 </div>
246
247 {/* Reading list modal */}
248 <Modal isOpen={listModalOpen} onClose={() => setListModalOpen(false)} title="Save to Reading List">
249 <div className="space-y-3">
250 {myLists.length > 0 ? (
251 <>
252 <p className="text-slate-400 text-sm mb-3">Select a list:</p>
253 {myLists.map(list => (
254 <button
255 key={list.list_id}
256 onClick={() => handleAddToList(list.list_id)}
257 className="w-full flex items-center justify-between p-3 bg-slate-800 rounded-xl border border-slate-700 hover:border-indigo-500/50 transition-colors"
258 >
259 <span className="text-white text-sm">{list.name}</span>
260 <span className="text-slate-500 text-xs">{list.stories.length} stories</span>
261 </button>
262 ))}
263 <div className="border-t border-slate-700 pt-3">
264 <p className="text-slate-400 text-sm mb-2">Or create a new list:</p>
265 <div className="flex gap-2">
266 <input
267 value={newListName}
268 onChange={e => setNewListName(e.target.value)}
269 placeholder="List name..."
270 className="flex-1 px-3 py-2 bg-slate-800 border border-slate-700 rounded-xl text-white text-sm placeholder-slate-500 focus:outline-none focus:border-indigo-500"
271 />
272 <Button size="sm" onClick={handleCreateList} disabled={!newListName.trim()}>
273 Create
274 </Button>
275 </div>
276 </div>
277 </>
278 ) : (
279 <div>
280 <p className="text-slate-400 text-sm mb-3">You don't have any reading lists yet. Create one:</p>
281 <div className="flex gap-2">
282 <input
283 value={newListName}
284 onChange={e => setNewListName(e.target.value)}
285 placeholder="My Favorites..."
286 className="flex-1 px-3 py-2 bg-slate-800 border border-slate-700 rounded-xl text-white text-sm placeholder-slate-500 focus:outline-none focus:border-indigo-500"
287 />
288 <Button size="sm" onClick={handleCreateList} disabled={!newListName.trim()}>
289 Create
290 </Button>
291 </div>
292 </div>
293 )}
294 </div>
295 </Modal>
296 </div>
297 )
298}
Note: See TracBrowser for help on using the repository browser.