Files
badminton-scoreboard/src/pages/TeamSelectionPage.tsx

161 lines
5.2 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import { Link } from 'react-router-dom'
import { getTeamDisplayName } from '../lib/match'
import type { LoadStatus, RoundGroup } from '../types'
type TeamSelectionPageProps = {
areaAInput: string
areaBInput: string
groups: RoundGroup[]
groupSource: 'idle' | 'db' | 'manual'
loadMessage: string
loadStatus: LoadStatus
selectedGroupId: number | null
targetDate: string
onAreaAInputChange: (value: string) => void
onAreaBInputChange: (value: string) => void
onGenerateManualGroups: () => void
onLoadGroupsFromDb: () => void
onSelectGroup: (groupId: number) => void
onTargetDateChange: (value: string) => void
onUseGroup: (groupId: number) => void
}
export function TeamSelectionPage({
areaAInput,
areaBInput,
groups,
groupSource,
loadMessage,
loadStatus,
selectedGroupId,
targetDate,
onAreaAInputChange,
onAreaBInputChange,
onGenerateManualGroups,
onLoadGroupsFromDb,
onSelectGroup,
onTargetDateChange,
onUseGroup,
}: TeamSelectionPageProps) {
const hasGroups = groups.length > 0
const showInlineStatus = loadStatus !== 'idle' && loadStatus !== 'loaded' && Boolean(loadMessage)
const sourceLabel =
groupSource === 'db' ? '資料庫載入' : groupSource === 'manual' ? '手動產生' : '尚未建立'
return (
<section className="page-grid">
{loadStatus === 'loaded' && loadMessage ? (
<div className="floating-status-bubble" role="status" aria-live="polite">
{loadMessage}
</div>
) : null}
<article className="panel">
<div className="selection-shell">
<div className="selection-toolbar">
<label className="field">
<span></span>
<input
type="date"
value={targetDate}
onChange={(event) => onTargetDateChange(event.target.value)}
/>
</label>
<div className="button-stack">
<button className="primary-button" type="button" onClick={onLoadGroupsFromDb}>
</button>
<button className="secondary-button" type="button" onClick={onGenerateManualGroups}>
</button>
</div>
</div>
{showInlineStatus ? (
<div className={`status-banner status-banner-${loadStatus}`}>{loadMessage}</div>
) : null}
<div className="double-grid">
<label className="field">
<span>A </span>
<textarea
placeholder="每行一位球員"
value={areaAInput}
onChange={(event) => onAreaAInputChange(event.target.value)}
/>
</label>
<label className="field">
<span>B </span>
<textarea
placeholder="每行一位球員"
value={areaBInput}
onChange={(event) => onAreaBInputChange(event.target.value)}
/>
</label>
</div>
<div className="selection-hint">
<span>{sourceLabel}</span>
</div>
</div>
</article>
<article className="panel">
<div className="group-head">
<div>
<p className="panel-kicker">Step 2</p>
<h2></h2>
</div>
{selectedGroupId ? <span className="winner-badge"> {selectedGroupId} </span> : null}
</div>
<div className="group-board">
{hasGroups ? (
groups.map((group) => (
<article
key={group.id}
className={`group-card ${selectedGroupId === group.id ? 'group-card-active' : ''}`}
>
<div className="group-head">
<div>
<p className="panel-kicker"> {group.id} </p>
<h3></h3>
</div>
<div className="group-actions">
<button
className="secondary-button"
type="button"
onClick={() => onSelectGroup(group.id)}
>
</button>
<Link className="primary-button inline-link" to="/scoreboard" onClick={() => onUseGroup(group.id)}>
</Link>
</div>
</div>
<div className="team-stage-grid">
{group.teams.map((team) => (
<article key={`${group.id}-${team.id}`} className="team-stage-card">
<span className="team-index"> {team.id}</span>
<div className="team-name">{getTeamDisplayName(team)}</div>
</article>
))}
</div>
</article>
))
) : (
<div className="empty-state">
<h3></h3>
<p> AB </p>
</div>
)}
</div>
</article>
</section>
)
}