making progress

This commit is contained in:
2026-05-07 02:06:54 +02:00
parent 7b7e2d399c
commit dea186c995
39 changed files with 2021 additions and 67 deletions
+109
View File
@@ -0,0 +1,109 @@
import { useState } from 'react'
import { useParams, Link } from 'react-router-dom'
import { useIssues, useCreateIssue, useCloseIssue, useReopenIssue } from '../api/queries/issues'
import { cn } from '../lib/utils'
import type { IssueState } from '../types/api'
export default function RepoIssuesPage() {
const { owner = '', repo = '' } = useParams<{ owner: string; repo: string }>()
const [state, setState] = useState<IssueState>('open')
const [showNew, setShowNew] = useState(false)
const { data: issues, isLoading } = useIssues(owner, repo, state)
const createIssue = useCreateIssue(owner, repo)
const closeIssue = useCloseIssue(owner, repo)
const reopenIssue = useReopenIssue(owner, repo)
const [title, setTitle] = useState('')
const [body, setBody] = useState('')
const handleCreate = async (e: React.FormEvent) => {
e.preventDefault()
if (!title.trim()) return
await createIssue.mutateAsync({ title: title.trim(), body })
setTitle(''); setBody(''); setShowNew(false)
}
return (
<div className="max-w-4xl mx-auto px-4 md:px-6 py-6">
<div className="flex items-center gap-1 text-sm mb-4">
<Link to={`/repos/${owner}/${repo}`} className="text-[#0052CC] hover:underline">{repo}</Link>
<span className="text-[#5E6C84]">/</span>
<span className="font-semibold text-[#172B4D]">Issues</span>
</div>
<div className="flex items-center justify-between mb-4">
<h1 className="text-xl font-semibold text-[#172B4D]">Issues</h1>
<button onClick={() => setShowNew(true)}
className="px-3 py-2 rounded bg-[#0052CC] text-white text-sm font-medium hover:bg-[#0065FF] min-h-[44px]">
New issue
</button>
</div>
{showNew && (
<form onSubmit={handleCreate} className="mb-6 p-5 border border-[#4C9AFF] rounded bg-white space-y-3">
<h2 className="text-sm font-semibold text-[#172B4D]">New Issue</h2>
<input value={title} onChange={e => setTitle(e.target.value)} placeholder="Title" required
className="w-full border border-[#DFE1E6] rounded px-3 py-2 text-sm focus:outline-none focus:border-[#4C9AFF]" />
<textarea value={body} onChange={e => setBody(e.target.value)} placeholder="Description (optional)" rows={4}
className="w-full border border-[#DFE1E6] rounded px-3 py-2 text-sm resize-none focus:outline-none focus:border-[#4C9AFF]" />
<div className="flex gap-2">
<button type="submit" disabled={createIssue.isPending || !title.trim()}
className="px-4 py-2 rounded bg-[#0052CC] text-white text-sm hover:bg-[#0065FF] disabled:opacity-50 min-h-[44px]">
{createIssue.isPending ? 'Submitting…' : 'Submit'}
</button>
<button type="button" onClick={() => setShowNew(false)}
className="px-4 py-2 rounded border border-[#DFE1E6] text-sm text-[#172B4D] hover:bg-[#F4F5F7] min-h-[44px]">
Cancel
</button>
</div>
</form>
)}
<div className="flex gap-1 mb-4 border-b border-[#DFE1E6]">
{(['open', 'closed'] as IssueState[]).map(s => {
const count = issues?.filter(i => i.state === s).length ?? 0
return (
<button key={s} onClick={() => setState(s)}
className={cn('px-4 py-2 text-sm font-medium capitalize border-b-2 -mb-px min-h-[44px]',
state === s ? 'border-[#0052CC] text-[#0052CC]' : 'border-transparent text-[#5E6C84] hover:text-[#172B4D]')}>
{s} {count > 0 && `(${count})`}
</button>
)
})}
</div>
{isLoading ? (
<p className="text-sm text-[#5E6C84] py-4">Loading</p>
) : !issues?.length ? (
<p className="text-sm text-[#5E6C84] py-8 text-center">No {state} issues.</p>
) : (
<div className="flex flex-col gap-2">
{issues.map(issue => (
<div key={issue.id}
className="flex items-start gap-3 p-4 border border-[#DFE1E6] rounded hover:bg-[#FAFBFC]">
<svg width="16" height="16" viewBox="0 0 16 16" fill={issue.state === 'open' ? '#00875A' : '#5E6C84'}
className="mt-0.5 shrink-0">
<path d="M8 1.5a6.5 6.5 0 1 0 0 13 6.5 6.5 0 0 0 0-13ZM0 8a8 8 0 1 1 16 0A8 8 0 0 1 0 8Zm9 3a1 1 0 1 1-2 0 1 1 0 0 1 2 0Zm-.25-6.25a.75.75 0 0 0-1.5 0v3.5a.75.75 0 0 0 1.5 0v-3.5Z"/>
</svg>
<div className="flex-1 min-w-0">
<p className="text-sm font-medium text-[#172B4D]">#{issue.number} {issue.title}</p>
<p className="text-xs text-[#5E6C84] mt-0.5">
opened by {issue.authorName} · {new Date(issue.createdAt).toLocaleDateString()}
</p>
</div>
<button
onClick={() => issue.state === 'open'
? closeIssue.mutate(issue.number)
: reopenIssue.mutate(issue.number)
}
className="text-xs px-3 py-1.5 rounded border border-[#DFE1E6] text-[#5E6C84] hover:bg-[#F4F5F7] shrink-0 min-h-[32px]">
{issue.state === 'open' ? 'Close' : 'Reopen'}
</button>
</div>
))}
</div>
)}
</div>
)
}