source: chapterx-frontend/src/pages/story/StoryDetailPage.tsx@ 7fbb91c

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

Added functional collaboration between users

  • Property mode set to 100644
File size: 12.6 KB
RevLine 
[b62cefc]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('')
[73b69b2]25 const [liveLikes, setLiveLikes] = useState<number | null>(null)
26 const [liveComments, setLiveComments] = useState<number | null>(null)
[b62cefc]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
[acf690c]43 const handleAddToList = async (listId: number) => {
[b62cefc]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 }
[acf690c]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 }
[b62cefc]70 setListModalOpen(false)
71 }
72
[acf690c]73 const handleCreateList = async () => {
[b62cefc]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(),
[acf690c]82 stories: [],
83 }
84 try {
85 const realListId = await createReadingList(newList)
86 await addStoryToList(realListId, {
[b62cefc]87 item_id: Date.now() + 1,
[acf690c]88 list_id: realListId,
[b62cefc]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,
[acf690c]94 })
95 addToast(`Created "${newListName}" and added story!`)
96 } catch {
97 addToast('Failed to create list.', 'error')
[b62cefc]98 }
99 setNewListName('')
100 setListModalOpen(false)
101 }
102
103 const isOwner = currentUser?.user_id === story.user_id
[7fbb91c]104 const isCollaborator = currentUser ? storyCollabs.some(c => c.user_id === currentUser.user_id) : false
[b62cefc]105
106 return (
107 <div className="max-w-5xl mx-auto px-4 py-8">
108 {/* Back */}
109 <button
110 onClick={() => navigate(-1)}
111 className="flex items-center gap-2 text-slate-400 hover:text-white mb-6 text-sm transition-colors"
112 >
113 <ArrowLeft size={16} />
114 Back
115 </button>
116
117 {/* Hero */}
118 <div className={`relative rounded-2xl overflow-hidden mb-8 bg-gradient-to-br ${gradient}`}>
119 <div className="absolute inset-0 bg-gradient-to-t from-slate-950/90 via-slate-950/30 to-transparent" />
120 <div className="relative p-8 sm:p-12">
121 <div className="flex flex-wrap gap-2 mb-4">
122 {story.genres.map(g => <GenreBadge key={g} genre={g} />)}
123 {story.mature_content && (
124 <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">
125 18+
126 </span>
127 )}
128 </div>
129 <h1 className="font-serif text-3xl sm:text-4xl font-bold text-white mb-3">{story.title}</h1>
130 <p className="text-slate-300 text-lg mb-6 leading-relaxed max-w-2xl">{story.short_description}</p>
131
132 <div className="flex items-center gap-4 flex-wrap">
133 <div className="flex items-center gap-2">
134 <Avatar name={story.author_username} size="sm" />
135 <span className="text-white text-sm font-medium">{story.author_username}</span>
136 </div>
137 <div className="flex items-center gap-1 text-slate-400 text-sm">
138 <Eye size={14} />
139 {story.total_views.toLocaleString()} views
140 </div>
141 <div className="flex items-center gap-1 text-slate-400 text-sm">
142 <BookOpen size={14} />
143 {story.total_chapters} chapters
144 </div>
145 <div className="flex items-center gap-1 text-slate-400 text-sm">
146 <Calendar size={14} />
147 {new Date(story.created_at).toLocaleDateString('en-US', { month: 'long', year: 'numeric' })}
148 </div>
149 </div>
150 </div>
151 </div>
152
153 <div className="grid grid-cols-1 lg:grid-cols-3 gap-8">
154 {/* Main content */}
155 <div className="lg:col-span-2 space-y-8">
156 {/* Action bar */}
157 <div className="flex items-center gap-3 flex-wrap">
[73b69b2]158 <LikeButton storyId={story.story_id} authorUserId={story.user_id} totalLikes={story.total_likes} onCountChange={setLiveLikes} />
[b62cefc]159 {currentUser && (
160 <Button variant="secondary" size="sm" onClick={() => setListModalOpen(true)}>
161 <BookmarkPlus size={14} />
162 Save to List
163 </Button>
164 )}
[7fbb91c]165 {(isOwner || isCollaborator) && (
[b62cefc]166 <Button variant="ghost" size="sm" onClick={() => navigate(`/writer/edit-story/${story.story_id}`)}>
167 Edit Story
168 </Button>
169 )}
170 </div>
171
172 {/* Chapters */}
173 <div>
174 <div className="flex items-center justify-between mb-4">
175 <h2 className="font-serif text-xl font-bold text-white">Chapters</h2>
[7fbb91c]176 {(isOwner || isCollaborator) && (
[b62cefc]177 <Button size="sm" onClick={() => navigate(`/writer/create-chapter/${story.story_id}`)}>
178 <Plus size={14} />
179 Add Chapter
180 </Button>
181 )}
182 </div>
183 <ChapterList chapters={storyChapters} storyId={story.story_id} />
184 </div>
185
186 {/* Comments */}
187 <div className="border-t border-slate-700 pt-8">
[73b69b2]188 <CommentSection storyId={story.story_id} authorUserId={story.user_id} onCountChange={setLiveComments} />
[b62cefc]189 </div>
190 </div>
191
192 {/* Sidebar */}
193 <div className="space-y-6">
194 {/* Story info */}
195 <div className="bg-slate-800 border border-slate-700 rounded-2xl p-5">
196 <h3 className="text-white font-semibold mb-4">Story Info</h3>
197 <div className="space-y-3 text-sm">
198 <div className="flex justify-between">
199 <span className="text-slate-400">Status</span>
200 <span className="text-white capitalize">{story.status}</span>
201 </div>
202 <div className="flex justify-between">
203 <span className="text-slate-400">Chapters</span>
204 <span className="text-white">{story.total_chapters}</span>
205 </div>
206 <div className="flex justify-between">
207 <span className="text-slate-400">Likes</span>
[73b69b2]208 <span className="text-white">{(liveLikes ?? story.total_likes).toLocaleString()}</span>
[b62cefc]209 </div>
210 <div className="flex justify-between">
211 <span className="text-slate-400">Views</span>
212 <span className="text-white">{story.total_views.toLocaleString()}</span>
213 </div>
214 <div className="flex justify-between">
215 <span className="text-slate-400">Comments</span>
[73b69b2]216 <span className="text-white">{liveComments ?? story.total_comments}</span>
[b62cefc]217 </div>
218 <div className="flex justify-between">
219 <span className="text-slate-400">Published</span>
220 <span className="text-white">{new Date(story.created_at).toLocaleDateString()}</span>
221 </div>
222 </div>
223 </div>
224
225 {/* Collaborators */}
226 {storyCollabs.length > 0 && (
227 <div className="bg-slate-800 border border-slate-700 rounded-2xl p-5">
228 <div className="flex items-center gap-2 mb-4">
229 <Users size={16} className="text-violet-400" />
230 <h3 className="text-white font-semibold">Collaborators</h3>
231 </div>
232 <div className="space-y-3">
233 {storyCollabs.map(c => (
234 <div key={c.collab_id} className="flex items-center gap-2">
235 <Avatar name={c.name} size="sm" />
236 <div>
237 <p className="text-white text-sm">@{c.username}</p>
238 <p className="text-slate-500 text-xs">{c.role}</p>
239 </div>
240 </div>
241 ))}
242 </div>
243 </div>
244 )}
245 </div>
246 </div>
247
248 {/* Reading list modal */}
249 <Modal isOpen={listModalOpen} onClose={() => setListModalOpen(false)} title="Save to Reading List">
250 <div className="space-y-3">
251 {myLists.length > 0 ? (
252 <>
253 <p className="text-slate-400 text-sm mb-3">Select a list:</p>
254 {myLists.map(list => (
255 <button
256 key={list.list_id}
257 onClick={() => handleAddToList(list.list_id)}
258 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"
259 >
260 <span className="text-white text-sm">{list.name}</span>
261 <span className="text-slate-500 text-xs">{list.stories.length} stories</span>
262 </button>
263 ))}
264 <div className="border-t border-slate-700 pt-3">
265 <p className="text-slate-400 text-sm mb-2">Or create a new list:</p>
266 <div className="flex gap-2">
267 <input
268 value={newListName}
269 onChange={e => setNewListName(e.target.value)}
270 placeholder="List name..."
271 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"
272 />
273 <Button size="sm" onClick={handleCreateList} disabled={!newListName.trim()}>
274 Create
275 </Button>
276 </div>
277 </div>
278 </>
279 ) : (
280 <div>
281 <p className="text-slate-400 text-sm mb-3">You don't have any reading lists yet. Create one:</p>
282 <div className="flex gap-2">
283 <input
284 value={newListName}
285 onChange={e => setNewListName(e.target.value)}
286 placeholder="My Favorites..."
287 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"
288 />
289 <Button size="sm" onClick={handleCreateList} disabled={!newListName.trim()}>
290 Create
291 </Button>
292 </div>
293 </div>
294 )}
295 </div>
296 </Modal>
297 </div>
298 )
299}
Note: See TracBrowser for help on using the repository browser.