import { useState, useEffect } from 'react' import { useParams, useNavigate } from 'react-router-dom' import api from '@/services/api' import type { Project, ProjectMember, Milestone } from '@/types' import dayjs from 'dayjs' export default function ProjectDetailPage() { const { id } = useParams() const navigate = useNavigate() const [project, setProject] = useState(null) const [members, setMembers] = useState([]) const [milestones, setMilestones] = useState([]) const [allProjects, setAllProjects] = useState([]) const [showAddMember, setShowAddMember] = useState(false) const [showAddMilestone, setShowAddMilestone] = useState(false) const [newMemberUserId, setNewMemberUserId] = useState(1) const [newMemberRole, setNewMemberRole] = useState('developer') const [newMilestoneTitle, setNewMilestoneTitle] = useState('') const [users, setUsers] = useState([]) const [roles, setRoles] = useState([]) const [editing, setEditing] = useState(false) const [editForm, setEditForm] = useState({ owner: '', repo: '', description: '', sub_projects: [] as string[], related_projects: [] as string[] }) useEffect(() => { api.get(`/projects/${id}`).then(({ data }) => { setProject(data) setEditForm({ owner: data.owner_name || data.owner || '', repo: data.repo || '', description: data.description || '', sub_projects: data.sub_projects || [], related_projects: data.related_projects || [], }) }) api.get(`/projects/${id}/members`).then(({ data }) => setMembers(data)) api.get(`/milestones?project_id=${id}`).then(({ data }) => setMilestones(data)) api.get('/projects').then(({ data }) => setAllProjects(data)) api.get('/users').then(r => setUsers(r.data)).catch(() => {}) api.get('/roles').then(r => setRoles(r.data)).catch(() => {}) api.get('/users').then(r => setUsers(r.data)).catch(() => {}) }, [id]) const handleMulti = (e: React.ChangeEvent, field: 'sub_projects' | 'related_projects') => { const values = Array.from(e.target.selectedOptions).map((o) => o.value) setEditForm({ ...editForm, [field]: values }) } const updateProject = async (e: React.FormEvent) => { e.preventDefault() const { data } = await api.patch(`/projects/${id}`, editForm) setProject(data) setEditing(false) } const addMember = async () => { if (!newMemberUserId) return await api.post(`/projects/${id}/members`, { user_id: newMemberUserId, role: newMemberRole }) setShowAddMember(false) api.get(`/projects/${id}/members`).then(({ data }) => setMembers(data)) } const removeMember = async (userId: number, role: string) => { // Prevent removing owner if (role === 'admin') { alert('Cannot remove project owner (admin)') return } if (!confirm('Remove this member?')) return await api.delete(`/projects/${id}/members/${userId}`) api.get(`/projects/${id}/members`).then(({ data }) => setMembers(data)) } const addMilestone = async () => { if (!newMilestoneTitle.trim()) return await api.post(`/projects/${id}/milestones`, { title: newMilestoneTitle, status: 'open' }) setShowAddMilestone(false) setNewMilestoneTitle('') api.get(`/projects/${id}/milestones`).then(({ data }) => setMilestones(data)).catch(() => {}) } const deleteProject = async () => { const confirmName = prompt(`Type the project name "${project?.name}" to confirm deletion:`) if (confirmName !== project?.name) { alert('Project name does not match. Deletion cancelled.') return } await api.delete(`/projects/${id}`) navigate('/projects') } if (!project) return
Loading...
const selectableProjects = allProjects.filter((p) => p.id !== project.id && p.project_code) return (
{editing ? (
{project.name}
{project.project_code && {project.project_code}} setEditForm({ ...editForm, description: e.target.value })} placeholder="Description" /> setEditForm({ ...editForm, repo: e.target.value })} placeholder="Repository URL" />
) : ( <>

๐Ÿ“ {project.name} {project.project_code && {project.project_code}}

{project.description || 'No description'}

{project.repo &&

๐Ÿ“ฆ {project.repo}

}
Owner: {project.owner_name || project.owner || "Unknown"}
)}

Members ({members.length})

{members.length > 0 ? (
{members.map((m) => ( {`User #${m.user_id} (${m.role})`} ))}
) : (

No members

)}

Milestones ({milestones.length})

{milestones.map((ms) => (
navigate(`/milestones/${ms.id}`)}> {ms.status} {ms.title} {ms.due_date && ยท Due {dayjs(ms.due_date).format('YYYY-MM-DD')}}
))} {milestones.length === 0 &&

No milestones

}
{showAddMember && (
setShowAddMember(false)}>
e.stopPropagation()}>

Add Member

)} {showAddMilestone && (
setShowAddMilestone(false)}>
e.stopPropagation()}>

New Milestone

setNewMilestoneTitle(e.target.value)} placeholder="Milestone title" />
)}
) }