making progress
This commit is contained in:
@@ -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>
|
||||
)
|
||||
}
|
||||
Reference in New Issue
Block a user